mirror of
https://github.com/vdr-projects/vdr.git
synced 2025-03-01 10:50:46 +00:00
- Added Norwegian language texts (thanks to Jørgen Tvedt). - Increased the usleep value in cDvbOsd::Cmd() to 5000 in order to work on systems with the KURT/utime-patch (thanks to Guido Fiala). - Changed the check whether the driver is loaded in runvdr to check for the 'dvb' module (the last one loaded). - Fixed repeat function with LIRC (thanks to Stefan Huelswitt). - Increased the upper limit for the symbol rate to 30000 (thanks to Ulrich Röder). - Made the position of the channel display configurable (thanks to Stefan Huelswitt). - Made the width and height of the OSD configurable (thanks to Stefan Huelswitt). - DiSEqC support can now be generally enabled/disabled in the Setup menu. This may be necessary if your multiswitch gets irritated by the default DiSEqC codes '0' (thanks to Markus Lang). - Fixed replaying in case there is no index file. - Fixed jumping to an editing mark when replay has been paused. - Avoiding unnecessary code execution in the replay progress display (thanks to Guido Fiala). - When entering time values the digits that still have to be entered are now shown as '-' (as in "1-:--"). - When setting an editing mark while the progress display is not active, the display will now be turned on for a short while to indicate the successful setting of the mark. - Updated 'channels.conf' for Premiere World (thanks to Helmut Schächner). Check your timers if you use this channels.conf file, since the sequence of several PW channels has been changed. - Changed the color of "Info" messages to "black on green" and that of the confirmation messages (like "Delete...") to "black on yellow". - Fixed display with DEBUG_OSD (it still crashes sometimes, esp. when replaying, but I can't seem to find what causes this... any ideas anybody?). - Avoiding audio/video distortions in 'Transfer Mode' by no longer actually tuning the primary interface (which can't receive this channel, anyway). Apparently the driver gets irritated when the channel is switched and a replay session is started immediately after that. - Increased timeout until reporting "video data stream broken" when recording. - Explicitly switching back to the previously active channel after ending a replay session (to have it shown correctly in case it was in 'Transfer Mode').
208 lines
5.1 KiB
C
208 lines
5.1 KiB
C
/*
|
|
* videodir.c: Functions to maintain a distributed video directory
|
|
*
|
|
* See the main source file 'vdr.c' for copyright information and
|
|
* how to reach the author.
|
|
*
|
|
* $Id: videodir.c 1.5 2001/05/01 09:48:57 kls Exp $
|
|
*/
|
|
|
|
#include "videodir.h"
|
|
#include <ctype.h>
|
|
#include <errno.h>
|
|
#include <fcntl.h>
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <sys/stat.h>
|
|
#include <unistd.h>
|
|
#include "tools.h"
|
|
|
|
const char *VideoDirectory = "/video";
|
|
|
|
class cVideoDirectory {
|
|
private:
|
|
char *name, *stored, *adjusted;
|
|
int length, number, digits;
|
|
public:
|
|
cVideoDirectory(void);
|
|
~cVideoDirectory();
|
|
uint FreeMB(void);
|
|
const char *Name(void) { return name ? name : VideoDirectory; }
|
|
const char *Stored(void) { return stored; }
|
|
int Length(void) { return length; }
|
|
bool IsDistributed(void) { return name != NULL; }
|
|
bool Next(void);
|
|
void Store(void);
|
|
const char *Adjust(const char *FileName);
|
|
};
|
|
|
|
cVideoDirectory::cVideoDirectory(void)
|
|
{
|
|
length = strlen(VideoDirectory);
|
|
name = (VideoDirectory[length - 1] == '0') ? strdup(VideoDirectory) : NULL;
|
|
stored = adjusted = NULL;
|
|
number = -1;
|
|
digits = 0;
|
|
}
|
|
|
|
cVideoDirectory::~cVideoDirectory()
|
|
{
|
|
delete name;
|
|
delete stored;
|
|
delete adjusted;
|
|
}
|
|
|
|
uint cVideoDirectory::FreeMB(void)
|
|
{
|
|
return FreeDiskSpaceMB(name ? name : VideoDirectory);
|
|
}
|
|
|
|
bool cVideoDirectory::Next(void)
|
|
{
|
|
if (name) {
|
|
if (number < 0) {
|
|
int l = length;
|
|
while (l-- > 0 && isdigit(name[l]))
|
|
;
|
|
l++;
|
|
digits = length - l;
|
|
int n = atoi(&name[l]);
|
|
if (n == 0)
|
|
number = n;
|
|
else
|
|
return false; // base video directory must end with zero
|
|
}
|
|
if (++number > 0) {
|
|
char buf[16];
|
|
if (sprintf(buf, "%0*d", digits, number) == digits) {
|
|
strcpy(&name[length - digits], buf);
|
|
return DirectoryOk(name);
|
|
}
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
void cVideoDirectory::Store(void)
|
|
{
|
|
if (name) {
|
|
delete stored;
|
|
stored = strdup(name);
|
|
}
|
|
}
|
|
|
|
const char *cVideoDirectory::Adjust(const char *FileName)
|
|
{
|
|
if (stored) {
|
|
delete adjusted;
|
|
adjusted = strdup(FileName);
|
|
return strncpy(adjusted, stored, length);
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
int OpenVideoFile(const char *FileName, int Flags)
|
|
{
|
|
const char *ActualFileName = FileName;
|
|
|
|
// Incoming name must be in base video directory:
|
|
if (strstr(FileName, VideoDirectory) != FileName) {
|
|
esyslog(LOG_ERR, "ERROR: %s not in %s", FileName, VideoDirectory);
|
|
errno = ENOENT; // must set 'errno' - any ideas for a better value?
|
|
return -1;
|
|
}
|
|
// Are we going to create a new file?
|
|
if ((Flags & O_CREAT) != 0) {
|
|
cVideoDirectory Dir;
|
|
if (Dir.IsDistributed()) {
|
|
// Find the directory with the most free space:
|
|
uint MaxFree = Dir.FreeMB();
|
|
while (Dir.Next()) {
|
|
uint Free = FreeDiskSpaceMB(Dir.Name());
|
|
if (Free > MaxFree) {
|
|
Dir.Store();
|
|
MaxFree = Free;
|
|
}
|
|
}
|
|
if (Dir.Stored()) {
|
|
ActualFileName = Dir.Adjust(FileName);
|
|
if (!MakeDirs(ActualFileName, false))
|
|
return -1; // errno has been set by MakeDirs()
|
|
if (symlink(ActualFileName, FileName) < 0) {
|
|
LOG_ERROR_STR(FileName);
|
|
return -1;
|
|
}
|
|
ActualFileName = strdup(ActualFileName); // must survive Dir!
|
|
}
|
|
}
|
|
}
|
|
int Result = open(ActualFileName, Flags, S_IRUSR | S_IWUSR | S_IRGRP | S_IROTH);
|
|
if (ActualFileName != FileName)
|
|
delete ActualFileName;
|
|
return Result;
|
|
}
|
|
|
|
int CloseVideoFile(int FileHandle)
|
|
{
|
|
// just in case we ever decide to do something special when closing the file!
|
|
return close(FileHandle);
|
|
}
|
|
|
|
bool RenameVideoFile(const char *OldName, const char *NewName)
|
|
{
|
|
// Only the base video directory entry will be renamed, leaving the
|
|
// possible symlinks untouched. Going through all the symlinks and disks
|
|
// would be unnecessary work - maybe later...
|
|
if (rename(OldName, NewName) == -1) {
|
|
LOG_ERROR_STR(OldName);
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
bool RemoveVideoFile(const char *FileName)
|
|
{
|
|
return RemoveFileOrDir(FileName, true);
|
|
}
|
|
|
|
bool VideoFileSpaceAvailable(unsigned int SizeMB)
|
|
{
|
|
cVideoDirectory Dir;
|
|
if (Dir.IsDistributed()) {
|
|
if (Dir.FreeMB() >= SizeMB * 2) // base directory needs additional space
|
|
return true;
|
|
while (Dir.Next()) {
|
|
if (Dir.FreeMB() >= SizeMB)
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
return Dir.FreeMB() >= SizeMB;
|
|
}
|
|
|
|
const char *PrefixVideoFileName(const char *FileName, char Prefix)
|
|
{
|
|
static char *PrefixedName = NULL;
|
|
|
|
if (!PrefixedName || strlen(PrefixedName) <= strlen(FileName))
|
|
PrefixedName = (char *)realloc(PrefixedName, strlen(FileName) + 2);
|
|
if (PrefixedName) {
|
|
strcpy(PrefixedName, VideoDirectory);
|
|
char *p = PrefixedName + strlen(PrefixedName);
|
|
*p++ = '/';
|
|
*p++ = Prefix;
|
|
strcpy(p, FileName + strlen(VideoDirectory) + 1);
|
|
}
|
|
return PrefixedName;
|
|
}
|
|
|
|
void RemoveEmptyVideoDirectories(void)
|
|
{
|
|
cVideoDirectory Dir;
|
|
do {
|
|
RemoveEmptyDirectories(Dir.Name());
|
|
} while (Dir.Next());
|
|
}
|
|
|