summaryrefslogtreecommitdiff
path: root/recstatus.c
blob: 408f1590e74e91c4368227c629a3aaa384a76545 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
/*                                                                  -*- c++ -*-
Copyright (C) 2004-2013 Christian Wieninger

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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
Or, point your browser to http://www.gnu.org/licenses/old-licenses/gpl-2.0.html

The author can be reached at cwieninger@gmx.de

The project's page is at http://winni.vdr-developer.org/epgsearch
*/

#include "epgsearchtools.h"
#include "recstatus.h"
#include "recdone_thread.h"
#include "conflictcheck_thread.h"
#include "epgsearchcfg.h"
#include <math.h>
#define ALLOWED_BREAK_INSECS 2

extern int gl_InfoConflict;
cTimersRecording TimersRecording;
cRecdoneThread RecdoneThread;

cRecStatusMonitor* gl_recStatusMonitor = NULL;

cRecStatusMonitor::cRecStatusMonitor()
{
}

void cRecStatusMonitor::Recording(const cDevice *Device, const char *Name, const char* Filename, bool On)
{
    time_t now = time(NULL);
    // insert new timers currently recording in TimersRecording
    if (On && Name) {
        if (EPGSearchConfig.checkTimerConflOnRecording)
            cConflictCheckThread::Init((cPluginEpgsearch*)cPluginManager::GetPlugin("epgsearch"), true);

        LOCK_TIMERS_READ;
        for (const cTimer *ti = Timers->First(); ti; ti = Timers->Next(ti))
            if (ti->Recording()) {
                // check if this is a new entry
                cRecDoneTimerObj *tiRFound = NULL;
                cMutexLock TimersRecordingLock(&TimersRecording);
                for (cRecDoneTimerObj *tiR = TimersRecording.First(); tiR; tiR = TimersRecording.Next(tiR))
                    if (tiR->timer == ti) {
                        tiRFound = tiR;
                        break;
                    }

                if (tiRFound) { // already handled, perhaps a resume
                    if (tiRFound->lastBreak > 0 && now - tiRFound->lastBreak <= ALLOWED_BREAK_INSECS) {
                        LogFile.Log(1, "accepting resume of '%s' on device %d", Name, Device->CardIndex());
                        tiRFound->lastBreak = 0;
                    }
                    continue;
                }

                cRecDoneTimerObj* timerObj = new cRecDoneTimerObj(ti, Device->DeviceNumber());
                TimersRecording.Add(timerObj);

                cSearchExt* search = TriggeredFromSearchTimer(ti);
                if (!search || (search->avoidRepeats == 0 && search->delMode == 0)) // ignore if not avoid repeats and no auto-delete
                    continue;

                bool vpsUsed = ti->HasFlags(tfVps) && ti->Event() && ti->Event()->Vps();
                LogFile.Log(1, "recording started '%s' on device %d (search timer '%s'); VPS used: %s", Name, Device->CardIndex(), search->search, vpsUsed ? "Yes" : "No");
                const cEvent* event = ti->Event();
                if (!event) {
                    event = GetEvent(ti);
                    if (event)
                        LogFile.Log(3, "timer had no event: assigning '%s'", event->Title());
                }
                if (!event) {
                    LogFile.Log(1, "no event for timer found! will be ignored in done list");
                    continue;
                }
                time_t now = time(NULL);
                if (vpsUsed || now < ti->StartTime() + 60) { // allow a delay of one minute
                    timerObj->recDone = new cRecDone(ti, event, search);
                    return;
                } else
                    LogFile.Log(1, "recording started too late! will be ignored");
            }
    }

    if (!On) {
        // must be done in a different thread because we hold timer and scheduling lock here
        RecdoneThread.SetFilename(Filename);  // push_back Filename for processing
        RecdoneThread.Start();
    }
}

int cRecStatusMonitor::TimerRecDevice(const cTimer* timer)
{
    if (!timer) return 0;
    cMutexLock TimersRecordingLock(&TimersRecording);
    for (cRecDoneTimerObj *tiR = TimersRecording.First(); tiR; tiR = TimersRecording.Next(tiR))
        if (tiR->timer == timer && timer->Recording()) return tiR->deviceNr + 1;
    return 0;
}