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-12-28 12:57:16 +01:00
|
|
|
* $Id: vdr.c 1.48 2000/12/25 09:43:08 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-11-11 10:39:27 +01:00
|
|
|
#include "i18n.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-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-15 17:38:11 +02:00
|
|
|
static int Interrupted = 0;
|
|
|
|
|
2000-09-15 13:58:36 +02:00
|
|
|
static void SignalHandler(int signum)
|
2000-04-15 17:38:11 +02:00
|
|
|
{
|
2000-09-15 13:58:36 +02:00
|
|
|
if (signum != SIGPIPE)
|
|
|
|
Interrupted = signum;
|
|
|
|
signal(signum, SignalHandler);
|
2000-04-15 17:38:11 +02:00
|
|
|
}
|
|
|
|
|
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-09-15 15:09:15 +02:00
|
|
|
const char *ConfigDirectory = NULL;
|
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-09-15 15:09:15 +02:00
|
|
|
{ "config", required_argument, NULL, 'c' },
|
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-09-15 15:09:15 +02:00
|
|
|
while ((c = getopt_long(argc, argv, "c:dhl:p:v:", long_options, &option_index)) != -1) {
|
2000-07-23 15:01:31 +02:00
|
|
|
switch (c) {
|
2000-09-15 15:09:15 +02:00
|
|
|
case 'c': ConfigDirectory = optarg;
|
|
|
|
break;
|
2000-07-23 15:36:43 +02:00
|
|
|
case 'd': DaemonMode = true; break;
|
2000-09-15 15:09:15 +02:00
|
|
|
case 'h': printf("Usage: vdr [OPTION]\n\n" // for easier orientation, this is column 80|
|
|
|
|
" -c DIR, --config=DIR read config files from DIR (default is to read them\n"
|
|
|
|
" from the video directory)\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;
|
2000-09-17 14:18:14 +02:00
|
|
|
while (optarg && *optarg && optarg[strlen(optarg) - 1] == '/')
|
|
|
|
optarg[strlen(optarg) - 1] = 0;
|
2000-07-28 13:44:31 +02:00
|
|
|
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) {
|
2000-09-20 16:49:49 +02:00
|
|
|
#if !defined(DEBUG_OSD) && !defined(REMOTE_KBD)
|
2000-07-23 15:36:43 +02:00
|
|
|
pid_t pid = fork();
|
|
|
|
if (pid < 0) {
|
2000-12-08 16:23:32 +01:00
|
|
|
fprintf(stderr, "%m\n");
|
|
|
|
esyslog(LOG_ERR, "ERROR: %m");
|
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
|
2000-09-20 16:49:49 +02:00
|
|
|
fprintf(stderr, "vdr: can't run in daemon mode with DEBUG_OSD or REMOTE_KBD on!\n");
|
2000-07-23 15:36:43 +02:00
|
|
|
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
|
|
|
// Configuration data:
|
|
|
|
|
2000-09-15 15:09:15 +02:00
|
|
|
if (!ConfigDirectory)
|
|
|
|
ConfigDirectory = VideoDirectory;
|
|
|
|
|
|
|
|
Setup.Load(AddDirectory(ConfigDirectory, "setup.conf"));
|
|
|
|
Channels.Load(AddDirectory(ConfigDirectory, "channels.conf"));
|
|
|
|
Timers.Load(AddDirectory(ConfigDirectory, "timers.conf"));
|
2000-11-11 16:38:41 +01:00
|
|
|
Commands.Load(AddDirectory(ConfigDirectory, "commands.conf"));
|
2000-07-15 16:35:18 +02:00
|
|
|
#ifdef REMOTE_LIRC
|
|
|
|
Keys.SetDummyValues();
|
|
|
|
#else
|
2000-10-29 14:00:00 +01:00
|
|
|
bool KeysLoaded = Keys.Load(AddDirectory(ConfigDirectory, KEYS_CONF));
|
2000-07-15 12:39:20 +02:00
|
|
|
#endif
|
2000-02-19 13:36:48 +01:00
|
|
|
|
2000-11-03 15:31:03 +01:00
|
|
|
// DVB interfaces:
|
|
|
|
|
|
|
|
if (!cDvbApi::Init())
|
|
|
|
abort();
|
|
|
|
|
2000-09-10 10:51:58 +02:00
|
|
|
cDvbApi::SetPrimaryDvbApi(Setup.PrimaryDVB);
|
|
|
|
|
2000-11-05 18:39:17 +01:00
|
|
|
Channels.SwitchTo(1);
|
2000-02-19 13:36:48 +01:00
|
|
|
|
2000-11-18 13:57:32 +01:00
|
|
|
cEITScanner EITScanner;
|
|
|
|
|
2000-10-29 14:00:00 +01:00
|
|
|
// User interface:
|
|
|
|
|
|
|
|
Interface = new cInterface(SVDRPport);
|
|
|
|
#ifndef REMOTE_LIRC
|
|
|
|
if (!KeysLoaded)
|
|
|
|
Interface->LearnKeys();
|
|
|
|
#endif
|
|
|
|
|
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-09-15 13:58:36 +02:00
|
|
|
if (signal(SIGPIPE, SignalHandler) == SIG_IGN) signal(SIGPIPE, SIG_IGN);
|
2000-04-15 17:38:11 +02:00
|
|
|
|
2000-07-23 15:01:31 +02:00
|
|
|
// Main program loop:
|
|
|
|
|
2000-09-10 14:56:18 +02:00
|
|
|
cOsdBase *Menu = NULL;
|
2000-04-29 15:57:42 +02:00
|
|
|
cReplayControl *ReplayControl = NULL;
|
2000-04-22 13:51:48 +02:00
|
|
|
int LastChannel = -1;
|
2000-11-05 18:39:17 +01:00
|
|
|
int PreviousChannel = cDvbApi::CurrentChannel();
|
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:
|
2000-11-18 13:57:32 +01:00
|
|
|
if (!EITScanner.Active() && cDvbApi::CurrentChannel() != LastChannel) {
|
2000-09-09 14:57:43 +02:00
|
|
|
if (!Menu)
|
2000-11-05 18:39:17 +01:00
|
|
|
Menu = new cDisplayChannel(cDvbApi::CurrentChannel(), LastChannel > 0);
|
2000-10-03 14:06:44 +02:00
|
|
|
PreviousChannel = LastChannel;
|
2000-11-05 18:39:17 +01:00
|
|
|
LastChannel = cDvbApi::CurrentChannel();
|
2000-04-22 13:51:48 +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-09-10 14:56:18 +02:00
|
|
|
cOsdBase **Interact = Menu ? &Menu : (cOsdBase **)&ReplayControl;
|
2000-10-08 12:24:30 +02:00
|
|
|
eKeys key = Interface->GetKey(!*Interact || !(*Interact)->NeedsFastResponse());
|
2000-11-18 13:57:32 +01:00
|
|
|
if (NORMALKEY(key) != kNone)
|
|
|
|
EITScanner.Activity();
|
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())
|
2000-11-11 10:39:27 +01:00
|
|
|
Interface->Error(tr("No free DVB device to record!"));
|
2000-05-27 16:10:47 +02:00
|
|
|
break;
|
2000-10-08 15:08:26 +02:00
|
|
|
case osRecordings:
|
|
|
|
DELETENULL(Menu);
|
|
|
|
DELETENULL(ReplayControl);
|
|
|
|
Menu = new cMenuRecordings;
|
|
|
|
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);
|
2000-11-11 10:39:27 +01:00
|
|
|
Interface->Info(tr("Switching primary DVB..."));
|
2000-09-10 10:51:58 +02:00
|
|
|
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-10-03 14:06:44 +02:00
|
|
|
// Toggle channels:
|
|
|
|
case k0:
|
2000-11-05 18:39:17 +01:00
|
|
|
if (PreviousChannel != cDvbApi::CurrentChannel())
|
2000-10-03 14:06:44 +02:00
|
|
|
Channels.SwitchTo(PreviousChannel);
|
|
|
|
break;
|
2000-09-10 14:56:18 +02:00
|
|
|
// Direct Channel Select:
|
2000-10-03 14:06:44 +02:00
|
|
|
case k1 ... k9:
|
2000-10-08 12:24:30 +02:00
|
|
|
if (!Interface->Recording())
|
2000-11-01 15:41:33 +01:00
|
|
|
Menu = new cDisplayChannel(key);
|
2000-04-29 15:57:42 +02:00
|
|
|
break;
|
2000-09-09 14:57:43 +02:00
|
|
|
// Left/Right rotates trough channel groups:
|
2000-10-08 11:39:11 +02:00
|
|
|
case kLeft|k_Repeat:
|
2000-09-09 14:57:43 +02:00
|
|
|
case kLeft:
|
2000-10-08 11:39:11 +02:00
|
|
|
case kRight|k_Repeat:
|
2000-10-08 12:24:30 +02:00
|
|
|
case kRight: if (!Interface->Recording()) {
|
2000-09-09 14:57:43 +02:00
|
|
|
int SaveGroup = CurrentGroup;
|
2000-10-08 11:39:11 +02:00
|
|
|
if (NORMALKEY(key) == kRight)
|
2000-09-09 14:57:43 +02:00
|
|
|
CurrentGroup = Channels.GetNextGroup(CurrentGroup) ;
|
|
|
|
else
|
|
|
|
CurrentGroup = Channels.GetPrevGroup(CurrentGroup < 1 ? 1 : CurrentGroup);
|
|
|
|
if (CurrentGroup < 0)
|
|
|
|
CurrentGroup = SaveGroup;
|
2000-11-01 15:41:33 +01:00
|
|
|
Menu = new cDisplayChannel(CurrentGroup, false, true);
|
2000-09-09 14:57:43 +02:00
|
|
|
}
|
|
|
|
break;
|
2000-04-22 13:51:48 +02:00
|
|
|
// Up/Down Channel Select:
|
2000-10-08 11:39:11 +02:00
|
|
|
case kUp|k_Repeat:
|
2000-04-15 17:38:11 +02:00
|
|
|
case kUp:
|
2000-10-08 11:39:11 +02:00
|
|
|
case kDown|k_Repeat:
|
2000-10-08 12:24:30 +02:00
|
|
|
case kDown: if (!Interface->Recording()) {
|
2000-11-05 18:39:17 +01:00
|
|
|
int n = cDvbApi::CurrentChannel() + (NORMALKEY(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-12-28 12:57:16 +01:00
|
|
|
if (!Menu) {
|
2000-11-18 13:57:32 +01:00
|
|
|
EITScanner.Process();
|
2000-12-28 12:57:16 +01:00
|
|
|
cVideoCutter::Active();
|
|
|
|
}
|
2000-04-15 17:38:11 +02:00
|
|
|
}
|
|
|
|
isyslog(LOG_INFO, "caught signal %d", Interrupted);
|
2000-12-28 12:57:16 +01:00
|
|
|
cVideoCutter::Stop();
|
2000-04-30 10:22:13 +02:00
|
|
|
delete Menu;
|
|
|
|
delete ReplayControl;
|
2000-10-08 12:24:30 +02:00
|
|
|
delete Interface;
|
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
|
|
|
}
|