mirror of
https://github.com/vdr-projects/vdr.git
synced 2025-03-01 10:50:46 +00:00
- Added Norwegian language texts (thanks to Jørgen Tvedt). - Increased the usleep value in cDvbOsd::Cmd() to 5000 in order to work on systems with the KURT/utime-patch (thanks to Guido Fiala). - Changed the check whether the driver is loaded in runvdr to check for the 'dvb' module (the last one loaded). - Fixed repeat function with LIRC (thanks to Stefan Huelswitt). - Increased the upper limit for the symbol rate to 30000 (thanks to Ulrich Röder). - Made the position of the channel display configurable (thanks to Stefan Huelswitt). - Made the width and height of the OSD configurable (thanks to Stefan Huelswitt). - DiSEqC support can now be generally enabled/disabled in the Setup menu. This may be necessary if your multiswitch gets irritated by the default DiSEqC codes '0' (thanks to Markus Lang). - Fixed replaying in case there is no index file. - Fixed jumping to an editing mark when replay has been paused. - Avoiding unnecessary code execution in the replay progress display (thanks to Guido Fiala). - When entering time values the digits that still have to be entered are now shown as '-' (as in "1-:--"). - 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 setting of the mark. - Updated 'channels.conf' for Premiere World (thanks to Helmut Schächner). Check your timers if you use this channels.conf file, since the sequence of several PW channels has been changed. - Changed the color of "Info" messages to "black on green" and that of the confirmation messages (like "Delete...") to "black on yellow". - Fixed display with DEBUG_OSD (it still crashes sometimes, esp. when replaying, but I can't seem to find what causes this... any ideas anybody?). - Avoiding audio/video distortions in 'Transfer Mode' by no longer actually tuning the primary interface (which can't receive this channel, anyway). Apparently the driver gets irritated when the channel is switched and a replay session is started immediately after that. - Increased timeout until reporting "video data stream broken" when recording. - Explicitly switching back to the previously active channel after ending a replay session (to have it shown correctly in case it was in 'Transfer Mode').
201 lines
4.0 KiB
C
201 lines
4.0 KiB
C
/*
|
|
* thread.c: A simple thread base class
|
|
*
|
|
* See the main source file 'vdr.c' for copyright information and
|
|
* how to reach the author.
|
|
*
|
|
* $Id: thread.c 1.9 2001/06/27 11:34:41 kls Exp $
|
|
*/
|
|
|
|
#include "thread.h"
|
|
#include <errno.h>
|
|
#include <signal.h>
|
|
#include <sys/wait.h>
|
|
#include <unistd.h>
|
|
#include "tools.h"
|
|
|
|
// --- cMutex ----------------------------------------------------------------
|
|
|
|
cMutex::cMutex(void)
|
|
{
|
|
lockingPid = 0;
|
|
locked = 0;
|
|
pthread_mutex_init(&mutex, NULL);
|
|
}
|
|
|
|
cMutex::~cMutex()
|
|
{
|
|
pthread_mutex_destroy(&mutex);
|
|
}
|
|
|
|
void cMutex::Lock(void)
|
|
{
|
|
if (getpid() != lockingPid || !locked)
|
|
pthread_mutex_lock(&mutex);
|
|
lockingPid = getpid();
|
|
locked++;
|
|
}
|
|
|
|
void cMutex::Unlock(void)
|
|
{
|
|
if (!--locked)
|
|
pthread_mutex_unlock(&mutex);
|
|
}
|
|
|
|
// --- cThread ---------------------------------------------------------------
|
|
|
|
// The signal handler is necessary to be able to use SIGIO to wake up any
|
|
// pending 'select()' call.
|
|
|
|
time_t cThread::lastPanic = 0;
|
|
int cThread::panicLevel = 0;
|
|
bool cThread::signalHandlerInstalled = false;
|
|
bool cThread::emergencyExitRequested = false;
|
|
|
|
cThread::cThread(void)
|
|
{
|
|
if (!signalHandlerInstalled) {
|
|
signal(SIGIO, SignalHandler);
|
|
signalHandlerInstalled = true;
|
|
}
|
|
running = false;
|
|
parentPid = threadPid = lockingPid = 0;
|
|
locked = 0;
|
|
}
|
|
|
|
cThread::~cThread()
|
|
{
|
|
}
|
|
|
|
void cThread::SignalHandler(int signum)
|
|
{
|
|
signal(signum, SignalHandler);
|
|
}
|
|
|
|
void *cThread::StartThread(cThread *Thread)
|
|
{
|
|
Thread->threadPid = getpid();
|
|
Thread->Action();
|
|
return NULL;
|
|
}
|
|
|
|
bool cThread::Start(void)
|
|
{
|
|
if (!running) {
|
|
running = true;
|
|
parentPid = getpid();
|
|
pthread_create(&thread, NULL, (void *(*) (void *))&StartThread, (void *)this);
|
|
usleep(10000); // otherwise calling Active() immediately after Start() causes a "pure virtual method called" error
|
|
}
|
|
return true; //XXX return value of pthread_create()???
|
|
}
|
|
|
|
bool cThread::Active(void)
|
|
{
|
|
if (threadPid) {
|
|
if (kill(threadPid, SIGIO) < 0) { // couldn't find another way of checking whether the thread is still running - any ideas?
|
|
if (errno == ESRCH)
|
|
threadPid = 0;
|
|
else
|
|
LOG_ERROR;
|
|
}
|
|
else
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
void cThread::Cancel(int WaitSeconds)
|
|
{
|
|
if (WaitSeconds > 0) {
|
|
for (time_t t0 = time(NULL) + WaitSeconds; time(NULL) < t0; ) {
|
|
if (!Active())
|
|
return;
|
|
usleep(10000);
|
|
}
|
|
esyslog(LOG_ERR, "ERROR: thread %d won't end (waited %d seconds) - cancelling it...", threadPid, WaitSeconds);
|
|
}
|
|
pthread_cancel(thread);
|
|
}
|
|
|
|
bool cThread::Lock(void)
|
|
{
|
|
if (!lockingPid || lockingPid != getpid()) {
|
|
Mutex.Lock();
|
|
lockingPid = getpid();
|
|
}
|
|
locked++;
|
|
return true;
|
|
}
|
|
|
|
void cThread::Unlock(void)
|
|
{
|
|
if (!--locked) {
|
|
lockingPid = 0;
|
|
Mutex.Unlock();
|
|
}
|
|
}
|
|
|
|
void cThread::WakeUp(void)
|
|
{
|
|
kill(parentPid, SIGIO); // makes any waiting 'select()' call return immediately
|
|
}
|
|
|
|
#define MAXPANICLEVEL 10
|
|
|
|
void cThread::RaisePanic(void)
|
|
{
|
|
if (lastPanic > 0) {
|
|
if (time(NULL) - lastPanic < 5)
|
|
panicLevel++;
|
|
else if (panicLevel > 0)
|
|
panicLevel--;
|
|
}
|
|
lastPanic = time(NULL);
|
|
if (panicLevel > MAXPANICLEVEL) {
|
|
esyslog(LOG_ERR, "ERROR: max. panic level exceeded");
|
|
EmergencyExit(true);
|
|
}
|
|
else
|
|
dsyslog(LOG_INFO, "panic level: %d", panicLevel);
|
|
}
|
|
|
|
bool cThread::EmergencyExit(bool Request)
|
|
{
|
|
if (!Request)
|
|
return emergencyExitRequested;
|
|
esyslog(LOG_ERR, "initiating emergency exit");
|
|
return emergencyExitRequested = true; // yes, it's an assignment, not a comparison!
|
|
}
|
|
|
|
// --- cThreadLock -----------------------------------------------------------
|
|
|
|
cThreadLock::cThreadLock(cThread *Thread)
|
|
{
|
|
thread = NULL;
|
|
locked = false;
|
|
Lock(Thread);
|
|
}
|
|
|
|
cThreadLock::~cThreadLock()
|
|
{
|
|
if (thread && locked)
|
|
thread->Unlock();
|
|
}
|
|
|
|
bool cThreadLock::Lock(cThread *Thread)
|
|
{
|
|
if (Thread && !thread) {
|
|
thread = Thread;
|
|
locked = Thread->Lock();
|
|
return locked;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
bool cThreadLock::Locked(void)
|
|
{
|
|
return locked;
|
|
}
|
|
|