Go to the documentation of this file.
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) // for static functions
56 bool on_host,
bool transcode_bfr_comm,
bool on_line_comm);
58 static std::chrono::seconds
eit_start_rand(
uint inputId, std::chrono::seconds eitTransportTimeout);
86 : m_eventThread(new
MThread(
"TVRecEvent", this)),
94 bool enter_power_save_mode)
96 LOG(VB_CHANNEL, LOG_INFO,
LOC + QString(
"CreateChannel(%1)")
109 startchannel, enter_power_save_mode,
m_rbFileExt, setchan);
117 LOG(VB_GENERAL, LOG_ERR,
LOC +
118 QString(
"CreateChannel(%1) failed due to VBOX not responding "
119 "to network check on inputid [%2]")
129 if (!CardUtil::IsSatIPPresent(
m_inputId))
132 LOG(VB_GENERAL, LOG_ERR,
LOC +
133 QString(
"CreateChannel(%1) failed due to SatIP box not responding "
134 "to network check on inputid [%2]")
161 LOG(VB_CHANNEL, LOG_ERR, QString(
"[%1] Failed to GetDevices")
170 if (startchannel.isEmpty())
174 LOG(VB_CHANNEL, LOG_ERR, QString(
"[%1] Failed to create channel "
226 LOG(VB_RECORD, LOG_INFO,
LOC +
"TeardownAll");
308 LOG(VB_RECORD, LOG_INFO,
LOC +
"Pending recording revoked on " +
309 QString(
"inputid [%1]").arg(rcinfo->
GetInputID()));
315 (*it).m_doNotAsk = (*it).m_canceled =
true;
320 LOG(VB_RECORD, LOG_INFO,
LOC +
321 QString(
"RecordPending on inputid [%1]").arg(rcinfo->
GetInputID()));
327 pending.
m_ask =
true;
343 for (
uint inputid : inputids)
378 LOG(VB_RECORD, LOG_INFO,
LOC +
379 QString(
"CancelNextRecording(%1) -- begin").arg(cancel));
384 LOG(VB_RECORD, LOG_INFO,
LOC + QString(
"CancelNextRecording(%1) -- "
385 "error, unknown recording").arg(cancel));
391 std::vector<unsigned int> &inputids = (*it).m_possibleConflicts;
392 for (
uint inputid : inputids)
394 LOG(VB_RECORD, LOG_INFO,
LOC +
395 QString(
"CancelNextRecording -- inputid 0x%1")
396 .arg((uint64_t)inputid,0,16));
403 LOG(VB_RECORD, LOG_INFO,
LOC +
404 QString(
"CancelNextRecording -- inputid [%1]")
411 (*it).m_canceled =
false;
414 LOG(VB_RECORD, LOG_INFO,
LOC +
415 QString(
"CancelNextRecording(%1) -- end").arg(cancel));
432 LOG(VB_RECORD, LOG_INFO,
LOC + QString(
"StartRecording(%1)")
458 .addSecs(post_roll_seconds);
460 QString msg = QString(
"updating recording: %1 %2 %3 %4")
465 LOG(VB_RECORD, LOG_INFO,
LOC + msg);
473 bool cancelNext =
false;
475 PendingMap::iterator it;
480 (*it).m_ask = (*it).m_doNotAsk =
false;
481 cancelNext = (*it).m_canceled;
502 LOG(VB_RECORD, LOG_INFO,
LOC +
503 "Checking input group recorders - begin");
509 std::vector<unsigned int> inputids2;
510 std::vector<TVState> states;
513 for (
uint inputid : inputids)
518 if (is_busy && !sourceid)
528 ((mplexid == 0 || mplexid == 32767) &&
532 inputids2.push_back(inputid);
537 for (
uint i = 0; (i < inputids2.size()) && ok; i++)
539 LOG(VB_RECORD, LOG_INFO,
LOC +
540 QString(
"Attempting to stop input [%1] in state %2")
547 LOG(VB_GENERAL, LOG_INFO,
LOC + QString(
"a [%1]: %2")
555 QString message = QString(
"QUIT_LIVETV %1").arg(inputids2[i]);
560 LOG(VB_RECORD, LOG_INFO,
LOC +
561 QString(
"Stopping recording on [%1], %2") .arg(inputids2[i])
562 .arg(success ?
"succeeded" :
"failed"));
576 LOG(VB_RECORD, LOG_INFO,
LOC +
"Checking input group recorders - done");
579 bool did_switch =
false;
583 did_switch = (
nullptr != ri2);
604 QString message = QString(
"LIVETV_EXITED");
626 LOG(VB_RECORD, LOG_WARNING,
LOC +
"Still failing.");
638 QString message = QString(
"LIVETV_WATCH %1 1").arg(
m_inputId);
644 else if (!did_switch)
646 QString msg = QString(
"Wanted to record: %1 %2 %3 %4\n\t\t\t")
654 msg +=
"But a user has canceled this recording";
659 msg += QString(
"But the current state is: %1")
666 msg += QString(
"\n\t\t\tCurrently recording: %1 %2 %3 %4")
673 LOG(VB_GENERAL, LOG_INFO,
LOC + msg);
722 LOG(VB_RECORD, LOG_INFO,
LOC +
723 QString(
"SetRecordingStatus(%1->%2) on line %3")
726 QString::number(line)));
745 if (now < m_curRecording->GetDesiredEndTime())
787 LOG(VB_GENERAL, LOG_ERR,
LOC +
788 QString(
"Unknown state in RemoveRecording: %1")
807 QString msg =
"Unknown state in RemovePlaying: %1";
824 LOG(VB_RECORD, LOG_INFO,
LOC + QString(
"StartedRecording(%1) fn(%2)")
856 LOG((recq->
IsDamaged()) ? VB_GENERAL : VB_RECORD, LOG_INFO,
857 LOC + QString(
"FinishedRecording(%1) %2 recq:\n%3")
859 (recq->
IsDamaged()) ?
"damaged" :
"good",
876 bool was_finished =
false;
877 static QMutex s_finRecLock;
878 static QHash<QString,QDateTime> s_finRecMap;
880 QMutexLocker locker(&s_finRecLock);
882 QDateTime expired = now.addSecs(-5LL * 60);
883 QHash<QString,QDateTime>::iterator it = s_finRecMap.begin();
884 while (it != s_finRecMap.end())
887 it = s_finRecMap.erase(it);
892 it = s_finRecMap.find(key);
893 if (it != s_finRecMap.end())
896 s_finRecMap[key] = now;
900 LOG(VB_RECORD, LOG_INFO,
LOC +
901 QString(
"FinishedRecording(%1) %2 quality"
902 "\n\t\t\ttitle: %3\n\t\t\t"
903 "in recgroup: %4 status: %5:%6 %7 %8")
905 is_good ?
"Good" :
"Bad",
911 was_finished?
"already_finished":
"finished_now"));
934 (VID_4K | VID_1080 | VID_720 | VID_DAMAGED |
935 VID_WIDESCREEN | VID_PROGRESSIVE,
936 ((avg_height > 2000) ? VID_4K :
937 ((avg_height > 1000) ? VID_1080 :
938 ((avg_height > 700) ? VID_720 : 0))) |
939 (progressive ? VID_PROGRESSIVE : 0) |
940 ((is_good) ? 0 : VID_DAMAGED) |
958 if (curRec->
IsLocal() && (fsize >= 1000) &&
968 if (recgrp !=
"LiveTV")
970 LOG(VB_RECORD, LOG_INFO,
LOC +
971 QString(
"FinishedRecording -- UPDATE_RECORDING_STATUS: %1")
974 MythEvent me(QString(
"UPDATE_RECORDING_STATUS %1 %2 %3 %4 %5")
988 QString message = QString(
"DONE_RECORDING %1 %2 %3")
994 QHash<QString,int>::iterator autoJob =
998 LOG(VB_GENERAL, LOG_INFO,
999 "autoRunJobs not initialized until FinishedRecording()");
1005 LOG(VB_JOBQUEUE, LOG_INFO, QString(
"AutoRunJobs 0x%1").arg(*autoJob,0,16));
1006 if ((recgrp ==
"LiveTV") || (fsize < 1000) ||
1020 #define TRANSITION(ASTATE,BSTATE) \
1021 ((m_internalState == (ASTATE)) && (m_desiredNextState == (BSTATE)))
1022 #define SET_NEXT() do { nextState = m_desiredNextState; changed = true; } while(false)
1023 #define SET_LAST() do { nextState = m_internalState; changed = true; } while(false)
1037 bool changed =
false;
1039 QString transMsg = QString(
" %1 to %2")
1044 LOG(VB_GENERAL, LOG_ERR,
LOC +
1045 "HandleStateChange(): Null transition" + transMsg);
1054 LOG(VB_EIT, LOG_INFO,
LOC + QString(
"Stop EIT scan on input %1").arg(
GetInputId()));
1093 QString msg = (changed) ?
"Changing from" :
"Unknown state transition:";
1094 LOG(VB_GENERAL, LOG_INFO,
LOC + msg + transMsg);
1142 LOG(VB_RECORD, LOG_INFO,
LOC + QString(
"TeardownRecorder(%1)")
1143 .arg((request_flags &
kFlagKillRec) ?
"kFlagKillRec" :
""));
1156 __FILE__, __LINE__);
1173 LOG(VB_FILE, LOG_INFO,
LOC +
"calling StopReads()");
1225 #endif // USING_V4L2
1232 "SELECT SUM(useeit) "
1233 "FROM videosource, capturecard "
1234 "WHERE videosource.sourceid = capturecard.sourceid AND"
1235 " capturecard.cardid = :INPUTID");
1244 return query.
value(0).toBool();
1252 "SELECT SUM(dishnet_eit) "
1253 "FROM videosource, capturecard "
1254 "WHERE videosource.sourceid = capturecard.sourceid AND"
1255 " capturecard.cardid = :INPUTID");
1264 return query.
value(0).toBool();
1274 "SELECT MAX(cardid) "
1275 "FROM capturecard ";
1285 return query.
value(0).toInt();
1292 auto timeout = std::chrono::seconds(
MythRandom(0, eitTransportTimeout.count() / 3));
1297 if (highest_input > 0)
1298 timeout += eitTransportTimeout * inputId / highest_input;
1333 __FILE__, __LINE__);
1339 LOG(VB_GENERAL, LOG_ERR,
LOC +
1340 "RunTV encountered fatal error, exiting event thread.");
1389 QString message = QString(
"QUIT_LIVETV %1").arg(
m_inputId);
1405 bool enable_ui =
true;
1421 LOG(VB_RECORD, LOG_INFO,
LOC +
1422 "Switching Buffer (" +
1423 QString(
"!has_rec(%1) && ").arg(has_rec) +
1424 QString(
"!rec_soon(%1) && (").arg(rec_soon) +
1437 LOG(VB_RECORD, LOG_INFO,
"Waiting for ringbuffer switch");
1445 LOG(VB_RECORD, LOG_INFO,
LOC +
"Enabling Full LiveTV UI.");
1446 QString message = QString(
"LIVETV_WATCH %1 0").arg(
m_inputId);
1469 LOG(VB_EIT, LOG_INFO,
LOC +
1470 "EIT scanning disabled for this input.");
1475 LOG(VB_EIT, LOG_INFO,
LOC +
1476 "EIT scanning disabled for all channels on this input.");
1485 bool allow_eit =
true;
1486 std::vector<unsigned int> inputids =
1489 for (
uint i = 0; i < inputids.size() && allow_eit; ++i)
1496 QDateTime::currentDateTime().addYears(1);
1500 LOG(VB_EIT, LOG_INFO,
LOC + QString(
1501 "Postponing EIT scan on input [%1] "
1502 "because input %2 is busy")
1563 while (!ok && (
t.elapsed() < time))
1587 std::chrono::milliseconds te = t2.
elapsed();
1588 if (!ok && te < 10ms)
1589 std::this_thread::sleep_for(10ms - te);
1602 LOG(VB_RECORD, LOG_INFO,
LOC +
"Deleting stale pending recording " +
1603 QString(
"[%1] '%2'")
1604 .arg((*it).m_info->GetInputID())
1605 .arg((*it).m_info->GetTitle()));
1607 delete (*it).m_info;
1622 LOG(VB_CHANNEL, LOG_INFO,
1623 LOC +
"Stopping active EIT scan for pending recording.");
1631 bool has_rec =
false;
1635 ((*it).m_info->GetInputID() ==
m_inputId) &&
1641 (*it).m_recordingStart);
1646 if (!(*it).m_ask && !(*it).m_doNotAsk)
1649 auto timeuntil = ((*it).m_doNotAsk) ?
1653 (*it).m_canceled =
true;
1655 QString query = QString(
"ASK_RECORDING %1 %2 %3 %4")
1657 .arg(timeuntil.count())
1658 .arg(has_rec ? 1 : 0)
1659 .arg((*it).m_hasLaterShowing ? 1 : 0);
1661 LOG(VB_GENERAL, LOG_INFO,
LOC + query);
1664 (*it).m_info->ToStringList(msg);
1668 (*it).m_ask = (*it).m_doNotAsk =
false;
1683 "SELECT videodevice, vbidevice, audiodevice, "
1684 " audioratelimit, cardtype, "
1685 " skipbtaudio, signal_timeout, channel_timeout, "
1686 " dvb_wait_for_seqstart, "
1688 " dvb_on_demand, dvb_tuning_delay, dvb_eitscan,"
1690 " firewire_speed, firewire_model, firewire_connection, "
1694 "WHERE cardid = :INPUTID");
1707 test = query.
value(0).toString();
1708 if (!test.isEmpty())
1711 test = query.
value(1).toString();
1712 if (!test.isEmpty())
1715 test = query.
value(2).toString();
1716 if (!test.isEmpty())
1721 test = query.
value(4).toString();
1722 if (!test.isEmpty())
1733 if (table_timeout < 1000)
1745 uint fireoff = dvboff + 3;
1746 firewire_opts.
m_speed = query.
value(fireoff + 0).toUInt();
1748 test = query.
value(fireoff + 1).toString();
1749 if (!test.isEmpty())
1754 parentid = query.
value(15).toUInt();
1771 pid_cache.push_back(item);
1780 bool vctpid_cached =
false;
1781 for (
const auto& pid : pid_cache)
1786 vctpid_cached =
true;
1790 return vctpid_cached;
1810 LOG(VB_RECORD, LOG_INFO,
LOC +
"Setting up table monitoring.");
1814 if (!sm || !dtvchan)
1816 LOG(VB_GENERAL, LOG_ERR,
LOC +
"Setting up table monitoring.");
1827 QString recording_type =
"all";
1833 recording_type = setting->
getValue();
1840 if ((
minor > 0) && (tuningmode ==
"atsc"))
1842 QString msg = QString(
"ATSC channel: %1_%2").arg(major).arg(
minor);
1843 LOG(VB_RECORD, LOG_INFO,
LOC + msg);
1864 LOG(VB_RECORD, LOG_INFO,
LOC +
1865 "Successfully set up ATSC table monitoring.");
1885 LOG(VB_RECORD, LOG_INFO,
LOC +
1886 QString(
"DVB service_id %1 on net_id %2 tsid %3")
1887 .arg(progNum).arg(netid).arg(tsid));
1907 LOG(VB_RECORD, LOG_INFO,
LOC +
1908 "Successfully set up DVB table monitoring.");
1923 QString msg = QString(
"MPEG program number: %1").arg(progNum);
1924 LOG(VB_RECORD, LOG_INFO,
LOC + msg);
1944 LOG(VB_RECORD, LOG_INFO,
LOC +
1945 "Successfully set up MPEG table monitoring.");
1956 for (
auto item = pid_cache.cbegin(); !ok && item != pid_cache.cend(); ++item)
1957 ok |= item->IsPermanent();
1962 QString msg =
"No valid DTV info, ATSC maj(%1) min(%2), MPEG pn(%3)";
1963 LOG(VB_GENERAL, LOG_ERR,
LOC + msg.arg(major).arg(
minor).arg(progNum));
1967 LOG(VB_RECORD, LOG_INFO,
LOC +
1968 "Successfully set up raw pid monitoring.");
1990 LOG(VB_RECORD, LOG_INFO,
LOC + QString(
"SetupSignalMonitor(%1, %2)")
1991 .arg(tablemon).arg(notify));
2014 LOG(VB_RECORD, LOG_INFO,
LOC +
"Signal monitor successfully created");
2019 LOG(VB_GENERAL, LOG_ERR,
LOC +
2020 "Failed to setup digital signal monitoring");
2047 LOG(VB_RECORD, LOG_INFO,
LOC +
"TeardownSignalMonitor() -- begin");
2052 if (dtvMon && dtvChan)
2056 if (!pid_cache.empty())
2066 LOG(VB_RECORD, LOG_INFO,
LOC +
"TeardownSignalMonitor() -- end");
2082 QString msg =
"SetSignalMonitoringRate(%1, %2)";
2083 LOG(VB_RECORD, LOG_INFO,
LOC +
2084 msg.arg(rate.count()).arg(notifyFrontend) +
"-- start");
2090 LOG(VB_GENERAL, LOG_ERR,
LOC +
2091 "Signal Monitoring is notsupported by your hardware.");
2097 LOG(VB_GENERAL, LOG_ERR,
LOC +
2098 "Signal can only be monitored in LiveTV Mode.");
2113 LOG(VB_RECORD, LOG_INFO,
LOC +
2114 msg.arg(rate.count()).arg(notifyFrontend) +
" -- end");
2142 query.
prepare(
"SELECT channel.channum, channel.callsign "
2144 "WHERE channel.chanid = :CHANID");
2146 if (!query.
exec() || !query.
next())
2152 QString channelname = query.
value(0).toString();
2153 QString callsign = query.
value(1).toString();
2156 "SELECT channel.channum "
2157 "FROM channel, capturecard "
2158 "WHERE deleted IS NULL AND "
2159 " ( channel.chanid = :CHANID OR "
2160 " ( channel.channum = :CHANNUM AND "
2161 " channel.callsign = :CALLSIGN ) "
2163 " channel.sourceid = capturecard.sourceid AND "
2164 " capturecard.cardid = :INPUTID");
2166 query.
bindValue(
":CHANNUM", channelname);
2174 else if (query.
size() > 0)
2176 msg =
"Found channel (%1) on current input[%2].";
2183 "SELECT channel.channum, capturecard.cardid "
2184 "FROM channel, capturecard "
2185 "WHERE deleted IS NULL AND "
2186 " ( channel.chanid = :CHANID OR "
2187 " ( channel.channum = :CHANNUM AND "
2188 " channel.callsign = :CALLSIGN ) "
2190 " channel.sourceid = capturecard.sourceid AND "
2191 " capturecard.cardid != :INPUTID");
2193 query.
bindValue(
":CHANNUM", channelname);
2201 else if (query.
next())
2203 msg = QString(
"Found channel (%1) on different input(%2).")
2204 .arg(query.
value(0).toString(), query.
value(1).toString());
2205 LOG(VB_RECORD, LOG_INFO,
LOC + msg);
2209 msg = QString(
"Did not find channel(%1) on any input.").arg(channelname);
2210 LOG(VB_RECORD, LOG_ERR,
LOC + msg);
2235 static QString
add_spacer(
const QString &channel,
const QString &spacer)
2237 QString chan = channel;
2238 if ((chan.length() >= 2) && !spacer.isEmpty())
2239 return chan.left(chan.length()-1) + spacer + chan.right(1);
2271 uint &complete_valid_channel_on_rec,
2272 bool &is_extra_char_useful,
2273 QString &needed_spacer)
const
2275 #if DEBUG_CHANNEL_PREFIX
2276 LOG(VB_GENERAL, LOG_DEBUG, QString(
"CheckChannelPrefix(%1)").arg(
prefix));
2279 static const std::array<const QString,5> s_spacers = {
"",
"_",
"-",
"#",
"." };
2282 QString basequery = QString(
2283 "SELECT channel.chanid, channel.channum, capturecard.cardid "
2284 "FROM channel, capturecard "
2285 "WHERE deleted IS NULL AND "
2286 " channel.channum LIKE '%1%' AND "
2287 " channel.sourceid = capturecard.sourceid");
2289 const std::array<const QString,2> inputquery
2291 QString(
" AND capturecard.cardid = '%1'").arg(
m_inputId),
2292 QString(
" AND capturecard.cardid != '%1'").arg(
m_inputId),
2295 std::vector<unsigned int> fchanid;
2296 std::vector<QString> fchannum;
2297 std::vector<unsigned int> finputid;
2298 std::vector<QString> fspacer;
2300 for (
const auto & str : inputquery)
2302 for (
const auto & spacer : s_spacers)
2305 prefix, (spacer ==
"_") ?
"\\_" : spacer);
2306 query.
prepare(basequery.arg(qprefix) + str);
2312 else if (query.
size())
2314 while (query.
next())
2316 fchanid.push_back(query.
value(0).toUInt());
2317 fchannum.push_back(query.
value(1).toString());
2318 finputid.push_back(query.
value(2).toUInt());
2319 fspacer.emplace_back(spacer);
2320 #if DEBUG_CHANNEL_PREFIX
2321 LOG(VB_GENERAL, LOG_DEBUG,
2322 QString(
"(%1,%2) Adding %3 rec %4")
2323 .arg(i).arg(j).arg(query.
value(1).toString(),6)
2324 .arg(query.
value(2).toUInt()));
2335 is_extra_char_useful =
false;
2336 complete_valid_channel_on_rec = 0;
2337 needed_spacer.clear();
2339 if (fchanid.empty())
2342 if (fchanid.size() == 1)
2344 needed_spacer = fspacer[0];
2347 complete_valid_channel_on_rec = (nc) ? 0 : finputid[0];
2348 is_extra_char_useful = nc;
2356 is_extra_char_useful =
false;
2357 for (
uint i = 0; (i < fchannum.size()) && !is_extra_char_useful; i++)
2360 #if DEBUG_CHANNEL_PREFIX
2361 LOG(VB_GENERAL, LOG_DEBUG, QString(
"is_extra_char_useful(%1!=%2): %3")
2363 .arg(is_extra_char_useful));
2370 for (
size_t i = 0; i < fchannum.size(); i++)
2372 if (fchannum[i] ==
prefix)
2374 complete_valid_channel_on_rec = finputid[i];
2380 if (complete_valid_channel_on_rec != 0)
2384 bool spacer_needed =
true;
2385 for (
uint i = 0; (i < fspacer.size() && spacer_needed); i++)
2386 spacer_needed = !fspacer[i].isEmpty();
2388 needed_spacer = fspacer[0];
2392 for (
size_t i = 0; i < ((is_extra_char_useful) ? 0 : fchanid.size()); i++)
2396 needed_spacer = fspacer[i];
2397 complete_valid_channel_on_rec = finputid[i];
2406 const QString &channum)
2412 if (!videoFilters.isEmpty())
2440 busy_input = &dummy;
2442 busy_input->
Clear();
2459 bool has_pending =
false;
2469 if (!busy_input->
m_inputId && has_pending)
2473 if (timeLeft <= time_buffer)
2596 long long bitrate = 0;
2599 bitrate = 10080000LL;
2603 bitrate = 20200000LL;
2607 bitrate = 22200000LL;
2611 bitrate = 10080000LL;
2739 MythEvent me(
"SCHEDULER_ADD_RECORDING", prog);
2766 LOG(VB_JOBQUEUE, LOG_INFO,
2767 QString(
"InitAutoRunJobs for %1, line %2 -> 0x%3")
2785 LOG(VB_GENERAL, LOG_INFO,
LOC +
2786 QString(
"SetLiveRecording(%1)").arg(recording));
2796 LOG(VB_GENERAL, LOG_INFO,
LOC +
"SetLiveRecording() -- cancel");
2804 LOG(VB_GENERAL, LOG_INFO,
LOC +
"SetLiveRecording() -- record");
2817 MythEvent me(QString(
"UPDATE_RECORDING_STATUS %1 %2 %3 %4 %5")
2834 LOG(VB_RECORD, LOG_INFO,
LOC +
2835 QString(
"StopLiveTV(void) curRec: 0x%1 pseudoRec: 0x%2")
2884 LOG(VB_GENERAL, LOG_ERR,
LOC +
2885 "PauseRecorder() called with no recorder");
2920 LOG(VB_GENERAL, LOG_ERR,
LOC +
2921 QString(
"Channel: \'%1\' was not found in the database.\n"
2922 "\t\tMost likely, the 'starting channel' for this "
2923 "Input Connection is invalid.\n"
2924 "\t\tCould not toggle favorite.").arg(channum));
2931 LOG(VB_RECORD, LOG_ERR,
LOC +
2932 QString(
"ToggleChannelFavorite: Invalid channel group name %1,")
2933 .arg(changroupname));
2940 LOG(VB_RECORD, LOG_ERR,
LOC +
"Unable to toggle channel favorite.");
2943 LOG(VB_RECORD, LOG_INFO,
LOC +
2944 QString(
"Toggled channel favorite.channum %1, chan group %2")
2945 .arg(channum, changroupname));
2963 return (ret < 0) ? -1 : ret / 655;
2983 return (ret < 0) ? -1 : ret / 655;
3017 QString origIn = input;
3018 LOG(VB_RECORD, LOG_INFO,
LOC +
"SetInput(" + input +
") -- begin");
3022 LOG(VB_RECORD, LOG_INFO,
LOC +
"SetInput() -- end no channel class");
3026 LOG(VB_RECORD, LOG_INFO,
LOC +
"SetInput(" + origIn +
":" + input +
3027 ") -- end nothing to do");
3045 LOG(VB_CHANNEL, LOG_INFO,
LOC +
3046 QString(
"SetChannel(%1) -- begin").arg(name));
3079 LOG(VB_CHANNEL, LOG_INFO,
LOC + QString(
"SetChannel(%1) -- end").arg(name));
3091 LOG(VB_CHANNEL, LOG_INFO,
LOC +
3092 QString(
"QueueEITChannelChange(%1) -- begin").arg(name));
3109 LOG(VB_CHANNEL, LOG_INFO,
LOC +
3110 QString(
"QueueEITChannelChange(%1) -- end --> %2").arg(name).arg(ok));
3116 QString &title, QString &subtitle,
3117 QString &desc, QString &category,
3118 QString &starttime, QString &endtime,
3119 QString &callsign, QString &iconpath,
3120 QString &channum,
uint &sourceChanid,
3121 QString &seriesid, QString &programid)
3123 QString compare =
"<=";
3124 QString sortorder =
"desc";
3129 chanid = sourceChanid;
3177 QString querystr = QString(
3178 "SELECT title, subtitle, description, category, "
3179 " starttime, endtime, callsign, icon, "
3180 " channum, seriesid, programid "
3181 "FROM program, channel "
3182 "WHERE program.chanid = channel.chanid AND "
3183 " channel.chanid = :CHANID AND "
3184 " starttime %1 :STARTTIME "
3185 "ORDER BY starttime %2 "
3186 "LIMIT 1").arg(compare, sortorder);
3191 query.
bindValue(
":STARTTIME", starttime);
3194 title = subtitle = desc = category =
"";
3195 starttime = endtime = callsign = iconpath =
"";
3196 channum = seriesid = programid =
"";
3204 else if (query.
next())
3206 title = query.
value(0).toString();
3207 subtitle = query.
value(1).toString();
3208 desc = query.
value(2).toString();
3209 category = query.
value(3).toString();
3210 starttime = query.
value(4).toString();
3211 endtime = query.
value(5).toString();
3212 callsign = query.
value(6).toString();
3213 iconpath = query.
value(7).toString();
3214 channum = query.
value(8).toString();
3215 seriesid = query.
value(9).toString();
3216 programid = query.
value(10).toString();
3217 sourceChanid = chanid;
3223 "SELECT channum, callsign, icon "
3225 "WHERE chanid = :CHANID");
3232 else if (query.
next())
3234 sourceChanid = chanid;
3235 channum = query.
value(0).toString();
3236 callsign = query.
value(1).toString();
3237 iconpath = query.
value(2).toString();
3242 QString &callsign, QString &channum,
3243 QString &channame, QString &xmltvid)
const
3250 if ((!chanid || !sourceid) && !
m_channel)
3261 "SELECT callsign, channum, name, xmltvid "
3263 "WHERE chanid = :CHANID");
3274 callsign = query.
value(0).toString();
3275 channum = query.
value(1).toString();
3276 channame = query.
value(2).toString();
3277 xmltvid = query.
value(3).toString();
3283 const QString& oldchannum,
3284 const QString& callsign,
const QString& channum,
3285 const QString& channame,
const QString& xmltvid)
3287 if (!chanid || !sourceid || channum.isEmpty())
3293 "SET callsign = :CALLSIGN, "
3294 " channum = :CHANNUM, "
3295 " name = :CHANNAME, "
3296 " xmltvid = :XMLTVID "
3297 "WHERE chanid = :CHANID AND "
3298 " sourceid = :SOURCEID");
3328 if (oldbuffer && (oldbuffer !=
Buffer))
3340 LOG(VB_GENERAL, LOG_INFO,
LOC +
"RingBufferChanged()");
3360 QString &input)
const
3387 if (
m_channel && !channum.isEmpty() && (channum.indexOf(
"NextChannel") >= 0))
3390 int dir = channum.right(channum.length() - 12).toInt();
3432 if (
mpeg->HasProgram(progNum))
3455 LOG(VB_RECORD, LOG_INFO,
LOC +
3456 "HandleTuning Request: " + request.
toString());
3463 LOG(VB_CHANNEL, LOG_INFO,
LOC +
"On same multiplex");
3477 LOG(VB_RECORD, LOG_INFO,
LOC +
3478 "No recorder yet, calling TuningFrequency");
3483 LOG(VB_RECORD, LOG_INFO,
LOC +
"Waiting for recorder pause..");
3496 LOG(VB_RECORD, LOG_INFO,
LOC +
3497 "Recorder paused, calling TuningFrequency");
3524 LOG(VB_RECORD, LOG_INFO,
LOC + QString(
"TuningShutdowns(%1)")
3551 if (sd && (sd != rec_sd))
3585 LOG(VB_RECORD, LOG_INFO,
LOC +
"Tearing down RingBuffer");
3613 LOG(VB_GENERAL, LOG_INFO,
LOC +
"TuningFrequency");
3633 if (request.
m_minorChan && (tuningmode ==
"atsc"))
3658 slist<<
"message"<<QObject::tr(
"On known multiplex...");
3672 if (!channum.isEmpty())
3685 LOG(VB_GENERAL, LOG_ERR,
LOC +
3686 QString(
"Failed to set channel to %1. Reverting to kState_None")
3695 LOG(VB_GENERAL, LOG_ERR,
LOC +
3696 QString(
"Failed to set channel to %1.").arg(channum));
3716 bool use_dr = use_sm && (livetv || antadj);
3717 bool has_dummy =
false;
3738 LOG(VB_GENERAL, LOG_ERR,
LOC +
"Failed to create RingBuffer 1");
3748 LOG(VB_RECORD, LOG_INFO,
LOC +
"Starting Signal Monitor");
3753 LOG(VB_GENERAL, LOG_ERR,
LOC +
"Failed to setup signal monitor");
3771 SetVideoStreamsRequired(0);
3798 LOG(VB_CHANNEL, LOG_DEBUG,
LOC +
3799 QString(
"Pre-fail start deadline: %1 "
3800 "Start recording deadline: %2 "
3801 "Good signal deadline: %3")
3803 .toString(
"hh:mm:ss.zzz"),
3805 .toString(
"hh:mm:ss.zzz"),
3807 .toString(
"hh:mm:ss.zzz")));
3831 LOG(VB_RECORD, LOG_INFO,
"DummyDTVRecorder -- started");
3857 bool keep_trying =
false;
3870 LOG(VB_RECORD, LOG_INFO,
LOC +
"TuningSignalCheck: Good signal");
3876 QString desc = tr(
"Good signal seen after %1 ms")
3885 tr(
"See 'Tuning timeout' in mythtv-setup "
3886 "for this input."));
3889 LOG(VB_GENERAL, LOG_WARNING,
LOC +
3890 QString(
"It took longer than %1 ms to get a signal lock. "
3891 "Keeping status of '%2'")
3894 LOG(VB_GENERAL, LOG_WARNING,
LOC +
3895 "See 'Tuning timeout' in mythtv-setup for this input");
3904 LOG(VB_GENERAL, LOG_ERR,
LOC +
"TuningSignalCheck: SignalMonitor " +
3917 LOG(VB_GENERAL, LOG_ERR,
LOC +
3918 "TuningSignalCheck: Hit pre-fail timeout");
3932 QString desc = tr(
"Taking more than %1 ms to get a lock.")
3940 tr(
"See 'Tuning timeout' in mythtv-setup "
3941 "for this input."));
3945 LOG(VB_GENERAL, LOG_WARNING,
LOC +
3946 QString(
"TuningSignalCheck: taking more than %1 ms to get a lock. "
3947 "marking this recording as '%2'.")
3950 LOG(VB_GENERAL, LOG_WARNING,
LOC +
3951 "See 'Tuning timeout' in mythtv-setup for this input");
3959 LOG(VB_RECORD, LOG_INFO,
LOC +
3960 QString(
"TuningSignalCheck: Still waiting. Will timeout @ %1")
3962 .toString(
"hh:mm:ss.zzz")));
3973 MythEvent me(QString(
"UPDATE_RECORDING_STATUS %1 %2 %3 %4 %5")
4005 LOG(VB_EIT, LOG_INFO,
LOC +
4006 "EIT scanning disabled for all sources on this input.");
4016 bool on_host,
bool transcode_bfr_comm,
bool on_line_comm)
4032 if ((!autoTrans) || (autoTrans->
getValue().toInt() == 0))
4056 !transcode_bfr_comm;
4080 QString profileName =
"Live TV";
4081 if (!tvchain && rec)
4084 QString profileRequested = profileName;
4089 LOG(VB_RECORD, LOG_INFO,
LOC +
4090 QString(
"Using profile '%1' to record")
4095 profileName =
"Default";
4098 LOG(VB_RECORD, LOG_INFO,
LOC +
4099 QString(
"Profile '%1' not found, using "
4100 "fallback profile '%2' to record")
4101 .arg(profileRequested, profileName));
4105 LOG(VB_RECORD, LOG_ERR,
LOC +
4106 QString(
"Profile '%1' not found, and unable "
4107 "to load fallback profile '%2'. Results "
4108 "may be unpredicable")
4109 .arg(profileRequested, profileName));
4121 LOG(VB_RECORD, LOG_INFO,
LOC +
"Starting Recorder");
4123 bool had_dummyrec =
false;
4129 had_dummyrec =
true;
4150 LOG(VB_GENERAL, LOG_ERR,
LOC +
"Failed to create RingBuffer 2");
4159 LOG(VB_GENERAL, LOG_INFO,
LOC + QString(
"rec->GetPathname(): '%1'")
4164 LOG(VB_GENERAL, LOG_ERR,
LOC +
4165 QString(
"RingBuffer '%1' not open...")
4175 LOG(VB_GENERAL, LOG_ERR,
LOC +
4176 QString(
"Failed to start recorder! ringBuffer is NULL\n"
4177 "\t\t\t\t Tuning request was %1\n")
4182 QString message = QString(
"QUIT_LIVETV %1").arg(
m_inputId);
4192 LOG(VB_GENERAL, LOG_INFO,
LOC +
"TuningNewRecorder - CreateRecorder()");
4201 LOG(VB_GENERAL, LOG_ERR,
LOC +
"Failed to initialize recorder!");
4209 LOG(VB_GENERAL, LOG_ERR,
LOC +
4210 QString(
"Failed to start recorder!\n"
4211 "\t\t\t\t Tuning request was %1\n")
4216 QString message = QString(
"QUIT_LIVETV %1").arg(
m_inputId);
4259 std::this_thread::sleep_for(5us);
4288 LOG(VB_RECORD, LOG_INFO,
LOC +
4289 QString(
"TuningNewRecorder -- UPDATE_RECORDING_STATUS: %1")
4291 MythEvent me(QString(
"UPDATE_RECORDING_STATUS %1 %2 %3 %4 %5")
4309 LOG(VB_RECORD, LOG_INFO,
LOC +
"Restarting Recorder");
4311 bool had_dummyrec =
false;
4322 had_dummyrec =
true;
4348 QString msg1 = QString(
"Recording: %1 %2 %3 %4")
4353 QString msg2 = QString(
"Recording: %1 %2 %3 %4")
4358 LOG(VB_RECORD, LOG_INFO,
LOC +
"Pseudo LiveTV recording starting." +
4359 "\n\t\t\t" + msg1 +
"\n\t\t\t" + msg2);
4376 LOG(VB_RECORD, LOG_INFO,
LOC + QString(
"SetFlags(%1) -> %2 @ %3:%4")
4385 LOG(VB_RECORD, LOG_INFO,
LOC + QString(
"ClearFlags(%1) -> %2 @ %3:%4")
4396 msg +=
"FrontendReady,";
4398 msg +=
"RunMainLoop,";
4400 msg +=
"ExitPlayer,";
4402 msg +=
"FinishRecording,";
4406 msg +=
"CancelNextRecording,";
4416 msg +=
"Recording,";
4429 msg +=
"AntennaAdjust,";
4432 msg +=
"PENDINGACTIONS,";
4436 msg +=
"WaitingForRecPause,";
4438 msg +=
"WaitingForSignal,";
4440 msg +=
"NeedToStartRecorder,";
4442 msg +=
"KillRingBuffer,";
4445 msg +=
"ANYRUNNING,";
4449 msg +=
"SignalMonitorRunning,";
4451 msg +=
"EITScannerRunning,";
4453 msg +=
"ANYRECRUNNING,";
4457 msg +=
"DummyRecorderRunning,";
4459 msg +=
"RecorderRunning,";
4463 msg +=
"RingBufferReady,";
4466 msg = QString(
"0x%1").arg(f,0,16);
4494 const QString & channum)
4496 LOG(VB_RECORD, LOG_INFO,
LOC +
"GetProgramRingBufferForLiveTV()");
4518 LOG(VB_GENERAL, LOG_ERR,
LOC +
4519 QString(
"Channel: \'%1\' was not found in the database.\n"
4520 "\t\tMost likely, the 'starting channel' for this "
4521 "Input Connection is invalid.\n"
4522 "\t\tCould not start livetv.").arg(channum));
4545 LOG(VB_GENERAL, LOG_ERR,
LOC +
"GetProgramRingBufferForLiveTV()"
4546 "\n\t\t\tProgramInfo is invalid."
4576 if (!(*
Buffer) || !(*Buffer)->IsOpen())
4578 LOG(VB_GENERAL, LOG_ERR,
LOC + QString(
"RingBuffer '%1' not open...")
4593 LOG(VB_RECORD, LOG_INFO,
LOC + QString(
"CreateLiveTVRingBuffer(%1)")
4610 LOG(VB_GENERAL, LOG_ERR,
LOC +
4611 QString(
"CreateLiveTVRingBuffer(%1) failed").arg(channum));
4638 bool discont,
bool set_rec)
4643 msg = QString(
" curRec(%1) curRec.size(%2)")
4647 LOG(VB_RECORD, LOG_INFO,
LOC +
4648 QString(
"SwitchLiveTVRingBuffer(discont %1, set_next_rec %2)")
4649 .arg(discont).arg(set_rec) + msg);
4707 LOG(VB_RECORD, LOG_INFO,
LOC +
"SwitchRecordingRingBuffer()");
4711 LOG(VB_RECORD, LOG_ERR,
LOC +
"SwitchRecordingRingBuffer -> "
4712 "already switching.");
4718 LOG(VB_RECORD, LOG_ERR,
LOC +
"SwitchRecordingRingBuffer -> "
4719 "invalid recorder.");
4725 LOG(VB_RECORD, LOG_ERR,
LOC +
"SwitchRecordingRingBuffer -> "
4726 "invalid recording.");
4732 LOG(VB_RECORD, LOG_ERR,
LOC +
"SwitchRecordingRingBuffer -> "
4733 "Not the same channel.");
4744 LOG(VB_RECORD, LOG_ERR,
LOC +
4745 QString(
"SwitchRecordingRingBuffer() -> "
4746 "cannot switch profile '%1' to '%2'")
4758 if (!buffer || !buffer->
IsOpen())
4765 LOG(VB_RECORD, LOG_ERR,
LOC +
"SwitchRecordingRingBuffer() -> "
4766 "Failed to create new RB.");
4775 LOG(VB_RECORD, LOG_INFO,
LOC +
"SwitchRecordingRingBuffer -> done");
4781 QMap<uint,TVRec*>::const_iterator it =
s_inputs.constFind(inputid);
4789 return QString(
"Program(%1) channel(%2) input(%3) flags(%4)")
bool CreateLiveTVRingBuffer(const QString &channum)
uint GetMajorChannel(void) const
Returns major channel, 0 if unknown.
bool isActive(void) const
@ BROWSE_LEFT
Fetch information on current channel in the past.
bool next(void)
Wrap QSqlQuery::next() so we can display the query results.
QSqlQuery wrapper that fetches a DB connection from the connection pool.
virtual void Close(void)=0
Closes the channel changing hardware to use.
void SetFlags(uint f, const QString &file, int line)
bool GetKeyframeDurations(int64_t start, int64_t end, frm_pos_map_t &map) const
std::chrono::milliseconds elapsed(void)
Returns milliseconds elapsed since last start() or restart()
void AddFlags(uint64_t _flags) override
bool IsReallyRecording(void)
Returns true if frontend can consider the recorder started.
QString MakeUniqueKey(void) const
Creates a unique string that can be used to identify an existing recording.
std::vector< uint > m_possibleConflicts
virtual bool Init(QString &startchannel, bool setchan)
QString toString(const QDateTime &raw_dt, uint format)
Returns formatted string representing the time.
static const uint kFlagNoRec
void RecordPending(const ProgramInfo *rcinfo, std::chrono::seconds secsleft, bool hasLater)
Tells TVRec "rcinfo" is the next pending recording.
static const uint kFlagErrored
void start(QThread::Priority p=QThread::InheritPriority)
Tell MThread to start running the thread in the near future.
QDateTime m_startRecordingDeadline
static void RemoveJobsFromMask(int jobs, int &mask)
static const uint kFlagEITScannerRunning
void FinishedRecording(bool allowReRecord)
If not a premature stop, adds program to history of recorded programs.
@ BROWSE_UP
Fetch information on previous channel.
static bool JobIsInMask(int job, int mask)
void SetPseudoLiveTVRecording(RecordingInfo *pi)
Sets the pseudo LiveTV RecordingInfo.
void SetVideoStreamsRequired(uint num)
void SetDVBService(uint network_id, uint transport_id, int service_id)
void StopEITEventProcessing(void)
Stops inserting Event Information Tables into DB.
bool WaitForEventThreadSleep(bool wake=true, std::chrono::milliseconds time=std::chrono::milliseconds::max())
static void error(const char *str,...)
virtual int DecrRef(void)
Decrements reference count and deletes on 0.
void SetRecordingStatus(RecStatus::Type status)
uint QueryMplexID(void) const
Queries multiplex any recording would be made on, zero if unknown.
ChannelChangeDirection
ChannelChangeDirection is an enumeration of possible channel changing directions.
static RecorderBase * CreateRecorder(TVRec *tvrec, ChannelBase *channel, RecordingProfile &profile, const GeneralDBOptions &genOpt)
bool ShouldSwitchToAnotherInput(const QString &chanid) const
Checks if named channel exists on current tuner, or another tuner.
static const uint kFlagRecording
final result desired is a timed recording
bool Save(bool sendSig=true)
virtual bool IsAllGood(void) const
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 QString GetInputName(uint inputid)
A QElapsedTimer based timer to replace use of QTime as a timer.
QMutex m_triggerEventSleepLock
virtual bool Open(void)=0
Opens the channel changing hardware for use.
TuningQueue m_tuningRequests
uint RemoteGetState(uint inputid)
virtual QString GetInputName(void) const
const QString kRecorderInUseID
void SetDuration(std::chrono::seconds Duration)
Contains a duration during which the notification will be displayed for. The duration is informative ...
void ApplyRecordRecID(void)
Sets recordid to match RecordingRule recordid.
void SetCaching(bool cacheTables)
virtual void SetRecordingID(uint _recordedid)
bool Init(void)
Performs instance initialization, returns true on success.
static bool IsSupported(const QString &cardtype)
bool wait(std::chrono::milliseconds time=std::chrono::milliseconds::max())
Wait for the MThread to exit, with a maximum timeout.
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 StartEITEventProcessing(ChannelBase *channel, EITSource *eitSource)
Start inserting Event Information Tables from the multiplex we happen to be tuned to into the databas...
Holds information on a TV Program one might wish to record.
void ToggleChannelFavorite(const QString &changroupname)
Toggles whether the current channel should be on our favorites list.
std::chrono::seconds m_eitTransportTimeout
uint64_t GetFilesize(void) const override
virtual bool IsOpen(void) const =0
Reports whether channel is already open.
bool QueryAverageScanProgressive(void) const
If present in recording this loads average video scan type of the main video stream from database's s...
virtual void Unpause(void)
Unpause tells recorder to unpause.
static const uint kFlagRingBufferReady
uint GetRecordingID(void) const
MythMediaBuffer * m_buffer
virtual bool IsErrored(void)=0
Tells us whether an unrecoverable error has been encountered.
QString GetID(void) const
QWaitCondition m_triggerLiveTVDir
MThread * m_eventThread
Event processing thread, runs TVRec::run().
static const uint kFlagRecorderRunning
This class is used as a container for messages.
static const uint kFlagRec
RecStatus::Type m_recStatus
static bool GetATSCChannel(uint sourceid, const QString &channum, uint &major, uint &minor)
QString SetInput(QString input)
Changes to the specified input.
def write(text, progress=True)
static int GetProgramNumber(uint sourceid, const QString &channum)
static const uint64_t kDTVSigMon_WaitForMGT
uint GetSourceID(void) const
Returns current source id.
static bool QueueRecordingJobs(const RecordingInfo &recinfo, int jobTypes=JOB_NONE)
QVariant value(int i) const
QString m_overRecordCategory
QString GetCategory(void) const
virtual void SetStreamData(MPEGStreamData *data)
bool GetProgramRingBufferForLiveTV(RecordingInfo **pginfo, MythMediaBuffer **Buffer, const QString &channum)
uint GetMinorChannel(void) const
Returns minor channel, 0 if unknown.
static void apply_broken_dvb_driver_crc_hack(ChannelBase *, MPEGStreamData *)
QString toString(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)
MPEGStreamData * GetStreamData()
Returns the MPEG stream data if it exists.
int TotalSize(void) const
bool exec(void)
Wrap QSqlQuery::exec() so we can display SQL.
RecordingInfo * m_curRecording
void SaveCommFlagged(CommFlagStatus flag)
Set "commflagged" field in "recorded" table to "flag".
void StopRecording(bool killFile=false)
Changes from a recording state to kState_None.
bool GetKeyframePositions(long long start, long long end, frm_pos_map_t &map) const
void start(void)
starts measuring elapsed time.
QDateTime GetScheduledEndTime(void) const
The scheduled end time of the program.
#define LOG(_MASK_, _LEVEL_, _QSTRING_)
long long GetFramesWritten(void)
Returns number of frames written to disk by recorder.
static bool GetDevices(uint inputid, uint &parentid, GeneralDBOptions &gen_opts, DVBDBOptions &dvb_opts, FireWireDBOptions &firewire_opts)
void UpdateInUseMark(bool force=false)
QString GetInput(void) const
Returns current input.
static uint GetMplexID(uint sourceid, const QString &channum)
static const uint kFlagDummyRecorderRunning
virtual QString GetChannelName(void) const
bool IsDamaged(void) const
void UpdateRecordingEnd(void)
Update information in the recorded table when the end-time of a recording is changed.
DTVChannel * GetDTVChannel(void)
static const uint kFlagPendingActions
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.
virtual void Pause(bool clear=true)
Pause tells recorder to pause, it should not block.
QDateTime GetRecordingEndTime(void) const
Approximate time the recording should have ended, did end, or is intended to end.
virtual int GetPictureAttribute(PictureAttribute) const
QString GetRecordingGroup(void) const
QString GetChainID(void)
Get the chainid of the livetv instance.
std::chrono::seconds m_overRecordSecCat
bool GetKeyframeDurations(long long start, long long end, frm_pos_map_t &map) const
bool QueryTuningInfo(QString &channum, QString &input) const
Returns the channel and input needed to record the program.
static void GetPidsToCache(DTVSignalMonitor *dtvMon, pid_cache_t &pid_cache)
QDateTime current(bool stripped)
Returns current Date and Time in UTC.
bool TuningOnSameMultiplex(TuningRequest &request)
void TeardownRecorder(uint request_flags)
Tears down the recorder.
@ kState_Error
Error State, if we ever try to enter this state errored is set.
QDateTime GetRecordingStartTime(void) const
Approximate time the recording started.
long long GetFilePosition(void)
Returns total number of bytes written by RingBuffer.
QString GetPathname(void) const
static const uint kFlagCancelNextRecording
QString toStringXML(void) const
virtual void Renumber(uint sourceid, const QString &oldChanNum, const QString &newChanNum)
Changes a channum if we have it cached anywhere.
RecordingInfo * SwitchRecordingRingBuffer(const RecordingInfo &rcinfo)
QDateTime GetRecordEndTime(const ProgramInfo *pi) const
Returns recording end time with proper post-roll.
static QString toString(RecStatus::Type recstatus, uint id)
Converts "recstatus" into a short (unreadable) string.
QDateTime m_recordEndTime
QString GetTuningMode(void) const
Returns tuning mode last set by SetTuningMode().
@ kState_ChangingState
This is a placeholder state which we never actually enter, but is returned by GetState() when we are ...
uint GetOriginalNetworkID(void) const
Returns DVB original_network_id, 0 if unknown.
ProgramInfo * GetProgramAt(int at) const
Returns program at the desired location.
@ kState_None
None State, this is the initial state in both TV and TVRec, it indicates that we are ready to change ...
void StopActiveScan(void)
static bool IsRequired(const QString &cardtype)
Returns true iff the card type supports signal monitoring.
void SetIgnoreCRC(bool haveCRCbug)
std::chrono::seconds m_overRecordSecNrml
void RecorderPaused(void)
This is a callback, called by the "recorder" instance when it has actually paused.
std::chrono::seconds secsInFuture(const QDateTime &future)
PendingMap m_pendingRecordings
#define TRANSITION(ASTATE, BSTATE)
static const uint kFlagAnyRecRunning
virtual bool InitPictureAttributes(void)
uint GetInputId(void) const
Returns the inputid.
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...
uint GetTransportID(void) const
Returns DVB transport_stream_id, 0 if unknown.
virtual void ReturnCachedTable(const PSIPTable *psip) const
static const uint kFlagWaitingForSignal
static const uint kFlagDetect
void SetPathname(const QString &pn)
void SaveVideoProperties(uint mask, uint video_property_flags)
void SetRecordingEndTime(const QDateTime &dt)
bool CreateChannel(const QString &startchannel, bool enter_power_save_mode)
QHash< QString, int > m_autoRunJobs
RecordingFile * GetRecordingFile() const
void PauseRecorder(void)
Tells "recorder" to pause, used for channel and input changes.
void NotifySchedulerOfRecording(RecordingInfo *rec)
Tell scheduler about the recording.
void SetDesiredChannel(int major, int minor)
static int GetChanID(int db_mplexid, int service_transport_id, int major_channel, int minor_channel, int program_number)
QDateTime GetScheduledStartTime(void) const
The scheduled start time of program.
static const uint64_t kDVBSigMon_WaitForPos
Wait for rotor to complete turning the antenna.
RecStatus::Type GetRecordingStatus(void) const
void SetRecordingStatus(RecStatus::Type new_status, int line, bool have_lock=false)
MarkTypes QueryAverageAspectRatio(void) const
static bool is_dishnet_eit(uint inputid)
void SetDishNetEIT(bool use_dishnet_eit)
void SetHostPrefix(const QString &prefix)
static QString GenMythURL(const QString &host=QString(), int port=0, QString path=QString(), const QString &storageGroup=QString())
bool RemoteStopRecording(uint inputid)
Abstract class providing a generic interface to tuning hardware.
void SetRingBuffer(MythMediaBuffer *Buffer)
Tells recorder to use an externally created ringbuffer.
uint TablePID(uint i) const
static bool IsEncoder(const QString &rawtype)
bool RemoteIsBusy(uint inputid, InputInfo &busy_input)
static const uint kFlagKillRec
close recorder, discard recording
void SetScheduledEndTime(const QDateTime &dt)
void SaveCachedPids(const pid_cache_t &pid_cache) const
Saves MPEG PIDs to cache to database.
int GetBackendServerPort(void)
Returns the locally defined backend control port.
RecorderBase * m_recorder
void SendSystemEvent(const QString &msg)
virtual void StoreInputChannels(void)
Saves current channel as the default channel for the current input.
long long GetMaxBitrate(void) const
Returns the maximum bits per second this recorder can produce.
void SendEvent(const MythEvent &event)
static const uint kFlagRunMainLoop
void SetRecordingRuleType(RecordingType type)
TVState RemoveRecording(TVState state) const
If "state" is kState_RecordingOnly or kState_WatchingLiveTV, returns a kState_None,...
RecordingInfo * m_pseudoLiveTVRecording
void SetRecordingType(const QString &recording_type)
static QString FlagToString(uint f)
static MSqlQueryInfo InitCon(ConnectionReuse _reuse=kNormalConnection)
Only use this in combination with MSqlQuery constructor.
QString GetTitle(void) const
void HandleStateChange(void)
Changes the internalState to the desiredNextState if possible.
std::vector< pid_cache_item_t > pid_cache_t
void SetInputType(const QString &type)
Encapsulates data about ATSC stream and emits events for most tables.
static void DBError(const QString &where, const MSqlQuery &query)
bool CheckChannel(const QString &name) const
Checks if named channel exists on current tuner.
static const uint64_t kDTVSigMon_WaitForPMT
virtual void Reset(void)=0
Reset the recorder to the startup state.
static const uint kFlagSignalMonitorRunning
void SetRecordingGroup(const QString &group)
bool SetupSignalMonitor(bool tablemon, bool EITscan, bool notify)
This creates a SignalMonitor instance and begins signal monitoring.
static const uint64_t kDTVSigMon_WaitForPAT
std::enable_if_t< std::chrono::__is_duration< T >::value, T > GetDurSetting(const QString &key, T defaultval=T::zero())
QString TuningGetChanNum(const TuningRequest &request, QString &input) const
static const uint kFlagEITScan
final result desired is an EIT Scan
static const uint64_t kDTVSigMon_WaitForSDT
void TeardownSignalMonitor(void)
If a SignalMonitor instance exists, the monitoring thread is stopped and the instance is deleted.
virtual RecordingQuality * GetRecordingQuality(const RecordingInfo *ri) const
Returns a report about the current recordings quality.
bool WaitForNextLiveTVDir(void)
void SetDesiredStartTime(const QDateTime &dt)
static const uint kFlagCloseRec
close recorder, keep recording
int GetAutoRunJobs(void) const
Returns a bitmap of which jobs are attached to this RecordingInfo.
uint GetSourceID(void) const
Encapsulates data about MPEG stream and emits events for each table.
QWaitCondition m_triggerEventSleepWait
double GetFrameRate(void) const
Returns the latest frame rate.
static bool JobIsNotInMask(int job, int mask)
static const uint kFlagKillRingBuffer
void ChangeState(TVState nextState)
Puts a state change on the nextState queue.
TVState m_desiredNextState
QString toString(Verbosity v=kLongDescription, const QString &sep=":", const QString &grp="\"") const
void run(void) override
Event handling method, contains event loop.
bool m_triggerEventLoopSignal
MPEGStreamData * GetStreamData(void) const
static QString add_spacer(const QString &channel, const QString &spacer)
Adds the spacer before the last character in chan.
virtual bool IsRecording(void)
Tells whether the StartRecorder() loop is running.
AutoExpireType GetAutoExpire(void) const
void TuningShutdowns(const TuningRequest &request)
This shuts down anything that needs to be shut down before handling the passed in tuning request.
QString m_liveTVStartChannel
friend class TuningRequest
virtual int GetVideoFd(void)=0
Returns file descriptor of recorder device.
virtual QString getValue(void) const
void ReloadAll(const QStringList &data=QStringList())
QDateTime m_eitScanStartTime
int GetPictureAttribute(PictureAttribute attr)
void TuningNewRecorder(MPEGStreamData *streamData)
Creates a recorder instance.
void StartedRecording(const QString &ext)
Inserts this RecordingInfo into the database as an existing recording.
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 QReadWriteLock s_inputsLock
static const uint kFlagWaitingForRecPause
This is a specialization of RecorderBase used to handle MPEG-2, MPEG-4, MPEG-4 AVC,...
@ CHANNEL_DIRECTION_FAVORITE
bool isConnected(void) const
Only updated once during object creation.
static const uint kFlagLiveTV
final result desired is LiveTV recording
void SetNextLiveTVDir(QString dir)
MythCoreContext * gCoreContext
This global variable contains the MythCoreContext instance for the app.
float GetFramerate(void)
Returns recordering frame rate from the recorder.
bool m_signalEventCmdSent
RecStatus::Type GetRecordingStatus(void) const
bool QueueEITChannelChange(const QString &name)
Queues up a channel change for the EITScanner.
std::chrono::seconds m_eitCrawlIdleStart
ATSCStreamData * GetATSCStreamData()
Returns the ATSC stream data if it exists.
QWaitCondition m_triggerEventLoopWait
virtual bool EnterPowerSavingMode(void)
Enters power saving mode if the card supports it.
@ BROWSE_DOWN
Fetch information on next channel.
static bool IsV4L(const QString &rawtype)
bool IsErrored(void) const
Returns true is "errored" is true, false otherwise.
int GetNumSetting(const QString &key, int defaultval=0)
virtual long long GetFramesWritten(void)=0
Returns number of frames written to disk.
void SaveAutoExpire(AutoExpireType autoExpire, bool updateDelete=false)
Set "autoexpire" field in "recorded" table to "autoExpire".
virtual void SetFd(int fd)
Sets file descriptor.
@ BROWSE_RIGHT
Fetch information on current channel in the future.
ProgramInfo * GetRecording(void)
Allocates and returns a ProgramInfo for the current recording.
void SpawnLiveTV(LiveTVChain *newchain, bool pip, QString startchan)
Tells TVRec to spawn a "Live TV" recorder.
void CancelNextRecording(bool cancel)
Tells TVRec to cancel the upcoming recording.
QMap< long long, long long > frm_pos_map_t
Frame # -> File offset map.
static SignalMonitor * Init(const QString &cardtype, int db_cardnum, ChannelBase *channel, bool release_stream)
virtual void StopRecording(void)
StopRecording() signals to the recorder that it should stop recording and exit cleanly.
bool m_triggerEventSleepSignal
static QString GetVideoFilters(uint sourceid, const QString &channum)
virtual void Initialize(void)=0
This is called between SetOptionsFromProfile() and run() to initialize any devices,...
QString GetSIStandard(void) const
Returns PSIP table standard: MPEG, DVB, ATSC, or OpenCable.
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)
virtual bool HasExtraSlowTuning(void) const
bool GetBoolSetting(const QString &key, bool defaultval=false)
void SetProgramNumber(int progNum)
This table tells the decoder on which PIDs to find other tables, and their sizes and each table's cur...
SignalMonitor * m_signalMonitor
@ BROWSE_FAVORITE
Fetch information on the next favorite channel.
int64_t GetKeyframePosition(uint64_t desired) const
Returns byte position in RingBuffer of a keyframe according to recorder.
static bool ToggleChannel(uint chanid, int changrpid, bool delete_chan)
@ kState_WatchingPreRecorded
Watching Pre-recorded is a TV only state for when we are watching a pre-existing recording.
uint GetInputID(void) const
void SetUpdateRate(std::chrono::milliseconds msec)
Sets the number of milliseconds between signal monitoring attempts in the signal monitoring thread.
void SavePositionMap(bool force=false, bool finished=false)
Save the seektable to the DB.
void SetChanID(uint _chanid)
Provides interface to the tuning hardware when using DVB drivers.
QString GetInputType(int pos=-1) const
uint GetChanID(void) const
This is the unique key used in the database to locate tuning information.
QMutex m_nextLiveTVDirLock
void CheckForRecGroupChange(void)
Check if frontend changed the recording group.
virtual bool SetChannelByString(const QString &chan)=0
RecordingType GetRecordingRuleType(void) const
void IgnoreEncrypted(bool ignore)
QDateTime m_signalEventCmdTimeout
void SetChannel(int major, int minor)
Holds information on recordings and videos.
std::chrono::seconds secsInPast(const QDateTime &past)
bool m_reachedRecordingDeadline
bool GetChannelInfo(uint &chanid, uint &sourceid, QString &callsign, QString &channum, QString &channame, QString &xmltvid) const
TVState
TVState is an enumeration of the states used by TV and TVRec.
void ToStringList(QStringList &list) const
Serializes ProgramInfo into a QStringList which can be passed over a socket.
void AddListener(SignalMonitorListener *listener)
V4LChannel * GetV4LChannel(void)
GeneralDBOptions m_genOpt
QString FindNextDirMostFree(void)
QRecursiveMutex m_pendingRecLock
void RingBufferChanged(MythMediaBuffer *Buffer, RecordingInfo *pginfo, RecordingQuality *recq)
static void AddJobsToMask(int jobs, int &mask)
void TuningRestartRecorder(void)
Restarts a stopped recorder or unpauses a paused recorder.
static constexpr std::chrono::milliseconds kSignalMonitoringRate
How many milliseconds the signal monitor should wait between checks.
bool CheckChannel(const QString &channum) const
uint QueryAverageHeight(void) const
If present in recording this loads average height of the main video stream from database's stream mar...
bool SetupDTVSignalMonitor(bool EITscan)
Tells DTVSignalMonitor what channel to look for.
static int GetChannelGroupId(const QString &changroupname)
static const uint kFlagAnyRunning
void SetNextRecording(const RecordingInfo *ri, MythMediaBuffer *Buffer)
Sets next recording info, to be applied as soon as practical.
std::chrono::milliseconds SetSignalMonitoringRate(std::chrono::milliseconds rate, int notifyFrontend=1)
Sets the signal monitoring rate.
static void GetPreviewImage(const ProgramInfo &pginfo, const QString &token)
Submit a request for the generation of a preview image.
void bindValue(const QString &placeholder, const QVariant &val)
Add a single binding.
static const uint kFlagFinishRecording
void SetTuningMode(const QString &tuning_mode)
Sets tuning mode: "mpeg", "dvb", "atsc", etc.
virtual void SetStreamData(MPEGStreamData *data)
Sets the MPEG stream data for DTVSignalMonitor to use, and connects the table signals to the monitor.
@ BROWSE_SAME
Fetch browse information on current channel and time.
This is the coordinating class of the Recorder Subsystem.
void AddHistory(bool resched=true, bool forcedup=false, bool future=false)
Adds recording history, creating "record" it if necessary.
void FinishedRecording(RecordingInfo *curRec, RecordingQuality *recq)
If not a premature stop, adds program to history of recorded programs.
uint m_signalMonitorCheckCnt
virtual void SetRotorTarget(float)
Sets rotor target pos from 0.0 to 1.0.
virtual void SetVideoFilters(QString &filters)=0
Tells recorder which filters to use.
bool IsOnSameMultiplex(void) const
This is a wrapper around QThread that does several additional things.
int GetProgramNumber(void) const
Returns program number in PAT, -1 if unknown.
void AppendNewProgram(ProgramInfo *pginfo, const QString &channum, const QString &inputname, bool discont)
TVState GetState(void) const
Returns the TVState of the recorder.
QDateTime m_signalMonitorDeadline
static uint GetSourceID(uint inputid)
std::chrono::milliseconds m_dvbTuningDelay
bool SwitchLiveTVRingBuffer(const QString &channum, bool discont, bool set_rec)
This class is intended to detect the presence of needed tables.
static bool StateIsPlaying(TVState state)
Returns true if we are in any state associated with a player.
virtual int GetInputID(void) const
@ kState_WatchingLiveTV
Watching LiveTV is the state for when we are watching a recording and the user has control over the c...
bool RemoteRecordPending(uint inputid, const ProgramInfo *pginfo, std::chrono::seconds secsleft, bool hasLater)
virtual void Start()
Start signal monitoring thread.
TuningRequest m_lastTuningRequest
void SetNotifyFrontend(bool notify)
Enables or disables frontend notification of the current signal value.
MThread * m_recorderThread
Recorder thread, runs RecorderBase::run().
void StartedRecording(RecordingInfo *curRec)
Inserts a "curRec" into the database.
void InitAutoRunJobs(RecordingInfo *rec, AutoRunInitType t, RecordingProfile *recpro, int line)
BrowseDirection
Used to request ProgramInfo for channel browsing.
void SendMythSystemRecEvent(const QString &msg, const RecordingInfo *pginfo)
uint GetFlags(void) const
virtual void AddListeningPID(uint pid, PIDPriority priority=kPIDPriorityNormal)
static bool GetInputInfo(InputInfo &input, std::vector< uint > *groupids=nullptr)
QString GetHostName(void)
TVRec(int _inputid)
Performs instance initialization not requiring access to database.
virtual uint GetSourceID(void) const
static bool IsEITCapable(const QString &rawtype)
static const uint kFlagAntennaAdjust
antenna adjusting mode (LiveTV without recording).
static const uint kFlagExitPlayer
QString StateToString(TVState state)
Returns a human readable QString representing a TVState.
Class providing a generic interface to digital tuning hardware.
long long GetKeyframePosition(long long desired) const
Returns closest keyframe position before the desired frame.
bool IsErrored(void) const
void SetRecordingRuleID(uint id)
MPEGStreamData * TuningSignalCheck(void)
This checks if we have a channel lock.
~TVRec(void) override
Stops the event and scanning threads and deletes any ChannelBase, RingBuffer, and RecorderBase instan...
virtual uint GetNextChannel(uint chanid, ChannelChangeDirection direction) const
void GetCachedPids(pid_cache_t &pid_cache) const
Returns cached MPEG PIDs for last tuned channel.
static int get_highest_input(void)
bool SetVideoFiltersForChannel(uint sourceid, const QString &channum)