Version 0.04

- Changed name from 'osm' to 'vdr' to avoid mixups with the 'oms' program that
  appears to be in use with DVD replay.
- Implemented a channel display in the top menu line.
- Implemented replay progress display (press "Ok" when replaying to bring it up).
- Implemented direct channel selecting by pressing the numeric keys.
- Added several 'const' keywords to please stricter compilers.
- The repeat function for the remote control no longer adapts dynamically
  to the timing of the RCU (this sometimes caused the repeat function to
  kick in too early).
- Channel selection is now blocked when recording or replaying.
- Improved process handling.
This commit is contained in:
Klaus Schmidinger 2000-04-24 18:00:00 +02:00
parent 37250a0568
commit 85b8e41e8b
26 changed files with 748 additions and 444 deletions

7
BUGS
View File

@ -14,10 +14,3 @@ Video Disk Recorder - Known Bugs
Haven't figured out yet how to ensure that it switches back to Haven't figured out yet how to ensure that it switches back to
the current channel. the current channel.
* Every now and then the on-screen display shows nothing but
"noise". If that occurs, I have to stop the 'osm' program
and do a 'make reload' for the card driver. After that it
works fine again.
Presumably this is a problem in the card driver or firmware?
Or could it be a problem with the hardware?
Does anybody else observe this?

18
HISTORY
View File

@ -1,5 +1,5 @@
Video Disk Recorder OSM Revision History Video Disk Recorder Revision History
---------------------------------------- ------------------------------------
2000-02-19: Version 0.01 (Initial revision). 2000-02-19: Version 0.01 (Initial revision).
@ -18,3 +18,17 @@ Video Disk Recorder OSM Revision History
able to store some 18 hours in full quality, so we don't really need that). able to store some 18 hours in full quality, so we don't really need that).
- Termination signals are now caught and the program cleans up before exiting. - Termination signals are now caught and the program cleans up before exiting.
- Support for CICAM. - Support for CICAM.
2000-04-24: Version 0.04
- Changed name from 'osm' to 'vdr' to avoid mixups with the 'oms' program that
appears to be in use with DVD replay.
- Implemented a channel display in the top menu line.
- Implemented replay progress display (press "Ok" when replaying to bring it up).
- Implemented direct channel selecting by pressing the numeric keys.
- Added several 'const' keywords to please stricter compilers.
- The repeat function for the remote control no longer adapts dynamically
to the timing of the RCU (this sometimes caused the repeat function to
kick in too early).
- Channel selection is now blocked when recording or replaying.
- Improved process handling.

96
INSTALL Normal file
View File

@ -0,0 +1,96 @@
Installation of the Video Disk Recorder
---------------------------------------
Compiling and running the program:
----------------------------------
Make sure the files from this package are located in a
directory that is "parallel" to the DVB directory of the
driver source for the Siemens DVB-S PCI card (refer to
http://linuxtv.org/dvb/siemens_dvb.html for more information
about that driver). For example, if the DVB driver was
extracted into the directory /home/kls/vdr/DVB, then this
package should be extracted into /home/kls/vdr/VDR.
This program requires the card driver version 0.04 or higher
to work properly.
After extracting the package, change into the VDR directory
and type 'make'. This should produce an executable file
named 'vdr', which can be run after the DVB driver has been
installed.
There are two macros you can use to customize the 'vdr' program
at compile time. Adding "DEBUG_REMOTE=1" to the 'make' call
will use the PC's keyboard as input device instead of the "Remote
Control Unit" (see http://www.cadsoft.de/people/kls/vdr/remote.htm).
Adding "DEBUG_OSD=1" will use the PC screen (or current window)
to display texts instead of the DVB card's on-screen display
interface. These modes are useful when testing new menus if you
only have a remote connection to the VDR (which, in my case, is
located in the living room and has neither a monitor nor a keyboard).
The video data directory:
-------------------------
All recordings are written into directories below "/video". Please
make sure this directory exists, and that the user who runs the 'vdr'
program has read and write access to that directory.
If you prefer a different location for your video files, you can change
the value of 'BaseDir' in recording.c.
Note that the file system need not be 64-bit proof, since the 'vdr'
program splits video files into chunks of about 1GB. You should use
a disk with several gigabytes of free space. One GB can store roughly
half an hour of video data.
Configuration files:
--------------------
There are three configuration files that hold information about
channels, remote control keys and timers. These files are currrently
assumed to be located in the directory from which the 'vdr' program
was started (this will become configurable later). The configuration
files can be edited with any text editor, or will be written by the
'vdr' program if any changes are made inside the on-screen menus.
The meaning of the data entries may still vary in future releases,
so for the moment please look at the source code (config.c) to see
the meaning of the various fields.
Learning the remote control keys:
---------------------------------
There is no default 'keys.conf' file, so if you compile the program
without 'DEBUG_REMOTE=1' you will have to go through a "teach-in"
session that allows the program to learn your remote control codes.
It will first attempt to determine the basic data transfer mode and
timing of your remote control unit, and then will ask you to press one
key after the other so that it can learn the various key codes. You will
at least need to provide an "Up" and a "Down" key, so that you can switch
channels. The rest of the key definitions is optional, but the more keys
you define, the more you will be able to navigate through the menus and
control recording/replaying.
If the program has been built with "DEBUG_REMOTE=1", it will use the
key configuration file 'keys-pc.conf', so that you won't loose data
when switching between normal and debug mode.
The default PC key assignments are:
Up, Down, Left, Right Crsr keys in numeric block
Menu '5' in numeric block
Ok Enter
Back Backspace
0..9 '0'..'9' in top row
Red, Green, Yellow, Blue 'F1'..'F4'
Record 'r'
Pause 'p'
Stop 's'
Begin 'B'
SearchForward 'f'
SearchBack 'b'
SkipForward 'PgDn' in numeric block
SkipBack 'PgUp' in numeric block
If you prefer different key assignments, simply delete the file
'keys-pc.conf' and restart 'vdr' to get into learning mode.

53
MANUAL
View File

@ -1,16 +1,55 @@
Video Disk Recorder User's Manual Video Disk Recorder User's Manual
--------------------------------- ---------------------------------
* Navigating through the On Screen Menus
The "Main" menu can be called up with the "Menu" key of your remote
control unit. The "Up" and "Down" keys are used to select a specific
item. The "Left" and "Right" keys can be used to change options, and
the numeric keys allow direct input of numeric data. The "Ok" key
confirms any changes (or switches to a channel in the "Channels" menu).
The "Back" key goes back one level in the menu structure, discarding
any changes that might have been made in the current menu.
In the "Timers" menu, the current timer can be enabled or disabled with
the "Right" or "Left" key, respectively (enabled timers are marked with ">").
"Ok" here opens the "Edit timer" menu.
Textual options, like channel names or recording file names, can be edited
by pressing the "Right" button (which puts brackets around the current
character as in "[R]TL"), selecting the desired character position with
"Left" and "Right", and changing the character with the "Up" and "Down"
keys. "Ok" then confirms the changes.
The "Red", "Green", "Yellow" and "Blue" buttons have special meanings
in the various menus and are listed at the bottom of the on-screen-display.
At any point in the menu system, pressing the "Menu" key again will
immediately leave the menu system.
* Selecting a Channel * Selecting a Channel
You can select a channel either by pressing the "Up" or "Down" key (while There are three ways to select a channel:
no On Screen Menu is displayed), or browsing through the channel list in
the menu and pressing "Ok" on the desired channel. 1. With no On Screen Menu displayed press the "Up" or "Down" key to switch
to the next higher or lower channel.
2. Press the "Menu" button to bring up the On Screen Menu, select "Channels"
and browse through the list with the "Up" and "Down" key; to switch to the
selected channel press "Ok".
2. Directly type in the channel number with the numeric keys ('0'..'9');
if no key is pressed for about half a second, the digits collected so
far will define the channel number.
After switching to a different channel the channel number and name, as well
as the current time are displayed at the top of the screen. This line
automatically goes away after about two seconds, or if any key is pressed.
To bring up the channel display without switching channels you can press
the "Ok" button.
* Instant Recording * Instant Recording
You can start recording the current channel by pressing the "Record" You can start recording the current channel by pressing the "Record"
button. This will create a timer event named "instant" that start button. This will create a timer event named "instant" that starts
at the current time and records for two hours. at the current time and records for two hours.
If you want to modify the recording time you need to edit the timer. If you want to modify the recording time you need to edit the timer.
Stop instant recording by disabling or deleting the timer. Stop instant recording by disabling or deleting the timer.
@ -23,6 +62,8 @@ Video Disk Recorder User's Manual
* Replay Control * Replay Control
The following keys have the listed meaning in Replay mode:
- "Begin" Positions to beginning of the recording and starts playback - "Begin" Positions to beginning of the recording and starts playback
from there. from there.
- "Pause" Halts playback at the current frame. Press again to continue - "Pause" Halts playback at the current frame. Press again to continue
@ -32,6 +73,10 @@ Video Disk Recorder User's Manual
- "Search" Runs playback forward or backward at a higher speed. Press - "Search" Runs playback forward or backward at a higher speed. Press
again to resume normal speed. again to resume normal speed.
- "Skip" Skips about 60 seconds forward or backward. - "Skip" Skips about 60 seconds forward or backward.
- "Ok" Brings up the replay progress display, which shows the date,
time and title of the recording, a progress bar and the
current and total time of the recording.
Press "Ok" again to turn off the progress display.
* Programming the Timer * Programming the Timer

View File

@ -1,12 +1,12 @@
# #
# Makefile for the On Screen Menu of the Video Disk Recorder # Makefile for the On Screen Menu of the Video Disk Recorder
# #
# See the main source file 'osm.c' for copyright information and # See the main source file 'vdr.c' for copyright information and
# how to reach the author. # how to reach the author.
# #
# $Id: Makefile 1.2 2000/03/05 13:51:57 kls Exp $ # $Id: Makefile 1.3 2000/04/24 09:44:10 kls Exp $
OBJS = config.o dvbapi.o interface.o menu.o osd.o recording.o remote.o tools.o osm.o OBJS = config.o dvbapi.o interface.o menu.o osd.o recording.o remote.o tools.o vdr.o
ifdef DEBUG_REMOTE ifdef DEBUG_REMOTE
DEFINES += -DDEBUG_REMOTE DEFINES += -DDEBUG_REMOTE
@ -19,20 +19,20 @@ endif
%.o: %.c %.o: %.c
g++ -g -O2 -Wall -c $(DEFINES) $< g++ -g -O2 -Wall -c $(DEFINES) $<
all: osm all: vdr
config.o : config.c config.h dvbapi.h interface.h tools.h config.o : config.c config.h dvbapi.h interface.h tools.h
dvbapi.o : dvbapi.c config.h dvbapi.h interface.h tools.h dvbapi.o : dvbapi.c config.h dvbapi.h interface.h tools.h
interface.o: interface.c config.h dvbapi.h interface.h remote.h tools.h interface.o: interface.c config.h dvbapi.h interface.h remote.h tools.h
menu.o : menu.c config.h dvbapi.h interface.h menu.h osd.h recording.h tools.h menu.o : menu.c config.h dvbapi.h interface.h menu.h osd.h recording.h tools.h
osd.o : osd.c config.h dvbapi.h interface.h osd.h tools.h osd.o : osd.c config.h dvbapi.h interface.h osd.h tools.h
osm.o : osm.c config.h dvbapi.h interface.h menu.h osd.h recording.h tools.h vdr.o : vdr.c config.h dvbapi.h interface.h menu.h osd.h recording.h tools.h
recording.o: recording.c config.h dvbapi.h interface.h recording.h tools.h recording.o: recording.c config.h dvbapi.h interface.h recording.h tools.h
remote.o : remote.c remote.h tools.h remote.o : remote.c remote.h tools.h
tools.o : tools.c tools.h tools.o : tools.c tools.h
osm: $(OBJS) vdr: $(OBJS)
g++ -g -O2 $(OBJS) -lncurses -o osm g++ -g -O2 $(OBJS) -lncurses -o vdr
clean: clean:
-rm $(OBJS) osm -rm $(OBJS) vdr

112
README
View File

@ -7,6 +7,12 @@ of the LinuxTV project (http://linuxtv.org).
For details about the "Video Disk Recorder" project please For details about the "Video Disk Recorder" project please
refer to http://www.cadsoft.de/people/kls/vdr. refer to http://www.cadsoft.de/people/kls/vdr.
There is also a remote control unit described on those
Web pages, which can be used within this program.
Please see the INSTALL file for details on how to install
this program on your computer.
The author can be contacted at kls@cadsoft.de. The author can be contacted at kls@cadsoft.de.
Yet another "set-top-box"? Yet another "set-top-box"?
@ -25,112 +31,6 @@ of commercial set-top-boxes usually are a lot more fancy than
the ones in this system, but here we have the full source code the ones in this system, but here we have the full source code
and can modify the menus in whatever way desired. and can modify the menus in whatever way desired.
Compiling and running the program:
----------------------------------
Make sure the files from this package are located in a
directory that is "parallel" to the DVB directory of the
driver source for the Siemens DVB-S PCI card (refer to
http://linuxtv.org/dvb/siemens_dvb.html for more information
about that driver). For example, if the DVB driver was
extracted into the directory /home/kls/vdr/DVB, then this
package should be extracted into /home/kls/vdr/OSM.
This program requires the card driver version 0.04 or higher
to work properly.
After extracting the package, change into the OSM directory
and type 'make'. This should produce an executable file
named 'osm', which can be run after the DVB driver has been
installed.
There are two macros you can use to customize the 'osm' program
at compile time. Adding "DEBUG_REMOTE=1" to the 'make' call
will use the PC's keyboard as input device instead of the "Remote
Control Unit" (see http://www.cadsoft.de/people/kls/vdr/remote.htm).
Adding "DEBUG_OSD=1" will use the PC screen (or current window)
to display texts instead of the DVB card's on-screen display
interface. These modes are useful when testing new menus if you
only have a remote connection to the VDR (which, in my case, is
located in the living room and has neither a monitor nor a keyboard).
Configuration files:
--------------------
There are three configuration files that hold information about
channels, remote control keys and timers. These files are currrently
assumed to be located in the directory from which the 'osm' program
was started (this will become configurable later). The configuration
files can be edited with any text editor, or will be written by the
'osm' program if any changes are made inside the on-screen menus.
The meaning of the data entries may still vary in future releases,
so for the moment please look at the source code (config.c) to see
the meaning of the various fields.
Learning the remote control keys:
---------------------------------
There is no default 'keys.conf' file, so if you compile the program
without 'DEBUG_REMOTE=1' you will have to go through a "teach-in"
session that allows the program to learn your remote control codes.
It will first attempt to determine the basic data transfer mode and
timing of your remote control unit, and then will ask you to press one
key after the other so that it can learn the various key codes. You will
at least need to provide an "Up" and a "Down" key, so that you can switch
channels. The rest of the key definitions is optional, but the more keys
you define, the more you will be able to navigate through the menus and
control recording/replaying.
If the program has been built with "DEBUG_REMOTE=1", it will use the
key configuration file 'keys-pc.conf', so that you won't loose data
when switching between normal and debug mode.
The default PC key assignments are:
Up, Down, Left, Right Crsr keys in numeric block
Menu '5' in numeric block
Ok Enter
Back Backspace
0..9 '0'..'9' in top row
Red, Green, Yellow, Blue 'F1'..'F4'
Record 'r'
Pause 'p'
Stop 's'
Begin 'B'
SearchForward 'f'
SearchBack 'b'
SkipForward 'PgDn' in numeric block
SkipBack 'PgUp' in numeric block
If you prefer different key assignments, simply delete the file
'keys-pc.conf' and restart 'osm' to get into learning mode.
Navigating through the On Screen Menus:
---------------------------------------
The "Main" menu can be called up with the "Menu" key of your remote
control unit. The "Up" and "Down" keys are used to select a specific
item. The "Left" and "Right" keys can be used to change options, and
the numeric keys allow direct input of numeric data. The "Ok" key
confirms any changes (or switches to a channel in the "Channels" menu).
The "Back" key goes back one level in the menu structure, discarding
any changes that might have been made in the current menu.
In the "Timers" menu, the current timer can be enabled or disabled with
the "Right" or "Left" key, respectively (enabled timers are marked with ">").
"Ok" here opens the "Edit timer" menu.
Textual options, like channel names or recording file names, can be edited
by pressing the "Right" button (which puts brackets around the current
character as in "[R]TL"), selecting the desired character position with
"Left" and "Right", and changing the character with the "Up" and "Down"
keys. "Ok" then confirms the changes.
The "Red", "Green", "Yellow" and "Blue" buttons have special meanings
in the various menus and are listed at the bottom of the on-screen-display.
At any point in the menu system, pressing the "Menu" key again will
immediately leave the menu system.
What do you think? What do you think?
------------------ ------------------

3
TODO
View File

@ -1,7 +1,6 @@
TODO list for the Video Disk Recorder project TODO list for the Video Disk Recorder project
--------------------------------------------- ---------------------------------------------
* Channel select via numeric keys.
* Make it work with two DVB-S PCI cards to allow simultaneous * Make it work with two DVB-S PCI cards to allow simultaneous
recording of one programme, while replaying another programme recording of one programme, while replaying another programme
(or maybe the same one, but time delayed). (or maybe the same one, but time delayed).
@ -10,7 +9,5 @@ TODO list for the Video Disk Recorder project
* Implement "on-disk editing" to allow "cutting out" of certain * Implement "on-disk editing" to allow "cutting out" of certain
scenes in order to archive them (or, reversely, cut out scenes in order to archive them (or, reversely, cut out
commercial breaks). commercial breaks).
* Implement on-screen display of replay progress (progress bar
and/or time index).
* Implement channel scanning. * Implement channel scanning.
* Better support for encrypted channels. * Better support for encrypted channels.

View File

@ -1,10 +1,10 @@
/* /*
* config.c: Configuration file handling * config.c: Configuration file handling
* *
* See the main source file 'osm.c' for copyright information and * See the main source file 'vdr.c' for copyright information and
* how to reach the author. * how to reach the author.
* *
* $Id: config.c 1.3 2000/04/15 12:48:00 kls Exp $ * $Id: config.c 1.5 2000/04/24 09:44:15 kls Exp $
*/ */
#include "config.h" #include "config.h"
@ -206,7 +206,6 @@ bool cChannel::Switch(void)
if (!DvbApi.Recording()) { if (!DvbApi.Recording()) {
isyslog(LOG_INFO, "switching to channel %d", Index() + 1); isyslog(LOG_INFO, "switching to channel %d", Index() + 1);
CurrentChannel = Index(); CurrentChannel = Index();
Interface.DisplayChannel(CurrentChannel + 1, name);
for (int i = 3; --i;) { for (int i = 3; --i;) {
if (DvbApi.SetChannel(frequency, polarization, diseqc, srate, vpid, apid, ca, pnr)) if (DvbApi.SetChannel(frequency, polarization, diseqc, srate, vpid, apid, ca, pnr))
return true; return true;

View File

@ -1,10 +1,10 @@
/* /*
* config.h: Configuration file handling * config.h: Configuration file handling
* *
* See the main source file 'osm.c' for copyright information and * See the main source file 'vdr.c' for copyright information and
* how to reach the author. * how to reach the author.
* *
* $Id: config.h 1.3 2000/04/15 12:44:23 kls Exp $ * $Id: config.h 1.4 2000/04/24 09:44:17 kls Exp $
*/ */
#ifndef __CONFIG_H #ifndef __CONFIG_H

171
dvbapi.c
View File

@ -1,21 +1,19 @@
/* /*
* dvbapi.c: Interface to the DVB driver * dvbapi.c: Interface to the DVB driver
* *
* See the main source file 'osm.c' for copyright information and * See the main source file 'vdr.c' for copyright information and
* how to reach the author. * how to reach the author.
* *
* $Id: dvbapi.c 1.3 2000/04/15 14:45:04 kls Exp $ * $Id: dvbapi.c 1.8 2000/04/24 15:30:35 kls Exp $
*/ */
#include "dvbapi.h" #include "dvbapi.h"
#include <errno.h> #include <errno.h>
#include <fcntl.h> #include <fcntl.h>
#include <signal.h>
#include <stdlib.h> #include <stdlib.h>
#include <sys/ioctl.h> #include <sys/ioctl.h>
#include <sys/stat.h> #include <sys/stat.h>
#include <sys/time.h> #include <sys/time.h>
#include <sys/wait.h>
#include <unistd.h> #include <unistd.h>
#include "interface.h" #include "interface.h"
#include "tools.h" #include "tools.h"
@ -59,7 +57,6 @@
#define RESUMEFILESUFFIX "/resume.vdr" #define RESUMEFILESUFFIX "/resume.vdr"
#define RECORDFILESUFFIX "/%03d.vdr" #define RECORDFILESUFFIX "/%03d.vdr"
#define RECORDFILESUFFIXLEN 20 // some additional bytes for safety... #define RECORDFILESUFFIXLEN 20 // some additional bytes for safety...
#define MAXPROCESSTIMEOUT 3 // seconds
// The number of frames to back up when resuming an interrupted replay session: // The number of frames to back up when resuming an interrupted replay session:
#define RESUMEBACKUP (10 * FRAMESPERSEC) #define RESUMEBACKUP (10 * FRAMESPERSEC)
@ -85,7 +82,7 @@ public:
int Last(void) { return last; } int Last(void) { return last; }
int GetResume(void) { return resume; } int GetResume(void) { return resume; }
bool StoreResume(int Index); bool StoreResume(int Index);
static char *Str(int Index); static char *Str(int Index, bool WithFrame = false);
}; };
cIndexFile::cIndexFile(const char *FileName, bool Record) cIndexFile::cIndexFile(const char *FileName, bool Record)
@ -256,7 +253,7 @@ bool cIndexFile::StoreResume(int Index)
return false; return false;
} }
char *cIndexFile::Str(int Index) char *cIndexFile::Str(int Index, bool WithFrame)
{ {
static char buffer[16]; static char buffer[16];
int f = (Index % FRAMESPERSEC) + 1; int f = (Index % FRAMESPERSEC) + 1;
@ -264,7 +261,7 @@ char *cIndexFile::Str(int Index)
int m = s / 60 % 60; int m = s / 60 % 60;
int h = s / 3600; int h = s / 3600;
s %= 60; s %= 60;
snprintf(buffer, sizeof(buffer), "%d:%02d:%02d.%02d", h, m, s, f); snprintf(buffer, sizeof(buffer), WithFrame ? "%d:%02d:%02d.%02d" : "%d:%02d:%02d", h, m, s, f);
return buffer; return buffer;
} }
@ -554,7 +551,7 @@ private:
public: public:
cRecordBuffer(int *InFile, const char *FileName); cRecordBuffer(int *InFile, const char *FileName);
virtual ~cRecordBuffer(); virtual ~cRecordBuffer();
int WriteWithTimeout(void); int WriteWithTimeout(bool EndIfEmpty = false);
}; };
cRecordBuffer::cRecordBuffer(int *InFile, const char *FileName) cRecordBuffer::cRecordBuffer(int *InFile, const char *FileName)
@ -720,13 +717,13 @@ int cRecordBuffer::Write(int Max)
return 0; return 0;
} }
int cRecordBuffer::WriteWithTimeout(void) int cRecordBuffer::WriteWithTimeout(bool EndIfEmpty)
{ {
int w, written = 0; int w, written = 0;
int t0 = time_ms(); int t0 = time_ms();
while ((w = Write()) > 0 && time_ms() - t0 < MAXRECORDWRITETIME) while ((w = Write()) > 0 && time_ms() - t0 < MAXRECORDWRITETIME)
written += w; written += w;
return w < 0 ? w : written; return w < 0 ? w : (written == 0 && EndIfEmpty ? -1 : written);
} }
// --- cReplayBuffer --------------------------------------------------------- // --- cReplayBuffer ---------------------------------------------------------
@ -752,6 +749,7 @@ public:
int Resume(void); int Resume(void);
bool Save(void); bool Save(void);
void SkipSeconds(int Seconds); void SkipSeconds(int Seconds);
void GetIndex(int &Current, int &Total);
}; };
cReplayBuffer::cReplayBuffer(int *OutFile, const char *FileName) cReplayBuffer::cReplayBuffer(int *OutFile, const char *FileName)
@ -840,7 +838,7 @@ void cReplayBuffer::SkipSeconds(int Seconds)
} }
Index += Seconds * FRAMESPERSEC; Index += Seconds * FRAMESPERSEC;
if (Index < 0) if (Index < 0)
Index = 1; Index = 1; // not '0', to allow GetNextIFrame() below to work!
uchar FileNumber; uchar FileNumber;
int FileOffset; int FileOffset;
if (index->GetNextIFrame(Index, false, &FileNumber, &FileOffset) >= 0) if (index->GetNextIFrame(Index, false, &FileNumber, &FileOffset) >= 0)
@ -849,6 +847,16 @@ void cReplayBuffer::SkipSeconds(int Seconds)
} }
} }
void cReplayBuffer::GetIndex(int &Current, int &Total)
{
if (index) {
Current = index->Get(fileNumber, fileOffset);
Total = index->Last();
}
else
Current = Total = -1;
}
void cReplayBuffer::SkipAudioBlocks(void) void cReplayBuffer::SkipAudioBlocks(void)
{ {
int Length; int Length;
@ -1001,8 +1009,10 @@ cDvbApi::cDvbApi(void)
memset(&colorPairs, 0, sizeof(colorPairs)); memset(&colorPairs, 0, sizeof(colorPairs));
start_color(); start_color();
leaveok(stdscr, TRUE); leaveok(stdscr, TRUE);
window = stdscr; window = NULL;
#endif #endif
lastProgress = -1;
replayTitle = NULL;
} }
cDvbApi::~cDvbApi() cDvbApi::~cDvbApi()
@ -1018,6 +1028,7 @@ cDvbApi::~cDvbApi()
endwin(); endwin();
#endif #endif
} }
delete replayTitle;
} }
#ifdef DEBUG_OSD #ifdef DEBUG_OSD
@ -1056,17 +1067,21 @@ void cDvbApi::Cmd(OSD_Command cmd, int color, int x0, int y0, int x1, int y1, co
void cDvbApi::Open(int w, int h) void cDvbApi::Open(int w, int h)
{ {
int d = (h < 0) ? MenuLines + h : 0;
h = abs(h);
cols = w; cols = w;
rows = h; rows = h;
#ifdef DEBUG_OSD #ifdef DEBUG_OSD
//XXX size... window = subwin(stdscr, h, w, d, 0);
syncok(window, TRUE);
#define B2C(b) (((b) * 1000) / 255) #define B2C(b) (((b) * 1000) / 255)
#define SETCOLOR(n, r, g, b, o) init_color(n, B2C(r), B2C(g), B2C(b)) #define SETCOLOR(n, r, g, b, o) init_color(n, B2C(r), B2C(g), B2C(b))
#else #else
w *= charWidth; w *= charWidth;
h *= lineHeight; h *= lineHeight;
int x = (720 - w) / 2; //TODO PAL vs. NTSC??? d *= lineHeight;
int y = (576 - h) / 2; int x = (720 - MenuColumns * charWidth) / 2; //TODO PAL vs. NTSC???
int y = (576 - MenuLines * lineHeight) / 2 + d;
Cmd(OSD_Open, 4, x, y, x + w - 1, y + h - 1); Cmd(OSD_Open, 4, x, y, x + w - 1, y + h - 1);
#define SETCOLOR(n, r, g, b, o) Cmd(OSD_SetColor, n, r, g, b, o) #define SETCOLOR(n, r, g, b, o) Cmd(OSD_SetColor, n, r, g, b, o)
#endif #endif
@ -1079,6 +1094,8 @@ void cDvbApi::Open(int w, int h)
SETCOLOR(clrCyan, 0x00, 0xFC, 0xFC, 255); SETCOLOR(clrCyan, 0x00, 0xFC, 0xFC, 255);
SETCOLOR(clrMagenta, 0xB0, 0x00, 0xFC, 255); SETCOLOR(clrMagenta, 0xB0, 0x00, 0xFC, 255);
SETCOLOR(clrWhite, 0xFC, 0xFC, 0xFC, 255); SETCOLOR(clrWhite, 0xFC, 0xFC, 0xFC, 255);
lastProgress = -1;
} }
void cDvbApi::Close(void) void cDvbApi::Close(void)
@ -1086,6 +1103,7 @@ void cDvbApi::Close(void)
#ifndef DEBUG_OSD #ifndef DEBUG_OSD
Cmd(OSD_Close); Cmd(OSD_Close);
#endif #endif
lastProgress = -1;
} }
void cDvbApi::Clear(void) void cDvbApi::Clear(void)
@ -1108,6 +1126,7 @@ void cDvbApi::Fill(int x, int y, int w, int h, eDvbColor color)
wmove(window, y + r, x); // ncurses wants 'y' before 'x'! wmove(window, y + r, x); // ncurses wants 'y' before 'x'!
whline(window, ' ', w); whline(window, ' ', w);
} }
wsyncup(window); // shouldn't be necessary because of 'syncok()', but w/o it doesn't work
#else #else
Cmd(OSD_FillBlock, color, x * charWidth, y * lineHeight, (x + w) * charWidth - 1, (y + h) * lineHeight - 1); Cmd(OSD_FillBlock, color, x * charWidth, y * lineHeight, (x + w) * charWidth - 1, (y + h) * lineHeight - 1);
#endif #endif
@ -1131,6 +1150,52 @@ void cDvbApi::Text(int x, int y, const char *s, eDvbColor colorFg, eDvbColor col
#endif #endif
} }
bool cDvbApi::ShowProgress(bool Initial)
{
int Current, Total;
if (GetIndex(&Current, &Total)) {
if (Initial) {
if (replayTitle)
Text(0, 0, replayTitle);
Text(-7, 2, cIndexFile::Str(Total));
}
#ifdef DEBUG_OSD
int p = cols * Current / Total;
Fill(0, 1, p, 1, clrGreen);
Fill(p, 1, cols - p, 1, clrWhite);
#else
int w = cols * charWidth;
int p = w * Current / Total;
if (p != lastProgress) {
int y1 = 1 * lineHeight;
int y2 = 2 * lineHeight - 1;
int x1, x2;
eDvbColor color;
if (lastProgress < p) {
x1 = lastProgress + 1;
x2 = p;
if (p >= w)
p = w - 1;
color = clrGreen;
}
else {
x1 = p + 1;
x2 = lastProgress;
color = clrWhite;
}
if (lastProgress < 0)
Cmd(OSD_FillBlock, clrWhite, 0, y1, w - 1, y2);
Cmd(OSD_FillBlock, color, x1, y1, x2, y2);
lastProgress = p;
}
#endif
Text(0, 2, cIndexFile::Str(Current));
return true;
}
return false;
}
bool cDvbApi::SetChannel(int FrequencyMHz, char Polarization, int Diseqc, int Srate, int Vpid, int Apid, int Ca, int Pnr) bool cDvbApi::SetChannel(int FrequencyMHz, char Polarization, int Diseqc, int Srate, int Vpid, int Apid, int Ca, int Pnr)
{ {
if (videoDev >= 0) { if (videoDev >= 0) {
@ -1160,35 +1225,17 @@ bool cDvbApi::SetChannel(int FrequencyMHz, char Polarization, int Diseqc, int Sr
return false; return false;
} }
void cDvbApi::KillProcess(pid_t pid)
{
pid_t Pid2Wait4 = pid;
for (time_t t0 = time(NULL); time(NULL) - t0 < MAXPROCESSTIMEOUT; ) {
int status;
pid_t pid = waitpid(Pid2Wait4, &status, WNOHANG);
if (pid < 0) {
if (errno != ECHILD)
LOG_ERROR;
return;
}
if (pid == Pid2Wait4)
return;
}
esyslog(LOG_ERR, "ERROR: process %d won't end (waited %d seconds) - terminating it...", Pid2Wait4, MAXPROCESSTIMEOUT);
if (kill(Pid2Wait4, SIGTERM) < 0) {
esyslog(LOG_ERR, "ERROR: process %d won't terminate (%s) - killing it...", Pid2Wait4, strerror(errno));
if (kill(Pid2Wait4, SIGKILL) < 0)
esyslog(LOG_ERR, "ERROR: process %d won't die (%s) - giving up", Pid2Wait4, strerror(errno));
}
}
bool cDvbApi::Recording(void) bool cDvbApi::Recording(void)
{ {
if (pidRecord && !CheckProcess(pidRecord))
pidRecord = 0;
return pidRecord; return pidRecord;
} }
bool cDvbApi::Replaying(void) bool cDvbApi::Replaying(void)
{ {
if (pidReplay && !CheckProcess(pidReplay))
pidReplay = 0;
return pidReplay; return pidReplay;
} }
@ -1239,6 +1286,7 @@ bool cDvbApi::StartRecord(const char *FileName)
dsyslog(LOG_INFO, "start recording process (pid=%d)", getpid()); dsyslog(LOG_INFO, "start recording process (pid=%d)", getpid());
isMainProcess = false; isMainProcess = false;
bool DataStreamBroken = false;
int fromMain = toRecordPipe[0]; int fromMain = toRecordPipe[0];
int toMain = fromRecordPipe[1]; int toMain = fromRecordPipe[1];
cRecordBuffer *Buffer = new cRecordBuffer(&videoDev, FileName); cRecordBuffer *Buffer = new cRecordBuffer(&videoDev, FileName);
@ -1251,21 +1299,26 @@ bool cDvbApi::StartRecord(const char *FileName)
struct timeval timeout; struct timeval timeout;
timeout.tv_sec = 1; timeout.tv_sec = 1;
timeout.tv_usec = 0; timeout.tv_usec = 0;
bool ForceEnd = false;
if (select(FD_SETSIZE, &set, NULL, NULL, &timeout) > 0) { if (select(FD_SETSIZE, &set, NULL, NULL, &timeout) > 0) {
if (FD_ISSET(videoDev, &set)) { if (FD_ISSET(videoDev, &set)) {
if (Buffer->Read() < 0) if (Buffer->Read() < 0)
break; break;
DataStreamBroken = false;
} }
if (FD_ISSET(fromMain, &set)) { if (FD_ISSET(fromMain, &set)) {
switch (readchar(fromMain)) { switch (readchar(fromMain)) {
case dvbStop: Buffer->Stop(); break; case dvbStop: Buffer->Stop();
break; ForceEnd = DataStreamBroken;
break;
} }
} }
} }
else else {
DataStreamBroken = true;
esyslog(LOG_ERR, "ERROR: video data stream broken"); esyslog(LOG_ERR, "ERROR: video data stream broken");
if (Buffer->WriteWithTimeout() < 0) }
if (Buffer->WriteWithTimeout(ForceEnd) < 0)
break; break;
} }
delete Buffer; delete Buffer;
@ -1309,7 +1362,7 @@ void cDvbApi::SetReplayMode(int Mode)
} }
} }
bool cDvbApi::StartReplay(const char *FileName) bool cDvbApi::StartReplay(const char *FileName, const char *Title)
{ {
if (Recording()) { if (Recording()) {
esyslog(LOG_ERR, "ERROR: StartReplay() called while recording - ignored!"); esyslog(LOG_ERR, "ERROR: StartReplay() called while recording - ignored!");
@ -1318,6 +1371,13 @@ bool cDvbApi::StartReplay(const char *FileName)
StopReplay(); StopReplay();
if (videoDev >= 0) { if (videoDev >= 0) {
lastProgress = -1;
delete replayTitle;
if (Title) {
if ((replayTitle = strdup(Title)) == NULL)
esyslog(LOG_ERR, "ERROR: StartReplay: can't copy title '%s'", Title);
}
// Check FileName: // Check FileName:
if (!FileName) { if (!FileName) {
@ -1361,7 +1421,7 @@ bool cDvbApi::StartReplay(const char *FileName)
bool FastRewind = false; bool FastRewind = false;
int ResumeIndex = Buffer->Resume(); int ResumeIndex = Buffer->Resume();
if (ResumeIndex >= 0) if (ResumeIndex >= 0)
isyslog(LOG_INFO, "resuming replay at index %d (%s)", ResumeIndex, cIndexFile::Str(ResumeIndex)); isyslog(LOG_INFO, "resuming replay at index %d (%s)", ResumeIndex, cIndexFile::Str(ResumeIndex, true));
for (;;) { for (;;) {
if (Buffer->Read() < 0) if (Buffer->Read() < 0)
break; break;
@ -1405,6 +1465,12 @@ bool cDvbApi::StartReplay(const char *FileName)
Buffer->SkipSeconds(Seconds); Buffer->SkipSeconds(Seconds);
} }
} }
case dvbGetIndex: {
int Current, Total;
Buffer->GetIndex(Current, Total);
writeint(toMain, Current);
writeint(toMain, Total);
}
break; break;
} }
} }
@ -1470,3 +1536,20 @@ void cDvbApi::Skip(int Seconds)
} }
} }
bool cDvbApi::GetIndex(int *Current, int *Total)
{
if (pidReplay) {
int total;
purge(fromReplay);
writechar(toReplay, dvbGetIndex);
if (readint(fromReplay, *Current) && readint(fromReplay, total)) {
if (Total)
*Total = total;
}
else
*Current = -1;
return *Current >= 0;
}
return false;
}

View File

@ -1,10 +1,10 @@
/* /*
* dvbapi.h: Interface to the DVB driver * dvbapi.h: Interface to the DVB driver
* *
* See the main source file 'osm.c' for copyright information and * See the main source file 'vdr.c' for copyright information and
* how to reach the author. * how to reach the author.
* *
* $Id: dvbapi.h 1.3 2000/04/15 13:36:10 kls Exp $ * $Id: dvbapi.h 1.8 2000/04/24 15:31:07 kls Exp $
*/ */
#ifndef __DVBAPI_H #ifndef __DVBAPI_H
@ -21,6 +21,9 @@ typedef unsigned char __u8;
#include <stdio.h> #include <stdio.h>
#include "../DVB/driver/dvb.h" #include "../DVB/driver/dvb.h"
#define MenuLines 15
#define MenuColumns 40
enum eDvbColor { clrBackground, enum eDvbColor { clrBackground,
#ifndef DEBUG_OSD #ifndef DEBUG_OSD
clrOBSOLETE, //FIXME apparently color '1' can't be used as FgColor with e.g. clrRed as BgColor??? clrOBSOLETE, //FIXME apparently color '1' can't be used as FgColor with e.g. clrRed as BgColor???
@ -66,6 +69,14 @@ public:
void ClrEol(int x, int y, eDvbColor color = clrBackground); void ClrEol(int x, int y, eDvbColor color = clrBackground);
void Text(int x, int y, const char *s, eDvbColor colorFg = clrWhite, eDvbColor colorBg = clrBackground); void Text(int x, int y, const char *s, eDvbColor colorFg = clrWhite, eDvbColor colorBg = clrBackground);
// Progress Display facilities
private:
int lastProgress;
char *replayTitle;
public:
bool ShowProgress(bool Initial = false);
// Channel facilities // Channel facilities
bool SetChannel(int FrequencyMHz, char Polarization, int Diseqc, int Srate, int Vpid, int Apid, int Ca, int Pnr); bool SetChannel(int FrequencyMHz, char Polarization, int Diseqc, int Srate, int Vpid, int Apid, int Ca, int Pnr);
@ -78,13 +89,13 @@ private:
dvbFastForward, dvbFastForward,
dvbFastRewind, dvbFastRewind,
dvbSkip, dvbSkip,
dvbGetIndex,
}; };
bool isMainProcess; bool isMainProcess;
pid_t pidRecord, pidReplay; pid_t pidRecord, pidReplay;
int fromRecord, toRecord; int fromRecord, toRecord;
int fromReplay, toReplay; int fromReplay, toReplay;
void SetReplayMode(int Mode); void SetReplayMode(int Mode);
void KillProcess(pid_t pid);
public: public:
bool Recording(void); bool Recording(void);
// Returns true if we are currently recording. // Returns true if we are currently recording.
@ -102,10 +113,11 @@ public:
// returned. // returned.
void StopRecord(void); void StopRecord(void);
// Stops the current recording session (if any). // Stops the current recording session (if any).
bool StartReplay(const char *FileName); bool StartReplay(const char *FileName, const char *Title = NULL);
// Starts replaying the given file. // Starts replaying the given file.
// If there is already a replay session active, it will be stopped // If there is already a replay session active, it will be stopped
// and the new file will be played back. // and the new file will be played back.
// If provided Title will be used in the progress display.
void StopReplay(void); void StopReplay(void);
// Stops the current replay session (if any). // Stops the current replay session (if any).
void PauseReplay(void); void PauseReplay(void);
@ -119,6 +131,7 @@ public:
// The sign of 'Seconds' determines the direction in which to skip. // The sign of 'Seconds' determines the direction in which to skip.
// Use a very large negative value to go all the way back to the // Use a very large negative value to go all the way back to the
// beginning of the recording. // beginning of the recording.
bool GetIndex(int *Current, int *Total = NULL);
}; };
#endif //__DVBAPI_H #endif //__DVBAPI_H

View File

@ -1,19 +1,16 @@
/* /*
* interface.c: Abstract user interface layer * interface.c: Abstract user interface layer
* *
* See the main source file 'osm.c' for copyright information and * See the main source file 'vdr.c' for copyright information and
* how to reach the author. * how to reach the author.
* *
* $Id: interface.c 1.3 2000/04/15 17:38:11 kls Exp $ * $Id: interface.c 1.6 2000/04/24 09:44:23 kls Exp $
*/ */
#include "interface.h" #include "interface.h"
#include <unistd.h> #include <unistd.h>
#include "remote.h" #include "remote.h"
#define MenuLines 15
#define MenuColumns 40
#ifndef DEBUG_REMOTE #ifndef DEBUG_REMOTE
cRcIo RcIo("/dev/ttyS1"); cRcIo RcIo("/dev/ttyS1");
#endif #endif
@ -26,6 +23,7 @@ cInterface::cInterface(void)
{ {
open = 0; open = 0;
cols[0] = 0; cols[0] = 0;
keyFromWait = kNone;
} }
void cInterface::Init(void) void cInterface::Init(void)
@ -35,10 +33,10 @@ void cInterface::Init(void)
#endif #endif
} }
void cInterface::Open(void) void cInterface::Open(int NumCols, int NumLines)
{ {
if (!open++) if (!open++)
DvbApi.Open(MenuColumns, MenuLines); DvbApi.Open(NumCols, NumLines);
} }
void cInterface::Close(void) void cInterface::Close(void)
@ -49,35 +47,45 @@ void cInterface::Close(void)
DvbApi.Close(); DvbApi.Close();
} }
unsigned int cInterface::GetCh(void) unsigned int cInterface::GetCh(bool Wait)
{ {
#ifdef DEBUG_REMOTE #ifdef DEBUG_REMOTE
timeout(Wait ? 1000 :10);
int c = getch(); int c = getch();
return (c > 0) ? c : 0; return (c > 0) ? c : 0;
#else #else
//XXX #ifdef DEBUG_OSD #ifdef DEBUG_OSD
//XXX wrefresh(window);//XXX timeout(0);
//XXX #endif getch(); // just to make 'ncurses' display the window:
unsigned int Command; #endif
return RcIo.GetCommand(&Command) ? Command : 0; if (Wait || RcIo.InputAvailable()) {
unsigned int Command;
return RcIo.GetCommand(&Command, NULL) ? Command : 0;
}
return 0;
#endif #endif
} }
eKeys cInterface::GetKey(void) eKeys cInterface::GetKey(bool Wait)
{ {
return Keys.Get(GetCh()); eKeys Key = keyFromWait != kNone ? keyFromWait : Keys.Get(GetCh(Wait));
keyFromWait = kNone;
return Key;
} }
eKeys cInterface::Wait(int Seconds) eKeys cInterface::Wait(int Seconds, bool KeepChar)
{ {
int t0 = time_ms(); int t0 = time_ms();
eKeys Key = kNone;
while (time_ms() - t0 < Seconds * 1000) { while (time_ms() - t0 < Seconds * 1000) {
eKeys Key = GetKey(); Key = GetKey();
if (Key != kNone) if (Key != kNone)
return Key; break;
} }
return kNone; if (KeepChar)
keyFromWait = Key;
return Key;
} }
void cInterface::Clear(void) void cInterface::Clear(void)
@ -312,8 +320,20 @@ void cInterface::LearnKeys(void)
void cInterface::DisplayChannel(int Number, const char *Name) void cInterface::DisplayChannel(int Number, const char *Name)
{ {
//TODO
#ifndef DEBUG_REMOTE #ifndef DEBUG_REMOTE
RcIo.Number(Number); RcIo.Number(Number);
#endif #endif
if (Name) {
Open(MenuColumns, 1);
char buffer[MenuColumns + 1];
snprintf(buffer, sizeof(buffer), "%d %s", Number, Name ? Name : "");
Write(0, 0, buffer);
time_t t = time(NULL);
struct tm *now = localtime(&t);
snprintf(buffer, sizeof(buffer), "%02d:%02d", now->tm_hour, now->tm_min);
Write(-5, 0, buffer);
if (Wait(2, true) == kOk)
GetKey();
Close();
}
} }

View File

@ -1,10 +1,10 @@
/* /*
* interface.h: Abstract user interface layer * interface.h: Abstract user interface layer
* *
* See the main source file 'osm.c' for copyright information and * See the main source file 'vdr.c' for copyright information and
* how to reach the author. * how to reach the author.
* *
* $Id: interface.h 1.3 2000/03/19 14:03:28 kls Exp $ * $Id: interface.h 1.7 2000/04/24 09:44:25 kls Exp $
*/ */
#ifndef __INTERFACE_H #ifndef __INTERFACE_H
@ -19,16 +19,17 @@ public:
private: private:
int open; int open;
int cols[MaxCols]; int cols[MaxCols];
unsigned int GetCh(void); eKeys keyFromWait;
unsigned int GetCh(bool Wait = true);
void QueryKeys(void); void QueryKeys(void);
void HelpButton(int Index, const char *Text, eDvbColor FgColor, eDvbColor BgColor); void HelpButton(int Index, const char *Text, eDvbColor FgColor, eDvbColor BgColor);
eKeys Wait(int Seconds = 1); eKeys Wait(int Seconds = 1, bool KeepChar = false);
public: public:
cInterface(void); cInterface(void);
void Init(void); void Init(void);
void Open(void); void Open(int NumCols = MenuColumns, int NumLines = MenuLines);
void Close(void); void Close(void);
eKeys GetKey(void); eKeys GetKey(bool Wait = true);
void Clear(void); void Clear(void);
void ClearEol(int x, int y, eDvbColor Color = clrBackground); void ClearEol(int x, int y, eDvbColor Color = clrBackground);
void SetCols(int *c); void SetCols(int *c);
@ -41,7 +42,7 @@ public:
bool Confirm(const char *s); bool Confirm(const char *s);
void Help(const char *Red, const char *Green = NULL, const char *Yellow = NULL, const char *Blue = NULL); void Help(const char *Red, const char *Green = NULL, const char *Yellow = NULL, const char *Blue = NULL);
void LearnKeys(void); void LearnKeys(void);
void DisplayChannel(int Number, const char *Name); void DisplayChannel(int Number, const char *Name = NULL);
}; };
extern cInterface Interface; extern cInterface Interface;

52
menu.c
View File

@ -1,10 +1,10 @@
/* /*
* menu.c: The actual menu implementations * menu.c: The actual menu implementations
* *
* See the main source file 'osm.c' for copyright information and * See the main source file 'vdr.c' for copyright information and
* how to reach the author. * how to reach the author.
* *
* $Id: menu.c 1.3 2000/04/15 15:07:36 kls Exp $ * $Id: menu.c 1.8 2000/04/24 15:32:11 kls Exp $
*/ */
#include "menu.h" #include "menu.h"
@ -428,7 +428,7 @@ void cMenuEditStrItem::Set(void)
char buf[1000]; char buf[1000];
if (pos >= 0) { if (pos >= 0) {
strncpy(buf, value, pos); strncpy(buf, value, pos);
char *s = value[pos] != ' ' ? value + pos + 1 : ""; const char *s = value[pos] != ' ' ? value + pos + 1 : "";
snprintf(buf + pos, sizeof(buf) - pos - 2, "[%c]%s", *(value + pos), s); snprintf(buf + pos, sizeof(buf) - pos - 2, "[%c]%s", *(value + pos), s);
SetValue(buf); SetValue(buf);
} }
@ -915,16 +915,7 @@ cMenuRecordingItem::cMenuRecordingItem(cRecording *Recording)
void cMenuRecordingItem::Set(void) void cMenuRecordingItem::Set(void)
{ {
char *buffer = NULL; SetText(recording->Title('\t'));
struct tm *t = localtime(&recording->start);
asprintf(&buffer, "%02d.%02d.%04d\t%02d:%02d\t%s",
t->tm_mday,
t->tm_mon + 1,
t->tm_year + 1900,
t->tm_hour,
t->tm_min,
recording->name);
SetText(buffer, false);
} }
// --- cMenuRecordings ------------------------------------------------------- // --- cMenuRecordings -------------------------------------------------------
@ -957,7 +948,7 @@ eOSState cMenuRecordings::Play(void)
cMenuRecordingItem *ri = (cMenuRecordingItem *)Get(Current()); cMenuRecordingItem *ri = (cMenuRecordingItem *)Get(Current());
if (ri) { if (ri) {
//XXX what if this recording's file is currently in use??? //XXX what if this recording's file is currently in use???
if (ri->recording->Play()) if (DvbApi.StartReplay(ri->recording->FileName(), ri->recording->Title()))
return osEnd; return osEnd;
} }
return osContinue; return osContinue;
@ -1022,3 +1013,36 @@ eOSState cMenuMain::ProcessKey(eKeys Key)
return state; return state;
} }
// --- cReplayDisplay --------------------------------------------------------
cReplayDisplay::cReplayDisplay(void)
{
Interface.Open(MenuColumns, -3);
shown = DvbApi.ShowProgress(true);
}
cReplayDisplay::~cReplayDisplay()
{
Interface.Close();
}
eKeys cReplayDisplay::ProcessKey(eKeys Key)
{
if (!DvbApi.Replaying())
return kOk; // will turn off replay display
shown = DvbApi.ShowProgress(!shown);
switch (Key) {
case kBegin:
case kPause:
case kStop:
case kSearchBack:
case kSearchForward:
case kSkipBack:
case kSkipForward: break; // will be done in main loop
case kMenu: break; // allow direct switching to menu
case kOk: break; // switches off replay display
default: Key = kNone; // ignore anything not explicitly known here
}
return Key;
}

13
menu.h
View File

@ -1,10 +1,10 @@
/* /*
* menu.h: The actual menu implementations * menu.h: The actual menu implementations
* *
* See the main source file 'osm.c' for copyright information and * See the main source file 'vdr.c' for copyright information and
* how to reach the author. * how to reach the author.
* *
* $Id: menu.h 1.2 2000/03/05 10:57:27 kls Exp $ * $Id: menu.h 1.5 2000/04/24 15:31:53 kls Exp $
*/ */
#ifndef _MENU_H #ifndef _MENU_H
@ -18,4 +18,13 @@ public:
virtual eOSState ProcessKey(eKeys Key); virtual eOSState ProcessKey(eKeys Key);
}; };
class cReplayDisplay {
private:
bool shown;
public:
cReplayDisplay(void);
~cReplayDisplay();
eKeys ProcessKey(eKeys Key);
};
#endif //_MENU_H #endif //_MENU_H

6
osd.c
View File

@ -1,10 +1,10 @@
/* /*
* osd.c: Abstract On Screen Display layer * osd.c: Abstract On Screen Display layer
* *
* See the main source file 'osm.c' for copyright information and * See the main source file 'vdr.c' for copyright information and
* how to reach the author. * how to reach the author.
* *
* $Id: osd.c 1.2 2000/02/27 17:23:07 kls Exp $ * $Id: osd.c 1.4 2000/04/24 09:44:31 kls Exp $
*/ */
#include "osd.h" #include "osd.h"
@ -35,7 +35,7 @@ cOsdItem::~cOsdItem()
delete text; delete text;
} }
void cOsdItem::SetText(char *Text, bool Copy) void cOsdItem::SetText(const char *Text, bool Copy)
{ {
delete text; delete text;
text = Copy ? strdup(Text) : Text; text = Copy ? strdup(Text) : Text;

10
osd.h
View File

@ -1,10 +1,10 @@
/* /*
* osd.h: Abstract On Screen Display layer * osd.h: Abstract On Screen Display layer
* *
* See the main source file 'osm.c' for copyright information and * See the main source file 'vdr.c' for copyright information and
* how to reach the author. * how to reach the author.
* *
* $Id: osd.h 1.2 2000/03/05 11:33:11 kls Exp $ * $Id: osd.h 1.4 2000/04/24 09:44:32 kls Exp $
*/ */
#ifndef __OSD_H #ifndef __OSD_H
@ -28,7 +28,7 @@ enum eOSState { osUnknown,
class cOsdItem : public cListObject { class cOsdItem : public cListObject {
private: private:
char *text; const char *text;
int offset; int offset;
eOSState state; eOSState state;
protected: protected:
@ -37,8 +37,8 @@ public:
cOsdItem(eOSState State = osUnknown); cOsdItem(eOSState State = osUnknown);
cOsdItem(char *Text, eOSState State = osUnknown); cOsdItem(char *Text, eOSState State = osUnknown);
virtual ~cOsdItem(); virtual ~cOsdItem();
void SetText(char *Text, bool Copy = true); void SetText(const char *Text, bool Copy = true);
char *Text(void) { return text; } const char *Text(void) { return text; }
void Display(int Offset = -1, bool Current = false); void Display(int Offset = -1, bool Current = false);
virtual void Set(void) {} virtual void Set(void) {}
virtual eOSState ProcessKey(eKeys Key); virtual eOSState ProcessKey(eKeys Key);

150
osm.c
View File

@ -1,150 +0,0 @@
/*
* osm.c: On Screen Menu for the Video Disk Recorder
*
* Copyright (C) 2000 Klaus Schmidinger
*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU General Public License
* as published by the Free Software Foundation; either version 2
* of the License, or (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
* Or, point your browser to http://www.gnu.org/copyleft/gpl.html
*
* The author can be reached at kls@cadsoft.de
*
* The project's page is at http://www.cadsoft.de/people/kls/vdr
*
* $Id: osm.c 1.3 2000/04/15 14:04:21 kls Exp $
*/
#include <signal.h>
#include "config.h"
#include "interface.h"
#include "menu.h"
#include "recording.h"
#include "tools.h"
#ifdef DEBUG_REMOTE
#define KEYS_CONF "keys-pc.conf"
#else
#define KEYS_CONF "keys.conf"
#endif
static int Interrupted = 0;
void SignalHandler(int signum)
{
Interrupted = signum;
}
int main(int argc, char *argv[])
{
openlog("vdr", LOG_PID | LOG_CONS, LOG_USER);
isyslog(LOG_INFO, "started");
Channels.Load("channels.conf");
Timers.Load("timers.conf");
if (!Keys.Load(KEYS_CONF))
Interface.LearnKeys();
Interface.Init();
cChannel::SwitchTo(CurrentChannel);
if (signal(SIGHUP, SignalHandler) == SIG_IGN) signal(SIGHUP, SIG_IGN);
if (signal(SIGINT, SignalHandler) == SIG_IGN) signal(SIGINT, SIG_IGN);
if (signal(SIGTERM, SignalHandler) == SIG_IGN) signal(SIGTERM, SIG_IGN);
cMenuMain *Menu = NULL;
cTimer *Timer = NULL;
cRecording *Recording = NULL;
while (!Interrupted) {
AssertFreeDiskSpace();
if (!Recording && !Timer && (Timer = cTimer::GetMatch()) != NULL) {
DELETENULL(Menu);
// make sure the timer won't be deleted:
Timer->SetRecording(true);
// switch to channel:
cChannel::SwitchTo(Timer->channel - 1);
// start recording:
Recording = new cRecording(Timer);
if (!Recording->Record())
DELETENULL(Recording);
}
if (Timer && !Timer->Matches()) {
// stop recording:
if (Recording) {
Recording->Stop();
DELETENULL(Recording);
}
// release timer:
Timer->SetRecording(false);
// clear single event timer:
if (Timer->IsSingleEvent()) {
DELETENULL(Menu); // must make sure no menu uses it
isyslog(LOG_INFO, "deleting timer %d", Timer->Index() + 1);
Timers.Del(Timer);
Timers.Save();
}
Timer = NULL;
}
eKeys key = Interface.GetKey();
if (Menu) {
switch (Menu->ProcessKey(key)) {
default: if (key != kMenu)
break;
case osBack:
case osEnd: DELETENULL(Menu);
break;
}
}
else {
switch (key) {
// Record/Replay Control:
case kBegin: DvbApi.Skip(-INT_MAX); break;
case kRecord: if (!DvbApi.Recording()) {
cTimer *timer = new cTimer(true);
Timers.Add(timer);
Timers.Save();
}
else
Interface.Error("Already recording!");
break;
case kPause: DvbApi.PauseReplay(); break;
case kStop: DvbApi.StopReplay(); break;
case kSearchBack: DvbApi.FastRewind(); break;
case kSearchForward: DvbApi.FastForward(); break;
case kSkipBack: DvbApi.Skip(-60); break;
case kSkipForward: DvbApi.Skip(60); break;
// Menu Control:
case kMenu: Menu = new cMenuMain;
Menu->Display();
break;
case kUp:
case kDown: {
int n = CurrentChannel + (key == kUp ? 1 : -1);
cChannel *channel = Channels.Get(n);
if (channel)
channel->Switch();
}
break;
default: break;
}
}
}
isyslog(LOG_INFO, "caught signal %d", Interrupted);
DvbApi.StopRecord();
DvbApi.StopReplay();
//TODO kill any remaining sub-processes!
isyslog(LOG_INFO, "exiting", Interrupted);
closelog();
return 0;
}

View File

@ -1,10 +1,10 @@
/* /*
* recording.h: Recording file handling * recording.h: Recording file handling
* *
* See the main source file 'osm.c' for copyright information and * See the main source file 'vdr.c' for copyright information and
* how to reach the author. * how to reach the author.
* *
* $Id: recording.c 1.2 2000/04/15 13:29:02 kls Exp $ * $Id: recording.c 1.6 2000/04/24 09:45:13 kls Exp $
*/ */
#define _GNU_SOURCE #define _GNU_SOURCE
@ -104,6 +104,7 @@ void AssertFreeDiskSpace(void)
cRecording::cRecording(const char *Name, time_t Start, int Priority, int LifeTime) cRecording::cRecording(const char *Name, time_t Start, int Priority, int LifeTime)
{ {
titleBuffer = NULL;
fileName = NULL; fileName = NULL;
name = strdup(Name); name = strdup(Name);
start = Start; start = Start;
@ -113,6 +114,7 @@ cRecording::cRecording(const char *Name, time_t Start, int Priority, int LifeTim
cRecording::cRecording(cTimer *Timer) cRecording::cRecording(cTimer *Timer)
{ {
titleBuffer = NULL;
fileName = NULL; fileName = NULL;
name = strdup(Timer->file); name = strdup(Timer->file);
start = Timer->StartTime(); start = Timer->StartTime();
@ -122,6 +124,7 @@ cRecording::cRecording(cTimer *Timer)
cRecording::cRecording(const char *FileName) cRecording::cRecording(const char *FileName)
{ {
titleBuffer = NULL;
fileName = strdup(FileName); fileName = strdup(FileName);
FileName += strlen(BaseDir) + 1; FileName += strlen(BaseDir) + 1;
char *p = strrchr(FileName, '/'); char *p = strrchr(FileName, '/');
@ -144,6 +147,7 @@ cRecording::cRecording(const char *FileName)
cRecording::~cRecording() cRecording::~cRecording()
{ {
delete titleBuffer;
delete fileName; delete fileName;
delete name; delete name;
} }
@ -157,6 +161,23 @@ const char *cRecording::FileName(void)
return fileName; return fileName;
} }
const char *cRecording::Title(char Delimiter)
{
delete titleBuffer;
titleBuffer = NULL;
struct tm *t = localtime(&start);
asprintf(&titleBuffer, "%02d.%02d.%04d%c%02d:%02d%c%s",
t->tm_mday,
t->tm_mon + 1,
t->tm_year + 1900,
Delimiter,
t->tm_hour,
t->tm_min,
Delimiter,
name);
return titleBuffer;
}
bool cRecording::Delete(void) bool cRecording::Delete(void)
{ {
bool result = true; bool result = true;
@ -180,21 +201,6 @@ bool cRecording::Remove(void)
return RemoveFileOrDir(FileName()); return RemoveFileOrDir(FileName());
} }
bool cRecording::Record(void)
{
return DvbApi.StartRecord(FileName());
}
bool cRecording::Play(void)
{
return DvbApi.StartReplay(FileName());
}
void cRecording::Stop(void)
{
DvbApi.StopRecord();
}
// --- cRecordings ----------------------------------------------------------- // --- cRecordings -----------------------------------------------------------
bool cRecordings::Load(bool Deleted) bool cRecordings::Load(bool Deleted)

View File

@ -1,10 +1,10 @@
/* /*
* recording.h: Recording file handling * recording.h: Recording file handling
* *
* See the main source file 'osm.c' for copyright information and * See the main source file 'vdr.c' for copyright information and
* how to reach the author. * how to reach the author.
* *
* $Id: recording.h 1.2 2000/04/14 15:12:42 kls Exp $ * $Id: recording.h 1.6 2000/04/24 09:45:49 kls Exp $
*/ */
#ifndef __RECORDING_H #ifndef __RECORDING_H
@ -12,15 +12,17 @@
#include <time.h> #include <time.h>
#include "config.h" #include "config.h"
#include "dvbapi.h"
#include "tools.h" #include "tools.h"
void AssertFreeDiskSpace(void); void AssertFreeDiskSpace(void);
class cRecording : public cListObject { class cRecording : public cListObject {
public: friend class cRecordings;
char *name; private:
char *titleBuffer;
char *fileName; char *fileName;
char *name;
public:
time_t start; time_t start;
int priority; int priority;
int lifetime; int lifetime;
@ -29,18 +31,13 @@ public:
cRecording(const char *FileName); cRecording(const char *FileName);
~cRecording(); ~cRecording();
const char *FileName(void); const char *FileName(void);
const char *Title(char Delimiter = ' ');
bool Delete(void); bool Delete(void);
// Changes the file name so that it will no longer be visible in the OSM // Changes the file name so that it will no longer be visible in the "Recordings" menu
// Returns false in case of error // Returns false in case of error
bool Remove(void); bool Remove(void);
// Actually removes the file from the disk // Actually removes the file from the disk
// Returns false in case of error // Returns false in case of error
bool Record(void);
// Starts recording of the file
bool Play(void);
// Starts playback of the file
void Stop(void);
// Stops recording or playback of the file
}; };
class cRecordings : public cList<cRecording> { class cRecordings : public cList<cRecording> {

View File

@ -1,10 +1,10 @@
/* /*
* remote.c: Interface to the Remote Control Unit * remote.c: Interface to the Remote Control Unit
* *
* See the main source file 'osm.c' for copyright information and * See the main source file 'vdr.c' for copyright information and
* how to reach the author. * how to reach the author.
* *
* $Id: remote.c 1.2 2000/04/15 16:00:51 kls Exp $ * $Id: remote.c 1.6 2000/04/24 09:45:56 kls Exp $
*/ */
#include "remote.h" #include "remote.h"
@ -17,6 +17,9 @@
#include <unistd.h> #include <unistd.h>
#include "tools.h" #include "tools.h"
#define REPEATLIMIT 100 // ms
#define REPEATDELAY 250 // ms
cRcIo::cRcIo(char *DeviceName) cRcIo::cRcIo(char *DeviceName)
{ {
dp = 0; dp = 0;
@ -25,7 +28,6 @@ cRcIo::cRcIo(char *DeviceName)
address = 0xFFFF; address = 0xFFFF;
t = 0; t = 0;
firstTime = lastTime = 0; firstTime = lastTime = 0;
minDelta = 0;
lastCommand = 0; lastCommand = 0;
if ((f = open(DeviceName, O_RDWR | O_NONBLOCK)) >= 0) { if ((f = open(DeviceName, O_RDWR | O_NONBLOCK)) >= 0) {
struct termios t; struct termios t;
@ -35,8 +37,11 @@ cRcIo::cRcIo(char *DeviceName)
if (tcsetattr(f, TCSAFLUSH, &t) == 0) if (tcsetattr(f, TCSAFLUSH, &t) == 0)
return; return;
} }
LOG_ERROR_STR(DeviceName);
close(f); close(f);
} }
else
LOG_ERROR_STR(DeviceName);
f = -1; f = -1;
} }
@ -46,9 +51,8 @@ cRcIo::~cRcIo()
close(f); close(f);
} }
int cRcIo::ReceiveByte(bool Wait) bool cRcIo::InputAvailable(bool Wait)
{ {
// Returns the byte if one was received within 1 second, -1 otherwise
if (f >= 0) { if (f >= 0) {
fd_set set; fd_set set;
struct timeval timeout; struct timeval timeout;
@ -56,13 +60,19 @@ int cRcIo::ReceiveByte(bool Wait)
timeout.tv_usec = Wait ? 0 : 10000; timeout.tv_usec = Wait ? 0 : 10000;
FD_ZERO(&set); FD_ZERO(&set);
FD_SET(f, &set); FD_SET(f, &set);
if (select(FD_SETSIZE, &set, NULL, NULL, &timeout) > 0) { if (select(FD_SETSIZE, &set, NULL, NULL, &timeout) > 0)
if (FD_ISSET(f, &set)) { return FD_ISSET(f, &set);
unsigned char b; }
if (read(f, &b, 1) == 1) return false;
return b; }
}
} int cRcIo::ReceiveByte(bool Wait)
{
// Returns the byte if one was received within a timeout, -1 otherwise
if (InputAvailable(Wait)) {
unsigned char b;
if (read(f, &b, 1) == 1)
return b;
} }
return -1; return -1;
} }
@ -112,7 +122,6 @@ bool cRcIo::SetCode(unsigned char Code, unsigned short Address)
{ {
code = Code; code = Code;
address = Address; address = Address;
minDelta = 200;
return SendCommand(code); return SendCommand(code);
} }
@ -157,12 +166,10 @@ bool cRcIo::GetCommand(unsigned int *Command, unsigned short *Address)
// let's have a timeout to avoid getting overrun by commands // let's have a timeout to avoid getting overrun by commands
int now = time_ms(); int now = time_ms();
int delta = now - lastTime; int delta = now - lastTime;
if (delta < minDelta)
minDelta = delta; // dynamically adjust to the smallest delta
lastTime = now; lastTime = now;
if (delta < minDelta * 1.3) { // if commands come in rapidly... if (delta < REPEATLIMIT) { // if commands come in rapidly...
if (now - firstTime < 250) if (now - firstTime < REPEATDELAY)
return false; // ...repeat function kicks in after 250ms return false; // ...repeat function kicks in after a short delay
return true; return true;
} }
} }
@ -209,12 +216,12 @@ bool cRcIo::Number(int n, bool Hex)
bool cRcIo::String(char *s) bool cRcIo::String(char *s)
{ {
char *chars = mode == modeH ? "0123456789ABCDEF" : "0123456789-EHLP "; const char *chars = mode == modeH ? "0123456789ABCDEF" : "0123456789-EHLP ";
int n = 0; int n = 0;
for (int i = 0; *s && i < 4; s++, i++) { for (int i = 0; *s && i < 4; s++, i++) {
n <<= 4; n <<= 4;
for (char *c = chars; *c; c++) { for (const char *c = chars; *c; c++) {
if (*c == *s) { if (*c == *s) {
n |= c - chars; n |= c - chars;
break; break;

View File

@ -1,10 +1,10 @@
/* /*
* remote.h: Interface to the Remote Control Unit * remote.h: Interface to the Remote Control Unit
* *
* See the main source file 'osm.c' for copyright information and * See the main source file 'vdr.c' for copyright information and
* how to reach the author. * how to reach the author.
* *
* $Id: remote.h 1.1 2000/02/19 13:36:48 kls Exp $ * $Id: remote.h 1.4 2000/04/24 09:46:00 kls Exp $
*/ */
#ifndef __REMOTE_H #ifndef __REMOTE_H
@ -19,7 +19,7 @@ private:
unsigned char dp, code, mode; unsigned char dp, code, mode;
unsigned short address; unsigned short address;
time_t t; time_t t;
int firstTime, lastTime, minDelta; int firstTime, lastTime;
unsigned int lastCommand; unsigned int lastCommand;
bool SendCommand(unsigned char Cmd); bool SendCommand(unsigned char Cmd);
int ReceiveByte(bool Wait = true); int ReceiveByte(bool Wait = true);
@ -29,6 +29,7 @@ public:
enum { modeH = 'h', modeB = 'b', modeS = 's' }; enum { modeH = 'h', modeB = 'b', modeS = 's' };
cRcIo(char *DeviceName); cRcIo(char *DeviceName);
~cRcIo(); ~cRcIo();
bool InputAvailable(bool Wait = false);
void Flush(int WaitSeconds = 0); void Flush(int WaitSeconds = 0);
bool SetCode(unsigned char Code, unsigned short Address); bool SetCode(unsigned char Code, unsigned short Address);
bool SetMode(unsigned char Mode); bool SetMode(unsigned char Mode);

View File

@ -1,5 +1,5 @@
1:15:MTWTF--:1828:1901:10:5:nano 0:15:MTWTF--:1828:1901:10:5:nano
1:3:M------:2110:2230:99:10:SevenDays 0:3:M------:2110:2230:99:10:SevenDays
1:10:-T-----:2058:2150:99:10:Quarks 1:10:-T-----:2058:2150:99:10:Quarks
1:3:---T---:2158:2300:99:10:Switch 1:3:---T---:2158:2300:99:10:Switch
1:2:----F--:2110:2155:99:10:Anke 1:2:----F--:2110:2155:99:10:Anke

60
tools.c
View File

@ -1,16 +1,17 @@
/* /*
* tools.c: Various tools * tools.c: Various tools
* *
* See the main source file 'osm.c' for copyright information and * See the main source file 'vdr.c' for copyright information and
* how to reach the author. * how to reach the author.
* *
* $Id: tools.c 1.3 2000/04/15 15:10:05 kls Exp $ * $Id: tools.c 1.7 2000/04/24 15:01:35 kls Exp $
*/ */
#define _GNU_SOURCE #define _GNU_SOURCE
#include "tools.h" #include "tools.h"
#include <dirent.h> #include <dirent.h>
#include <errno.h> #include <errno.h>
#include <signal.h>
#include <stdlib.h> #include <stdlib.h>
#include <string.h> #include <string.h>
#include <sys/stat.h> #include <sys/stat.h>
@ -21,6 +22,17 @@
int SysLogLevel = 3; int SysLogLevel = 3;
bool DataAvailable(int filedes)
{
fd_set set;
FD_ZERO(&set);
FD_SET(filedes, &set);
struct timeval timeout;
timeout.tv_sec = 0;
timeout.tv_usec = 10000;
return select(FD_SETSIZE, &set, NULL, NULL, &timeout) > 0 && FD_ISSET(filedes, &set);
}
void writechar(int filedes, char c) void writechar(int filedes, char c)
{ {
write(filedes, &c, sizeof(c)); write(filedes, &c, sizeof(c));
@ -40,8 +52,13 @@ char readchar(int filedes)
bool readint(int filedes, int &n) bool readint(int filedes, int &n)
{ {
//XXX timeout!! return DataAvailable(filedes) && read(filedes, &n, sizeof(n)) == sizeof(n);
return read(filedes, &n, sizeof(n)); }
void purge(int filedes)
{
while (DataAvailable(filedes))
readchar(filedes);
} }
char *readline(FILE *f) char *readline(FILE *f)
@ -135,6 +152,41 @@ bool RemoveFileOrDir(const char *FileName)
return false; return false;
} }
bool CheckProcess(pid_t pid)
{
pid_t Pid2Check = pid;
int status;
pid = waitpid(Pid2Check, &status, WNOHANG);
if (pid < 0) {
if (errno != ECHILD)
LOG_ERROR;
return false;
}
return true;
}
void KillProcess(pid_t pid, int Timeout)
{
pid_t Pid2Wait4 = pid;
for (time_t t0 = time(NULL); time(NULL) - t0 < Timeout; ) {
int status;
pid_t pid = waitpid(Pid2Wait4, &status, WNOHANG);
if (pid < 0) {
if (errno != ECHILD)
LOG_ERROR;
return;
}
if (pid == Pid2Wait4)
return;
}
esyslog(LOG_ERR, "ERROR: process %d won't end (waited %d seconds) - terminating it...", Pid2Wait4, Timeout);
if (kill(Pid2Wait4, SIGTERM) < 0) {
esyslog(LOG_ERR, "ERROR: process %d won't terminate (%s) - killing it...", Pid2Wait4, strerror(errno));
if (kill(Pid2Wait4, SIGKILL) < 0)
esyslog(LOG_ERR, "ERROR: process %d won't die (%s) - giving up", Pid2Wait4, strerror(errno));
}
}
// --- cListObject ----------------------------------------------------------- // --- cListObject -----------------------------------------------------------
cListObject::cListObject(void) cListObject::cListObject(void)

12
tools.h
View File

@ -1,17 +1,20 @@
/* /*
* tools.h: Various tools * tools.h: Various tools
* *
* See the main source file 'osm.c' for copyright information and * See the main source file 'vdr.c' for copyright information and
* how to reach the author. * how to reach the author.
* *
* $Id: tools.h 1.3 2000/04/15 15:09:47 kls Exp $ * $Id: tools.h 1.7 2000/04/24 15:01:49 kls Exp $
*/ */
#ifndef __TOOLS_H #ifndef __TOOLS_H
#define __TOOLS_H #define __TOOLS_H
#include <errno.h>
#include <stdio.h> #include <stdio.h>
#include <syslog.h> #include <syslog.h>
#include <sys/wait.h>
#include <sys/types.h>
extern int SysLogLevel; extern int SysLogLevel;
@ -23,18 +26,23 @@ extern int SysLogLevel;
#define LOG_ERROR_STR(s) esyslog(LOG_ERR, "ERROR: %s: %s", s, strerror(errno)); #define LOG_ERROR_STR(s) esyslog(LOG_ERR, "ERROR: %s: %s", s, strerror(errno));
#define SECSINDAY 86400 #define SECSINDAY 86400
#define MAXPROCESSTIMEOUT 3 // seconds
#define DELETENULL(p) (delete (p), p = NULL) #define DELETENULL(p) (delete (p), p = NULL)
bool DataAvailable(int filedes);
void writechar(int filedes, char c); void writechar(int filedes, char c);
void writeint(int filedes, int n); void writeint(int filedes, int n);
char readchar(int filedes); char readchar(int filedes);
bool readint(int filedes, int &n); bool readint(int filedes, int &n);
void purge(int filedes);
char *readline(FILE *f); char *readline(FILE *f);
int time_ms(void); int time_ms(void);
void delay_ms(int ms); void delay_ms(int ms);
bool MakeDirs(const char *FileName, bool IsDirectory = false); bool MakeDirs(const char *FileName, bool IsDirectory = false);
bool RemoveFileOrDir(const char *FileName); bool RemoveFileOrDir(const char *FileName);
bool CheckProcess(pid_t pid);
void KillProcess(pid_t pid, int Timeout = MAXPROCESSTIMEOUT);
class cListObject { class cListObject {
private: private:

189
vdr.c Normal file
View File

@ -0,0 +1,189 @@
/*
* vdr.c: Video Disk Recorder main program
*
* Copyright (C) 2000 Klaus Schmidinger
*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU General Public License
* as published by the Free Software Foundation; either version 2
* of the License, or (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
* Or, point your browser to http://www.gnu.org/copyleft/gpl.html
*
* The author can be reached at kls@cadsoft.de
*
* The project's page is at http://www.cadsoft.de/people/kls/vdr
*
* $Id: vdr.c 1.12 2000/04/24 13:36:39 kls Exp $
*/
#include <signal.h>
#include "config.h"
#include "interface.h"
#include "menu.h"
#include "recording.h"
#include "tools.h"
#ifdef DEBUG_REMOTE
#define KEYS_CONF "keys-pc.conf"
#else
#define KEYS_CONF "keys.conf"
#endif
#define DIRECTCHANNELTIMEOUT 500 //ms
static int Interrupted = 0;
void SignalHandler(int signum)
{
Interrupted = signum;
}
int main(int argc, char *argv[])
{
openlog("vdr", LOG_PID | LOG_CONS, LOG_USER);
isyslog(LOG_INFO, "started");
Channels.Load("channels.conf");
Timers.Load("timers.conf");
if (!Keys.Load(KEYS_CONF))
Interface.LearnKeys();
Interface.Init();
cChannel::SwitchTo(CurrentChannel);
if (signal(SIGHUP, SignalHandler) == SIG_IGN) signal(SIGHUP, SIG_IGN);
if (signal(SIGINT, SignalHandler) == SIG_IGN) signal(SIGINT, SIG_IGN);
if (signal(SIGTERM, SignalHandler) == SIG_IGN) signal(SIGTERM, SIG_IGN);
cMenuMain *Menu = NULL;
cReplayDisplay *ReplayDisplay = NULL;
cTimer *Timer = NULL;
int dcTime = 0, dcNumber = 0;
int LastChannel = -1;
while (!Interrupted) {
// Channel display:
if (CurrentChannel != LastChannel) {
if (!Menu && !ReplayDisplay) {
cChannel *channel = Channels.Get(CurrentChannel);
if (channel)
Interface.DisplayChannel(CurrentChannel + 1, channel->name);
}
LastChannel = CurrentChannel;
}
// Direct Channel Select (action):
if (dcNumber) {
Interface.DisplayChannel(dcNumber);
if (time_ms() - dcTime > DIRECTCHANNELTIMEOUT) {
cChannel::SwitchTo(dcNumber - 1);
dcNumber = 0;
LastChannel = -1; // in case an invalid channel number was entered!
}
}
// Timer Processing:
else {
AssertFreeDiskSpace();
if (!Timer && (Timer = cTimer::GetMatch()) != NULL) {
DELETENULL(Menu);
DELETENULL(ReplayDisplay);
// make sure the timer won't be deleted:
Timer->SetRecording(true);
// switch to channel:
cChannel::SwitchTo(Timer->channel - 1);
// start recording:
cRecording Recording(Timer);
DvbApi.StartRecord(Recording.FileName());
}
if (Timer && !Timer->Matches()) {
// stop recording:
DvbApi.StopRecord();
// release timer:
Timer->SetRecording(false);
// clear single event timer:
if (Timer->IsSingleEvent()) {
DELETENULL(Menu); // must make sure no menu uses it
isyslog(LOG_INFO, "deleting timer %d", Timer->Index() + 1);
Timers.Del(Timer);
Timers.Save();
}
Timer = NULL;
}
}
// User Input:
eKeys key = Interface.GetKey(!ReplayDisplay);
if (Menu) {
switch (Menu->ProcessKey(key)) {
default: if (key != kMenu)
break;
case osBack:
case osEnd: DELETENULL(Menu);
break;
}
}
else if (!ReplayDisplay || (key = ReplayDisplay->ProcessKey(key)) != kNone) {
switch (key) {
// Direct Channel Select (input):
case k0: case k1: case k2: case k3: case k4: case k5: case k6: case k7: case k8: case k9:
{
if (!(DvbApi.Recording() || DvbApi.Replaying())) {
dcNumber = dcNumber * 10 + key - k0;
dcTime = time_ms();
}
}
// Record/Replay Control:
case kBegin: DvbApi.Skip(-INT_MAX); break;
case kRecord: if (!(DvbApi.Recording() || DvbApi.Replaying())) {
cTimer *timer = new cTimer(true);
Timers.Add(timer);
Timers.Save();
}
break;
case kPause: DvbApi.PauseReplay(); break;
case kStop: DELETENULL(ReplayDisplay);
DvbApi.StopReplay();
break;
case kSearchBack: DvbApi.FastRewind(); break;
case kSearchForward: DvbApi.FastForward(); break;
case kSkipBack: DvbApi.Skip(-60); break;
case kSkipForward: DvbApi.Skip(60); break;
// Menu Control:
case kMenu: DELETENULL(ReplayDisplay);
Menu = new cMenuMain;
Menu->Display();
break;
// Up/Down Channel Select:
case kUp:
case kDown: if (!(DvbApi.Recording() || DvbApi.Replaying())) {
int n = CurrentChannel + (key == kUp ? 1 : -1);
cChannel *channel = Channels.Get(n);
if (channel)
channel->Switch();
}
break;
// Viewing Control:
case kOk: if (ReplayDisplay)
DELETENULL(ReplayDisplay);
else if (DvbApi.Replaying())
ReplayDisplay = new cReplayDisplay;
else
LastChannel = -1; break; // forces channel display
default: break;
}
}
}
isyslog(LOG_INFO, "caught signal %d", Interrupted);
DvbApi.StopRecord();
DvbApi.StopReplay();
isyslog(LOG_INFO, "exiting");
closelog();
return 0;
}