13#include "libmythbase/mythconfig.h"
46#define DEBUG_CHANNEL_PREFIX 0
48#define LOC QString("TVRec[%1]: ").arg(m_inputId)
49#define LOC2 QString("TVRec[%1]: ").arg(inputid)
57 bool on_host,
bool transcode_bfr_comm,
bool on_line_comm);
59static std::chrono::seconds
eit_start_rand(
uint inputId, std::chrono::seconds eitTransportTimeout);
87 : m_eventThread(new
MThread(
"TVRecEvent", this)),
95 bool enter_power_save_mode)
97 LOG(VB_CHANNEL, LOG_INFO,
LOC + QString(
"CreateChannel(%1)")
110 startchannel, enter_power_save_mode,
m_rbFileExt, setchan);
118 LOG(VB_GENERAL, LOG_ERR,
LOC +
119 QString(
"CreateChannel(%1) failed due to VBOX not responding "
120 "to network check on inputid [%2]")
130 if (!CardUtil::IsSatIPPresent(
m_inputId))
133 LOG(VB_GENERAL, LOG_ERR,
LOC +
134 QString(
"CreateChannel(%1) failed due to SatIP box not responding "
135 "to network check on inputid [%2]")
162 LOG(VB_CHANNEL, LOG_ERR,
LOC +
163 QString(
"Failed to GetDevices for input %1")
172 if (startchannel.isEmpty())
176 LOG(VB_CHANNEL, LOG_ERR,
LOC +
177 QString(
"Failed to create channel instance for %1")
235 LOG(VB_RECORD, LOG_INFO,
LOC +
"TeardownAll");
319 LOG(VB_RECORD, LOG_INFO,
LOC +
"Pending recording revoked on " +
320 QString(
"inputid [%1]").arg(rcinfo->
GetInputID()));
326 (*it).m_doNotAsk =
true;
327 (*it).m_canceled =
true;
332 LOG(VB_RECORD, LOG_INFO,
LOC +
333 QString(
"RecordPending on inputid [%1]").arg(rcinfo->
GetInputID()));
339 pending.
m_ask =
true;
355 for (
uint inputid : inputids)
390 LOG(VB_RECORD, LOG_INFO,
LOC +
391 QString(
"CancelNextRecording(%1) -- begin").arg(cancel));
396 LOG(VB_RECORD, LOG_INFO,
LOC + QString(
"CancelNextRecording(%1) -- "
397 "error, unknown recording").arg(cancel));
403 std::vector<unsigned int> &inputids = (*it).m_possibleConflicts;
404 for (
uint inputid : inputids)
406 LOG(VB_RECORD, LOG_INFO,
LOC +
407 QString(
"CancelNextRecording -- inputid 0x%1")
408 .arg((uint64_t)inputid,0,16));
415 LOG(VB_RECORD, LOG_INFO,
LOC +
416 QString(
"CancelNextRecording -- inputid [%1]")
423 (*it).m_canceled =
false;
426 LOG(VB_RECORD, LOG_INFO,
LOC +
427 QString(
"CancelNextRecording(%1) -- end").arg(cancel));
444 LOG(VB_RECORD, LOG_INFO,
LOC + QString(
"StartRecording(%1)")
470 .addSecs(post_roll_seconds);
472 QString msg = QString(
"updating recording: %1 %2 %3 %4")
477 LOG(VB_RECORD, LOG_INFO,
LOC + msg);
485 bool cancelNext =
false;
492 (*it).m_ask = (*it).m_doNotAsk =
false;
493 cancelNext = (*it).m_canceled;
514 LOG(VB_RECORD, LOG_INFO,
LOC +
515 "Checking input group recorders - begin");
521 std::vector<unsigned int> inputids2;
522 std::vector<TVState> states;
525 for (
uint inputid : inputids)
530 if (is_busy && !sourceid)
540 ((mplexid == 0 || mplexid == 32767) &&
544 inputids2.push_back(inputid);
549 for (
uint i = 0; (i < inputids2.size()) && ok; i++)
551 LOG(VB_RECORD, LOG_INFO,
LOC +
552 QString(
"Attempting to stop input [%1] in state %2")
559 LOG(VB_GENERAL, LOG_INFO,
LOC + QString(
"a [%1]: %2")
567 QString message = QString(
"QUIT_LIVETV %1").arg(inputids2[i]);
572 LOG(VB_RECORD, LOG_INFO,
LOC +
573 QString(
"Stopping recording on [%1], %2") .arg(inputids2[i])
574 .arg(success ?
"succeeded" :
"failed"));
588 LOG(VB_RECORD, LOG_INFO,
LOC +
"Checking input group recorders - done");
591 bool did_switch =
false;
595 did_switch = (
nullptr != ri2);
616 QString message = QString(
"LIVETV_EXITED");
638 LOG(VB_RECORD, LOG_WARNING,
LOC +
"Still failing.");
650 QString message = QString(
"LIVETV_WATCH %1 1").arg(
m_inputId);
656 else if (!did_switch)
658 QString msg = QString(
"Wanted to record: %1 %2 %3 %4\n\t\t\t")
666 msg +=
"But a user has canceled this recording";
671 msg += QString(
"But the current state is: %1")
678 msg += QString(
"\n\t\t\tCurrently recording: %1 %2 %3 %4")
685 LOG(VB_GENERAL, LOG_INFO,
LOC + msg);
734 LOG(VB_RECORD, LOG_INFO,
LOC +
735 QString(
"SetRecordingStatus(%1->%2) on line %3")
738 QString::number(line)));
757 if (now < m_curRecording->GetDesiredEndTime())
799 LOG(VB_GENERAL, LOG_ERR,
LOC +
800 QString(
"Unknown state in RemoveRecording: %1")
819 QString msg =
"Unknown state in RemovePlaying: %1";
836 LOG(VB_RECORD, LOG_INFO,
LOC + QString(
"StartedRecording(%1) fn(%2)")
868 LOG((recq->
IsDamaged()) ? VB_GENERAL : VB_RECORD, LOG_INFO,
869 LOC + QString(
"FinishedRecording(%1) %2 recq:\n%3")
871 (recq->
IsDamaged()) ?
"damaged" :
"good",
888 bool was_finished =
false;
889 static QMutex s_finRecLock;
890 static QHash<QString,QDateTime> s_finRecMap;
892 QMutexLocker locker(&s_finRecLock);
894 QDateTime expired = now.addSecs(-5LL * 60);
895 QHash<QString,QDateTime>::iterator it = s_finRecMap.begin();
896 while (it != s_finRecMap.end())
899 it = s_finRecMap.erase(it);
904 it = s_finRecMap.find(key);
905 if (it != s_finRecMap.end())
908 s_finRecMap[key] = now;
912 LOG(VB_RECORD, LOG_INFO,
LOC +
913 QString(
"FinishedRecording(%1) %2 quality"
914 "\n\t\t\ttitle: %3\n\t\t\t"
915 "in recgroup: %4 status: %5:%6 %7 %8")
917 is_good ?
"Good" :
"Bad",
923 was_finished?
"already_finished":
"finished_now"));
947 if (avg_height > 2000)
949 else if (avg_height > 1000)
951 else if (avg_height > 700)
954 flags |= VID_PROGRESSIVE;
956 flags |= VID_DAMAGED;
959 flags |= VID_WIDESCREEN;
962 (VID_4K | VID_1080 | VID_720 | VID_DAMAGED |
963 VID_WIDESCREEN | VID_PROGRESSIVE, flags);
979 if (curRec->
IsLocal() && (fsize >= 1000) &&
989 LOG(VB_RECORD, LOG_INFO,
LOC +
990 QString(
"FinishedRecording -- UPDATE_RECORDING_STATUS: %1")
993 MythEvent me(QString(
"UPDATE_RECORDING_STATUS %1 %2 %3 %4 %5")
1006 QString message = QString(
"DONE_RECORDING %1 %2 %3")
1012 QHash<QString,int>::iterator autoJob =
1016 LOG(VB_GENERAL, LOG_INFO,
1017 "autoRunJobs not initialized until FinishedRecording()");
1023 LOG(VB_JOBQUEUE, LOG_INFO, QString(
"AutoRunJobs 0x%1").arg(*autoJob,0,16));
1024 if ((recgrp ==
"LiveTV") || (fsize < 1000) ||
1038#define TRANSITION(ASTATE,BSTATE) \
1039 ((m_internalState == (ASTATE)) && (m_desiredNextState == (BSTATE)))
1040#define SET_NEXT() do { nextState = m_desiredNextState; changed = true; } while(false)
1041#define SET_LAST() do { nextState = m_internalState; changed = true; } while(false)
1055 bool changed =
false;
1057 QString transMsg = QString(
" %1 to %2")
1062 LOG(VB_GENERAL, LOG_ERR,
LOC +
1063 "HandleStateChange(): Null transition" + transMsg);
1072 LOG(VB_EIT, LOG_INFO,
LOC + QString(
"Stop EIT scan on input %1").arg(
GetInputId()));
1091 auto *tv_rec =
s_inputs.value(input);
1094 LOG(VB_EIT, LOG_INFO,
LOC +
1095 QString(
"Stop EIT scan active on conflicting input %1")
1097 tv_rec->m_scanner->StopActiveScan();
1139 QString msg = (changed) ?
"Changing from" :
"Unknown state transition:";
1140 LOG(VB_GENERAL, LOG_INFO,
LOC + msg + transMsg);
1187 LOG(VB_RECORD, LOG_INFO,
LOC + QString(
"TeardownRecorder(%1)")
1188 .arg((request_flags &
kFlagKillRec) ?
"kFlagKillRec" :
""));
1201 __FILE__, __LINE__);
1218 LOG(VB_FILE, LOG_INFO,
LOC +
"calling StopReads()");
1278 "SELECT SUM(useeit) "
1279 "FROM videosource, capturecard "
1280 "WHERE videosource.sourceid = capturecard.sourceid AND"
1281 " capturecard.cardid = :INPUTID");
1290 return query.
value(0).toBool();
1298 "SELECT SUM(dishnet_eit) "
1299 "FROM videosource, capturecard "
1300 "WHERE videosource.sourceid = capturecard.sourceid AND"
1301 " capturecard.cardid = :INPUTID");
1310 return query.
value(0).toBool();
1319 "SELECT MAX(cardid) "
1320 "FROM capturecard ");
1328 return query.
value(0).toInt();
1335 auto timeout = std::chrono::seconds(
MythRandom(0, eitTransportTimeout.count() / 3));
1340 if (highest_input > 0)
1341 timeout += eitTransportTimeout * inputId / highest_input;
1374 __FILE__, __LINE__);
1380 LOG(VB_GENERAL, LOG_ERR,
LOC +
1381 "RunTV encountered fatal error, exiting event thread.");
1430 QString message = QString(
"QUIT_LIVETV %1").arg(
m_inputId);
1448 bool enable_ui =
true;
1463 LOG(VB_RECORD, LOG_INFO,
LOC +
1464 "Switching Buffer (" +
1465 QString(
"!has_rec(%1) && ").arg(has_rec) +
1466 QString(
"!rec_soon(%1) && (").arg(rec_soon) +
1479 LOG(VB_RECORD, LOG_INFO,
"Waiting for ringbuffer switch");
1489 LOG(VB_RECORD, LOG_INFO,
LOC +
"Enabling Full LiveTV UI.");
1490 QString message = QString(
"LIVETV_WATCH %1 0").arg(
m_inputId);
1508 bool conflicting_input =
false;
1514 LOG(VB_EIT, LOG_INFO,
LOC +
1515 QString(
"EIT scanning disabled for input %1")
1521 LOG(VB_EIT, LOG_INFO,
LOC +
1522 QString(
"EIT scanning disabled for video source %1")
1528 LOG(VB_EIT, LOG_INFO,
LOC +
1529 QString(
"EIT scanning enabled for input %1 connected to video source %2 '%3'")
1536 bool allow_eit =
true;
1539 for (
uint i = 0; i < inputids.size() && allow_eit; ++i)
1548 for (
auto input : inputids)
1550 auto *tv_rec =
s_inputs.value(input);
1551 if (tv_rec && tv_rec->m_scanner)
1553 conflicting_input =
true;
1556 LOG(VB_EIT, LOG_INFO,
LOC +
1557 QString(
"EIT scan on conflicting input %1").arg(input));
1559 busy_input.
m_inputId = tv_rec->m_inputId;
1568 LOG(VB_EIT, LOG_INFO,
LOC +
1569 QString(
"Start EIT active scan on input %1")
1574 if (conflicting_input)
1581 const int seconds_postpone = 300;
1582 LOG(VB_EIT, LOG_INFO,
LOC +
1583 QString(
"Postponing EIT scan on input %1 for %2 seconds because input %3 is busy")
1596 LOG(VB_EIT, LOG_INFO,
LOC +
1597 QString(
"Stop EIT scan on input %1 to allow scan on a conflicting input")
1662 while (!ok && (
t.elapsed() < time))
1686 std::chrono::milliseconds te = t2.
elapsed();
1687 if (!ok && te < 10ms)
1688 std::this_thread::sleep_for(10ms - te);
1701 LOG(VB_RECORD, LOG_INFO,
LOC +
"Deleting stale pending recording " +
1702 QString(
"[%1] '%2'")
1703 .arg((*it).m_info->GetInputID())
1704 .arg((*it).m_info->GetTitle()));
1706 delete (*it).m_info;
1721 LOG(VB_CHANNEL, LOG_INFO,
1722 LOC +
"Stopping active EIT scan for pending recording.");
1730 bool has_rec =
false;
1734 ((*it).m_info->GetInputID() ==
m_inputId) &&
1740 (*it).m_recordingStart);
1745 if (!(*it).m_ask && !(*it).m_doNotAsk)
1748 auto timeuntil = ((*it).m_doNotAsk) ?
1752 (*it).m_canceled =
true;
1754 QString query = QString(
"ASK_RECORDING %1 %2 %3 %4")
1756 .arg(timeuntil.count())
1757 .arg(has_rec ? 1 : 0)
1758 .arg((*it).m_hasLaterShowing ? 1 : 0);
1760 LOG(VB_GENERAL, LOG_INFO,
LOC + query);
1763 (*it).m_info->ToStringList(msg);
1767 (*it).m_ask = (*it).m_doNotAsk =
false;
1782 "SELECT videodevice, vbidevice, audiodevice, "
1783 " audioratelimit, cardtype, "
1784 " skipbtaudio, signal_timeout, channel_timeout, "
1785 " dvb_wait_for_seqstart, "
1787 " dvb_on_demand, dvb_tuning_delay, dvb_eitscan,"
1789 " firewire_speed, firewire_model, firewire_connection, "
1793 "WHERE cardid = :INPUTID");
1806 test = query.
value(0).toString();
1807 if (!test.isEmpty())
1810 test = query.
value(1).toString();
1811 if (!test.isEmpty())
1814 test = query.
value(2).toString();
1815 if (!test.isEmpty())
1820 test = query.
value(4).toString();
1821 if (!test.isEmpty())
1832 if (table_timeout < 1000)
1844 uint fireoff = dvboff + 3;
1845 firewire_opts.
m_speed = query.
value(fireoff + 0).toUInt();
1847 test = query.
value(fireoff + 1).toString();
1848 if (!test.isEmpty())
1853 parentid = query.
value(15).toUInt();
1870 pid_cache.push_back(item);
1879 bool vctpid_cached =
false;
1880 for (
const auto& pid : pid_cache)
1885 vctpid_cached =
true;
1890 return vctpid_cached;
1910 LOG(VB_RECORD, LOG_INFO,
LOC +
"Setting up table monitoring.");
1914 if (!sm || !dtvchan)
1916 LOG(VB_GENERAL, LOG_ERR,
LOC +
"Setting up table monitoring.");
1927 QString recording_type =
"all";
1933 recording_type = setting->
getValue();
1940 if ((
minor > 0) && (tuningmode ==
"atsc"))
1942 QString msg = QString(
"ATSC channel: %1_%2").arg(major).arg(
minor);
1943 LOG(VB_RECORD, LOG_INFO,
LOC + msg);
1964 LOG(VB_RECORD, LOG_INFO,
LOC +
1965 "Successfully set up ATSC table monitoring.");
1985 LOG(VB_RECORD, LOG_INFO,
LOC +
1986 QString(
"DVB service_id %1 on net_id %2 tsid %3")
1987 .arg(progNum).arg(netid).arg(tsid));
2007 LOG(VB_RECORD, LOG_INFO,
LOC +
2008 "Successfully set up DVB table monitoring.");
2023 QString msg = QString(
"MPEG program number: %1").arg(progNum);
2024 LOG(VB_RECORD, LOG_INFO,
LOC + msg);
2044 LOG(VB_RECORD, LOG_INFO,
LOC +
2045 "Successfully set up MPEG table monitoring.");
2056 for (
auto item = pid_cache.cbegin(); !ok && item != pid_cache.cend(); ++item)
2057 ok |= item->IsPermanent();
2062 QString msg =
"No valid DTV info, ATSC maj(%1) min(%2), MPEG pn(%3)";
2063 LOG(VB_GENERAL, LOG_ERR,
LOC + msg.arg(major).arg(
minor).arg(progNum));
2067 LOG(VB_RECORD, LOG_INFO,
LOC +
2068 "Successfully set up raw pid monitoring.");
2090 LOG(VB_RECORD, LOG_INFO,
LOC + QString(
"SetupSignalMonitor(%1, %2)")
2091 .arg(tablemon).arg(notify));
2114 LOG(VB_RECORD, LOG_INFO,
LOC +
"Signal monitor successfully created");
2119 LOG(VB_GENERAL, LOG_ERR,
LOC +
2120 "Failed to setup digital signal monitoring");
2147 LOG(VB_RECORD, LOG_INFO,
LOC +
"TeardownSignalMonitor() -- begin");
2152 if (dtvMon && dtvChan)
2156 if (!pid_cache.empty())
2166 LOG(VB_RECORD, LOG_INFO,
LOC +
"TeardownSignalMonitor() -- end");
2182 QString msg =
"SetSignalMonitoringRate(%1, %2)";
2183 LOG(VB_RECORD, LOG_INFO,
LOC +
2184 msg.arg(rate.count()).arg(notifyFrontend) +
"-- start");
2190 LOG(VB_GENERAL, LOG_ERR,
LOC +
2191 "Signal Monitoring is notsupported by your hardware.");
2197 LOG(VB_GENERAL, LOG_ERR,
LOC +
2198 "Signal can only be monitored in LiveTV Mode.");
2213 LOG(VB_RECORD, LOG_INFO,
LOC +
2214 msg.arg(rate.count()).arg(notifyFrontend) +
" -- end");
2242 query.
prepare(
"SELECT channel.channum, channel.callsign "
2244 "WHERE channel.chanid = :CHANID");
2246 if (!query.
exec() || !query.
next())
2252 QString channelname = query.
value(0).toString();
2253 QString callsign = query.
value(1).toString();
2256 "SELECT channel.channum "
2257 "FROM channel, capturecard "
2258 "WHERE deleted IS NULL AND "
2259 " ( channel.chanid = :CHANID OR "
2260 " ( channel.channum = :CHANNUM AND "
2261 " channel.callsign = :CALLSIGN ) "
2263 " channel.sourceid = capturecard.sourceid AND "
2264 " capturecard.cardid = :INPUTID");
2266 query.
bindValue(
":CHANNUM", channelname);
2274 else if (query.
size() > 0)
2276 msg =
"Found channel (%1) on current input[%2].";
2283 "SELECT channel.channum, capturecard.cardid "
2284 "FROM channel, capturecard "
2285 "WHERE deleted IS NULL AND "
2286 " ( channel.chanid = :CHANID OR "
2287 " ( channel.channum = :CHANNUM AND "
2288 " channel.callsign = :CALLSIGN ) "
2290 " channel.sourceid = capturecard.sourceid AND "
2291 " capturecard.cardid != :INPUTID");
2293 query.
bindValue(
":CHANNUM", channelname);
2301 else if (query.
next())
2303 msg = QString(
"Found channel (%1) on different input(%2).")
2304 .arg(query.
value(0).toString(), query.
value(1).toString());
2305 LOG(VB_RECORD, LOG_INFO,
LOC + msg);
2309 msg = QString(
"Did not find channel(%1) on any input.").arg(channelname);
2310 LOG(VB_RECORD, LOG_ERR,
LOC + msg);
2335static QString
add_spacer(
const QString &channel,
const QString &spacer)
2337 QString chan = channel;
2338 if ((chan.length() >= 2) && !spacer.isEmpty())
2339 return chan.left(chan.length()-1) + spacer + chan.right(1);
2371 uint &complete_valid_channel_on_rec,
2372 bool &is_extra_char_useful,
2373 QString &needed_spacer)
const
2375#if DEBUG_CHANNEL_PREFIX
2376 LOG(VB_GENERAL, LOG_DEBUG, QString(
"CheckChannelPrefix(%1)").arg(
prefix));
2379 static const std::array<const QString,5> s_spacers = {
"",
"_",
"-",
"#",
"." };
2382 QString basequery = QString(
2383 "SELECT channel.chanid, channel.channum, capturecard.cardid "
2384 "FROM channel, capturecard "
2385 "WHERE deleted IS NULL AND "
2386 " channel.channum LIKE '%1%' AND "
2387 " channel.sourceid = capturecard.sourceid");
2389 const std::array<const QString,2> inputquery
2391 QString(
" AND capturecard.cardid = '%1'").arg(
m_inputId),
2392 QString(
" AND capturecard.cardid != '%1'").arg(
m_inputId),
2395 std::vector<unsigned int> fchanid;
2396 std::vector<QString> fchannum;
2397 std::vector<unsigned int> finputid;
2398 std::vector<QString> fspacer;
2400 for (
const auto & str : inputquery)
2402 for (
const auto & spacer : s_spacers)
2405 prefix, (spacer ==
"_") ?
"\\_" : spacer);
2406 query.
prepare(basequery.arg(qprefix) + str);
2412 else if (query.
size())
2414 while (query.
next())
2416 fchanid.push_back(query.
value(0).toUInt());
2417 fchannum.push_back(query.
value(1).toString());
2418 finputid.push_back(query.
value(2).toUInt());
2419 fspacer.emplace_back(spacer);
2420#if DEBUG_CHANNEL_PREFIX
2421 LOG(VB_GENERAL, LOG_DEBUG,
2422 QString(
"(%1,%2) Adding %3 rec %4")
2423 .arg(i).arg(j).arg(query.
value(1).toString(),6)
2424 .arg(query.
value(2).toUInt()));
2435 is_extra_char_useful =
false;
2436 complete_valid_channel_on_rec = 0;
2437 needed_spacer.clear();
2439 if (fchanid.empty())
2442 if (fchanid.size() == 1)
2444 needed_spacer = fspacer[0];
2447 complete_valid_channel_on_rec = (nc) ? 0 : finputid[0];
2448 is_extra_char_useful = nc;
2456 is_extra_char_useful =
false;
2457 for (
uint i = 0; (i < fchannum.size()) && !is_extra_char_useful; i++)
2460#if DEBUG_CHANNEL_PREFIX
2461 LOG(VB_GENERAL, LOG_DEBUG, QString(
"is_extra_char_useful(%1!=%2): %3")
2463 .arg(is_extra_char_useful));
2470 for (
size_t i = 0; i < fchannum.size(); i++)
2472 if (fchannum[i] ==
prefix)
2474 complete_valid_channel_on_rec = finputid[i];
2480 if (complete_valid_channel_on_rec != 0)
2484 bool spacer_needed =
true;
2485 for (
uint i = 0; (i < fspacer.size() && spacer_needed); i++)
2486 spacer_needed = !fspacer[i].isEmpty();
2488 needed_spacer = fspacer[0];
2492 for (
size_t i = 0; i < ((is_extra_char_useful) ? 0 : fchanid.size()); i++)
2496 needed_spacer = fspacer[i];
2497 complete_valid_channel_on_rec = finputid[i];
2506 const QString &channum)
2512 if (!videoFilters.isEmpty())
2540 busy_input = &dummy;
2542 busy_input->
Clear();
2559 bool has_pending =
false;
2569 if (!busy_input->
m_inputId && has_pending)
2573 if (timeLeft <= time_buffer)
2696 long long bitrate = 0;
2699 bitrate = 10080000LL;
2703 bitrate = 20200000LL;
2707 bitrate = 22200000LL;
2711 bitrate = 10080000LL;
2839 MythEvent me(
"SCHEDULER_ADD_RECORDING", prog);
2866 LOG(VB_JOBQUEUE, LOG_INFO,
2867 QString(
"InitAutoRunJobs for %1, line %2 -> 0x%3")
2885 LOG(VB_GENERAL, LOG_INFO,
LOC +
2886 QString(
"SetLiveRecording(%1)").arg(recording));
2894 LOG(VB_GENERAL, LOG_INFO,
LOC +
"SetLiveRecording() -- cancel");
2902 LOG(VB_GENERAL, LOG_INFO,
LOC +
"SetLiveRecording() -- record");
2915 MythEvent me(QString(
"UPDATE_RECORDING_STATUS %1 %2 %3 %4 %5")
2932 LOG(VB_RECORD, LOG_INFO,
LOC +
2933 QString(
"StopLiveTV(void) curRec: 0x%1 pseudoRec: 0x%2")
2982 LOG(VB_GENERAL, LOG_ERR,
LOC +
2983 "PauseRecorder() called with no recorder");
3018 LOG(VB_GENERAL, LOG_ERR,
LOC +
3019 QString(
"Channel: \'%1\' was not found in the database.\n"
3020 "\t\tMost likely, the 'starting channel' for this "
3021 "Input Connection is invalid.\n"
3022 "\t\tCould not toggle favorite.").arg(channum));
3029 LOG(VB_RECORD, LOG_ERR,
LOC +
3030 QString(
"ToggleChannelFavorite: Invalid channel group name %1,")
3031 .arg(changroupname));
3038 LOG(VB_RECORD, LOG_ERR,
LOC +
"Unable to toggle channel favorite.");
3041 LOG(VB_RECORD, LOG_INFO,
LOC +
3042 QString(
"Toggled channel favorite.channum %1, chan group %2")
3043 .arg(channum, changroupname));
3061 return (ret < 0) ? -1 : ret / 655;
3081 return (ret < 0) ? -1 : ret / 655;
3115 QString origIn = input;
3116 LOG(VB_RECORD, LOG_INFO,
LOC +
"SetInput(" + input +
") -- begin");
3120 LOG(VB_RECORD, LOG_INFO,
LOC +
"SetInput() -- end no channel class");
3124 LOG(VB_RECORD, LOG_INFO,
LOC +
"SetInput(" + origIn +
":" + input +
3125 ") -- end nothing to do");
3143 LOG(VB_CHANNEL, LOG_INFO,
LOC +
3144 QString(
"SetChannel(%1) -- begin").arg(name));
3177 LOG(VB_CHANNEL, LOG_INFO,
LOC + QString(
"SetChannel(%1) -- end").arg(name));
3189 LOG(VB_CHANNEL, LOG_INFO,
LOC +
3190 QString(
"QueueEITChannelChange(%1)").arg(name));
3207 LOG(VB_CHANNEL, LOG_DEBUG,
LOC +
3208 QString(
"QueueEITChannelChange(%1) %2")
3209 .arg(name, ok ?
"done" :
"failed"));
3215 QString &title, QString &subtitle,
3216 QString &desc, QString &category,
3217 QString &starttime, QString &endtime,
3218 QString &callsign, QString &iconpath,
3219 QString &channum,
uint &sourceChanid,
3220 QString &seriesid, QString &programid)
3222 QString compare =
"<=";
3223 QString sortorder =
"desc";
3228 chanid = sourceChanid;
3276 QString querystr = QString(
3277 "SELECT title, subtitle, description, category, "
3278 " starttime, endtime, callsign, icon, "
3279 " channum, seriesid, programid "
3280 "FROM program, channel "
3281 "WHERE program.chanid = channel.chanid AND "
3282 " channel.chanid = :CHANID AND "
3283 " starttime %1 :STARTTIME "
3284 "ORDER BY starttime %2 "
3285 "LIMIT 1").arg(compare, sortorder);
3290 query.
bindValue(
":STARTTIME", starttime);
3293 title = subtitle = desc = category =
"";
3294 starttime = endtime = callsign = iconpath =
"";
3295 channum = seriesid = programid =
"";
3303 else if (query.
next())
3305 title = query.
value(0).toString();
3306 subtitle = query.
value(1).toString();
3307 desc = query.
value(2).toString();
3308 category = query.
value(3).toString();
3309 starttime = query.
value(4).toString();
3310 endtime = query.
value(5).toString();
3311 callsign = query.
value(6).toString();
3312 iconpath = query.
value(7).toString();
3313 channum = query.
value(8).toString();
3314 seriesid = query.
value(9).toString();
3315 programid = query.
value(10).toString();
3316 sourceChanid = chanid;
3322 "SELECT channum, callsign, icon "
3324 "WHERE chanid = :CHANID");
3331 else if (query.
next())
3333 sourceChanid = chanid;
3334 channum = query.
value(0).toString();
3335 callsign = query.
value(1).toString();
3336 iconpath = query.
value(2).toString();
3341 QString &callsign, QString &channum,
3342 QString &channame, QString &xmltvid)
const
3349 if ((!chanid || !sourceid) && !
m_channel)
3360 "SELECT callsign, channum, name, xmltvid "
3362 "WHERE chanid = :CHANID");
3373 callsign = query.
value(0).toString();
3374 channum = query.
value(1).toString();
3375 channame = query.
value(2).toString();
3376 xmltvid = query.
value(3).toString();
3382 const QString& oldchannum,
3383 const QString& callsign,
const QString& channum,
3384 const QString& channame,
const QString& xmltvid)
3386 if (!chanid || !sourceid || channum.isEmpty())
3392 "SET callsign = :CALLSIGN, "
3393 " channum = :CHANNUM, "
3394 " name = :CHANNAME, "
3395 " xmltvid = :XMLTVID "
3396 "WHERE chanid = :CHANID AND "
3397 " sourceid = :SOURCEID");
3427 if (oldbuffer && (oldbuffer !=
Buffer))
3439 LOG(VB_GENERAL, LOG_INFO,
LOC +
"RingBufferChanged()");
3461 QString &input)
const
3488 if (
m_channel && !channum.isEmpty() && (channum.indexOf(
"NextChannel") >= 0))
3491 int dir = channum.right(channum.length() - 12).toInt();
3533 if (
mpeg->HasProgram(progNum))
3556 LOG(VB_RECORD, LOG_INFO,
LOC +
3557 "HandleTuning Request: " + request.
toString());
3564 LOG(VB_CHANNEL, LOG_INFO,
LOC +
"On same multiplex");
3578 LOG(VB_RECORD, LOG_INFO,
LOC +
3579 "No recorder yet, calling TuningFrequency");
3584 LOG(VB_RECORD, LOG_INFO,
LOC +
"Waiting for recorder pause..");
3597 LOG(VB_RECORD, LOG_INFO,
LOC +
3598 "Recorder paused, calling TuningFrequency");
3606 if (streamData ==
nullptr)
3629 LOG(VB_RECORD, LOG_INFO,
LOC + QString(
"TuningShutdowns(%1)")
3656 if (sd && (sd != rec_sd))
3690 LOG(VB_RECORD, LOG_INFO,
LOC +
"Tearing down RingBuffer");
3718 LOG(VB_GENERAL, LOG_INFO,
LOC + QString(
"TuningFrequency(%1)")
3739 if (request.
m_minorChan && (tuningmode ==
"atsc"))
3764 slist<<
"message"<<QObject::tr(
"On known multiplex...");
3778 if (!channum.isEmpty())
3791 LOG(VB_GENERAL, LOG_ERR,
LOC +
3792 QString(
"Failed to set channel to %1. Reverting to kState_None")
3801 LOG(VB_GENERAL, LOG_ERR,
LOC +
3802 QString(
"Failed to set channel to %1.").arg(channum));
3821 bool use_dr = use_sm && (livetv || antadj);
3822 bool has_dummy =
false;
3843 LOG(VB_GENERAL, LOG_ERR,
LOC +
"Failed to create RingBuffer 1");
3853 LOG(VB_RECORD, LOG_INFO,
LOC +
"Starting Signal Monitor");
3858 LOG(VB_GENERAL, LOG_ERR,
LOC +
"Failed to setup signal monitor");
3876 SetVideoStreamsRequired(0);
3903 LOG(VB_CHANNEL, LOG_DEBUG,
LOC +
3904 QString(
"Pre-fail start deadline: %1 "
3905 "Start recording deadline: %2 "
3906 "Good signal deadline: %3")
3908 .toString(
"hh:mm:ss.zzz"),
3910 .toString(
"hh:mm:ss.zzz"),
3912 .toString(
"hh:mm:ss.zzz")));
3936 LOG(VB_RECORD, LOG_INFO,
"DummyDTVRecorder -- started");
3962 bool keep_trying =
false;
3975 LOG(VB_RECORD, LOG_INFO,
LOC +
"TuningSignalCheck: Good signal");
3981 QString desc = tr(
"Good signal seen after %1 ms")
3990 tr(
"See 'Tuning timeout' in mythtv-setup "
3991 "for this input."));
3994 LOG(VB_GENERAL, LOG_WARNING,
LOC +
3995 QString(
"It took longer than %1 ms to get a signal lock. "
3996 "Keeping status of '%2'")
3999 LOG(VB_GENERAL, LOG_WARNING,
LOC +
4000 "See 'Tuning timeout' in mythtv-setup for this input");
4009 LOG(VB_GENERAL, LOG_ERR,
LOC +
"TuningSignalCheck: SignalMonitor " +
4022 LOG(VB_GENERAL, LOG_ERR,
LOC +
4023 "TuningSignalCheck: Hit pre-fail timeout");
4037 QString desc = tr(
"Taking more than %1 ms to get a lock.")
4045 tr(
"See 'Tuning timeout' in mythtv-setup "
4046 "for this input."));
4050 LOG(VB_GENERAL, LOG_WARNING,
LOC +
4051 QString(
"TuningSignalCheck: taking more than %1 ms to get a lock. "
4052 "marking this recording as '%2'.")
4055 LOG(VB_GENERAL, LOG_WARNING,
LOC +
4056 "See 'Tuning timeout' in mythtv-setup for this input");
4064 LOG(VB_RECORD, LOG_INFO,
LOC +
4065 QString(
"TuningSignalCheck: Still waiting. Will timeout @ %1")
4067 .toString(
"hh:mm:ss.zzz")));
4078 MythEvent me(QString(
"UPDATE_RECORDING_STATUS %1 %2 %3 %4 %5")
4116 LOG(VB_EIT, LOG_INFO,
LOC +
4117 QString(
"EIT scanning disabled for video source %1")
4126 bool on_host,
bool transcode_bfr_comm,
bool on_line_comm)
4142 if ((!autoTrans) || (autoTrans->
getValue().toInt() == 0))
4166 !transcode_bfr_comm;
4190 QString profileName =
"Live TV";
4191 if (!tvchain && rec)
4194 QString profileRequested = profileName;
4199 LOG(VB_RECORD, LOG_INFO,
LOC +
4200 QString(
"Using profile '%1' to record")
4205 profileName =
"Default";
4208 LOG(VB_RECORD, LOG_INFO,
LOC +
4209 QString(
"Profile '%1' not found, using "
4210 "fallback profile '%2' to record")
4211 .arg(profileRequested, profileName));
4215 LOG(VB_RECORD, LOG_ERR,
LOC +
4216 QString(
"Profile '%1' not found, and unable "
4217 "to load fallback profile '%2'. Results "
4218 "may be unpredicable")
4219 .arg(profileRequested, profileName));
4249 LOG(VB_GENERAL, LOG_ERR,
LOC +
"Failed to create RingBuffer 2");
4258 QString pathname = (*rec)->GetPathname();
4259 LOG(VB_GENERAL, LOG_INFO,
LOC + QString(
"rec->GetPathname(): '%1'")
4264 LOG(VB_GENERAL, LOG_ERR,
LOC +
4265 QString(
"RingBuffer '%1' not open...")
4275 LOG(VB_GENERAL, LOG_ERR,
LOC +
4276 QString(
"Failed to start recorder! ringBuffer is NULL\n"
4277 "\t\t\t\t Tuning request was %1\n")
4282 QString message = QString(
"QUIT_LIVETV %1").arg(
m_inputId);
4292 LOG(VB_GENERAL, LOG_INFO,
LOC +
"TuningNewRecorder - CreateRecorder()");
4301 LOG(VB_GENERAL, LOG_ERR,
LOC +
"Failed to initialize recorder!");
4309 LOG(VB_GENERAL, LOG_ERR,
LOC +
4310 QString(
"Failed to start recorder!\n"
4311 "\t\t\t\t Tuning request was %1\n")
4316 QString message = QString(
"QUIT_LIVETV %1").arg(
m_inputId);
4359 std::this_thread::sleep_for(5us);
4387 LOG(VB_RECORD, LOG_INFO,
LOC +
"Starting Recorder");
4389 bool had_dummyrec =
false;
4395 had_dummyrec =
true;
4413 LOG(VB_RECORD, LOG_INFO,
LOC +
4414 QString(
"TuningNewRecorder -- UPDATE_RECORDING_STATUS: %1")
4416 MythEvent me(QString(
"UPDATE_RECORDING_STATUS %1 %2 %3 %4 %5")
4434 LOG(VB_RECORD, LOG_INFO,
LOC +
"Restarting Recorder");
4436 bool had_dummyrec =
false;
4447 had_dummyrec =
true;
4473 QString msg1 = QString(
"Recording: %1 %2 %3 %4")
4478 QString msg2 = QString(
"Recording: %1 %2 %3 %4")
4483 LOG(VB_RECORD, LOG_INFO,
LOC +
"Pseudo LiveTV recording starting." +
4484 "\n\t\t\t" + msg1 +
"\n\t\t\t" + msg2);
4501 LOG(VB_RECORD, LOG_INFO,
LOC + QString(
"SetFlags(%1) -> %2 @ %3:%4")
4510 LOG(VB_RECORD, LOG_INFO,
LOC + QString(
"ClearFlags(%1) -> %2 @ %3:%4")
4521 msg +=
"FrontendReady,";
4523 msg +=
"RunMainLoop,";
4525 msg +=
"ExitPlayer,";
4527 msg +=
"FinishRecording,";
4531 msg +=
"CancelNextRecording,";
4541 msg +=
"Recording,";
4554 msg +=
"AntennaAdjust,";
4557 msg +=
"PENDINGACTIONS,";
4561 msg +=
"WaitingForRecPause,";
4563 msg +=
"WaitingForSignal,";
4565 msg +=
"NeedToStartRecorder,";
4567 msg +=
"KillRingBuffer,";
4570 msg +=
"ANYRUNNING,";
4574 msg +=
"SignalMonitorRunning,";
4576 msg +=
"EITScannerRunning,";
4578 msg +=
"ANYRECRUNNING,";
4582 msg +=
"DummyRecorderRunning,";
4584 msg +=
"RecorderRunning,";
4588 msg +=
"RingBufferReady,";
4591 msg = QString(
"0x%1").arg(f,0,16);
4619 const QString & channum)
4621 LOG(VB_RECORD, LOG_INFO,
LOC +
"GetProgramRingBufferForLiveTV()");
4643 LOG(VB_GENERAL, LOG_ERR,
LOC +
4644 QString(
"Channel: \'%1\' was not found in the database.\n"
4645 "\t\tMost likely, the 'starting channel' for this "
4646 "Input Connection is invalid.\n"
4647 "\t\tCould not start livetv.").arg(channum));
4670 LOG(VB_GENERAL, LOG_ERR,
LOC +
"GetProgramRingBufferForLiveTV()"
4671 "\n\t\t\tProgramInfo is invalid."
4701 if (!(*
Buffer) || !(*Buffer)->IsOpen())
4703 LOG(VB_GENERAL, LOG_ERR,
LOC + QString(
"RingBuffer '%1' not open...")
4718 LOG(VB_RECORD, LOG_INFO,
LOC + QString(
"CreateLiveTVRingBuffer(%1)")
4735 LOG(VB_GENERAL, LOG_ERR,
LOC +
4736 QString(
"CreateLiveTVRingBuffer(%1) failed").arg(channum));
4763 bool discont,
bool set_rec)
4768 msg = QString(
" curRec(%1) curRec.size(%2)")
4772 LOG(VB_RECORD, LOG_INFO,
LOC +
4773 QString(
"SwitchLiveTVRingBuffer(discont %1, set_next_rec %2)")
4774 .arg(discont).arg(set_rec) + msg);
4832 LOG(VB_RECORD, LOG_INFO,
LOC +
"SwitchRecordingRingBuffer()");
4836 LOG(VB_RECORD, LOG_ERR,
LOC +
"SwitchRecordingRingBuffer -> "
4837 "already switching.");
4843 LOG(VB_RECORD, LOG_ERR,
LOC +
"SwitchRecordingRingBuffer -> "
4844 "invalid recorder.");
4850 LOG(VB_RECORD, LOG_ERR,
LOC +
"SwitchRecordingRingBuffer -> "
4851 "invalid recording.");
4857 LOG(VB_RECORD, LOG_ERR,
LOC +
"SwitchRecordingRingBuffer -> "
4858 "Not the same channel.");
4869 LOG(VB_RECORD, LOG_ERR,
LOC +
4870 QString(
"SwitchRecordingRingBuffer() -> "
4871 "cannot switch profile '%1' to '%2'")
4883 if (!buffer || !buffer->
IsOpen())
4890 LOG(VB_RECORD, LOG_ERR,
LOC +
"SwitchRecordingRingBuffer() -> "
4891 "Failed to create new RB.");
4900 LOG(VB_RECORD, LOG_INFO,
LOC +
"SwitchRecordingRingBuffer -> done");
4906 QMap<uint,TVRec*>::const_iterator it =
s_inputs.constFind(inputid);
4914 LOG(VB_RECORD, LOG_INFO,
LOC + QString(
"enable:%1").arg(enable));
4941 return QString(
"Program(%1) channel(%2) input(%3) flags(%4)")
std::vector< pid_cache_item_t > pid_cache_t
Encapsulates data about ATSC stream and emits events for most tables.
const MasterGuideTable * GetCachedMGT(bool current=true) const
void SetDesiredChannel(int major, int minor)
static bool GetInputInfo(InputInfo &input, std::vector< uint > *groupids=nullptr)
static QString GetStartChannel(uint inputid)
static bool IsEITCapable(const QString &rawtype)
static QString GetInputName(uint inputid)
static bool IsV4L(const QString &rawtype)
static uint GetSourceID(uint inputid)
static bool IsEncoder(const QString &rawtype)
static std::vector< uint > GetConflictingInputs(uint inputid)
static bool IsChannelReusable(const QString &rawtype)
Abstract class providing a generic interface to tuning hardware.
virtual QString GetInputName(void) const
virtual int ChangePictureAttribute(PictureAdjustType, PictureAttribute, bool)
virtual uint GetSourceID(void) const
virtual bool Open(void)=0
Opens the channel changing hardware for use.
virtual uint GetNextChannel(uint chanid, ChannelChangeDirection direction) const
virtual void StoreInputChannels(void)
Saves current channel as the default channel for the current input.
virtual void Close(void)=0
Closes the channel changing hardware to use.
virtual bool IsOpen(void) const =0
Reports whether channel is already open.
virtual QString GetChannelName(void) const
virtual void Renumber(uint sourceid, const QString &oldChanNum, const QString &newChanNum)
Changes a channum if we have it cached anywhere.
bool CheckChannel(const QString &channum) const
virtual int GetPictureAttribute(PictureAttribute) const
virtual int GetChanID(void) const
virtual void SetFd(int fd)
Sets file descriptor.
virtual bool Init(QString &startchannel, bool setchan)
virtual int GetInputID(void) const
static ChannelBase * CreateChannel(TVRec *tvrec, const GeneralDBOptions &genOpt, const DVBDBOptions &dvbOpt, const FireWireDBOptions &fwOpt, const QString &startchannel, bool enter_power_save_mode, QString &rbFileExt, bool setchan)
virtual bool InitPictureAttributes(void)
virtual bool SetChannelByString(const QString &chan)=0
static bool ToggleChannel(uint chanid, int changrpid, bool delete_chan)
static int GetChannelGroupId(const QString &changroupname)
static uint GetMplexID(uint sourceid, const QString &channum)
static int GetChanID(int db_mplexid, int service_transport_id, int major_channel, int minor_channel, int program_number)
static int GetProgramNumber(uint sourceid, const QString &channum)
static QString GetVideoFilters(uint sourceid, const QString &channum)
static bool IsOnSameMultiplex(uint srcid, const QString &new_channum, const QString &old_channum)
static QString GetChanNum(int chan_id)
Returns the channel-number string of the given channel.
static bool GetATSCChannel(uint sourceid, const QString &channum, uint &major, uint &minor)
Class providing a generic interface to digital tuning hardware.
int GetProgramNumber(void) const
Returns program number in PAT, -1 if unknown.
void SaveCachedPids(const pid_cache_t &pid_cache) const
Saves MPEG PIDs to cache to database.
QString GetTuningMode(void) const
Returns tuning mode last set by SetTuningMode().
uint GetTransportID(void) const
Returns DVB transport_stream_id, 0 if unknown.
QString GetSIStandard(void) const
Returns PSIP table standard: MPEG, DVB, ATSC, or OpenCable.
void SetTuningMode(const QString &tuning_mode)
Sets tuning mode: "mpeg", "dvb", "atsc", etc.
uint GetMajorChannel(void) const
Returns major channel, 0 if unknown.
uint GetMinorChannel(void) const
Returns minor channel, 0 if unknown.
void GetCachedPids(pid_cache_t &pid_cache) const
Returns cached MPEG PIDs for last tuned channel.
QString GetSuggestedTuningMode(bool is_live_tv) const
Returns suggested tuning mode: "mpeg", "dvb", or "atsc".
virtual bool EnterPowerSavingMode(void)
Enters power saving mode if the card supports it.
uint GetOriginalNetworkID(void) const
Returns DVB original_network_id, 0 if unknown.
This is a specialization of RecorderBase used to handle MPEG-2, MPEG-4, MPEG-4 AVC,...
MPEGStreamData * GetStreamData(void) const
virtual void SetStreamData(MPEGStreamData *data)
This class is intended to detect the presence of needed tables.
void SetDVBService(uint network_id, uint transport_id, int service_id)
void AddFlags(uint64_t _flags) override
ATSCStreamData * GetATSCStreamData()
Returns the ATSC stream data if it exists.
void SetChannel(int major, int minor)
virtual void SetRotorTarget(float)
Sets rotor target pos from 0.0 to 1.0.
void IgnoreEncrypted(bool ignore)
MPEGStreamData * GetStreamData()
Returns the MPEG stream data if it exists.
void SetProgramNumber(int progNum)
virtual void SetStreamData(MPEGStreamData *data)
Sets the MPEG stream data for DTVSignalMonitor to use, and connects the table signals to the monitor.
Provides interface to the tuning hardware when using DVB drivers.
std::chrono::milliseconds m_dvbTuningDelay
void SetDishNetEIT(bool use_dishnet_eit)
Acts as glue between ChannelBase, EITSource and EITHelper.
void StopEITEventProcessing(void)
Stops inserting Event Information Tables into DB.
void StartEITEventProcessing(ChannelBase *channel, EITSource *eitSource)
Start inserting Event Information Tables from the multiplex we happen to be tuned to into the databas...
void StartActiveScan(TVRec *rec, std::chrono::seconds max_seconds_per_multiplex)
Start active EIT scan.
void StopActiveScan(void)
Stop active EIT scan.
static bool QueueRecordingJobs(const RecordingInfo &recinfo, int jobTypes=JOB_NONE)
static void RemoveJobsFromMask(int jobs, int &mask)
static void AddJobsToMask(int jobs, int &mask)
static bool JobIsInMask(int job, int mask)
static bool QueueJob(int jobType, uint chanid, const QDateTime &recstartts, const QString &args="", const QString &comment="", QString host="", int flags=0, int status=JOB_QUEUED, QDateTime schedruntime=QDateTime())
static bool JobIsNotInMask(int job, int mask)
Keeps track of recordings in a current LiveTV instance.
int TotalSize(void) const
QString GetID(void) const
void FinishedRecording(ProgramInfo *pginfo)
void ReloadAll(const QStringList &data=QStringList())
void SetInputType(const QString &type)
ProgramInfo * GetProgramAt(int at) const
Returns program at the desired location.
void SetHostPrefix(const QString &prefix)
QString GetInputType(int pos=-1) const
void AppendNewProgram(ProgramInfo *pginfo, const QString &channum, const QString &inputname, bool discont)
Encapsulates data about MPEG stream and emits events for each table.
void SetRecordingType(const QString &recording_type)
virtual void ReturnCachedTable(const PSIPTable *psip) const
void SetIgnoreCRC(bool haveCRCbug)
void SetVideoStreamsRequired(uint num)
void SetCaching(bool cacheTables)
virtual void AddListeningPID(uint pid, PIDPriority priority=kPIDPriorityNormal)
QSqlQuery wrapper that fetches a DB connection from the connection pool.
bool prepare(const QString &query)
QSqlQuery::prepare() is not thread safe in Qt <= 3.3.2.
QVariant value(int i) const
bool isActive(void) const
bool isConnected(void) const
Only updated once during object creation.
bool exec(void)
Wrap QSqlQuery::exec() so we can display SQL.
void bindValue(const QString &placeholder, const QVariant &val)
Add a single binding.
bool next(void)
Wrap QSqlQuery::next() so we can display the query results.
static MSqlQueryInfo InitCon(ConnectionReuse _reuse=kNormalConnection)
Only use this in combination with MSqlQuery constructor.
This is a wrapper around QThread that does several additional things.
void start(QThread::Priority p=QThread::InheritPriority)
Tell MThread to start running the thread in the near future.
bool wait(std::chrono::milliseconds time=std::chrono::milliseconds::max())
Wait for the MThread to exit, with a maximum timeout.
This table tells the decoder on which PIDs to find other tables, and their sizes and each table's cur...
uint TableType(uint i) const
uint TablePID(uint i) const
QString GetHostName(void)
QString GetSetting(const QString &key, const QString &defaultval="")
void SendSystemEvent(const QString &msg)
int GetBackendServerPort(void)
Returns the locally defined backend control port.
void dispatch(const MythEvent &event)
static QString GenMythURL(const QString &host=QString(), int port=0, QString path=QString(), const QString &storageGroup=QString())
int GetNumSetting(const QString &key, int defaultval=0)
void SendEvent(const MythEvent &event)
std::enable_if_t< std::chrono::__is_duration< T >::value, T > GetDurSetting(const QString &key, T defaultval=T::zero())
bool GetBoolSetting(const QString &key, bool defaultval=false)
static void DBError(const QString &where, const MSqlQuery &query)
T dequeue()
Removes item from front of list and returns a copy. O(1).
void enqueue(const T &d)
Adds item to the back of the list. O(1).
This class is used as a container for messages.
void SetDuration(std::chrono::seconds Duration)
Contains a duration during which the notification will be displayed for. The duration is informative ...
A QElapsedTimer based timer to replace use of QTime as a timer.
std::chrono::milliseconds elapsed(void)
Returns milliseconds elapsed since last start() or restart()
void start(void)
starts measuring elapsed time.
std::vector< uint > m_possibleConflicts
QDateTime m_recordingStart
static void GetPreviewImage(const ProgramInfo &pginfo, const QString &token)
Submit a request for the generation of a preview image.
Holds information on recordings and videos.
uint GetChanID(void) const
This is the unique key used in the database to locate tuning information.
uint GetRecordingRuleID(void) const
bool IsSameProgramWeakCheck(const ProgramInfo &other) const
Checks for duplicate using only title, chanid and startts.
QString toString(Verbosity v=kLongDescription, const QString &sep=":", const QString &grp="\"") const
bool QueryTuningInfo(QString &channum, QString &input) const
Returns the channel and input needed to record the program.
void UpdateInUseMark(bool force=false)
QString GetRecordingGroup(void) const
void SaveAutoExpire(AutoExpireType autoExpire, bool updateDelete=false)
Set "autoexpire" field in "recorded" table to "autoExpire".
void SetRecordingRuleType(RecordingType type)
uint GetRecordingID(void) const
QDateTime GetScheduledEndTime(void) const
The scheduled end time of the program.
uint QueryMplexID(void) const
Queries multiplex any recording would be made on, zero if unknown.
void SetRecordingStatus(RecStatus::Type status)
void SetRecordingGroup(const QString &group)
uint GetSourceID(void) const
void SetScheduledEndTime(const QDateTime &dt)
void SetRecordingStartTime(const QDateTime &dt)
void SaveVideoProperties(uint mask, uint video_property_flags)
QString GetTitle(void) const
uint QueryAverageHeight(void) const
If present in recording this loads average height of the main video stream from database's stream mar...
void MarkAsInUse(bool inuse, const QString &usedFor="")
Tracks a recording's in use status, to prevent deletion and to allow the storage scheduler to perform...
QDateTime GetRecordingStartTime(void) const
Approximate time the recording started.
bool QueryAverageScanProgressive(void) const
If present in recording this loads average video scan type of the main video stream from database's s...
QDateTime GetScheduledStartTime(void) const
The scheduled start time of program.
void SetChanID(uint _chanid)
bool IsCommercialFree(void) const
MarkTypes QueryAverageAspectRatio(void) const
void SetRecordingRuleID(uint id)
QString MakeUniqueKey(void) const
Creates a unique string that can be used to identify an existing recording.
QString GetPathname(void) const
uint GetInputID(void) const
void ToStringList(QStringList &list) const
Serializes ProgramInfo into a QStringList which can be passed over a socket.
void SetRecordingEndTime(const QDateTime &dt)
void SetStorageGroup(const QString &group)
RecStatus::Type GetRecordingStatus(void) const
QDateTime GetRecordingEndTime(void) const
Approximate time the recording should have ended, did end, or is intended to end.
void SaveCommFlagged(CommFlagStatus flag)
Set "commflagged" field in "recorded" table to "flag".
QString GetSubtitle(void) const
QString GetCategory(void) const
virtual void SetRecordingID(uint _recordedid)
void SetPathname(const QString &pn)
RecordingType GetRecordingRuleType(void) const
QString QueryRecordingGroup(void) const
Query recgroup from recorded.
static QString toString(RecStatus::Type recstatus, uint id)
Converts "recstatus" into a short (unreadable) string.
virtual void Reset(void)=0
Reset the recorder to the startup state.
virtual bool IsRecording(void)
Tells whether the StartRecorder() loop is running.
virtual void Pause(bool clear=true)
Pause tells recorder to pause, it should not block.
virtual void SetVideoFilters(QString &filters)=0
Tells recorder which filters to use.
virtual int GetVideoFd(void)=0
Returns file descriptor of recorder device.
virtual void Unpause(void)
Unpause tells recorder to unpause.
virtual bool CheckForRingBufferSwitch(void)
If requested, switch to new RingBuffer/ProgramInfo objects.
virtual void Initialize(void)=0
This is called between SetOptionsFromProfile() and run() to initialize any devices,...
void SetRecording(const RecordingInfo *pginfo)
Changes the Recording from the one set initially with SetOptionsFromProfile().
double GetFrameRate(void) const
Returns the latest frame rate.
bool GetKeyframePositions(long long start, long long end, frm_pos_map_t &map) const
void SetNextRecording(const RecordingInfo *ri, MythMediaBuffer *Buffer)
Sets next recording info, to be applied as soon as practical.
virtual void StopRecording(void)
StopRecording() signals to the recorder that it should stop recording and exit cleanly.
virtual RecordingQuality * GetRecordingQuality(const RecordingInfo *ri) const
Returns a report about the current recordings quality.
bool GetKeyframeDurations(long long start, long long end, frm_pos_map_t &map) const
virtual long long GetFramesWritten(void)=0
Returns number of frames written to disk.
long long GetKeyframePosition(long long desired) const
Returns closest keyframe position before the desired frame.
virtual bool IsPaused(bool holding_lock=false) const
Returns true iff recorder is paused.
void SavePositionMap(bool force=false, bool finished=false)
Save the seektable to the DB.
static RecorderBase * CreateRecorder(TVRec *tvrec, ChannelBase *channel, RecordingProfile &profile, const GeneralDBOptions &genOpt)
virtual bool IsErrored(void)=0
Tells us whether an unrecoverable error has been encountered.
void SetRingBuffer(MythMediaBuffer *Buffer)
Tells recorder to use an externally created ringbuffer.
Holds information on a TV Program one might wish to record.
void FinishedRecording(bool allowReRecord)
If not a premature stop, adds program to history of recorded programs.
void SetDesiredStartTime(const QDateTime &dt)
RecordingRule * GetRecordingRule(void)
Returns the "record" field, creating it if necessary.
void ApplyRecordRecGroupChange(const QString &newrecgroup)
Sets the recording group, both in this RecordingInfo and in the database.
int GetAutoRunJobs(void) const
Returns a bitmap of which jobs are attached to this RecordingInfo.
void AddHistory(bool resched=true, bool forcedup=false, bool future=false)
Adds recording history, creating "record" it if necessary.
void ApplyRecordRecID(void)
Sets recordid to match RecordingRule recordid.
void UpdateRecordingEnd(void)
Update information in the recorded table when the end-time of a recording is changed.
uint64_t GetFilesize(void) const override
void StartedRecording(const QString &ext)
Inserts this RecordingInfo into the database as an existing recording.
void SetDesiredEndTime(const QDateTime &dt)
RecordingFile * GetRecordingFile() const
bool IsDamaged(void) const
QString toStringXML(void) const
AutoExpireType GetAutoExpire(void) const
bool Save(bool sendSig=true)
virtual int DecrRef(void)
Decrements reference count and deletes on 0.
virtual int IncrRef(void)
Increments reference count.
static void Init()
Initializes the some static constants needed by SignalMonitorValue.
void AddListener(SignalMonitorListener *listener)
static SignalMonitor * Init(const QString &cardtype, int db_cardnum, ChannelBase *channel, bool release_stream)
static const uint64_t kDVBSigMon_WaitForPos
Wait for rotor to complete turning the antenna.
static const uint64_t kDTVSigMon_WaitForSDT
bool IsErrored(void) const
virtual bool HasExtraSlowTuning(void) const
virtual bool IsAllGood(void) const
static bool IsSupported(const QString &cardtype)
static const uint64_t kDTVSigMon_WaitForPMT
void SetUpdateRate(std::chrono::milliseconds msec)
Sets the number of milliseconds between signal monitoring attempts in the signal monitoring thread.
static const uint64_t kDTVSigMon_WaitForPAT
void SetNotifyFrontend(bool notify)
Enables or disables frontend notification of the current signal value.
static const uint64_t kDTVSigMon_WaitForMGT
static bool IsRequired(const QString &cardtype)
Returns true iff the card type supports signal monitoring.
virtual void Start()
Start signal monitoring thread.
static QString GetSourceName(uint sourceid)
virtual QString getValue(void) const
QString FindNextDirMostFree(void)
This is the coordinating class of the Recorder Subsystem.
bool SwitchLiveTVRingBuffer(const QString &channum, bool discont, bool set_rec)
V4LChannel * GetV4LChannel(void)
bool ShouldSwitchToAnotherInput(const QString &chanid) const
Checks if named channel exists on current tuner, or another tuner.
bool GetChannelInfo(uint &chanid, uint &sourceid, QString &callsign, QString &channum, QString &channame, QString &xmltvid) const
static const uint kFlagRecording
final result desired is a timed recording
static const uint kFlagAnyRecRunning
TuningQueue m_tuningRequests
bool SetupDTVSignalMonitor(bool EITscan)
Tells DTVSignalMonitor what channel to look for.
QHash< QString, int > m_autoRunJobs
bool GetKeyframePositions(int64_t start, int64_t end, frm_pos_map_t &map) const
Returns byte position in RingBuffer of a keyframes according to recorder.
static bool GetDevices(uint inputid, uint &parentid, GeneralDBOptions &gen_opts, DVBDBOptions &dvb_opts, FireWireDBOptions &firewire_opts)
void NotifySchedulerOfRecording(RecordingInfo *rec)
Tell scheduler about the recording.
void PauseRecorder(void)
Tells "recorder" to pause, used for channel and input changes.
DTVChannel * GetDTVChannel(void)
uint GetFlags(void) const
bool CreateChannel(const QString &startchannel, bool enter_power_save_mode)
void FinishedRecording(RecordingInfo *curRec, RecordingQuality *recq)
If not a premature stop, adds program to history of recorded programs.
QDateTime m_signalEventCmdTimeout
static const uint kFlagErrored
TVState RemoveRecording(TVState state) const
If "state" is kState_RecordingOnly or kState_WatchingLiveTV, returns a kState_None,...
bool SetVideoFiltersForChannel(uint sourceid, const QString &channum)
void SetPseudoLiveTVRecording(RecordingInfo *pi)
Sets the pseudo LiveTV RecordingInfo.
static const uint kFlagEITScannerRunning
TVRec(int _inputid)
Performs instance initialization not requiring access to database.
bool SetChannelInfo(uint chanid, uint sourceid, const QString &oldchannum, const QString &callsign, const QString &channum, const QString &channame, const QString &xmltvid)
QString GetChainID(void)
Get the chainid of the livetv instance.
void TuningFrequency(const TuningRequest &request)
Performs initial tuning required for any tuning event.
QMutex m_nextLiveTVDirLock
static const uint kFlagExitPlayer
void SetFlags(uint f, const QString &file, int line)
static const uint kFlagFinishRecording
static const uint kFlagEITScan
final result desired is an EIT Scan
TVState m_desiredNextState
static bool StateIsRecording(TVState state)
Returns true if "state" is kState_RecordingOnly, or kState_WatchingLiveTV.
QDateTime m_recordEndTime
~TVRec(void) override
Stops the event and scanning threads and deletes any ChannelBase, RingBuffer, and RecorderBase instan...
void SetChannel(const QString &name, uint requestType=kFlagDetect)
Changes to a named channel on the current tuner.
bool Init(void)
Performs instance initialization, returns true on success.
GeneralDBOptions m_genOpt
void TuningNewRecorder(MPEGStreamData *streamData)
Creates a recorder instance.
RecordingInfo * m_curRecording
bool QueueEITChannelChange(const QString &name)
Queues up a channel change for the EITScanner.
std::chrono::seconds m_eitTransportTimeout
QString m_liveTVStartChannel
QString m_overRecordCategory
ProgramInfo * GetRecording(void)
Allocates and returns a ProgramInfo for the current recording.
friend class TuningRequest
QWaitCondition m_triggerEventSleepWait
void SetNextLiveTVDir(QString dir)
RecStatus::Type GetRecordingStatus(void) const
static const uint kFlagFrontendReady
static const uint kFlagKillRingBuffer
long long GetMaxBitrate(void) const
Returns the maximum bits per second this recorder can produce.
RecStatus::Type StartRecording(ProgramInfo *pginfo)
Tells TVRec to Start recording the program "rcinfo" as soon as possible.
bool WaitForEventThreadSleep(bool wake=true, std::chrono::milliseconds time=std::chrono::milliseconds::max())
You MUST HAVE the stateChange-lock locked when you call this method!
static const uint kFlagRecorderRunning
void ToggleChannelFavorite(const QString &changroupname)
Toggles whether the current channel should be on our favorites list.
std::chrono::seconds m_eitCrawlIdleStart
void HandleStateChange(void)
Changes the internalState to the desiredNextState if possible.
QRecursiveMutex m_pendingRecLock
void RecorderPaused(void)
This is a callback, called by the "recorder" instance when it has actually paused.
QDateTime m_eitScanStopTime
static const uint kFlagKillRec
close recorder, discard recording
QMutex m_triggerEventLoopLock
bool m_reachedRecordingDeadline
int ChangePictureAttribute(PictureAdjustType type, PictureAttribute attr, bool direction)
Returns current value [0,100] if it succeeds, -1 otherwise.
TuningRequest m_lastTuningRequest
bool IsErrored(void) const
Returns true is "errored" is true, false otherwise.
QDateTime m_eitScanStartTime
RecordingInfo * SwitchRecordingRingBuffer(const RecordingInfo &rcinfo)
long long GetFramesWritten(void)
Returns number of frames written to disk by recorder.
void StartedRecording(RecordingInfo *curRec)
Inserts a "curRec" into the database.
QDateTime m_preFailDeadline
void TuningShutdowns(const TuningRequest &request)
This shuts down anything that needs to be shut down before handling the passed in tuning request.
void RecordPending(const ProgramInfo *rcinfo, std::chrono::seconds secsleft, bool hasLater)
Tells TVRec "rcinfo" is the next pending recording.
bool CheckChannelPrefix(const QString &prefix, uint &complete_valid_channel_on_rec, bool &is_extra_char_useful, QString &needed_spacer) const
Checks a prefix against the channels in the DB.
void StopLiveTV(void)
Tells TVRec to stop a "Live TV" recorder.
PendingMap m_pendingRecordings
void GetNextProgram(BrowseDirection direction, QString &title, QString &subtitle, QString &desc, QString &category, QString &starttime, QString &endtime, QString &callsign, QString &iconpath, QString &channum, uint &chanid, QString &seriesid, QString &programid)
RecorderBase * m_recorder
long long GetFilePosition(void)
Returns total number of bytes written by RingBuffer.
static const uint kFlagRunMainLoop
bool m_triggerEventSleepSignal
static const uint kFlagRingBufferReady
SignalMonitor * m_signalMonitor
std::chrono::seconds m_eitScanPeriod
void TeardownRecorder(uint request_flags)
Tears down the recorder.
bool IsBusy(InputInfo *busy_input=nullptr, std::chrono::seconds time_buffer=5s) const
Returns true if the recorder is busy, or will be within the next time_buffer seconds.
void CheckForRecGroupChange(void)
Check if frontend changed the recording group.
QString TuningGetChanNum(const TuningRequest &request, QString &input) const
FireWireDBOptions m_fwOpt
void run(void) override
Event handling method, contains event loop.
static const uint kFlagRec
QWaitCondition m_triggerEventLoopWait
static const uint kFlagAnyRunning
void SetRecordingStatus(RecStatus::Type new_status, int line, bool have_lock=false)
void CancelNextRecording(bool cancel)
Tells TVRec to cancel the upcoming recording.
uint m_signalMonitorCheckCnt
MythMediaBuffer * m_buffer
static const uint kFlagLiveTV
final result desired is LiveTV recording
QWaitCondition m_triggerLiveTVDir
static const uint kFlagCancelNextRecording
void SetRingBuffer(MythMediaBuffer *Buffer)
Sets "ringBuffer", deleting any existing RingBuffer.
QString GetInput(void) const
Returns current input.
QDateTime m_startRecordingDeadline
static TVRec * GetTVRec(uint inputid)
void SetLiveRecording(int recording)
Tells the Scheduler about changes to the recording status of the LiveTV recording.
static QReadWriteLock s_inputsLock
TVState RemovePlaying(TVState state) const
Returns TVState that would remove the playing, but potentially keep recording if we are watching an i...
void RingBufferChanged(MythMediaBuffer *Buffer, RecordingInfo *pginfo, RecordingQuality *recq)
static const uint kFlagDummyRecorderRunning
QRecursiveMutex m_stateChangeLock
RecordingInfo * m_pseudoLiveTVRecording
bool WaitForNextLiveTVDir(void)
static const uint kFlagSignalMonitorRunning
static QString FlagToString(uint f)
TVState GetState(void) const
Returns the TVState of the recorder.
bool CreateLiveTVRingBuffer(const QString &channum)
static const uint kFlagPendingActions
bool m_triggerEventLoopSignal
void TuningRestartRecorder(void)
Restarts a stopped recorder or unpauses a paused recorder.
DTVRecorder * GetDTVRecorder(void)
int GetPictureAttribute(PictureAttribute attr)
std::chrono::seconds m_overRecordSecNrml
bool SetupSignalMonitor(bool tablemon, bool EITscan, bool notify)
This creates a SignalMonitor instance and begins signal monitoring.
QMutex m_triggerEventSleepLock
volatile bool m_switchingBuffer
static const uint kFlagAntennaAdjust
antenna adjusting mode (LiveTV without recording).
uint GetInputId(void) const
Returns the inputid.
void InitAutoRunJobs(RecordingInfo *rec, AutoRunInitType t, RecordingProfile *recpro, int line)
static const uint kFlagNoRec
bool m_signalEventCmdSent
static const uint kFlagCloseRec
close recorder, keep recording
static const uint kFlagDetect
QDateTime m_signalMonitorDeadline
QDateTime GetRecordEndTime(const ProgramInfo *pi) const
Returns recording end time with proper post-roll.
RecStatus::Type m_recStatus
void HandlePendingRecordings(void)
static QMap< uint, TVRec * > s_inputs
void SpawnLiveTV(LiveTVChain *newchain, bool pip, QString startchan)
Tells TVRec to spawn a "Live TV" recorder.
void HandleTuning(void)
Handles all tuning events.
void StopRecording(bool killFile=false)
Changes from a recording state to kState_None.
void EnableActiveScan(bool enable)
std::chrono::milliseconds SetSignalMonitoringRate(std::chrono::milliseconds rate, int notifyFrontend=1)
Sets the signal monitoring rate.
void ClearFlags(uint f, const QString &file, int line)
bool IsReallyRecording(void)
Returns true if frontend can consider the recorder started.
int64_t GetKeyframePosition(uint64_t desired) const
Returns byte position in RingBuffer of a keyframe according to recorder.
static constexpr std::chrono::milliseconds kSignalMonitoringRate
How many milliseconds the signal monitor should wait between checks.
static const uint kFlagNeedToStartRecorder
bool TuningOnSameMultiplex(TuningRequest &request)
MThread * m_recorderThread
Recorder thread, runs RecorderBase::run().
std::chrono::seconds m_overRecordSecCat
bool CheckChannel(const QString &name) const
Checks if named channel exists on current tuner.
static const uint kFlagWaitingForSignal
void TeardownSignalMonitor(void)
If a SignalMonitor instance exists, the monitoring thread is stopped and the instance is deleted.
MPEGStreamData * TuningSignalCheck(void)
This checks if we have a channel lock.
bool GetKeyframeDurations(int64_t start, int64_t end, frm_pos_map_t &map) const
std::vector< uint > m_eitInputs
void ChangeState(TVState nextState)
Puts a state change on the nextState queue.
bool GetProgramRingBufferForLiveTV(RecordingInfo **pginfo, MythMediaBuffer **Buffer, const QString &channum)
bool TuningNewRecorderReal(MPEGStreamData *streamData, RecordingInfo **rec, RecordingProfile &profile, bool had_dummyrec)
Helper function for TVRec::TuningNewRecorder.
MThread * m_eventThread
Event processing thread, runs TVRec::run().
bool HasFlags(uint f) const
float GetFramerate(void)
Returns recordering frame rate from the recorder.
DTVSignalMonitor * GetDTVSignalMonitor(void)
uint GetSourceID(void) const
Returns current source id.
static const uint kFlagWaitingForRecPause
QString SetInput(QString input)
Changes to the specified input.
static bool StateIsPlaying(TVState state)
Returns true if we are in any state associated with a player.
QString LoadProfile(void *tvchain, RecordingInfo *rec, RecordingProfile &profile) const
bool IsOnSameMultiplex(void) const
RecordingInfo * m_program
QString toString(void) const
Implements tuning for TV cards using the V4L driver API, both versions 1 and 2.
MythCoreContext * gCoreContext
This global variable contains the MythCoreContext instance for the app.
#define LOG(_MASK_, _LEVEL_, _QSTRING_)
Convenience inline random number generator functions.
void SendMythSystemRecEvent(const QString &msg, const RecordingInfo *pginfo)
std::chrono::seconds secsInPast(const QDateTime &past)
QString toString(const QDateTime &raw_dt, uint format)
Returns formatted string representing the time.
std::chrono::seconds secsInFuture(const QDateTime &future)
QDateTime current(bool stripped)
Returns current Date and Time in UTC.
uint32_t MythRandom()
generate 32 random bits
def write(text, progress=True)
const QString kRecorderInUseID
QMap< long long, long long > frm_pos_map_t
Frame # -> File offset map.
QString StateToString(TVState state)
Returns a human readable QString representing a TVState.
BrowseDirection
Used to request ProgramInfo for channel browsing.
@ BROWSE_SAME
Fetch browse information on current channel and time.
@ BROWSE_RIGHT
Fetch information on current channel in the future.
@ BROWSE_LEFT
Fetch information on current channel in the past.
@ BROWSE_UP
Fetch information on previous channel.
@ BROWSE_FAVORITE
Fetch information on the next favorite channel.
@ BROWSE_DOWN
Fetch information on next channel.
ChannelChangeDirection
ChannelChangeDirection is an enumeration of possible channel changing directions.
@ CHANNEL_DIRECTION_FAVORITE
TVState
TVState is an enumeration of the states used by TV and TVRec.
@ kState_None
None State, this is the initial state in both TV and TVRec, it indicates that we are ready to change ...
@ kState_RecordingOnly
Recording Only is a TVRec only state for when we are recording a program, but there is no one current...
@ kState_WatchingLiveTV
Watching LiveTV is the state for when we are watching a recording and the user has control over the c...
@ kState_Error
Error State, if we ever try to enter this state errored is set.
@ kState_WatchingPreRecorded
Watching Pre-recorded is a TV only state for when we are watching a pre-existing recording.
@ kState_ChangingState
This is a placeholder state which we never actually enter, but is returned by GetState() when we are ...
static std::chrono::seconds eit_start_rand(uint inputId, std::chrono::seconds eitTransportTimeout)
static void GetPidsToCache(DTVSignalMonitor *dtvMon, pid_cache_t &pid_cache)
static int get_highest_input(void)
static void apply_broken_dvb_driver_crc_hack(ChannelBase *, MPEGStreamData *)
static int init_jobs(const RecordingInfo *rec, RecordingProfile &profile, bool on_host, bool transcode_bfr_comm, bool on_line_comm)
static bool get_use_eit(uint inputid)
static QString add_spacer(const QString &channel, const QString &spacer)
Adds the spacer before the last character in chan.
static bool ApplyCachedPids(DTVSignalMonitor *dtvMon, const DTVChannel *channel)
static bool is_dishnet_eit(uint inputid)
#define TRANSITION(ASTATE, BSTATE)
bool RemoteStopRecording(uint inputid)
bool RemoteIsBusy(uint inputid, InputInfo &busy_input)
bool RemoteRecordPending(uint inputid, const ProgramInfo *pginfo, std::chrono::seconds secsleft, bool hasLater)
uint RemoteGetState(uint inputid)