vdr/shutdown.c
Klaus Schmidinger 83d7a4b783 Version 1.5.16
- Fixed setting the current subtitle track in Transfer-Mode (reported by Petri Helin,
  thanks to Reinhard Nissl for pointing out that cDevice::Transferring() doesn't
  return the right value in the early stage of channel switching).
- Updated the Danish OSD texts (thanks to Mogens Elneff).
- Updated the Dutch OSD texts (thanks to Carel Willemse).
- Fixed various spelling errors and improved PLUGINS.html (thanks to Ville Skyttä).
- Updated the Italian OSD texts (thanks to Diego Pierotto).
- Fixed the format in cRemote::Put() to use "%016llX" instead of "%016LX" (thanks
  to Ludwig Nussel for pointing this out).
- Revised the fix of calculating the scrollbar height in the skins. The scrollbar
  handle now always has a height that is at least the width of the scrollbar.
- When switching the device that's used for live viewing away from the current
  transponder in favor of an upcoming recording, an attempt is now made to stay
  on the current channel by possibly going into 'Transfer Mode'.
- The CAM menu now stays open as long as it is automatically updated, even without
  pressing a remote control key. This is important when updating the CAM firmware
  from the transponder.
- No longer trying to switch to an available channel if the primary device has
  no valid programme and a menu is open (avoids interference with the CAM in
  case a CAM menu is open).
- Increased the valid range of the "Subtitle offset" setup option to -100...100
  (thanks to Rolf Ahrenberg).
- Added the backslash ('\') to the list of characters that need to be escaped
  when executing external commands (thanks to Peter Bieringer for reporting this one).
2008-02-24 18:00:00 +01:00

265 lines
7.0 KiB
C

/*
* shutdown.c: Handling of shutdown and inactivity
*
* See the main source file 'vdr.c' for copyright information and
* how to reach the author.
*
* Original version written by Udo Richter <udo_richter@gmx.de>.
*
* $Id: shutdown.c 1.5 2008/02/24 10:29:00 kls Exp $
*/
#include "shutdown.h"
#include <stdio.h>
#include <stdlib.h>
#include <sys/types.h>
#include <sys/wait.h>
#include "channels.h"
#include "config.h"
#include "cutter.h"
#include "i18n.h"
#include "interface.h"
#include "menu.h"
#include "plugin.h"
#include "timers.h"
#include "tools.h"
cShutdownHandler ShutdownHandler;
cCountdown::cCountdown(void)
{
timeout = 0;
counter = 0;
timedOut = false;
message = NULL;
}
void cCountdown::Start(const char *Message, int Seconds)
{
timeout = time(NULL) + Seconds;
counter = -1;
timedOut = false;
message = Message;
Update();
}
void cCountdown::Cancel(void)
{
if (timeout) {
timeout = 0;
timedOut = false;
Skins.Message(mtStatus, NULL);
}
}
bool cCountdown::Done(void)
{
if (timedOut) {
Cancel();
return true;
}
return false;
}
bool cCountdown::Update(void)
{
if (timeout) {
int NewCounter = (timeout - time(NULL) + 9) / 10;
if (NewCounter <= 0)
timedOut = true;
if (counter != NewCounter) {
counter = NewCounter;
char time[10];
snprintf(time, sizeof(time), "%d:%d0", counter > 0 ? counter / 6 : 0, counter > 0 ? counter % 6 : 0);
cString Message = cString::sprintf(message, time);
Skins.Message(mtStatus, Message);
return true;
}
}
return false;
}
cShutdownHandler::cShutdownHandler(void)
{
activeTimeout = 0;
retry = 0;
shutdownCommand = NULL;
exitCode = -1;
emergencyExitRequested = false;
}
cShutdownHandler::~cShutdownHandler()
{
free(shutdownCommand);
}
void cShutdownHandler::RequestEmergencyExit(void)
{
if (Setup.EmergencyExit) {
esyslog("initiating emergency exit");
emergencyExitRequested = true;
Exit(1);
}
else
dsyslog("emergency exit request ignored according to setup");
}
void cShutdownHandler::CheckManualStart(int ManualStart)
{
time_t Delta = Setup.NextWakeupTime ? Setup.NextWakeupTime - time(NULL) : 0;
if (!Setup.NextWakeupTime || abs(Delta) > ManualStart) {
// Apparently the user started VDR manually
dsyslog("assuming manual start of VDR");
// Set inactive after MinUserInactivity
SetUserInactiveTimeout();
}
else {
// Set inactive from now on
dsyslog("scheduled wakeup time in %ld minutes, assuming automatic start of VDR", Delta / 60);
SetUserInactive();
}
}
void cShutdownHandler::SetShutdownCommand(const char *ShutdownCommand)
{
free(shutdownCommand);
shutdownCommand = ShutdownCommand ? strdup(ShutdownCommand) : NULL;
}
void cShutdownHandler::CallShutdownCommand(time_t WakeupTime, int Channel, const char *File, bool UserShutdown)
{
time_t Delta = WakeupTime ? WakeupTime - time(NULL) : 0;
cString cmd = cString::sprintf("%s %ld %ld %d \"%s\" %d", shutdownCommand, WakeupTime, Delta, Channel, *strescape(File, "\\\"$"), UserShutdown);
isyslog("executing '%s'", *cmd);
int Status = SystemExec(cmd, true);
if (!WIFEXITED(Status) || WEXITSTATUS(Status))
esyslog("SystemExec() failed with status %d", Status);
else {
Setup.NextWakeupTime = WakeupTime; // Remember this wakeup time for comparison on reboot
Setup.Save();
}
}
void cShutdownHandler::SetUserInactiveTimeout(int Seconds, bool Force)
{
if (!Setup.MinUserInactivity && !Force) {
activeTimeout = 0;
return;
}
if (Seconds < 0)
Seconds = Setup.MinUserInactivity * 60;
activeTimeout = time(NULL) + Seconds;
}
bool cShutdownHandler::ConfirmShutdown(bool Interactive)
{
if (!Interactive && !cRemote::Enabled())
return false;
if (!shutdownCommand) {
if (Interactive)
Skins.Message(mtError, tr("Can't shutdown - option '-s' not given!"));
return false;
}
if (cCutter::Active()) {
if (!Interactive || !Interface->Confirm(tr("Editing - shut down anyway?")))
return false;
}
cTimer *timer = Timers.GetNextActiveTimer();
time_t Next = timer ? timer->StartTime() : 0;
time_t Delta = timer ? Next - time(NULL) : 0;
if (cRecordControls::Active() || (Next && Delta <= 0)) {
// VPS recordings in timer end margin may cause Delta <= 0
if (!Interactive || !Interface->Confirm(tr("Recording - shut down anyway?")))
return false;
}
else if (Next && Delta <= Setup.MinEventTimeout * 60) {
// Timer within Min Event Timeout
if (!Interactive)
return false;
cString buf = cString::sprintf(tr("Recording in %ld minutes, shut down anyway?"), Delta / 60);
if (!Interface->Confirm(buf))
return false;
}
if (cPluginManager::Active(Interactive ? tr("shut down anyway?") : NULL))
return false;
cPlugin *Plugin = cPluginManager::GetNextWakeupPlugin();
Next = Plugin ? Plugin->WakeupTime() : 0;
Delta = Next ? Next - time(NULL) : 0;
if (Next && Delta <= Setup.MinEventTimeout * 60) {
// Plugin wakeup within Min Event Timeout
if (!Interactive)
return false;
cString buf = cString::sprintf(tr("Plugin %s wakes up in %ld min, continue?"), Plugin->Name(), Delta / 60);
if (!Interface->Confirm(buf))
return false;
}
return true;
}
bool cShutdownHandler::ConfirmRestart(bool Interactive)
{
if (cCutter::Active()) {
if (!Interactive || !Interface->Confirm(tr("Editing - restart anyway?")))
return false;
}
cTimer *timer = Timers.GetNextActiveTimer();
time_t Next = timer ? timer->StartTime() : 0;
time_t Delta = timer ? Next - time(NULL) : 0;
if (cRecordControls::Active() || (Next && Delta <= 0)) {
// VPS recordings in timer end margin may cause Delta <= 0
if (!Interactive || !Interface->Confirm(tr("Recording - restart anyway?")))
return false;
}
if (cPluginManager::Active(Interactive ? tr("restart anyway?") : NULL))
return false;
return true;
}
bool cShutdownHandler::DoShutdown(bool Force)
{
time_t Now = time(NULL);
cTimer *timer = Timers.GetNextActiveTimer();
cPlugin *Plugin = cPluginManager::GetNextWakeupPlugin();
time_t Next = timer ? timer->StartTime() : 0;
time_t NextPlugin = Plugin ? Plugin->WakeupTime() : 0;
if (NextPlugin && (!Next || Next > NextPlugin)) {
Next = NextPlugin;
timer = NULL;
}
time_t Delta = Next ? Next - Now : 0;
if (Next && Delta < Setup.MinEventTimeout * 60) {
if (!Force)
return false;
Delta = Setup.MinEventTimeout * 60;
Next = Now + Delta;
timer = NULL;
dsyslog("reboot at %s", *TimeToString(Next));
}
if (Next && timer) {
dsyslog("next timer event at %s", *TimeToString(Next));
CallShutdownCommand(Next, timer->Channel()->Number(), timer->File(), Force);
}
else if (Next && Plugin) {
CallShutdownCommand(Next, 0, Plugin->Name(), Force);
dsyslog("next plugin wakeup at %s", *TimeToString(Next));
}
else
CallShutdownCommand(Next, 0, "", Force); // Next should always be 0 here. Just for safety, pass it.
return true;
}