Version 0.98

- Completed storing the current audio volume in the setup.conf file (thanks
  to Andy Grobb).
- Fixed closing the progress display with the "Back" key when in trick mode
  and Setup.ShowReplayMode is enabled (thanks to Stefan Huelswitt).
- New SVDRP commands LSTR and DELR to list and delete recordings (thanks to
  Thomas Heiligenmann).
- Fixed a crash when pressing the '2' button while replaying a DVD.
- Updated 'channels.conf' for the "Bundesliga" channels of Premiere World
  (thanks to Mel Schchner).
- Changed the tuning code to use FrontendInfo to detect the type of DVB card.
- Removed the recursion stuff from cThread (cMutex already does this).
- Fixed handling the repeat function in the channel display.
- Avoiding multiple EPG entries for the same event (thanks to Rolf Hakenes
  for some valuable information on how to do this).
- A recording on the primary interface can now be stopped to make it continue
  on an other free DVB card (if one is free at the moment). See MANUAL for
  details.
- Added some missing teletext PIDs (thanks to Norbert Schmidt).
- Added PTS to the converted PCM audio when replaying a DVD (thanks to Andreas
  Schultz). Now the audio and video of a DVD replayed over the DVB card's A/V
  out should always be in sync.
- Fixed handling the "Power" key in case Setup.MinUserInactivity is set to 0 to
  disable automatic shutdown.
- Added a fifth parameter to the 'shutdown' call that indicates the reason for
  this shutdown request (see INSTALL).
- Fixed releasing 'index' memory after recording or playback.
- Fixed ejecting a DVD while it is being replayed.
- Removed all video overlay stuff from cDvbApi and SVDRP. Guido Fiala's new
  'kvdr' version 0.4 now does these things itself. As a consequence of this you
  will now need to use kvdr 0.4 or later.
- The device /dev/video is now opened only if necessary (to GRAB an image),
  allowing other programs (like 'kvdr', for instance) to use that device.
This commit is contained in:
Klaus Schmidinger 2001-11-04 18:00:00 +01:00
parent 8465398c6d
commit 6e1fd83555
20 changed files with 732 additions and 776 deletions

View File

@ -23,6 +23,8 @@ Guido Fiala <gfiala@s.netic.de>
for implementing the SVDRP command 'HITK' for implementing the SVDRP command 'HITK'
for implementing image grabbing for implementing image grabbing
for implementing overlay capabilities (see his 'kvdr' tool at http://www.s.netic.de/gfiala) for implementing overlay capabilities (see his 'kvdr' tool at http://www.s.netic.de/gfiala)
(overlay capabilities have been removed again in VDR 0.98, since kvdr version 0.4
now does these things itself)
for making the replay progress display avoid unnecessary code execution for making the replay progress display avoid unnecessary code execution
Robert Schneider <Robert.Schneider@lotus.com> Robert Schneider <Robert.Schneider@lotus.com>
@ -109,12 +111,13 @@ Ulrich R
27500 27500
for his support in keeping the Premiere World channels up to date in 'channels.conf' for his support in keeping the Premiere World channels up to date in 'channels.conf'
Helmut Schächner <schaechner@yahoo.com> Mel Schächner <schaechner@yahoo.com>
for his support in keeping the Premiere World channels up to date in 'channels.conf' for his support in keeping the Premiere World channels up to date in 'channels.conf'
Andreas Schultz <aschultz@warp10.net> Andreas Schultz <aschultz@warp10.net>
for adding support for replaying DVDs (much of this was derived from for adding support for replaying DVDs (much of this was derived from
dvdplayer-0.5 by Matjaz Thaler <matjaz.thaler@guest.arnes.si>) dvdplayer-0.5 by Matjaz Thaler <matjaz.thaler@guest.arnes.si>)
for adding PTS to the converted PCM audio when replaying a DVD
Aaron Holtzman Aaron Holtzman
for writing 'ac3dec' for writing 'ac3dec'
@ -150,3 +153,12 @@ Andreas Share <a.share@t-online.de>
Simon Bauschulte <SemiSchwabe@Brutzel.de> Simon Bauschulte <SemiSchwabe@Brutzel.de>
for his support in keeping the Premiere World channels up to date in 'channels.conf' for his support in keeping the Premiere World channels up to date in 'channels.conf'
Andy Grobb <Charly98@01019freenet.de>
for completing storing the current audio volume in the setup.conf file
Thomas Heiligenmann <thomas@heiligenmann.de>
for implementing the SVDRP commands LSTR and DELR
Norbert Schmidt <nschmidt-nrw@t-online.de>
for filling in some missing teletext PIDs

39
HISTORY
View File

@ -595,7 +595,7 @@ Video Disk Recorder Revision History
- When setting an editing mark while the progress display is not active, the - 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 display will now be turned on for a short while to indicate the successful
setting of the mark. setting of the mark.
- Updated 'channels.conf' for Premiere World (thanks to Helmut Schächner). - Updated 'channels.conf' for Premiere World (thanks to Mel Schächner).
Check your timers if you use this channels.conf file, since the sequence of Check your timers if you use this channels.conf file, since the sequence of
several PW channels has been changed. several PW channels has been changed.
- Changed the color of "Info" messages to "black on green" and that of the - Changed the color of "Info" messages to "black on green" and that of the
@ -825,8 +825,43 @@ Video Disk Recorder Revision History
- The menu timeout now also works when pressing the "Back" button during replay - The menu timeout now also works when pressing the "Back" button during replay
to enter the "Recordings" menu. to enter the "Recordings" menu.
- Updated 'channels.conf' for the "Bundesliga" channels of Premiere World - Updated 'channels.conf' for the "Bundesliga" channels of Premiere World
(thanks to Helmut Schächner). (thanks to Mel Schächner).
- Fixed reading timers.conf and channels.conf that contain blanks after numeric - Fixed reading timers.conf and channels.conf that contain blanks after numeric
values. values.
- Fixed handling trick modes near the beginning and end of a recording. - Fixed handling trick modes near the beginning and end of a recording.
- Pressing the "Back" button while replaying a DVD now leads to the DVD menu. - Pressing the "Back" button while replaying a DVD now leads to the DVD menu.
2001-11-04: Version 0.98
- Completed storing the current audio volume in the setup.conf file (thanks
to Andy Grobb).
- Fixed closing the progress display with the "Back" key when in trick mode
and Setup.ShowReplayMode is enabled (thanks to Stefan Huelswitt).
- New SVDRP commands LSTR and DELR to list and delete recordings (thanks to
Thomas Heiligenmann).
- Fixed a crash when pressing the '2' button while replaying a DVD.
- Updated 'channels.conf' for the "Bundesliga" channels of Premiere World
(thanks to Mel Schächner).
- Changed the tuning code to use FrontendInfo to detect the type of DVB card.
- Removed the recursion stuff from cThread (cMutex already does this).
- Fixed handling the repeat function in the channel display.
- Avoiding multiple EPG entries for the same event (thanks to Rolf Hakenes
for some valuable information on how to do this).
- A recording on the primary interface can now be stopped to make it continue
on an other free DVB card (if one is free at the moment). See MANUAL for
details.
- Added some missing teletext PIDs (thanks to Norbert Schmidt).
- Added PTS to the converted PCM audio when replaying a DVD (thanks to Andreas
Schultz). Now the audio and video of a DVD replayed over the DVB card's A/V
out should always be in sync.
- Fixed handling the "Power" key in case Setup.MinUserInactivity is set to 0 to
disable automatic shutdown.
- Added a fifth parameter to the 'shutdown' call that indicates the reason for
this shutdown request (see INSTALL).
- Fixed releasing 'index' memory after recording or playback.
- Fixed ejecting a DVD while it is being replayed.
- Removed all video overlay stuff from cDvbApi and SVDRP. Guido Fiala's new
'kvdr' version 0.4 now does these things itself. As a consequence of this you
will now need to use kvdr 0.4 or later.
- The device /dev/video is now opened only if necessary (to GRAB an image),
allowing other programs (like 'kvdr', for instance) to use that device.

10
INSTALL
View File

@ -25,7 +25,7 @@ directory ../DVD (seen from the VDR directory). Adjust the definition
of DVDDIR in the Makefile if necessary. of DVDDIR in the Makefile if necessary.
You can find 'libdvdread' at You can find 'libdvdread' at
http://www.dtek.chalmers.se/groups/dvd/downloads.html http://www.dtek.chalmers.se/groups/dvd/downloads.shtml
If you want to replay CSS encrypted DVDs you also need to get the 'libdvdcss' If you want to replay CSS encrypted DVDs you also need to get the 'libdvdcss'
library from library from
@ -127,7 +127,7 @@ active, the user has been inactive for at least MinUserInactivity minutes
and the next timer event is at least MinEventTimeout minutes in the future and the next timer event is at least MinEventTimeout minutes in the future
(see the Setup parameters in MANUAL). (see the Setup parameters in MANUAL).
The command given in the '-s' option will be called with four parameters. The command given in the '-s' option will be called with five parameters.
The first one is the time (in UTC) of the next timer event (as a time_t The first one is the time (in UTC) of the next timer event (as a time_t
type number), and the second one is the number of seconds from the current type number), and the second one is the number of seconds from the current
time until the next timer event. Your program can choose which one to use time until the next timer event. Your program can choose which one to use
@ -153,6 +153,12 @@ contains the file name of the recording as defined in the timer (or an empty
string if no timer is present). These can be used by the shutdown program to string if no timer is present). These can be used by the shutdown program to
show that information on some display interface etc. show that information on some display interface etc.
The fifth parameter indicates the reason why the shutdown was requested.
'0' means this is an automatic shutdown due to some timeout, while '1' means
that this is a user requested shutdown (resulting from pressing the "Power"
key). The shutdown program may use this information to decide whether or
not to actually perform the system shutdown.
If a timer is currently recording, the parameters will reflect the start If a timer is currently recording, the parameters will reflect the start
time of that timer. This means that the first parameter will be a time in time of that timer. This means that the first parameter will be a time in
the past, and the second parameter will be a negative number. This only the past, and the second parameter will be a negative number. This only

14
MANUAL
View File

@ -310,6 +310,20 @@ Video Disk Recorder User's Manual
A timer can also be programmed by pressing the "Red" button on the "Schedule", A timer can also be programmed by pressing the "Red" button on the "Schedule",
"Now", "Next" or "Event" menus. "Now", "Next" or "Event" menus.
* Stopping a recording on the primary DVB interface
If the primary DVB interface is currently recording, the user can't switch
the channel or replay another recording on that interface. However, if there
is an other DVB interface that is currently not recording and provides the
necessary conditional access facilities to continue the recording that is
currently being performed on the primary DVB interface, the Main menu will
contain an option that allows you to stop recording on the primary DVB
interface. Select that option to stop the ongoing recording and thus free the
primary DVB interface to allow channel switching or replaying. The interrupted
recording will be continued on an other free DVB interface. There may be a
short discontinuity at that point when replaying that recording later, so you
may want to place such an action for instance in a commercial break.
* Parameters in the "Setup" menu * Parameters in the "Setup" menu
Select "Setup" from the "Main" menu to enter the setup menu. From there you can Select "Setup" from the "Main" menu to enter the setup menu. From there you can

View File

@ -2,19 +2,19 @@ RTL:12188:h:0:27500:163:104:105:0:12003
Sat.1:12480:v:0:27500:1791:1792:34:0:46 Sat.1:12480:v:0:27500:1791:1792:34:0:46
Pro-7:12480:v:0:27500:255:256;257:32:0:898 Pro-7:12480:v:0:27500:255:256;257:32:0:898
RTL2:12188:h:0:27500:166:128:68:0:12020 RTL2:12188:h:0:27500:166:128:68:0:12020
ARD:11837:h:0:27500:101:102:0:0:28106 ARD:11837:h:0:27500:101:102:104:0:28106
BR3:11837:h:0:27500:201:202:0:0:28107 BR3:11837:h:0:27500:201:202:204:0:28107
Hessen-3:11837:h:0:27500:301:302:0:0:28108 Hessen-3:11837:h:0:27500:301:302:0:0:28108
N3:12110:h:0:27500:2401:2402:0:0:28224 N3:12110:h:0:27500:2401:2402:2404:0:28224
SR3:11837:h:0:27500:501:502:0:0:28110 SR3:11837:h:0:27500:501:502:504:0:28110
WDR:11837:h:0:27500:601:602:0:0:28111 WDR:11837:h:0:27500:601:602:0:0:28111
BR-alpha:11837:h:0:27500:701:702:0:0:28112 BR-alpha:11837:h:0:27500:701:702:704:0:28112
SWR BW:11837:h:0:27500:801:802:0:0:28113 SWR BW:11837:h:0:27500:801:802:804:0:28113
Phoenix:11837:h:0:27500:901:902:0:0:28114 Phoenix:11837:h:0:27500:901:902:904:0:28114
ZDF:11954:h:0:27500:110:120:130:0:28006 ZDF:11954:h:0:27500:110:120:130:0:28006
3sat:11954:h:0:27500:210:220:230:0:28007 3sat:11954:h:0:27500:210:220:230:0:28007
KiKa:11954:h:0:27500:310:320:0:0:28008 KiKa:11954:h:0:27500:310:320:330:0:28008
arte:11836:h:0:27500:401:402:0:0:28109 arte:11836:h:0:27500:401:402:404:0:28109
ORF1:12692:h:0:22000:160:161:165:3:13001 ORF1:12692:h:0:22000:160:161:165:3:13001
ORF2:12692:h:0:22000:500:501:505:3:13002 ORF2:12692:h:0:22000:500:501:505:3:13002
ORF Sat:11954:h:0:27500:506:507:0:0:28010 ORF Sat:11954:h:0:27500:506:507:0:0:28010
@ -42,10 +42,10 @@ EinsFestival:12110:h:0:27500:201:202:0:0:28202
EinsMuXx:12110:h:0:27500:301:302:0:0:28203 EinsMuXx:12110:h:0:27500:301:302:0:0:28203
ZDF Theaterkanal:11954:h:0:27500:1110:1120:0:0:28016 ZDF Theaterkanal:11954:h:0:27500:1110:1120:0:0:28016
ZDF.doku:11954:h:0:27500:660:670:0:0:28014 ZDF.doku:11954:h:0:27500:660:670:0:0:28014
MDR:12110:h:0:27500:401:402:0:0:28204 MDR:12110:h:0:27500:401:402:404:0:28204
NICK-PARAMOUNT:12246:v:0:27500:167:108:0:0:29312 NICK-PARAMOUNT:12246:v:0:27500:167:108:0:0:29312
ORB:12110:h:0:27500:501:502:0:0:28205 ORB:12110:h:0:27500:501:502:504:0:28205
B1:12110:h:0:27500:601:602:0:0:28206 B1:12110:h:0:27500:601:602:604:0:28206
ARD Online-Kanal:12722:h:0:22000:0:701:0:0:0 ARD Online-Kanal:12722:h:0:22000:0:701:0:0:0
:Premiere World :Premiere World
Premiere World:11797:h:0:27500:255:256:32:0:8 Premiere World:11797:h:0:27500:255:256:32:0:8
@ -117,14 +117,14 @@ Cockpitkanal:11720:h:0:27500:2559:2560:0:3:242
Boxengasse:11720:h:0:27500:2047:2048:0:3:240 Boxengasse:11720:h:0:27500:2047:2048:0:3:240
:Premiere World Bundesliga :Premiere World Bundesliga
Superdom:11758:h:0:27500:2815:8192:0:3:18 Superdom:11758:h:0:27500:2815:8192:0:3:18
BuLi-Konferenz:11758:h:0:27500:3071:3072,3073:0:3:215 BuLi-Konferenz:11758:h:0:27500:3327:3328,3329:0:3:215
BuLi-Spiel 1:11719:h:0:27500:255:256,257:0:3:17 BuLi-Spiel 1:11719:h:0:27500:255:256,257:0:3:17
BuLi-Spiel 2:11719:h:0:27500:2047:2048,2049:0:3:240 BuLi-Spiel 2:11719:h:0:27500:2047:2048,2049:0:3:240
BuLi-Spiel 3:11719:h:0:27500:3327:3328,3329:0:3:241 BuLi-Spiel 3:11719:h:0:27500:2303:2304,2305:0:3:241
BuLi-Spiel 4:11719:h:0:27500:2303:2304,2305:0:3:242 BuLi-Spiel 4:11719:h:0:27500:2559:2560,2561:0:3:242
BuLi-Spiel 5:11719:h:0:27500:3583:3584,3585:0:3:243 BuLi-Spiel 5:11719:h:0:27500:2815:2816,2817:0:3:243
BuLi-Spiel 6:11719:h:0:27500:2559:2560,2561:0:3:244 BuLi-Spiel 6:11719:h:0:27500:3071:3072,3073:0:3:244
BuLi-Spiel 7:11758:h:0:27500:2815:2816,2817:0:3:214 BuLi-Spiel 7:11758:h:0:27500:3071:3072,3073:0:3:214
: :
TV Niepokalanow:11876:h:0:27500:305:321:0:0:20601 TV Niepokalanow:11876:h:0:27500:305:321:0:0:20601
Mosaico:11934:v:0:27500:165:100:0:0:29010 Mosaico:11934:v:0:27500:165:100:0:0:29010

View File

@ -4,7 +4,7 @@
* See the main source file 'vdr.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.84 2001/10/07 15:13:23 kls Exp $ * $Id: config.h 1.85 2001/10/27 09:56:04 kls Exp $
*/ */
#ifndef __CONFIG_H #ifndef __CONFIG_H
@ -18,7 +18,7 @@
#include "eit.h" #include "eit.h"
#include "tools.h" #include "tools.h"
#define VDRVERSION "0.97" #define VDRVERSION "0.98"
#define MAXPRIORITY 99 #define MAXPRIORITY 99
#define MAXLIFETIME 99 #define MAXLIFETIME 99

941
dvbapi.c

File diff suppressed because it is too large Load Diff

View File

@ -4,7 +4,7 @@
* See the main source file 'vdr.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.53 2001/09/23 11:01:46 kls Exp $ * $Id: dvbapi.h 1.58 2001/11/04 11:39:42 kls Exp $
*/ */
#ifndef __DVBAPI_H #ifndef __DVBAPI_H
@ -34,12 +34,6 @@
#include "eit.h" #include "eit.h"
#include "thread.h" #include "thread.h"
// Overlay facilities
#define MAXCLIPRECTS 100
typedef struct CRect {
signed short x, y, width, height;
};
#define FRAMESPERSEC 25 #define FRAMESPERSEC 25
// The maximum file size is limited by the range that can be covered // The maximum file size is limited by the range that can be covered
@ -89,7 +83,7 @@ class cDvbApi {
#endif //DVDSUPPORT #endif //DVDSUPPORT
friend class cTransferBuffer; friend class cTransferBuffer;
private: private:
int videoDev; FrontendType frontendType;
int fd_osd, fd_frontend, fd_sec, fd_dvr, fd_audio, fd_video, fd_demuxa1, fd_demuxa2, fd_demuxd1, fd_demuxd2, fd_demuxv, fd_demuxt; int fd_osd, fd_frontend, fd_sec, fd_dvr, fd_audio, fd_video, fd_demuxa1, fd_demuxa2, fd_demuxd1, fd_demuxd2, fd_demuxv, fd_demuxt;
int vPid, aPid1, aPid2, dPid1, dPid2; int vPid, aPid1, aPid2, dPid1, dPid2;
bool SetPid(int fd, dmxPesType_t PesType, int Pid, dmxOutput_t Output); bool SetPid(int fd, dmxPesType_t PesType, int Pid, dmxOutput_t Output);
@ -154,21 +148,6 @@ public:
bool GrabImage(const char *FileName, bool Jpeg = true, int Quality = -1, int SizeX = -1, int SizeY = -1); bool GrabImage(const char *FileName, bool Jpeg = true, int Quality = -1, int SizeX = -1, int SizeY = -1);
// Overlay facilities
private:
bool ovlStat, ovlGeoSet, ovlFbSet;
int ovlSizeX, ovlSizeY, ovlPosX, ovlPosY, ovlBpp, ovlPalette, ovlClips, ovlClipCount;
int ovlFbSizeX, ovlFbSizeY;
__u16 ovlBrightness, ovlColour, ovlHue, ovlContrast;
struct video_clip ovlClipRects[MAXCLIPRECTS];
public:
bool OvlF(int SizeX, int SizeY, int FbAddr, int Bpp, int Palette);
bool OvlG(int SizeX, int SizeY, int PosX, int PosY);
bool OvlC(int ClipCount, CRect *Cr);
bool OvlP(__u16 Brightness, __u16 Color, __u16 Hue, __u16 Contrast);
bool OvlO(bool Value);
// On Screen Display facilities // On Screen Display facilities
private: private:
@ -233,8 +212,6 @@ private:
cPlayBuffer *replayBuffer; cPlayBuffer *replayBuffer;
int ca; int ca;
int priority; int priority;
int Ca(void) { return ca; }
// Returns the ca of the current recording session (0..MAXDVBAPI).
int Priority(void) { return priority; } int Priority(void) { return priority; }
// Returns the priority of the current recording session (0..MAXPRIORITY), // Returns the priority of the current recording session (0..MAXPRIORITY),
// or -1 if no recording is currently active. // or -1 if no recording is currently active.
@ -243,6 +220,8 @@ private:
void SetModeReplay(void); void SetModeReplay(void);
void SetModeNormal(bool FromRecording); void SetModeNormal(bool FromRecording);
public: public:
int Ca(void) { return ca; }
// Returns the ca of the current recording session (0..MAXDVBAPI).
int SecondsToFrames(int Seconds); int SecondsToFrames(int Seconds);
// Returns the number of frames corresponding to the given number of seconds. // Returns the number of frames corresponding to the given number of seconds.
bool Recording(void); bool Recording(void);
@ -330,6 +309,7 @@ public:
void SetVolume(int Volume, bool Absolute = false); void SetVolume(int Volume, bool Absolute = false);
// Sets the volume to the given value, either absolutely or relative to // Sets the volume to the given value, either absolutely or relative to
// the current volume. // the current volume.
static int CurrentVolume(void) { return PrimaryDvbApi ? PrimaryDvbApi->volume : 0; }
}; };
class cEITScanner { class cEITScanner {

37
eit.c
View File

@ -16,7 +16,7 @@
* the Free Software Foundation; either version 2 of the License, or * * the Free Software Foundation; either version 2 of the License, or *
* (at your option) any later version. * * (at your option) any later version. *
* * * *
* $Id: eit.c 1.28 2001/10/19 13:13:25 kls Exp $ * $Id: eit.c 1.29 2001/10/28 13:51:22 kls Exp $
***************************************************************************/ ***************************************************************************/
#include "eit.h" #include "eit.h"
@ -189,6 +189,7 @@ cEventInfo::cEventInfo(unsigned short serviceid, unsigned short eventid)
bIsPresent = bIsFollowing = false; bIsPresent = bIsFollowing = false;
lDuration = 0; lDuration = 0;
tTime = 0; tTime = 0;
uTableID = 0;
uEventID = eventid; uEventID = eventid;
uServiceID = serviceid; uServiceID = serviceid;
nChannelNumber = 0; nChannelNumber = 0;
@ -231,6 +232,12 @@ bool cEventInfo::IsFollowing() const
{ {
return bIsFollowing; return bIsFollowing;
} }
void cEventInfo::SetTableID(unsigned char tableid)
{
uTableID = tableid;
}
/** */ /** */
void cEventInfo::SetFollowing(bool foll) void cEventInfo::SetFollowing(bool foll)
{ {
@ -246,6 +253,12 @@ const char * cEventInfo::GetDate() const
return szDate; return szDate;
} }
const unsigned char cEventInfo::GetTableID(void) const
{
return uTableID;
}
/** */ /** */
const char * cEventInfo::GetTimeString() const const char * cEventInfo::GetTimeString() const
{ {
@ -545,21 +558,26 @@ unsigned short cSchedule::GetServiceID() const
return uServiceID; return uServiceID;
} }
/** */ /** */
const cEventInfo * cSchedule::GetEvent(unsigned short uEventID) const const cEventInfo * cSchedule::GetEvent(unsigned short uEventID, time_t tTime) const
{ {
// Returns either the event info with the given uEventID or, if that one can't
// be found, the one with the given tTime (or NULL if neither can be found)
cEventInfo *pe = Events.First(); cEventInfo *pe = Events.First();
cEventInfo *pt = NULL;
while (pe != NULL) while (pe != NULL)
{ {
if (pe->GetEventID() == uEventID) if (pe->GetEventID() == uEventID)
return pe; return pe;
if (tTime > 0 && pe->GetTime() == tTime) // 'tTime < 0' is apparently used with NVOD channels
pt = pe;
pe = Events.Next(pe); pe = Events.Next(pe);
} }
return NULL; return pt;
} }
/** */ /** */
const cEventInfo * cSchedule::GetEvent(time_t tTime) const const cEventInfo * cSchedule::GetEventAround(time_t tTime) const
{ {
cEventInfo *pe = Events.First(); cEventInfo *pe = Events.First();
while (pe != NULL) while (pe != NULL)
@ -759,7 +777,7 @@ int cEIT::ProcessEIT(unsigned char *buffer)
if (!rEvent) if (!rEvent)
break; break;
} }
pEvent = (cEventInfo *)pSchedule->GetEvent((unsigned short)VdrProgramInfo->EventID); pEvent = (cEventInfo *)pSchedule->GetEvent((unsigned short)VdrProgramInfo->EventID, VdrProgramInfo->StartTime);
if (!pEvent) { if (!pEvent) {
// If we don't have that event ID yet, we create a new one. // If we don't have that event ID yet, we create a new one.
// Otherwise we copy the information into the existing event anyway, because the data might have changed. // Otherwise we copy the information into the existing event anyway, because the data might have changed.
@ -767,6 +785,14 @@ int cEIT::ProcessEIT(unsigned char *buffer)
pEvent = (cEventInfo *)pSchedule->GetEvent((unsigned short)VdrProgramInfo->EventID); pEvent = (cEventInfo *)pSchedule->GetEvent((unsigned short)VdrProgramInfo->EventID);
if (!pEvent) if (!pEvent)
break; break;
pEvent->SetTableID(tid);
}
else {
// We have found an existing event, either through its event ID or its start time.
// If the new event comes from a table that belongs to an "other TS" and the existing
// one comes from a "actual TS" table, lets skip it.
if ((tid == 0x4F || tid == 0x60) && (pEvent->GetTableID() == 0x4E || pEvent->GetTableID() == 0x50))
continue;
} }
if (rEvent) { if (rEvent) {
pEvent->SetTitle(rEvent->GetTitle()); pEvent->SetTitle(rEvent->GetTitle());
@ -774,6 +800,7 @@ int cEIT::ProcessEIT(unsigned char *buffer)
pEvent->SetExtendedDescription(rEvent->GetExtendedDescription()); pEvent->SetExtendedDescription(rEvent->GetExtendedDescription());
} }
else { else {
pEvent->SetTableID(tid);
pEvent->SetTitle(VdrProgramInfo->ShortName); pEvent->SetTitle(VdrProgramInfo->ShortName);
pEvent->SetSubtitle(VdrProgramInfo->ShortText); pEvent->SetSubtitle(VdrProgramInfo->ShortText);
pEvent->SetExtendedDescription(VdrProgramInfo->ExtendedName); pEvent->SetExtendedDescription(VdrProgramInfo->ExtendedName);

9
eit.h
View File

@ -16,7 +16,7 @@
* the Free Software Foundation; either version 2 of the License, or * * the Free Software Foundation; either version 2 of the License, or *
* (at your option) any later version. * * (at your option) any later version. *
* * * *
* $Id: eit.h 1.11 2001/09/22 11:43:21 kls Exp $ * $Id: eit.h 1.12 2001/10/28 12:33:10 kls Exp $
***************************************************************************/ ***************************************************************************/
#ifndef __EIT_H #ifndef __EIT_H
@ -29,6 +29,7 @@ class cEventInfo : public cListObject {
friend class cSchedule; friend class cSchedule;
friend class cEIT; friend class cEIT;
private: private:
unsigned char uTableID; // Table ID this event came from
unsigned short uServiceID; // Service ID of program for that event unsigned short uServiceID; // Service ID of program for that event
bool bIsFollowing; // true if this is the next event on this channel bool bIsFollowing; // true if this is the next event on this channel
bool bIsPresent; // true if this is the present event running bool bIsPresent; // true if this is the present event running
@ -40,6 +41,7 @@ private:
time_t tTime; // Start time time_t tTime; // Start time
int nChannelNumber; // the actual channel number from VDR's channel list (used in cMenuSchedule for sorting by channel number) int nChannelNumber; // the actual channel number from VDR's channel list (used in cMenuSchedule for sorting by channel number)
protected: protected:
void SetTableID(unsigned char tableid);
void SetFollowing(bool foll); void SetFollowing(bool foll);
void SetPresent(bool pres); void SetPresent(bool pres);
void SetTitle(const char *string); void SetTitle(const char *string);
@ -52,6 +54,7 @@ protected:
cEventInfo(unsigned short serviceid, unsigned short eventid); cEventInfo(unsigned short serviceid, unsigned short eventid);
public: public:
~cEventInfo(); ~cEventInfo();
const unsigned char GetTableID(void) const;
const char *GetTimeString(void) const; const char *GetTimeString(void) const;
const char *GetEndTimeString(void) const; const char *GetEndTimeString(void) const;
const char *GetDate(void) const; const char *GetDate(void) const;
@ -90,8 +93,8 @@ public:
const cEventInfo *GetPresentEvent(void) const; const cEventInfo *GetPresentEvent(void) const;
const cEventInfo *GetFollowingEvent(void) const; const cEventInfo *GetFollowingEvent(void) const;
unsigned short GetServiceID(void) const; unsigned short GetServiceID(void) const;
const cEventInfo *GetEvent(unsigned short uEventID) const; const cEventInfo *GetEvent(unsigned short uEventID, time_t tTime = 0) const;
const cEventInfo *GetEvent(time_t tTime) const; const cEventInfo *GetEventAround(time_t tTime) const;
const cEventInfo *GetEventNumber(int n) const { return Events.Get(n); } const cEventInfo *GetEventNumber(int n) const { return Events.Get(n); }
int NumEvents(void) const { return Events.Count(); } int NumEvents(void) const { return Events.Count(); }
void Dump(FILE *f, const char *Prefix = "") const; void Dump(FILE *f, const char *Prefix = "") const;

11
i18n.c
View File

@ -4,7 +4,7 @@
* See the main source file 'vdr.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: i18n.c 1.44 2001/09/30 11:31:43 kls Exp $ * $Id: i18n.c 1.45 2001/10/28 16:04:58 kls Exp $
* *
* Slovenian translations provided by Miha Setina <mihasetina@softhome.net> * Slovenian translations provided by Miha Setina <mihasetina@softhome.net>
* Italian translations provided by Alberto Carraro <bertocar@tin.it> * Italian translations provided by Alberto Carraro <bertocar@tin.it>
@ -376,6 +376,15 @@ const tPhrase Phrases[] = {
"Arrêter l'enregistrement?", "Arrêter l'enregistrement?",
"Stoppe opptak?", "Stoppe opptak?",
}, },
{ "on primary interface",
"auf dem primären Interface",
"", // TODO
"", // TODO
"", // TODO
"", // TODO
"", // TODO
"", // TODO
},
{ "Cancel editing?", { "Cancel editing?",
"Schneiden abbrechen?", "Schneiden abbrechen?",
"Zelite prekiniti urejanje?", "Zelite prekiniti urejanje?",

96
menu.c
View File

@ -4,7 +4,7 @@
* See the main source file 'vdr.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.131 2001/10/21 14:28:14 kls Exp $ * $Id: menu.c 1.139 2001/11/04 10:37:18 kls Exp $
*/ */
#include "menu.h" #include "menu.h"
@ -1808,11 +1808,15 @@ eOSState cMenuCommands::ProcessKey(eKeys Key)
// --- cMenuMain ------------------------------------------------------------- // --- cMenuMain -------------------------------------------------------------
#define STOP_RECORDING tr(" Stop recording ") #define STOP_RECORDING tr(" Stop recording ")
#define ON_PRIMARY_INTERFACE tr("on primary interface")
cMenuMain::cMenuMain(bool Replaying, eOSState State) cMenuMain::cMenuMain(bool Replaying, eOSState State)
:cOsdMenu(tr("Main")) :cOsdMenu(tr("Main"))
{ {
digit = 0; digit = 0;
// Basic menu items:
Add(new cOsdItem(hk(tr("Schedule")), osSchedule)); Add(new cOsdItem(hk(tr("Schedule")), osSchedule));
Add(new cOsdItem(hk(tr("Channels")), osChannels)); Add(new cOsdItem(hk(tr("Channels")), osChannels));
Add(new cOsdItem(hk(tr("Timers")), osTimers)); Add(new cOsdItem(hk(tr("Timers")), osTimers));
@ -1824,8 +1828,20 @@ cMenuMain::cMenuMain(bool Replaying, eOSState State)
Add(new cOsdItem(hk(tr("Setup")), osSetup)); Add(new cOsdItem(hk(tr("Setup")), osSetup));
if (Commands.Count()) if (Commands.Count())
Add(new cOsdItem(hk(tr("Commands")), osCommands)); Add(new cOsdItem(hk(tr("Commands")), osCommands));
// Replay control:
if (Replaying) if (Replaying)
Add(new cOsdItem(tr(" Stop replaying"), osStopReplay)); Add(new cOsdItem(tr(" Stop replaying"), osStopReplay));
// Record control:
if (cRecordControls::StopPrimary()) {
char *buffer = NULL;
asprintf(&buffer, "%s%s", STOP_RECORDING, ON_PRIMARY_INTERFACE);
Add(new cOsdItem(buffer, osStopRecord));
}
const char *s = NULL; const char *s = NULL;
while ((s = cRecordControls::GetInstantId(s)) != NULL) { while ((s = cRecordControls::GetInstantId(s)) != NULL) {
char *buffer = NULL; char *buffer = NULL;
@ -1833,8 +1849,14 @@ cMenuMain::cMenuMain(bool Replaying, eOSState State)
Add(new cOsdItem(buffer, osStopRecord)); Add(new cOsdItem(buffer, osStopRecord));
delete buffer; delete buffer;
} }
// Editing control:
if (cVideoCutter::Active()) if (cVideoCutter::Active())
Add(new cOsdItem(tr(" Cancel editing"), osCancelEdit)); Add(new cOsdItem(tr(" Cancel editing"), osCancelEdit));
// Color buttons:
const char *DVDbutton = const char *DVDbutton =
#ifdef DVDSUPPORT #ifdef DVDSUPPORT
cDVD::DiscOk() ? tr("Eject") : NULL; cDVD::DiscOk() ? tr("Eject") : NULL;
@ -1845,6 +1867,9 @@ cMenuMain::cMenuMain(bool Replaying, eOSState State)
Display(); Display();
lastActivity = time(NULL); lastActivity = time(NULL);
SetHasHotkeys(); SetHasHotkeys();
// Initial submenus:
switch (State) { switch (State) {
case osRecordings: AddSubMenu(new cMenuRecordings); break; case osRecordings: AddSubMenu(new cMenuRecordings); break;
#ifdef DVDSUPPORT #ifdef DVDSUPPORT
@ -1882,7 +1907,11 @@ eOSState cMenuMain::ProcessKey(eKeys Key)
case osStopRecord: if (Interface->Confirm(tr("Stop recording?"))) { case osStopRecord: if (Interface->Confirm(tr("Stop recording?"))) {
cOsdItem *item = Get(Current()); cOsdItem *item = Get(Current());
if (item) { if (item) {
cRecordControls::Stop(item->Text() + strlen(STOP_RECORDING)); const char *s = item->Text() + strlen(STOP_RECORDING);
if (strcmp(s, ON_PRIMARY_INTERFACE) == 0)
cRecordControls::StopPrimary(true);
else
cRecordControls::Stop(item->Text() + strlen(STOP_RECORDING));
return osEnd; return osEnd;
} }
} }
@ -1908,6 +1937,14 @@ eOSState cMenuMain::ProcessKey(eKeys Key)
#ifdef DVDSUPPORT #ifdef DVDSUPPORT
case kYellow: if (!HasSubMenu()) { case kYellow: if (!HasSubMenu()) {
if (cDVD::DiscOk()) { if (cDVD::DiscOk()) {
// We need to stop replaying a DVD before ejecting,
// otherwise the replay thread crashes. Currently
// checking LastReplayed() is pretty much the only way
// of finding out whether we are currently replaying a DVD
// (i.e. if LastReplayed() returns non-NULL, we are either
// replaying a normal recording, or nothing at all):
if (!cReplayControl::LastReplayed())
cDvbApi::PrimaryDvbApi->StopReplay();
cDVD::Eject(); cDVD::Eject();
state = osEnd; state = osEnd;
} }
@ -2062,7 +2099,9 @@ eOSState cDisplayChannel::ProcessKey(eKeys Key)
} }
} }
break; break;
case kLeft|k_Repeat:
case kLeft: case kLeft:
case kRight|k_Repeat:
case kRight: case kRight:
withInfo = false; withInfo = false;
if (group < 0) { if (group < 0) {
@ -2072,7 +2111,7 @@ eOSState cDisplayChannel::ProcessKey(eKeys Key)
} }
if (group >= 0) { if (group >= 0) {
int SaveGroup = group; int SaveGroup = group;
if (Key == kRight) if (NORMALKEY(Key) == kRight)
group = Channels.GetNextGroup(group) ; group = Channels.GetNextGroup(group) ;
else else
group = Channels.GetPrevGroup(group < 1 ? 1 : group); group = Channels.GetPrevGroup(group < 1 ? 1 : group);
@ -2101,8 +2140,10 @@ eOSState cDisplayChannel::ProcessKey(eKeys Key)
case kOk: if (group >= 0) case kOk: if (group >= 0)
Channels.SwitchTo(Channels.Get(Channels.GetNextNormal(group))->number); Channels.SwitchTo(Channels.Get(Channels.GetNextNormal(group))->number);
return osEnd; return osEnd;
default: Interface->PutKey(Key); default: if (NORMALKEY(Key) == kUp || NORMALKEY(Key) == kDown || (Key & (k_Repeat | k_Release)) == 0) {
return osEnd; Interface->PutKey(Key);
return osEnd;
}
}; };
if (time_ms() - lastTime < INFOTIMEOUT) { if (time_ms() - lastTime < INFOTIMEOUT) {
DisplayInfo(); DisplayInfo();
@ -2166,7 +2207,7 @@ bool cRecordControl::GetEventInfo(void)
if (Schedules) { if (Schedules) {
const cSchedule *Schedule = Schedules->GetSchedule(channel->pnr); const cSchedule *Schedule = Schedules->GetSchedule(channel->pnr);
if (Schedule) { if (Schedule) {
eventInfo = Schedule->GetEvent(Time); eventInfo = Schedule->GetEventAround(Time);
if (eventInfo) { if (eventInfo) {
if (seconds > 0) if (seconds > 0)
dsyslog(LOG_INFO, "got EPG info after %d seconds", seconds); dsyslog(LOG_INFO, "got EPG info after %d seconds", seconds);
@ -2260,6 +2301,19 @@ void cRecordControls::Stop(cDvbApi *DvbApi)
} }
} }
bool cRecordControls::StopPrimary(bool DoIt)
{
if (cDvbApi::PrimaryDvbApi->Recording()) {
cDvbApi *dvbApi = cDvbApi::GetDvbApi(cDvbApi::PrimaryDvbApi->Ca(), 0);
if (dvbApi) {
if (DoIt)
Stop(cDvbApi::PrimaryDvbApi);
return true;
}
}
return false;
}
const char *cRecordControls::GetInstantId(const char *LastInstantId) const char *cRecordControls::GetInstantId(const char *LastInstantId)
{ {
for (int i = 0; i < MAXDVBAPI; i++) { for (int i = 0; i < MAXDVBAPI; i++) {
@ -2354,7 +2408,8 @@ cReplayControl::cReplayControl(void)
timeSearchActive = false; timeSearchActive = false;
if (fileName) { if (fileName) {
marks.Load(fileName); marks.Load(fileName);
dvbApi->StartReplay(fileName); if (!dvbApi->StartReplay(fileName))
Interface->Error(tr("Channel locked (recording)!"));
} }
#ifdef DVDSUPPORT #ifdef DVDSUPPORT
else if (dvd) else if (dvd)
@ -2418,10 +2473,7 @@ void cReplayControl::Hide(void)
if (visible) { if (visible) {
Interface->Close(); Interface->Close();
needsFastResponse = visible = false; needsFastResponse = visible = false;
if (!modeOnly) modeOnly = false;
ShowMode();
else
modeOnly = false;
} }
} }
@ -2664,15 +2716,18 @@ void cReplayControl::MarkMove(bool Forward)
void cReplayControl::EditCut(void) void cReplayControl::EditCut(void)
{ {
Hide(); if (fileName) {
if (!cVideoCutter::Active()) { Hide();
if (!cVideoCutter::Start(fileName)) if (!cVideoCutter::Active()) {
Interface->Error(tr("Can't start editing process!")); if (!cVideoCutter::Start(fileName))
Interface->Error(tr("Can't start editing process!"));
else
Interface->Info(tr("Editing process started"));
}
else else
Interface->Info(tr("Editing process started")); Interface->Error(tr("Editing process already active!"));
ShowMode();
} }
else
Interface->Error(tr("Editing process already active!"));
} }
void cReplayControl::EditTest(void) void cReplayControl::EditTest(void)
@ -2700,6 +2755,7 @@ eOSState cReplayControl::ProcessKey(eKeys Key)
if (visible) { if (visible) {
if (timeoutShow && time(NULL) > timeoutShow) { if (timeoutShow && time(NULL) > timeoutShow) {
Hide(); Hide();
ShowMode();
timeoutShow = 0; timeoutShow = 0;
} }
else if (!modeOnly) else if (!modeOnly)
@ -2749,8 +2805,10 @@ eOSState cReplayControl::ProcessKey(eKeys Key)
switch (Key) { switch (Key) {
// Menu control: // Menu control:
case kMenu: Hide(); return osMenu; // allow direct switching to menu case kMenu: Hide(); return osMenu; // allow direct switching to menu
case kOk: if (visible && !modeOnly) case kOk: if (visible && !modeOnly) {
Hide(); Hide();
DoShowMode = true;
}
else else
Show(); Show();
break; break;

3
menu.h
View File

@ -4,7 +4,7 @@
* See the main source file 'vdr.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.33 2001/10/21 14:26:01 kls Exp $ * $Id: menu.h 1.34 2001/10/28 15:21:04 kls Exp $
*/ */
#ifndef _MENU_H #ifndef _MENU_H
@ -92,6 +92,7 @@ public:
static bool Start(cTimer *Timer = NULL); static bool Start(cTimer *Timer = NULL);
static void Stop(const char *InstantId); static void Stop(const char *InstantId);
static void Stop(cDvbApi *DvbApi); static void Stop(cDvbApi *DvbApi);
static bool StopPrimary(bool DoIt = false);
static const char *GetInstantId(const char *LastInstantId); static const char *GetInstantId(const char *LastInstantId);
static void Process(time_t t); static void Process(time_t t);
static bool Active(void); static bool Active(void);

View File

@ -7,7 +7,7 @@
* Parts of this file were inspired by the 'ringbuffy.c' from the * Parts of this file were inspired by the 'ringbuffy.c' from the
* LinuxDVB driver (see linuxtv.org). * LinuxDVB driver (see linuxtv.org).
* *
* $Id: ringbuffer.c 1.4 2001/08/05 12:17:45 kls Exp $ * $Id: ringbuffer.c 1.5 2001/11/03 09:50:46 kls Exp $
*/ */
#include "ringbuffer.h" #include "ringbuffer.h"
@ -215,9 +215,10 @@ int cRingBufferLinear::Get(uchar *Data, int Count)
// --- cFrame ---------------------------------------------------------------- // --- cFrame ----------------------------------------------------------------
cFrame::cFrame(const uchar *Data, int Count, int Index) cFrame::cFrame(const uchar *Data, int Count, eFrameType Type, int Index)
{ {
count = Count; count = Count;
type = Type;
index = Index; index = Index;
data = new uchar[count]; data = new uchar[count];
if (data) if (data)

View File

@ -4,7 +4,7 @@
* See the main source file 'vdr.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: ringbuffer.h 1.4 2001/08/05 11:12:06 kls Exp $ * $Id: ringbuffer.h 1.5 2001/11/03 10:41:33 kls Exp $
*/ */
#ifndef __RINGBUFFER_H #ifndef __RINGBUFFER_H
@ -75,18 +75,22 @@ public:
virtual ~cRingBufferLinear(); virtual ~cRingBufferLinear();
}; };
enum eFrameType { ftUnknown, ftVideo, ftAudio, ftDolby };
class cFrame { class cFrame {
friend class cRingBufferFrame; friend class cRingBufferFrame;
private: private:
cFrame *next; cFrame *next;
uchar *data; uchar *data;
int count; int count;
eFrameType type;
int index; int index;
public: public:
cFrame(const uchar *Data, int Count, int Index = -1); cFrame(const uchar *Data, int Count, eFrameType = ftUnknown, int Index = -1);
~cFrame(); ~cFrame();
const uchar *Data(void) const { return data; } const uchar *Data(void) const { return data; }
int Count(void) const { return count; } int Count(void) const { return count; }
eFrameType Type(void) const { return type; }
int Index(void) const { return index; } int Index(void) const { return index; }
}; };

183
svdrp.c
View File

@ -10,7 +10,7 @@
* and interact with the Video Disk Recorder - or write a full featured * and interact with the Video Disk Recorder - or write a full featured
* graphical interface that sits on top of an SVDRP connection. * graphical interface that sits on top of an SVDRP connection.
* *
* $Id: svdrp.c 1.25 2001/10/07 15:13:42 kls Exp $ * $Id: svdrp.c 1.27 2001/11/04 11:25:05 kls Exp $
*/ */
#include "svdrp.h" #include "svdrp.h"
@ -27,6 +27,7 @@
#include <sys/time.h> #include <sys/time.h>
#include <unistd.h> #include <unistd.h>
#include "config.h" #include "config.h"
#include "dvbapi.h"
#include "interface.h" #include "interface.h"
#include "tools.h" #include "tools.h"
@ -120,6 +121,12 @@ const char *HelpPages[] = {
" it returns the current channel number and name.", " it returns the current channel number and name.",
"DELC <number>\n" "DELC <number>\n"
" Delete channel.", " Delete channel.",
"DELR <number>\n"
" Delete the recording with the given number. Before a recording can be\n"
" deleted, an LSTR command must have been executed in order to retrieve\n"
" the recording numbers. The numbers don't change during subsequent DELR\n"
" commands. CAUTION: THERE IS NO CONFIRMATION PROMPT WHEN DELETING A\n"
" RECORDING - BE SURE YOU KNOW WHAT YOU ARE DOING!",
"DELT <number>\n" "DELT <number>\n"
" Delete timer.", " Delete timer.",
"GRAB <filename> [ jpeg | pnm [ <quality> [ <sizex> <sizey> ] ] ]\n" "GRAB <filename> [ jpeg | pnm [ <quality> [ <sizex> <sizey> ] ] ]\n"
@ -137,6 +144,9 @@ const char *HelpPages[] = {
" containing the given string as part of their name are listed.", " containing the given string as part of their name are listed.",
"LSTE\n" "LSTE\n"
" List EPG data.", " List EPG data.",
"LSTR [ <number> ]\n"
" List recordings. Without option, all recordings are listed. Otherwise\n"
" the summary for the given recording is listed.",
"LSTT [ <number> ]\n" "LSTT [ <number> ]\n"
" List timers. Without option, all timers are listed. Otherwise\n" " List timers. Without option, all timers are listed. Otherwise\n"
" only the given timer is listed.", " only the given timer is listed.",
@ -174,16 +184,6 @@ const char *HelpPages[] = {
" zero, this means that the timer is currently recording and has started\n" " zero, this means that the timer is currently recording and has started\n"
" at the given time. The first value in the resulting line is the number\n" " at the given time. The first value in the resulting line is the number\n"
" of the timer.", " of the timer.",
"OVLF <sizex> <sizey> <fbaddr> <bpp> <palette>\n"
" Set the size, address depth and palette of the overlay.",
"OVLG <sizex> <sizey> <posx> <posy>\n"
" Set the size and position of the overlay.",
"OVLC <clipcount> <base16-CRect-array>\n"
" Set the overlay clipping rectangles.",
"OVLP <brightness> <colour> <hue> <contrast>\n"
" Set the picture parameters for the overlay.",
"OVLO 0 | 1\n"
" Switch the overlay on or off.",
"UPDT <settings>\n" "UPDT <settings>\n"
" Updates a timer. Settings must be in the same format as returned\n" " Updates a timer. Settings must be in the same format as returned\n"
" by the LSTT command. If a timer with the same channel, day, start\n" " by the LSTT command. If a timer with the same channel, day, start\n"
@ -278,7 +278,6 @@ bool cSVDRP::Send(const char *s, int length)
if (wbytes < 0) { if (wbytes < 0) {
LOG_ERROR; LOG_ERROR;
file.Close(); file.Close();
cDvbApi::PrimaryDvbApi->OvlO(false);
} }
else //XXX while...??? else //XXX while...???
esyslog(LOG_ERR, "Wrote %d bytes to client while expecting %d\n", wbytes, length); esyslog(LOG_ERR, "Wrote %d bytes to client while expecting %d\n", wbytes, length);
@ -380,6 +379,27 @@ void cSVDRP::CmdDELC(const char *Option)
Reply(502, "DELC not yet implemented"); Reply(502, "DELC not yet implemented");
} }
void cSVDRP::CmdDELR(const char *Option)
{
if (*Option) {
if (isnumber(Option)) {
cRecording *recording = Recordings.Get(strtol(Option, NULL, 10) - 1);
if (recording) {
if (recording->Delete())
Reply(250, "Recording \"%s\" deleted", Option);
else
Reply(554, "Error while deleting recording!");
}
else
Reply(550, "Recording \"%s\" not found%s", Option, Recordings.Count() ? "" : " (use LSTR before deleting)");
}
else
Reply(501, "Error in recording number \"%s\"", Option);
}
else
Reply(501, "Missing recording number");
}
void cSVDRP::CmdDELT(const char *Option) void cSVDRP::CmdDELT(const char *Option)
{ {
if (*Option) { if (*Option) {
@ -589,6 +609,38 @@ void cSVDRP::CmdLSTE(const char *Option)
Reply(451, "Can't get EPG data"); Reply(451, "Can't get EPG data");
} }
void cSVDRP::CmdLSTR(const char *Option)
{
bool recordings = Recordings.Load();
if (*Option) {
if (isnumber(Option)) {
cRecording *recording = Recordings.Get(strtol(Option, NULL, 10) - 1);
if (recording) {
if (recording->Summary()) {
char *summary = strdup(recording->Summary());
Reply(250, "%s", strreplace(summary,'\n','|'));
delete summary;
}
else
Reply(550, "No summary availabe");
}
else
Reply(550, "Recording \"%s\" not found", Option);
}
else
Reply(501, "Error in recording number \"%s\"", Option);
}
else if (recordings) {
cRecording *recording = Recordings.First();
while (recording) {
Reply(recording == Recordings.Last() ? 250 : -250, "%d %s", recording->Index() + 1, recording->Title(' ', true));
recording = Recordings.Next(recording);
}
}
else
Reply(550, "No recordings available");
}
void cSVDRP::CmdLSTT(const char *Option) void cSVDRP::CmdLSTT(const char *Option)
{ {
if (*Option) { if (*Option) {
@ -767,106 +819,6 @@ void cSVDRP::CmdNEXT(const char *Option)
Reply(550, "No active timers"); Reply(550, "No active timers");
} }
void cSVDRP::CmdOVLF(const char *Option)
{
if (*Option) {
int SizeX = 0, SizeY = 0, Bpp = 0, Palette = 0, FbAddr = 0;
if (5 == sscanf(Option, "%d %d %x %d %d", &SizeX, &SizeY, &FbAddr, &Bpp, &Palette)) {
//somehow_set_overlay_geometry;
if (cDvbApi::PrimaryDvbApi->OvlF(SizeX, SizeY, FbAddr, Bpp, Palette))
Reply(250, "Overlay framebuffer set");
else
Reply(451, "Illegal overlay framebuffer settings");
}
else
Reply(501, "Could not parse overlay framebuffer settings");
}
else
Reply(501, "Missing overlay framebuffer settings");
}
void cSVDRP::CmdOVLG(const char *Option)
{
if (*Option) {
int SizeX = 0, SizeY = 0, PosX = 0, PosY = 0;
if (4 == sscanf(Option, "%d %d %d %d", &SizeX, &SizeY, &PosX, &PosY)) {
//somehow_set_overlay_geometry;
if (cDvbApi::PrimaryDvbApi->OvlG(SizeX, SizeY, PosX, PosY))
Reply(250, "Overlay geometry set");
else
Reply(451, "Illegal overlay geometry settings");
}
else
Reply(501, "Could not parse overlay geometry settings");
}
else
Reply(501, "Missing overlay geometry settings");
}
void cSVDRP::CmdOVLC(const char *Option)
{
if (*Option) {
int ClipCount = 0;
unsigned char s[2 * MAXCLIPRECTS * sizeof(CRect) + 2];
if (2 == sscanf(Option, "%d %s", &ClipCount, s)) {
// Base16-decoding of CRect-array:
unsigned char *p = (unsigned char*)ovlClipRects;
int i = 0, size = sizeof(CRect)*ClipCount;
for (int j = 0; i < size; i++) {
p[i] = (s[j++] - 65);
p[i] += (s[j++] - 65) << 4;
}
if (((unsigned)ClipCount == (i / sizeof(CRect))) && (ClipCount >= 0)) {
// apply it:
if (cDvbApi::PrimaryDvbApi->OvlC(ClipCount, ovlClipRects))
Reply(250, "Overlay-Clipping set");
else
Reply(451, "Illegal overlay clipping settings");
return;
}
}
Reply(501, "Error parsing Overlay-Clipping settings");
}
else
Reply(501, "Missing Clipping settings");
}
void cSVDRP::CmdOVLP(const char *Option)
{
if (*Option) {
int Brightness = 0, Colour = 0, Hue = 0, Contrast = 0;
if (4 == sscanf(Option, "%d %d %d %d", &Brightness, &Colour, &Hue, &Contrast)) {
//somehow_set_overlay_picture_settings;
if (cDvbApi::PrimaryDvbApi->OvlP(Brightness, Colour, Hue, Contrast))
Reply(250, "Overlay picture settings set");
else
Reply(451, "Illegal overlay picture settings");
}
else
Reply(501, "Could not parse overlay picture settings");
}
else
Reply(501, "Missing overlay picture settings");
}
void cSVDRP::CmdOVLO(const char *Option)
{
if (*Option) {
int Value;
if (1 == sscanf(Option, "%d", &Value)) {
//somehow_set_overlay_picture_settings;
if (cDvbApi::PrimaryDvbApi->OvlO(Value))
Reply(250, "Overlay capture set");
else
Reply(451, "Error setting overlay capture");
}
else
Reply(501, "Could not parse status");
}
else
Reply(501, "Missing overlay capture status");
}
void cSVDRP::CmdUPDT(const char *Option) void cSVDRP::CmdUPDT(const char *Option)
{ {
if (*Option) { if (*Option) {
@ -910,12 +862,14 @@ void cSVDRP::Execute(char *Cmd)
s = skipspace(s); s = skipspace(s);
if (CMD("CHAN")) CmdCHAN(s); if (CMD("CHAN")) CmdCHAN(s);
else if (CMD("DELC")) CmdDELC(s); else if (CMD("DELC")) CmdDELC(s);
else if (CMD("DELR")) CmdDELR(s);
else if (CMD("DELT")) CmdDELT(s); else if (CMD("DELT")) CmdDELT(s);
else if (CMD("GRAB")) CmdGRAB(s); else if (CMD("GRAB")) CmdGRAB(s);
else if (CMD("HELP")) CmdHELP(s); else if (CMD("HELP")) CmdHELP(s);
else if (CMD("HITK")) CmdHITK(s); else if (CMD("HITK")) CmdHITK(s);
else if (CMD("LSTC")) CmdLSTC(s); else if (CMD("LSTC")) CmdLSTC(s);
else if (CMD("LSTE")) CmdLSTE(s); else if (CMD("LSTE")) CmdLSTE(s);
else if (CMD("LSTR")) CmdLSTR(s);
else if (CMD("LSTT")) CmdLSTT(s); else if (CMD("LSTT")) CmdLSTT(s);
else if (CMD("MESG")) CmdMESG(s); else if (CMD("MESG")) CmdMESG(s);
else if (CMD("MODC")) CmdMODC(s); else if (CMD("MODC")) CmdMODC(s);
@ -925,11 +879,6 @@ void cSVDRP::Execute(char *Cmd)
else if (CMD("NEWC")) CmdNEWC(s); else if (CMD("NEWC")) CmdNEWC(s);
else if (CMD("NEWT")) CmdNEWT(s); else if (CMD("NEWT")) CmdNEWT(s);
else if (CMD("NEXT")) CmdNEXT(s); else if (CMD("NEXT")) CmdNEXT(s);
else if (CMD("OVLF")) CmdOVLF(s);
else if (CMD("OVLG")) CmdOVLG(s);
else if (CMD("OVLC")) CmdOVLC(s);
else if (CMD("OVLP")) CmdOVLP(s);
else if (CMD("OVLO")) CmdOVLO(s);
else if (CMD("UPDT")) CmdUPDT(s); else if (CMD("UPDT")) CmdUPDT(s);
else if (CMD("QUIT")) Close(); else if (CMD("QUIT")) Close();
else Reply(500, "Command unrecognized: \"%s\"", Cmd); else Reply(500, "Command unrecognized: \"%s\"", Cmd);

13
svdrp.h
View File

@ -4,13 +4,13 @@
* See the main source file 'vdr.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: svdrp.h 1.11 2001/09/14 14:35:34 kls Exp $ * $Id: svdrp.h 1.13 2001/11/04 11:20:46 kls Exp $
*/ */
#ifndef __SVDRP_H #ifndef __SVDRP_H
#define __SVDRP_H #define __SVDRP_H
#include "dvbapi.h" #include "recording.h"
#include "tools.h" #include "tools.h"
class cSocket { class cSocket {
@ -30,7 +30,7 @@ class cSVDRP {
private: private:
cSocket socket; cSocket socket;
cFile file; cFile file;
CRect ovlClipRects[MAXCLIPRECTS]; cRecordings Recordings;
uint numChars; uint numChars;
char cmdLine[MAXPARSEBUFFER]; char cmdLine[MAXPARSEBUFFER];
char *message; char *message;
@ -40,12 +40,14 @@ private:
void Reply(int Code, const char *fmt, ...); void Reply(int Code, const char *fmt, ...);
void CmdCHAN(const char *Option); void CmdCHAN(const char *Option);
void CmdDELC(const char *Option); void CmdDELC(const char *Option);
void CmdDELR(const char *Option);
void CmdDELT(const char *Option); void CmdDELT(const char *Option);
void CmdGRAB(const char *Option); void CmdGRAB(const char *Option);
void CmdHELP(const char *Option); void CmdHELP(const char *Option);
void CmdHITK(const char *Option); void CmdHITK(const char *Option);
void CmdLSTC(const char *Option); void CmdLSTC(const char *Option);
void CmdLSTE(const char *Option); void CmdLSTE(const char *Option);
void CmdLSTR(const char *Option);
void CmdLSTT(const char *Option); void CmdLSTT(const char *Option);
void CmdMESG(const char *Option); void CmdMESG(const char *Option);
void CmdMODC(const char *Option); void CmdMODC(const char *Option);
@ -55,11 +57,6 @@ private:
void CmdNEWC(const char *Option); void CmdNEWC(const char *Option);
void CmdNEWT(const char *Option); void CmdNEWT(const char *Option);
void CmdNEXT(const char *Option); void CmdNEXT(const char *Option);
void CmdOVLF(const char *Option);
void CmdOVLG(const char *Option);
void CmdOVLC(const char *Option);
void CmdOVLP(const char *Option);
void CmdOVLO(const char *Option);
void CmdUPDT(const char *Option); void CmdUPDT(const char *Option);
void Execute(char *Cmd); void Execute(char *Cmd);
public: public:

View File

@ -4,7 +4,7 @@
* See the main source file 'vdr.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: thread.c 1.15 2001/10/21 12:25:31 kls Exp $ * $Id: thread.c 1.16 2001/10/27 13:23:06 kls Exp $
*/ */
#include "thread.h" #include "thread.h"
@ -99,8 +99,7 @@ cThread::cThread(void)
signalHandlerInstalled = true; signalHandlerInstalled = true;
} }
running = false; running = false;
parentPid = threadPid = lockingPid = 0; parentPid = threadPid = 0;
locked = 0;
} }
cThread::~cThread() cThread::~cThread()
@ -159,24 +158,6 @@ void cThread::Cancel(int WaitSeconds)
pthread_cancel(thread); pthread_cancel(thread);
} }
bool cThread::Lock(void)
{
if (getpid() != lockingPid || !locked) {
Mutex.Lock();
lockingPid = getpid();
}
locked++;
return true;
}
void cThread::Unlock(void)
{
if (!--locked) {
lockingPid = 0;
Mutex.Unlock();
}
}
void cThread::WakeUp(void) void cThread::WakeUp(void)
{ {
kill(parentPid, SIGIO); // makes any waiting 'select()' call return immediately kill(parentPid, SIGIO); // makes any waiting 'select()' call return immediately
@ -228,17 +209,13 @@ bool cThreadLock::Lock(cThread *Thread)
{ {
if (Thread && !thread) { if (Thread && !thread) {
thread = Thread; thread = Thread;
locked = Thread->Lock(); Thread->Lock();
return locked; locked = true;
return true;
} }
return false; return false;
} }
bool cThreadLock::Locked(void)
{
return locked;
}
// --- cPipe ----------------------------------------------------------------- // --- cPipe -----------------------------------------------------------------
// cPipe::Open() and cPipe::Close() are based on code originally received from // cPipe::Open() and cPipe::Close() are based on code originally received from

View File

@ -4,7 +4,7 @@
* See the main source file 'vdr.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: thread.h 1.10 2001/10/20 10:25:19 kls Exp $ * $Id: thread.h 1.11 2001/10/27 13:22:20 kls Exp $
*/ */
#ifndef __THREAD_H #ifndef __THREAD_H
@ -45,9 +45,8 @@ class cThread {
friend class cThreadLock; friend class cThreadLock;
private: private:
pthread_t thread; pthread_t thread;
cMutex Mutex; cMutex mutex;
pid_t parentPid, threadPid, lockingPid; pid_t parentPid, threadPid;
int locked;
bool running; bool running;
static time_t lastPanic; static time_t lastPanic;
static int panicLevel; static int panicLevel;
@ -55,8 +54,8 @@ private:
static bool signalHandlerInstalled; static bool signalHandlerInstalled;
static void SignalHandler(int signum); static void SignalHandler(int signum);
static void *StartThread(cThread *Thread); static void *StartThread(cThread *Thread);
bool Lock(void); void Lock(void) { mutex.Lock(); }
void Unlock(void); void Unlock(void) { mutex.Unlock(); }
protected: protected:
void WakeUp(void); void WakeUp(void);
virtual void Action(void) = 0; virtual void Action(void) = 0;
@ -84,7 +83,6 @@ public:
cThreadLock(cThread *Thread = NULL); cThreadLock(cThread *Thread = NULL);
~cThreadLock(); ~cThreadLock();
bool Lock(cThread *Thread); bool Lock(cThread *Thread);
bool Locked(void);
}; };
#define LOCK_THREAD cThreadLock ThreadLock(this) #define LOCK_THREAD cThreadLock ThreadLock(this)

10
vdr.c
View File

@ -22,7 +22,7 @@
* *
* The project's page is at http://www.cadsoft.de/people/kls/vdr * The project's page is at http://www.cadsoft.de/people/kls/vdr
* *
* $Id: vdr.c 1.86 2001/10/20 11:18:38 kls Exp $ * $Id: vdr.c 1.89 2001/11/03 12:23:45 kls Exp $
*/ */
#include <getopt.h> #include <getopt.h>
@ -483,7 +483,7 @@ int main(int argc, char *argv[])
time_t Now = time(NULL); time_t Now = time(NULL);
if (Now - LastActivity > ACTIVITYTIMEOUT) { if (Now - LastActivity > ACTIVITYTIMEOUT) {
// Shutdown: // Shutdown:
if (Shutdown && Setup.MinUserInactivity && Now - LastActivity > Setup.MinUserInactivity * 60) { if (Shutdown && (Setup.MinUserInactivity || LastActivity == 1) && Now - LastActivity > Setup.MinUserInactivity * 60) {
cTimer *timer = Timers.GetNextActiveTimer(); cTimer *timer = Timers.GetNextActiveTimer();
time_t Next = timer ? timer->StartTime() : 0; time_t Next = timer ? timer->StartTime() : 0;
time_t Delta = timer ? Next - Now : 0; time_t Delta = timer ? Next - Now : 0;
@ -503,11 +503,12 @@ int main(int argc, char *argv[])
dsyslog(LOG_INFO, "next timer event at %s", ctime(&Next)); dsyslog(LOG_INFO, "next timer event at %s", ctime(&Next));
if (WatchdogTimeout > 0) if (WatchdogTimeout > 0)
signal(SIGALRM, SIG_IGN); signal(SIGALRM, SIG_IGN);
if (Interface->Confirm(tr("Press any key to cancel shutdown"), LastActivity == 1 ? 5 : SHUTDOWNWAIT, true)) { bool UserShutdown = key == kPower;
if (Interface->Confirm(tr("Press any key to cancel shutdown"), UserShutdown ? 5 : SHUTDOWNWAIT, true)) {
int Channel = timer ? timer->channel : 0; int Channel = timer ? timer->channel : 0;
const char *File = timer ? timer->file : ""; const char *File = timer ? timer->file : "";
char *cmd; char *cmd;
asprintf(&cmd, "%s %ld %ld %d '%s'", Shutdown, Next, Delta, Channel, File); asprintf(&cmd, "%s %ld %ld %d '%s' %d", Shutdown, Next, Delta, Channel, File, UserShutdown);
isyslog(LOG_INFO, "executing '%s'", cmd); isyslog(LOG_INFO, "executing '%s'", cmd);
SystemExec(cmd); SystemExec(cmd);
delete cmd; delete cmd;
@ -529,6 +530,7 @@ int main(int argc, char *argv[])
if (Interrupted) if (Interrupted)
isyslog(LOG_INFO, "caught signal %d", Interrupted); isyslog(LOG_INFO, "caught signal %d", Interrupted);
Setup.CurrentChannel = cDvbApi::CurrentChannel(); Setup.CurrentChannel = cDvbApi::CurrentChannel();
Setup.CurrentVolume = cDvbApi::CurrentVolume();
Setup.Save(); Setup.Save();
cVideoCutter::Stop(); cVideoCutter::Stop();
delete Menu; delete Menu;