2000-02-19 13:36:48 +01:00
|
|
|
/*
|
2000-04-24 09:46:05 +02:00
|
|
|
* vdr.c: Video Disk Recorder main program
|
2000-02-19 13:36:48 +01:00
|
|
|
*
|
|
|
|
* 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
|
|
|
|
*
|
2000-09-10 10:51:58 +02:00
|
|
|
* $Id: vdr.c 1.29 2000/09/10 10:42:32 kls Exp $
|
2000-02-19 13:36:48 +01:00
|
|
|
*/
|
|
|
|
|
2000-07-23 15:01:31 +02:00
|
|
|
#include <getopt.h>
|
2000-04-15 17:38:11 +02:00
|
|
|
#include <signal.h>
|
2000-07-23 15:01:31 +02:00
|
|
|
#include <stdlib.h>
|
2000-07-23 15:36:43 +02:00
|
|
|
#include <unistd.h>
|
2000-02-19 13:36:48 +01:00
|
|
|
#include "config.h"
|
2000-05-01 16:29:46 +02:00
|
|
|
#include "dvbapi.h"
|
2000-02-19 13:36:48 +01:00
|
|
|
#include "interface.h"
|
|
|
|
#include "menu.h"
|
2000-03-11 11:22:37 +01:00
|
|
|
#include "recording.h"
|
2000-07-23 15:01:31 +02:00
|
|
|
#include "svdrp.h"
|
2000-02-19 13:36:48 +01:00
|
|
|
#include "tools.h"
|
2000-07-29 15:21:42 +02:00
|
|
|
#include "videodir.h"
|
2000-02-19 13:36:48 +01:00
|
|
|
|
2000-07-15 12:39:20 +02:00
|
|
|
#ifdef REMOTE_KBD
|
2000-02-19 13:36:48 +01:00
|
|
|
#define KEYS_CONF "keys-pc.conf"
|
|
|
|
#else
|
|
|
|
#define KEYS_CONF "keys.conf"
|
|
|
|
#endif
|
|
|
|
|
2000-04-16 15:50:21 +02:00
|
|
|
#define DIRECTCHANNELTIMEOUT 500 //ms
|
|
|
|
|
2000-04-15 17:38:11 +02:00
|
|
|
static int Interrupted = 0;
|
|
|
|
|
|
|
|
void SignalHandler(int signum)
|
|
|
|
{
|
|
|
|
Interrupted = signum;
|
|
|
|
}
|
|
|
|
|
2000-09-10 10:51:58 +02:00
|
|
|
static eKeys ShowChannel(int Number, bool Switched, bool Group = false)
|
2000-09-09 14:57:43 +02:00
|
|
|
{
|
|
|
|
cChannel *channel = Group ? Channels.Get(Number) : Channels.GetByNumber(Number);
|
|
|
|
if (channel)
|
2000-09-10 10:51:58 +02:00
|
|
|
return Interface.DisplayChannel(channel->number, channel->name, !Switched || Setup.ShowInfoOnChSwitch);
|
2000-09-09 14:57:43 +02:00
|
|
|
return kNone;
|
|
|
|
}
|
|
|
|
|
2000-02-19 13:36:48 +01:00
|
|
|
int main(int argc, char *argv[])
|
|
|
|
{
|
2000-07-23 15:01:31 +02:00
|
|
|
// Command line options:
|
|
|
|
|
|
|
|
#define DEFAULTSVDRPPORT 2001
|
|
|
|
|
|
|
|
int SVDRPport = DEFAULTSVDRPPORT;
|
2000-07-23 15:36:43 +02:00
|
|
|
bool DaemonMode = false;
|
2000-07-23 15:01:31 +02:00
|
|
|
|
|
|
|
static struct option long_options[] = {
|
2000-07-23 15:36:43 +02:00
|
|
|
{ "daemon", no_argument, NULL, 'd' },
|
|
|
|
{ "help", no_argument, NULL, 'h' },
|
2000-07-29 19:03:09 +02:00
|
|
|
{ "log", required_argument, NULL, 'l' },
|
2000-07-23 15:36:43 +02:00
|
|
|
{ "port", required_argument, NULL, 'p' },
|
2000-07-28 13:44:31 +02:00
|
|
|
{ "video", required_argument, NULL, 'v' },
|
2000-07-23 15:01:31 +02:00
|
|
|
{ 0 }
|
|
|
|
};
|
|
|
|
|
|
|
|
int c;
|
|
|
|
int option_index = 0;
|
2000-07-29 19:03:09 +02:00
|
|
|
while ((c = getopt_long(argc, argv, "dhl:p:v:", long_options, &option_index)) != -1) {
|
2000-07-23 15:01:31 +02:00
|
|
|
switch (c) {
|
2000-07-23 15:36:43 +02:00
|
|
|
case 'd': DaemonMode = true; break;
|
2000-07-23 15:01:31 +02:00
|
|
|
case 'h': printf("Usage: vdr [OPTION]\n\n"
|
2000-07-29 19:03:09 +02:00
|
|
|
" -h, --help display this help and exit\n"
|
|
|
|
" -d, --daemon run in daemon mode\n"
|
|
|
|
" -l LEVEL, --log=LEVEL set log level (default: 3)\n"
|
|
|
|
" 0 = no logging, 1 = errors only,\n"
|
|
|
|
" 2 = errors and info, 3 = errors, info and debug\n"
|
|
|
|
" -p PORT, --port=PORT use PORT for SVDRP (default: %d)\n"
|
|
|
|
" 0 turns off SVDRP\n"
|
|
|
|
" -v DIR, --video=DIR use DIR as video directory (default is %s)\n"
|
2000-07-23 15:01:31 +02:00
|
|
|
"\n"
|
2000-07-28 13:44:31 +02:00
|
|
|
"Report bugs to <vdr-bugs@cadsoft.de>\n",
|
|
|
|
DEFAULTSVDRPPORT,
|
|
|
|
VideoDirectory
|
2000-07-23 15:01:31 +02:00
|
|
|
);
|
|
|
|
return 0;
|
|
|
|
break;
|
2000-07-29 19:03:09 +02:00
|
|
|
case 'l': if (isnumber(optarg)) {
|
|
|
|
int l = atoi(optarg);
|
|
|
|
if (0 <= l && l <= 3) {
|
|
|
|
SysLogLevel = l;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
fprintf(stderr, "vdr: invalid log level: %s\n", optarg);
|
|
|
|
abort();
|
|
|
|
break;
|
2000-07-23 15:01:31 +02:00
|
|
|
case 'p': if (isnumber(optarg))
|
2000-07-29 19:03:09 +02:00
|
|
|
SVDRPport = atoi(optarg);
|
2000-07-23 15:01:31 +02:00
|
|
|
else {
|
|
|
|
fprintf(stderr, "vdr: invalid port number: %s\n", optarg);
|
2000-07-28 13:44:31 +02:00
|
|
|
abort();
|
2000-07-23 15:01:31 +02:00
|
|
|
}
|
|
|
|
break;
|
2000-07-28 13:44:31 +02:00
|
|
|
case 'v': VideoDirectory = optarg;
|
|
|
|
break;
|
2000-07-23 15:01:31 +02:00
|
|
|
default: abort();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Log file:
|
|
|
|
|
2000-07-29 19:03:09 +02:00
|
|
|
if (SysLogLevel > 0)
|
|
|
|
openlog("vdr", LOG_PID | LOG_CONS, LOG_USER);
|
2000-07-23 15:36:43 +02:00
|
|
|
|
2000-07-28 13:44:31 +02:00
|
|
|
// Check the video directory:
|
|
|
|
|
2000-07-29 15:21:42 +02:00
|
|
|
if (!DirectoryOk(VideoDirectory, true)) {
|
2000-07-28 13:44:31 +02:00
|
|
|
fprintf(stderr, "vdr: can't access video directory %s\n", VideoDirectory);
|
|
|
|
abort();
|
|
|
|
}
|
|
|
|
|
2000-07-23 15:36:43 +02:00
|
|
|
// Daemon mode:
|
|
|
|
|
|
|
|
if (DaemonMode) {
|
|
|
|
#ifndef DEBUG_OSD
|
|
|
|
pid_t pid = fork();
|
|
|
|
if (pid < 0) {
|
|
|
|
fprintf(stderr, "%s\n", strerror(errno));
|
2000-07-29 19:03:09 +02:00
|
|
|
esyslog(LOG_ERR, "ERROR: %s", strerror(errno));
|
2000-07-28 13:44:31 +02:00
|
|
|
abort();
|
2000-07-23 15:36:43 +02:00
|
|
|
}
|
|
|
|
if (pid != 0)
|
|
|
|
return 0; // initial program immediately returns
|
|
|
|
fclose(stdin);
|
|
|
|
fclose(stdout);
|
|
|
|
fclose(stderr);
|
|
|
|
#else
|
|
|
|
fprintf(stderr, "vdr: can't run in daemon mode with DEBUG_OSD on!\n");
|
|
|
|
abort();
|
|
|
|
#endif
|
|
|
|
}
|
2000-07-29 18:19:49 +02:00
|
|
|
isyslog(LOG_INFO, "VDR version %s started", VDRVERSION);
|
2000-02-19 13:36:48 +01:00
|
|
|
|
2000-07-23 15:01:31 +02:00
|
|
|
// DVB interfaces:
|
|
|
|
|
2000-05-01 16:29:46 +02:00
|
|
|
if (!cDvbApi::Init())
|
2000-07-28 13:44:31 +02:00
|
|
|
abort();
|
2000-05-01 16:29:46 +02:00
|
|
|
|
2000-07-23 15:01:31 +02:00
|
|
|
// Configuration data:
|
|
|
|
|
2000-09-10 10:51:58 +02:00
|
|
|
Setup.Load("setup.conf");
|
2000-02-19 13:36:48 +01:00
|
|
|
Channels.Load("channels.conf");
|
|
|
|
Timers.Load("timers.conf");
|
2000-07-15 16:35:18 +02:00
|
|
|
#ifdef REMOTE_LIRC
|
|
|
|
Keys.SetDummyValues();
|
|
|
|
#else
|
2000-02-19 13:36:48 +01:00
|
|
|
if (!Keys.Load(KEYS_CONF))
|
|
|
|
Interface.LearnKeys();
|
2000-07-15 12:39:20 +02:00
|
|
|
#endif
|
2000-02-19 13:36:48 +01:00
|
|
|
Interface.Init();
|
|
|
|
|
2000-09-10 10:51:58 +02:00
|
|
|
cDvbApi::SetPrimaryDvbApi(Setup.PrimaryDVB);
|
|
|
|
|
2000-09-09 14:57:43 +02:00
|
|
|
Channels.SwitchTo(CurrentChannel);
|
2000-02-19 13:36:48 +01:00
|
|
|
|
2000-07-23 15:01:31 +02:00
|
|
|
// Signal handlers:
|
|
|
|
|
2000-04-15 17:38:11 +02:00
|
|
|
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);
|
|
|
|
|
2000-07-23 15:01:31 +02:00
|
|
|
// Main program loop:
|
|
|
|
|
|
|
|
cSVDRP *SVDRP = SVDRPport ? new cSVDRP(SVDRPport) : NULL;
|
2000-02-19 13:36:48 +01:00
|
|
|
cMenuMain *Menu = NULL;
|
2000-04-29 15:57:42 +02:00
|
|
|
cReplayControl *ReplayControl = NULL;
|
2000-04-16 15:50:21 +02:00
|
|
|
int dcTime = 0, dcNumber = 0;
|
2000-04-22 13:51:48 +02:00
|
|
|
int LastChannel = -1;
|
2000-02-19 13:36:48 +01:00
|
|
|
|
2000-04-15 17:38:11 +02:00
|
|
|
while (!Interrupted) {
|
2000-04-22 13:51:48 +02:00
|
|
|
// Channel display:
|
|
|
|
if (CurrentChannel != LastChannel) {
|
2000-09-09 14:57:43 +02:00
|
|
|
if (!Menu)
|
2000-09-10 10:51:58 +02:00
|
|
|
ShowChannel(CurrentChannel, LastChannel > 0);
|
2000-04-22 13:51:48 +02:00
|
|
|
LastChannel = CurrentChannel;
|
|
|
|
}
|
2000-04-16 15:50:21 +02:00
|
|
|
// Direct Channel Select (action):
|
2000-04-29 15:57:42 +02:00
|
|
|
if (dcNumber && time_ms() - dcTime > DIRECTCHANNELTIMEOUT) {
|
2000-09-09 14:57:43 +02:00
|
|
|
Channels.SwitchTo(dcNumber);
|
2000-04-29 15:57:42 +02:00
|
|
|
dcNumber = 0;
|
|
|
|
LastChannel = -1; // in case an invalid channel number was entered!
|
2000-04-15 17:38:11 +02:00
|
|
|
}
|
2000-05-01 16:29:46 +02:00
|
|
|
// Timers and Recordings:
|
|
|
|
if (!Menu) {
|
2000-04-30 10:22:13 +02:00
|
|
|
cTimer *Timer = cTimer::GetMatch();
|
|
|
|
if (Timer) {
|
2000-05-01 16:29:46 +02:00
|
|
|
if (!cRecordControls::Start(Timer)) {
|
|
|
|
//TODO need to do something to prevent the timer from hitting over and over again...
|
|
|
|
}
|
2000-04-15 17:38:11 +02:00
|
|
|
}
|
2000-05-01 16:29:46 +02:00
|
|
|
cRecordControls::Process();
|
2000-04-15 17:38:11 +02:00
|
|
|
}
|
2000-04-16 15:50:21 +02:00
|
|
|
// User Input:
|
2000-04-29 15:57:42 +02:00
|
|
|
cOsdBase **Interact = Menu ? (cOsdBase **)&Menu : (cOsdBase **)&ReplayControl;
|
2000-04-30 10:22:13 +02:00
|
|
|
eKeys key = Interface.GetKey(!*Interact || !(*Interact)->NeedsFastResponse());
|
2000-04-29 15:57:42 +02:00
|
|
|
if (*Interact) {
|
|
|
|
switch ((*Interact)->ProcessKey(key)) {
|
|
|
|
case osMenu: DELETENULL(Menu);
|
2000-05-01 16:29:46 +02:00
|
|
|
Menu = new cMenuMain(ReplayControl);
|
2000-04-29 15:57:42 +02:00
|
|
|
break;
|
2000-05-27 16:10:47 +02:00
|
|
|
case osRecord: DELETENULL(Menu);
|
|
|
|
if (!cRecordControls::Start())
|
|
|
|
Interface.Error("No free DVB device to record!");
|
|
|
|
break;
|
2000-04-29 15:57:42 +02:00
|
|
|
case osReplay: DELETENULL(Menu);
|
|
|
|
DELETENULL(ReplayControl);
|
|
|
|
ReplayControl = new cReplayControl;
|
|
|
|
break;
|
2000-05-01 16:29:46 +02:00
|
|
|
case osStopReplay:
|
|
|
|
DELETENULL(*Interact);
|
|
|
|
DELETENULL(ReplayControl);
|
|
|
|
break;
|
2000-09-10 10:51:58 +02:00
|
|
|
case osSwitchDvb:
|
|
|
|
DELETENULL(*Interact);
|
|
|
|
Interface.Info("Switching primary DVB...");
|
|
|
|
cDvbApi::SetPrimaryDvbApi(Setup.PrimaryDVB);
|
|
|
|
break;
|
2000-04-15 17:38:11 +02:00
|
|
|
case osBack:
|
2000-04-29 15:57:42 +02:00
|
|
|
case osEnd: DELETENULL(*Interact);
|
|
|
|
break;
|
|
|
|
default: ;
|
2000-04-15 17:38:11 +02:00
|
|
|
}
|
2000-02-19 13:36:48 +01:00
|
|
|
}
|
2000-04-29 15:57:42 +02:00
|
|
|
else {
|
2000-04-15 17:38:11 +02:00
|
|
|
switch (key) {
|
2000-04-16 15:50:21 +02:00
|
|
|
// Direct Channel Select (input):
|
|
|
|
case k0: case k1: case k2: case k3: case k4: case k5: case k6: case k7: case k8: case k9:
|
2000-04-29 15:57:42 +02:00
|
|
|
{
|
2000-05-01 16:29:46 +02:00
|
|
|
if (!Interface.Recording()) {
|
2000-04-29 15:57:42 +02:00
|
|
|
dcNumber = dcNumber * 10 + key - k0;
|
|
|
|
dcTime = time_ms();
|
|
|
|
Interface.DisplayChannel(dcNumber);
|
|
|
|
}
|
2000-04-22 09:56:33 +02:00
|
|
|
}
|
2000-04-29 15:57:42 +02:00
|
|
|
break;
|
2000-09-09 14:57:43 +02:00
|
|
|
// Left/Right rotates trough channel groups:
|
|
|
|
case kLeft:
|
|
|
|
case kRight: if (!Interface.Recording()) {
|
|
|
|
int SaveGroup = CurrentGroup;
|
|
|
|
if (key == kRight)
|
|
|
|
CurrentGroup = Channels.GetNextGroup(CurrentGroup) ;
|
|
|
|
else
|
|
|
|
CurrentGroup = Channels.GetPrevGroup(CurrentGroup < 1 ? 1 : CurrentGroup);
|
|
|
|
if (CurrentGroup < 0)
|
|
|
|
CurrentGroup = SaveGroup;
|
2000-09-10 10:51:58 +02:00
|
|
|
if (ShowChannel(CurrentGroup, false, true) == kOk)
|
2000-09-09 14:57:43 +02:00
|
|
|
Channels.SwitchTo(Channels.Get(Channels.GetNextNormal(CurrentGroup))->number);
|
|
|
|
}
|
|
|
|
break;
|
2000-04-22 13:51:48 +02:00
|
|
|
// Up/Down Channel Select:
|
2000-04-15 17:38:11 +02:00
|
|
|
case kUp:
|
2000-05-01 16:29:46 +02:00
|
|
|
case kDown: if (!Interface.Recording()) {
|
2000-04-24 10:36:14 +02:00
|
|
|
int n = CurrentChannel + (key == kUp ? 1 : -1);
|
2000-09-09 14:57:43 +02:00
|
|
|
cChannel *channel = Channels.GetByNumber(n);
|
2000-04-24 10:36:14 +02:00
|
|
|
if (channel)
|
|
|
|
channel->Switch();
|
|
|
|
}
|
2000-04-15 17:38:11 +02:00
|
|
|
break;
|
2000-05-01 16:29:46 +02:00
|
|
|
// Menu Control:
|
|
|
|
case kMenu: Menu = new cMenuMain(ReplayControl); break;
|
2000-04-22 13:51:48 +02:00
|
|
|
// Viewing Control:
|
2000-04-29 15:57:42 +02:00
|
|
|
case kOk: LastChannel = -1; break; // forces channel display
|
2000-04-15 17:38:11 +02:00
|
|
|
default: break;
|
|
|
|
}
|
2000-02-19 13:36:48 +01:00
|
|
|
}
|
2000-07-23 15:01:31 +02:00
|
|
|
if (SVDRP)
|
|
|
|
SVDRP->Process();//TODO lock menu vs. SVDRP?
|
2000-04-15 17:38:11 +02:00
|
|
|
}
|
|
|
|
isyslog(LOG_INFO, "caught signal %d", Interrupted);
|
2000-04-30 10:22:13 +02:00
|
|
|
delete Menu;
|
|
|
|
delete ReplayControl;
|
2000-07-23 15:01:31 +02:00
|
|
|
delete SVDRP;
|
2000-05-01 16:29:46 +02:00
|
|
|
cDvbApi::Cleanup();
|
2000-04-16 13:54:16 +02:00
|
|
|
isyslog(LOG_INFO, "exiting");
|
2000-07-29 19:03:09 +02:00
|
|
|
if (SysLogLevel > 0)
|
|
|
|
closelog();
|
2000-04-15 17:38:11 +02:00
|
|
|
return 0;
|
2000-02-19 13:36:48 +01:00
|
|
|
}
|