X-Git-Url: https://git.cweiske.de/enigma2.git/blobdiff_plain/4bc08995411e21f3564f09e136809be68ddf96a8..a19aef8b3244f753b02e5a06e7d3a185a424949f:/lib/service/servicedvb.cpp diff --git a/lib/service/servicedvb.cpp b/lib/service/servicedvb.cpp index 0c6d72bd..124916b8 100644 --- a/lib/service/servicedvb.cpp +++ b/lib/service/servicedvb.cpp @@ -6,162 +6,1507 @@ #include #include +#include +#include +#include + +#include +#include +#include +#include + +#include + +#include +#include + +#ifndef BYTE_ORDER +#error no byte order defined! +#endif + +#define TSPATH "/media/hdd" + +class eStaticServiceDVBInformation: public iStaticServiceInformation +{ + DECLARE_REF(eStaticServiceDVBInformation); +public: + RESULT getName(const eServiceReference &ref, std::string &name); + int getLength(const eServiceReference &ref); +}; + +DEFINE_REF(eStaticServiceDVBInformation); + +RESULT eStaticServiceDVBInformation::getName(const eServiceReference &ref, std::string &name) +{ + eServiceReferenceDVB &service = (eServiceReferenceDVB&)ref; + if ( !ref.name.empty() ) + { + if (service.getParentTransportStreamID().get()) // linkage subservice + { + ePtr service_center; + if (!eServiceCenter::getInstance(service_center)) + { + eServiceReferenceDVB parent = service; + parent.setTransportStreamID( service.getParentTransportStreamID() ); + parent.setServiceID( service.getParentServiceID() ); + parent.setParentTransportStreamID(eTransportStreamID(0)); + parent.setParentServiceID(eServiceID(0)); + parent.name=""; + ePtr service_info; + if (!service_center->info(parent, service_info)) + { + if (!service_info->getName(parent, name)) + { + // just show short name + unsigned int pos = name.find("\xc2\x86"); + if ( pos != std::string::npos ) + name.erase(0, pos+2); + pos = name.find("\xc2\x87"); + if ( pos != std::string::npos ) + name.erase(pos); + name+=" - "; + } + } + } + } + else + name=""; + name += ref.name; + return 0; + } + else + return -1; +} + +int eStaticServiceDVBInformation::getLength(const eServiceReference &ref) +{ + return -1; +} + +class eStaticServiceDVBBouquetInformation: public iStaticServiceInformation +{ + DECLARE_REF(eStaticServiceDVBBouquetInformation); +public: + RESULT getName(const eServiceReference &ref, std::string &name); + int getLength(const eServiceReference &ref); +}; + +DEFINE_REF(eStaticServiceDVBBouquetInformation); + +RESULT eStaticServiceDVBBouquetInformation::getName(const eServiceReference &ref, std::string &name) +{ + ePtr db; + ePtr res; + + int err; + if ((err = eDVBResourceManager::getInstance(res)) != 0) + { + eDebug("eStaticServiceDVBBouquetInformation::getName failed.. no resource manager!"); + return err; + } + if ((err = res->getChannelList(db)) != 0) + { + eDebug("eStaticServiceDVBBouquetInformation::getName failed.. no channel list!"); + return err; + } + + eBouquet *bouquet=0; + if ((err = db->getBouquet(ref, bouquet)) != 0) + { + eDebug("eStaticServiceDVBBouquetInformation::getName failed.. getBouquet failed!"); + return -1; + } + + if ( bouquet && bouquet->m_bouquet_name.length() ) + { + name = bouquet->m_bouquet_name; + return 0; + } + else + return -1; +} + +int eStaticServiceDVBBouquetInformation::getLength(const eServiceReference &ref) +{ + return -1; +} + +class eStaticServiceDVBPVRInformation: public iStaticServiceInformation +{ + DECLARE_REF(eStaticServiceDVBPVRInformation); + eServiceReference m_ref; + eDVBMetaParser m_parser; +public: + eStaticServiceDVBPVRInformation(const eServiceReference &ref); + RESULT getName(const eServiceReference &ref, std::string &name); + int getLength(const eServiceReference &ref); + + int getInfo(const eServiceReference &ref, int w); + std::string getInfoString(const eServiceReference &ref,int w); +}; + +DEFINE_REF(eStaticServiceDVBPVRInformation); + +eStaticServiceDVBPVRInformation::eStaticServiceDVBPVRInformation(const eServiceReference &ref) +{ + m_ref = ref; + m_parser.parseFile(ref.path); +} + +RESULT eStaticServiceDVBPVRInformation::getName(const eServiceReference &ref, std::string &name) +{ + ASSERT(ref == m_ref); + name = m_parser.m_name.size() ? m_parser.m_name : ref.path; + return 0; +} + +int eStaticServiceDVBPVRInformation::getLength(const eServiceReference &ref) +{ + ASSERT(ref == m_ref); + + eDVBTSTools tstools; + + if (tstools.openFile(ref.path.c_str())) + return 0; + + pts_t len; + if (tstools.calcLen(len)) + return 0; + + return len / 90000; +} + +int eStaticServiceDVBPVRInformation::getInfo(const eServiceReference &ref, int w) +{ + switch (w) + { + case iServiceInformation::sDescription: + return iServiceInformation::resIsString; + case iServiceInformation::sTimeCreate: + if (m_parser.m_time_create) + return m_parser.m_time_create; + else + return iServiceInformation::resNA; + default: + return iServiceInformation::resNA; + } +} + +std::string eStaticServiceDVBPVRInformation::getInfoString(const eServiceReference &ref,int w) +{ + switch (w) + { + case iServiceInformation::sDescription: + return m_parser.m_description; + default: + return ""; + } +} + +class eDVBPVRServiceOfflineOperations: public iServiceOfflineOperations +{ + DECLARE_REF(eDVBPVRServiceOfflineOperations); + eServiceReferenceDVB m_ref; +public: + eDVBPVRServiceOfflineOperations(const eServiceReference &ref); + + RESULT deleteFromDisk(int simulate); + RESULT getListOfFilenames(std::list &); +}; + +DEFINE_REF(eDVBPVRServiceOfflineOperations); + +eDVBPVRServiceOfflineOperations::eDVBPVRServiceOfflineOperations(const eServiceReference &ref): m_ref((const eServiceReferenceDVB&)ref) +{ +} + +RESULT eDVBPVRServiceOfflineOperations::deleteFromDisk(int simulate) +{ + if (simulate) + return 0; + else + { + std::list res; + if (getListOfFilenames(res)) + return -1; + + /* TODO: deferred removing.. */ + for (std::list::iterator i(res.begin()); i != res.end(); ++i) + { + eDebug("Removing %s...", i->c_str()); + ::unlink(i->c_str()); + } + + return 0; + } +} + +RESULT eDVBPVRServiceOfflineOperations::getListOfFilenames(std::list &res) +{ + res.clear(); + res.push_back(m_ref.path); + res.push_back(m_ref.path + ".meta"); + res.push_back(m_ref.path + ".ap"); + res.push_back(m_ref.path + ".cuts"); + return 0; +} + DEFINE_REF(eServiceFactoryDVB) -eServiceFactoryDVB::eServiceFactoryDVB() +eServiceFactoryDVB::eServiceFactoryDVB() +{ + ePtr sc; + + eServiceCenter::getPrivInstance(sc); + if (sc) + sc->addServiceFactory(eServiceFactoryDVB::id, this); +} + +eServiceFactoryDVB::~eServiceFactoryDVB() +{ + ePtr sc; + + eServiceCenter::getPrivInstance(sc); + if (sc) + sc->removeServiceFactory(eServiceFactoryDVB::id); +} + +DEFINE_REF(eDVBServiceList); + +eDVBServiceList::eDVBServiceList(const eServiceReference &parent): m_parent(parent) +{ +} + +eDVBServiceList::~eDVBServiceList() +{ +} + +RESULT eDVBServiceList::startQuery() +{ + ePtr db; + ePtr res; + + int err; + if ((err = eDVBResourceManager::getInstance(res)) != 0) + { + eDebug("no resource manager"); + return err; + } + if ((err = res->getChannelList(db)) != 0) + { + eDebug("no channel list"); + return err; + } + + ePtr q; + + if (!m_parent.path.empty()) + { + eDVBChannelQuery::compile(q, m_parent.path); + if (!q) + { + eDebug("compile query failed"); + return err; + } + } + + if ((err = db->startQuery(m_query, q, m_parent)) != 0) + { + eDebug("startQuery failed"); + return err; + } + + return 0; +} + +RESULT eDVBServiceList::getContent(PyObject *list, bool sorted) +{ + eServiceReferenceDVB ref; + + if (!m_query || !list || !PyList_Check(list)) + return -1; + + std::list tmplist; + + while (!m_query->getNextResult(ref)) + tmplist.push_back(ref); + + if (sorted) + tmplist.sort(iListableServiceCompare(this)); + + for (std::list::iterator it(tmplist.begin()); + it != tmplist.end(); ++it) + { + PyObject *refobj = New_eServiceReference(*it); + PyList_Append(list, refobj); + Py_DECREF(refobj); + } + return 0; +} + +RESULT eDVBServiceList::getContent(std::list &list, bool sorted) +{ + eServiceReferenceDVB ref; + + if (!m_query) + return -1; + + while (!m_query->getNextResult(ref)) + list.push_back(ref); + + if (sorted) + list.sort(iListableServiceCompare(this)); + + return 0; +} + +RESULT eDVBServiceList::getNext(eServiceReference &ref) +{ + if (!m_query) + return -1; + + return m_query->getNextResult((eServiceReferenceDVB&)ref); +} + +int eDVBServiceList::compareLessEqual(const eServiceReference &a, const eServiceReference &b) +{ + return m_query->compareLessEqual((const eServiceReferenceDVB&)a, (const eServiceReferenceDVB&)b); +} + +RESULT eDVBServiceList::startEdit(ePtr &res) +{ + if (m_parent.flags & eServiceReference::flagDirectory) // bouquet + { + ePtr db; + ePtr resm; + + if (eDVBResourceManager::getInstance(resm) || resm->getChannelList(db)) + return -1; + + if (db->getBouquet(m_parent, m_bouquet) != 0) + return -1; + + res = this; + + return 0; + } + res = 0; + return -1; +} + +RESULT eDVBServiceList::addService(eServiceReference &ref) +{ + if (!m_bouquet) + return -1; + return m_bouquet->addService(ref); +} + +RESULT eDVBServiceList::removeService(eServiceReference &ref) +{ + if (!m_bouquet) + return -1; + return m_bouquet->removeService(ref); +} + +RESULT eDVBServiceList::moveService(eServiceReference &ref, int pos) +{ + if (!m_bouquet) + return -1; + return m_bouquet->moveService(ref, pos); +} + +RESULT eDVBServiceList::flushChanges() +{ + if (!m_bouquet) + return -1; + return m_bouquet->flushChanges(); +} + +RESULT eDVBServiceList::setListName(const std::string &name) +{ + if (!m_bouquet) + return -1; + return m_bouquet->setListName(name); +} + +RESULT eServiceFactoryDVB::play(const eServiceReference &ref, ePtr &ptr) +{ + ePtr service; + int r = lookupService(service, ref); + if (r) + service = 0; + // check resources... + ptr = new eDVBServicePlay(ref, service); + return 0; +} + +RESULT eServiceFactoryDVB::record(const eServiceReference &ref, ePtr &ptr) +{ + if (ref.path.empty()) + { + ptr = new eDVBServiceRecord((eServiceReferenceDVB&)ref); + return 0; + } else + { + ptr = 0; + return -1; + } +} + +RESULT eServiceFactoryDVB::list(const eServiceReference &ref, ePtr &ptr) +{ + ePtr list = new eDVBServiceList(ref); + if (list->startQuery()) + { + ptr = 0; + return -1; + } + + ptr = list; + return 0; +} + +RESULT eServiceFactoryDVB::info(const eServiceReference &ref, ePtr &ptr) +{ + /* is a listable service? */ + if ((ref.flags & eServiceReference::flagDirectory) == eServiceReference::flagDirectory) // bouquet + { + if ( !ref.name.empty() ) // satellites or providers list + ptr = new eStaticServiceDVBInformation; + else // a dvb bouquet + ptr = new eStaticServiceDVBBouquetInformation; + } + else if (!ref.path.empty()) /* do we have a PVR service? */ + ptr = new eStaticServiceDVBPVRInformation(ref); + else // normal dvb service + { + ePtr service; + if (lookupService(service, ref)) // no eDVBService avail for this reference ( Linkage Services... ) + ptr = new eStaticServiceDVBInformation; + else + /* eDVBService has the iStaticServiceInformation interface, so we pass it here. */ + ptr = service; + } + return 0; +} + +RESULT eServiceFactoryDVB::offlineOperations(const eServiceReference &ref, ePtr &ptr) +{ + if (ref.path.empty()) + { + ptr = 0; + return -1; + } else + { + ptr = new eDVBPVRServiceOfflineOperations(ref); + return 0; + } +} + +RESULT eServiceFactoryDVB::lookupService(ePtr &service, const eServiceReference &ref) +{ + // TODO: handle the listing itself + // if (ref.... == -1) .. return "... bouquets ..."; + // could be also done in another serviceFactory (with seperate ID) to seperate actual services and lists + // TODO: cache + ePtr db; + ePtr res; + + int err; + if ((err = eDVBResourceManager::getInstance(res)) != 0) + { + eDebug("no resource manager"); + return err; + } + if ((err = res->getChannelList(db)) != 0) + { + eDebug("no channel list"); + return err; + } + + /* we are sure to have a ..DVB reference as the info() call was forwarded here according to it's ID. */ + if ((err = db->getService((eServiceReferenceDVB&)ref, service)) != 0) + { + eDebug("getService failed!"); + return err; + } + + return 0; +} + +eDVBServicePlay::eDVBServicePlay(const eServiceReference &ref, eDVBService *service): + m_reference(ref), m_dvb_service(service), m_is_paused(0) +{ + m_is_pvr = !ref.path.empty(); + + m_timeshift_enabled = m_timeshift_active = 0; + m_skipmode = 0; + + CONNECT(m_service_handler.serviceEvent, eDVBServicePlay::serviceEvent); + CONNECT(m_service_handler_timeshift.serviceEvent, eDVBServicePlay::serviceEventTimeshift); + CONNECT(m_event_handler.m_eit_changed, eDVBServicePlay::gotNewEvent); + + m_cuesheet_changed = 0; + + if (m_is_pvr) + loadCuesheet(); +} + +eDVBServicePlay::~eDVBServicePlay() +{ +} + +void eDVBServicePlay::gotNewEvent() +{ +#if 0 + // debug only + ePtr m_event_now, m_event_next; + getEvent(m_event_now, 0); + getEvent(m_event_next, 1); + + if (m_event_now) + eDebug("now running: %s (%d seconds :)", m_event_now->m_event_name.c_str(), m_event_now->m_duration); + if (m_event_next) + eDebug("next running: %s (%d seconds :)", m_event_next->m_event_name.c_str(), m_event_next->m_duration); +#endif + m_event((iPlayableService*)this, evUpdatedEventInfo); +} + +void eDVBServicePlay::serviceEvent(int event) +{ + switch (event) + { + case eDVBServicePMTHandler::eventTuned: + { + ePtr m_demux; + if (!m_service_handler.getDataDemux(m_demux)) + { + eServiceReferenceDVB &ref = (eServiceReferenceDVB&) m_reference; + int sid = ref.getParentServiceID().get(); + if (!sid) + sid = ref.getServiceID().get(); + if ( ref.getParentTransportStreamID().get() && + ref.getParentTransportStreamID() != ref.getTransportStreamID() ) + m_event_handler.startOther(m_demux, sid); + else + m_event_handler.start(m_demux, sid); + } + break; + } + case eDVBServicePMTHandler::eventTuneFailed: + { + eDebug("DVB service failed to tune"); + m_event((iPlayableService*)this, evTuneFailed); + break; + } + case eDVBServicePMTHandler::eventNewProgramInfo: + { + eDebug("eventNewProgramInfo %d %d", m_timeshift_enabled, m_timeshift_active); + if (m_timeshift_enabled) + updateTimeshiftPids(); + if (!m_timeshift_active) + updateDecoder(); + if (m_first_program_info && m_is_pvr) + { + m_first_program_info = 0; + seekTo(0); + } + m_event((iPlayableService*)this, evUpdatedInfo); + break; + } + case eDVBServicePMTHandler::eventEOF: + m_event((iPlayableService*)this, evEOF); + break; + case eDVBServicePMTHandler::eventSOF: + m_event((iPlayableService*)this, evSOF); + break; + } +} + +void eDVBServicePlay::serviceEventTimeshift(int event) +{ + switch (event) + { + case eDVBServicePMTHandler::eventNewProgramInfo: + if (m_timeshift_active) + updateDecoder(); + break; + case eDVBServicePMTHandler::eventEOF: + switchToLive(); + break; + } +} + +RESULT eDVBServicePlay::start() +{ + int r; + /* in pvr mode, we only want to use one demux. in tv mode, we're using + two (one for decoding, one for data source), as we must be prepared + to start recording from the data demux. */ + m_cue = new eCueSheet(); + m_first_program_info = 1; + r = m_service_handler.tune((eServiceReferenceDVB&)m_reference, m_is_pvr, m_cue); + m_event(this, evStart); + m_event((iPlayableService*)this, evSeekableStatusChanged); + return 0; +} + +RESULT eDVBServicePlay::stop() +{ + stopTimeshift(); /* in case timeshift was enabled, remove buffer etc. */ + + m_service_handler_timeshift.free(); + m_service_handler.free(); + + if (m_is_pvr && m_cuesheet_changed) + saveCuesheet(); + + return 0; +} + +RESULT eDVBServicePlay::connectEvent(const Slot2 &event, ePtr &connection) +{ + connection = new eConnection((iPlayableService*)this, m_event.connect(event)); + return 0; +} + +RESULT eDVBServicePlay::pause(ePtr &ptr) +{ + /* note: we check for timeshift to be enabled, + not neccessary active. if you pause when timeshift + is not active, you should activate it when unpausing */ + if ((!m_is_pvr) && (!m_timeshift_enabled)) + { + ptr = 0; + return -1; + } + + ptr = this; + return 0; +} + +RESULT eDVBServicePlay::setSlowMotion(int ratio) +{ + if (m_decoder) + return m_decoder->setSlowMotion(ratio); + else + return -1; +} + +RESULT eDVBServicePlay::setFastForward(int ratio) +{ + int skipmode, ffratio; + + if (ratio > 8) + { + skipmode = ratio; + ffratio = 1; + } else if (ratio > 0) + { + skipmode = 0; + ffratio = ratio; + } else if (!ratio) + { + skipmode = 0; + ffratio = 0; + } else // if (ratio < 0) + { + skipmode = ratio; + ffratio = 1; + } + + if (m_skipmode != skipmode) + { + eDebug("setting cue skipmode to %d", skipmode); + if (m_cue) + m_cue->setSkipmode(skipmode * 90000); /* convert to 90000 per second */ + } + + m_skipmode = skipmode; + + if (!m_decoder) + return -1; + + return m_decoder->setFastForward(ffratio); +} + +RESULT eDVBServicePlay::seek(ePtr &ptr) +{ + if (m_is_pvr || m_timeshift_enabled) + { + ptr = this; + return 0; + } + + ptr = 0; + return -1; +} + + /* TODO: when timeshift is enabled but not active, this doesn't work. */ +RESULT eDVBServicePlay::getLength(pts_t &len) +{ + ePtr pvr_channel; + + if ((m_timeshift_enabled ? m_service_handler_timeshift : m_service_handler).getPVRChannel(pvr_channel)) + return -1; + + return pvr_channel->getLength(len); +} + +RESULT eDVBServicePlay::pause() +{ + if (!m_is_paused && m_decoder) + { + m_is_paused = 1; + return m_decoder->freeze(0); + } else + return -1; +} + +RESULT eDVBServicePlay::unpause() +{ + if (m_is_paused && m_decoder) + { + m_is_paused = 0; + return m_decoder->unfreeze(); + } else + return -1; +} + +RESULT eDVBServicePlay::seekTo(pts_t to) +{ + eDebug("eDVBServicePlay::seekTo: jump %lld", to); + + if (!m_decode_demux) + return -1; + + ePtr pvr_channel; + + if ((m_timeshift_enabled ? m_service_handler_timeshift : m_service_handler).getPVRChannel(pvr_channel)) + return -1; + + if (!m_cue) + return -1; + + m_cue->seekTo(0, to); + return 0; +} + +RESULT eDVBServicePlay::seekRelative(int direction, pts_t to) { - ePtr sc; + eDebug("eDVBServicePlay::seekRelative: jump %d, %lld", direction, to); - eServiceCenter::getInstance(sc); - if (sc) - sc->addServiceFactory(eServiceFactoryDVB::id, this); + if (!m_decode_demux) + return -1; + + ePtr pvr_channel; + + if ((m_timeshift_enabled ? m_service_handler_timeshift : m_service_handler).getPVRChannel(pvr_channel)) + return -1; + + to *= direction; + + if (!m_cue) + return 0; + + m_cue->seekTo(1, to); + return 0; } -eServiceFactoryDVB::~eServiceFactoryDVB() +RESULT eDVBServicePlay::getPlayPosition(pts_t &pos) { - ePtr sc; + ePtr pvr_channel; - eServiceCenter::getInstance(sc); - if (sc) - sc->removeServiceFactory(eServiceFactoryDVB::id); + if (!m_decode_demux) + return -1; + + if ((m_timeshift_enabled ? m_service_handler_timeshift : m_service_handler).getPVRChannel(pvr_channel)) + return -1; + + int r = 0; + + /* if there is a decoder, use audio or video PTS */ + if (m_decoder) + { + r = m_decoder->getPTS(0, pos); + if (r) + return r; + } + + /* fixup */ + return pvr_channel->getCurrentPosition(m_decode_demux, pos, m_decoder ? 1 : 0); } -RESULT eServiceFactoryDVB::play(const eServiceReference &ref, ePtr &ptr) +RESULT eDVBServicePlay::setTrickmode(int trick) { - // check resources... - ptr = new eDVBServicePlay(ref); + if (m_decoder) + m_decoder->setTrickmode(trick); + return 0; +} + +RESULT eDVBServicePlay::isCurrentlySeekable() +{ + return m_is_pvr || m_timeshift_active; +} + +RESULT eDVBServicePlay::frontendStatusInfo(ePtr &ptr) +{ + ptr = this; + return 0; +} + +RESULT eDVBServicePlay::info(ePtr &ptr) +{ + ptr = this; + return 0; +} + +RESULT eDVBServicePlay::audioTracks(ePtr &ptr) +{ + ptr = this; + return 0; +} + +RESULT eDVBServicePlay::subServices(ePtr &ptr) +{ + ptr = this; return 0; } -RESULT eServiceFactoryDVB::record(const eServiceReference &, ePtr &ptr) +RESULT eDVBServicePlay::timeshift(ePtr &ptr) { ptr = 0; + if (m_timeshift_enabled || !m_is_pvr) + { + if (!m_timeshift_enabled) + { + /* we need enough diskspace */ + struct statfs fs; + if (statfs(TSPATH "/.", &fs) < 0) + { + eDebug("statfs failed!"); + return -2; + } + + if (((off_t)fs.f_bavail) * ((off_t)fs.f_bsize) < 1024*1024*1024LL) + { + eDebug("not enough diskspace for timeshift! (less than 1GB)"); + return -3; + } + } + ptr = this; + return 0; + } return -1; } -RESULT eServiceFactoryDVB::list(const eServiceReference &, ePtr &ptr) +RESULT eDVBServicePlay::cueSheet(ePtr &ptr) { + if (m_is_pvr) + { + ptr = this; + return 0; + } ptr = 0; return -1; } -eDVBServicePlay::eDVBServicePlay(const eServiceReference &ref): - m_reference(ref) +RESULT eDVBServicePlay::getName(std::string &name) { - CONNECT(m_serviceHandler.serviceEvent, eDVBServicePlay::serviceEvent); - eDebug("DVB start (play)"); + if (m_is_pvr) + { + ePtr i = new eStaticServiceDVBPVRInformation(m_reference); + return i->getName(m_reference, name); + } + if (m_dvb_service) + { + m_dvb_service->getName(m_reference, name); + if (name.empty()) + name = "(...)"; + } + else if (!m_reference.name.empty()) + eStaticServiceDVBInformation().getName(m_reference, name); + else + name = "DVB service"; + return 0; } -eDVBServicePlay::~eDVBServicePlay() +RESULT eDVBServicePlay::getEvent(ePtr &evt, int nownext) { - eDebug("DVB stop (play)"); + return m_event_handler.getEvent(evt, nownext); } -void eDVBServicePlay::serviceEvent(int event) +int eDVBServicePlay::getInfo(int w) { - eDebug("service event %d", event); - switch (event) - { - case eDVBServicePMTHandler::eventNewProgramInfo: + eDVBServicePMTHandler::program program; + + if (m_service_handler.getProgramInfo(program)) + return -1; + + switch (w) { - int vpid = -1, apid = -1, pcrpid = -1; - eDVBServicePMTHandler::program program; - if (m_serviceHandler.getProgramInfo(program)) - eDebug("getting program info failed."); - else + case sAspect: + if (!program.videoStreams.empty() && program.videoStreams[0].component_tag != -1) { - eDebugNoNewLine("have %d video stream(s)", program.videoStreams.size()); - if (!program.videoStreams.empty()) - { - eDebugNoNewLine(" ("); - for (std::vector::const_iterator - i(program.videoStreams.begin()); - i != program.videoStreams.end(); ++i) - { - if (vpid == -1) - vpid = i->pid; - if (i != program.videoStreams.begin()) - eDebugNoNewLine(", "); - eDebugNoNewLine("%04x", i->pid); - } - eDebugNoNewLine(")"); - } - eDebugNoNewLine(", and %d audio stream(s)", program.audioStreams.size()); - if (!program.audioStreams.empty()) + ePtr evt; + if (!m_event_handler.getEvent(evt, 0)) { - eDebugNoNewLine(" ("); - for (std::vector::const_iterator - i(program.audioStreams.begin()); - i != program.audioStreams.end(); ++i) + ePtr data; + if (!evt->getComponentData(data, program.videoStreams[0].component_tag)) { - if (apid == -1) - apid = i->pid; - if (i != program.audioStreams.begin()) - eDebugNoNewLine(", "); - eDebugNoNewLine("%04x", i->pid); + if ( data->getStreamContent() == 1 ) + { + switch(data->getComponentType()) + { + // SD + case 1: // 4:3 SD PAL + case 2: + case 3: // 16:9 SD PAL + case 4: // > 16:9 PAL + case 5: // 4:3 SD NTSC + case 6: + case 7: // 16:9 SD NTSC + case 8: // > 16:9 NTSC + + // HD + case 9: // 4:3 HD PAL + case 0xA: + case 0xB: // 16:9 HD PAL + case 0xC: // > 16:9 HD PAL + case 0xD: // 4:3 HD NTSC + case 0xE: + case 0xF: // 16:9 HD NTSC + case 0x10: // > 16:9 HD PAL + return data->getComponentType(); + } + } } - eDebugNoNewLine(")"); } - eDebug(", and the pcr pid is %04x", program.pcrPid); - if (program.pcrPid != 0x1fff) - pcrpid = program.pcrPid; } - - if (!m_decoder) + return -1; + case sIsCrypted: return program.isCrypted; + case sVideoPID: if (program.videoStreams.empty()) return -1; return program.videoStreams[0].pid; + case sAudioPID: if (program.audioStreams.empty()) return -1; return program.audioStreams[m_current_audio_stream].pid; + case sPCRPID: return program.pcrPid; + case sPMTPID: return program.pmtPid; + case sTXTPID: return program.textPid; + case sSID: return ((const eServiceReferenceDVB&)m_reference).getServiceID().get(); + case sONID: return ((const eServiceReferenceDVB&)m_reference).getOriginalNetworkID().get(); + case sTSID: return ((const eServiceReferenceDVB&)m_reference).getTransportStreamID().get(); + case sNamespace: return ((const eServiceReferenceDVB&)m_reference).getDVBNamespace().get(); + case sProvider: if (!m_dvb_service) return -1; return -2; + default: + return -1; + } +} + +std::string eDVBServicePlay::getInfoString(int w) +{ + switch (w) + { + case sProvider: + if (!m_dvb_service) return ""; + return m_dvb_service->m_provider_name; + default: + return ""; + } +} + +int eDVBServicePlay::getNumberOfTracks() +{ + eDVBServicePMTHandler::program program; + if (m_service_handler.getProgramInfo(program)) + return 0; + return program.audioStreams.size(); +} + +RESULT eDVBServicePlay::selectTrack(unsigned int i) +{ + int ret = selectAudioStream(i); + + if (m_decoder->start()) + return -5; + + return ret; +} + +RESULT eDVBServicePlay::getTrackInfo(struct iAudioTrackInfo &info, unsigned int i) +{ + eDVBServicePMTHandler::program program; + + if (m_service_handler.getProgramInfo(program)) + return -1; + + if (i >= program.audioStreams.size()) + return -2; + + if (program.audioStreams[i].type == eDVBServicePMTHandler::audioStream::atMPEG) + info.m_description = "MPEG"; + else if (program.audioStreams[i].type == eDVBServicePMTHandler::audioStream::atAC3) + info.m_description = "AC3"; + else if (program.audioStreams[i].type == eDVBServicePMTHandler::audioStream::atDTS) + info.m_description = "DTS"; + else + info.m_description = "???"; + + if (program.audioStreams[i].component_tag != -1) + { + ePtr evt; + if (!m_event_handler.getEvent(evt, 0)) { - ePtr demux; - m_serviceHandler.getDemux(demux); - if (demux) - demux->getMPEGDecoder(m_decoder); + ePtr data; + if (!evt->getComponentData(data, program.audioStreams[i].component_tag)) + info.m_language = data->getText(); } + } + + if (info.m_language.empty()) + info.m_language = program.audioStreams[i].language_code; + + return 0; +} + +int eDVBServicePlay::selectAudioStream(int i) +{ + eDVBServicePMTHandler::program program; + + if (m_service_handler.getProgramInfo(program)) + return -1; + + if ((unsigned int)i >= program.audioStreams.size()) + return -2; + + if (!m_decoder) + return -3; + + if (m_decoder->setAudioPID(program.audioStreams[i].pid, program.audioStreams[i].type)) + return -4; - if (m_decoder) + if (m_dvb_service && !m_is_pvr) + { + if (program.audioStreams[i].type == eDVBAudio::aMPEG) { - m_decoder->setVideoPID(vpid); - m_decoder->setAudioPID(apid, 0); - m_decoder->setSyncPCR(pcrpid); - m_decoder->start(); + m_dvb_service->setCachePID(eDVBService::cAPID, program.audioStreams[i].pid); + m_dvb_service->setCachePID(eDVBService::cAC3PID, -1); + } else + { + m_dvb_service->setCachePID(eDVBService::cAPID, -1); + m_dvb_service->setCachePID(eDVBService::cAC3PID, program.audioStreams[i].pid); } - - break; - } } + + m_current_audio_stream = i; + + return 0; } -RESULT eDVBServicePlay::start() +int eDVBServicePlay::getFrontendInfo(int w) { - eDebug("starting DVB service"); - return m_serviceHandler.tune((eServiceReferenceDVB&)m_reference); + if (m_is_pvr) + return 0; + eUsePtr channel; + if(m_service_handler.getChannel(channel)) + return 0; + ePtr fe; + if(channel->getFrontend(fe)) + return 0; + return fe->readFrontendData(w); } -RESULT eDVBServicePlay::stop() +int eDVBServicePlay::getNumberOfSubservices() { - eDebug("stopping.."); + ePtr evt; + if (!m_event_handler.getEvent(evt, 0)) + return evt->getNumOfLinkageServices(); return 0; } -RESULT eDVBServicePlay::connectEvent(const Slot2 &event, ePtr &connection) +RESULT eDVBServicePlay::getSubservice(eServiceReference &sub, unsigned int n) { + ePtr evt; + if (!m_event_handler.getEvent(evt, 0)) + { + if (!evt->getLinkageService(sub, m_reference, n)) + return 0; + } + sub.type=eServiceReference::idInvalid; return -1; } -RESULT eDVBServicePlay::getIPausableService(ePtr &ptr) +RESULT eDVBServicePlay::startTimeshift() { - // not yet possible, maybe later... - ptr = 0; - return -1; + ePtr demux; + + eDebug("Start timeshift!"); + + if (m_timeshift_enabled) + return -1; + + /* start recording with the data demux. */ + if (m_service_handler.getDataDemux(demux)) + return -2; + + demux->createTSRecorder(m_record); + if (!m_record) + return -3; + + char templ[]=TSPATH "/timeshift.XXXXXX"; + m_timeshift_fd = mkstemp(templ); + m_timeshift_file = templ; + + eDebug("recording to %s", templ); + + if (m_timeshift_fd < 0) + { + m_record = 0; + return -4; + } + + m_record->setTargetFD(m_timeshift_fd); + + m_timeshift_enabled = 1; + + updateTimeshiftPids(); + m_record->start(); + + return 0; } -RESULT eDVBServicePlay::getIServiceInformation(ePtr &ptr) +RESULT eDVBServicePlay::stopTimeshift() { - ptr = this; + if (!m_timeshift_enabled) + return -1; + + switchToLive(); + + m_timeshift_enabled = 0; + + m_record->stop(); + m_record = 0; + + close(m_timeshift_fd); + eDebug("remove timeshift file"); + remove(m_timeshift_file.c_str()); + return 0; } -RESULT eDVBServicePlay::getName(const eServiceReference &ref, std::string &name) +int eDVBServicePlay::isTimeshiftActive() { - name = "DVB service"; - return 0; + return m_timeshift_enabled && m_timeshift_active; +} + +RESULT eDVBServicePlay::activateTimeshift() +{ + if (!m_timeshift_enabled) + return -1; + + if (!m_timeshift_active) + { + switchToTimeshift(); + return 0; + } + + return -2; +} + +PyObject *eDVBServicePlay::getCutList() +{ + PyObject *list = PyList_New(0); + + for (std::multiset::iterator i(m_cue_entries.begin()); i != m_cue_entries.end(); ++i) + { + PyObject *tuple = PyTuple_New(2); + PyTuple_SetItem(tuple, 0, PyLong_FromLongLong(i->where)); + PyTuple_SetItem(tuple, 1, PyInt_FromLong(i->what)); + PyList_Append(list, tuple); + Py_DECREF(tuple); + } + + return list; +} + +void eDVBServicePlay::setCutList(PyObject *list) +{ + if (!PyList_Check(list)) + return; + int size = PyList_Size(list); + int i; + + m_cue_entries.clear(); + + for (i=0; i pids_to_record; + pids_to_record.insert(0); // PAT + if (program.pmtPid != -1) + pids_to_record.insert(program.pmtPid); // PMT + + if (program.textPid != -1) + pids_to_record.insert(program.textPid); // Videotext + + for (std::vector::const_iterator + i(program.videoStreams.begin()); + i != program.videoStreams.end(); ++i) + pids_to_record.insert(i->pid); + + for (std::vector::const_iterator + i(program.audioStreams.begin()); + i != program.audioStreams.end(); ++i) + pids_to_record.insert(i->pid); + + std::set new_pids, obsolete_pids; + + std::set_difference(pids_to_record.begin(), pids_to_record.end(), + m_pids_active.begin(), m_pids_active.end(), + std::inserter(new_pids, new_pids.begin())); + + std::set_difference( + m_pids_active.begin(), m_pids_active.end(), + pids_to_record.begin(), pids_to_record.end(), + std::inserter(new_pids, new_pids.begin()) + ); + + for (std::set::iterator i(new_pids.begin()); i != new_pids.end(); ++i) + m_record->addPID(*i); + + for (std::set::iterator i(obsolete_pids.begin()); i != obsolete_pids.end(); ++i) + m_record->removePID(*i); + } +} + +void eDVBServicePlay::switchToLive() +{ + if (!m_timeshift_active) + return; + + m_decoder = 0; + m_decode_demux = 0; + /* free the timeshift service handler, we need the resources */ + m_service_handler_timeshift.free(); + m_timeshift_active = 0; + + m_event((iPlayableService*)this, evSeekableStatusChanged); + + updateDecoder(); +} + +void eDVBServicePlay::switchToTimeshift() +{ + if (m_timeshift_active) + return; + + m_decode_demux = 0; + m_decoder = 0; + + m_timeshift_active = 1; + + m_event((iPlayableService*)this, evSeekableStatusChanged); + + eServiceReferenceDVB r = (eServiceReferenceDVB&)m_reference; + r.path = m_timeshift_file; + + m_service_handler_timeshift.tune(r, 1, m_cue); /* use the decoder demux for everything */ +} + +void eDVBServicePlay::updateDecoder() +{ + int vpid = -1, apid = -1, apidtype = -1, pcrpid = -1, tpid = -1; + eDVBServicePMTHandler &h = m_timeshift_active ? m_service_handler_timeshift : m_service_handler; + + eDVBServicePMTHandler::program program; + if (h.getProgramInfo(program)) + eDebug("getting program info failed."); + else + { + eDebugNoNewLine("have %d video stream(s)", program.videoStreams.size()); + if (!program.videoStreams.empty()) + { + eDebugNoNewLine(" ("); + for (std::vector::const_iterator + i(program.videoStreams.begin()); + i != program.videoStreams.end(); ++i) + { + if (vpid == -1) + vpid = i->pid; + if (i != program.videoStreams.begin()) + eDebugNoNewLine(", "); + eDebugNoNewLine("%04x", i->pid); + } + eDebugNoNewLine(")"); + } + eDebugNoNewLine(", and %d audio stream(s)", program.audioStreams.size()); + if (!program.audioStreams.empty()) + { + eDebugNoNewLine(" ("); + for (std::vector::const_iterator + i(program.audioStreams.begin()); + i != program.audioStreams.end(); ++i) + { + if (apid == -1) + { + apid = i->pid; + apidtype = i->type; + } + if (i != program.audioStreams.begin()) + eDebugNoNewLine(", "); + eDebugNoNewLine("%04x", i->pid); + } + eDebugNoNewLine(")"); + } + eDebugNoNewLine(", and the pcr pid is %04x", program.pcrPid); + pcrpid = program.pcrPid; + eDebug(", and the text pid is %04x", program.textPid); + tpid = program.textPid; + } + + if (!m_decoder) + { + h.getDecodeDemux(m_decode_demux); + if (m_decode_demux) + m_decode_demux->getMPEGDecoder(m_decoder); + if (m_cue) + m_cue->setDecodingDemux(m_decode_demux, m_decoder); + } + + if (m_decoder) + { + m_decoder->setVideoPID(vpid); + m_current_audio_stream = 0; + m_decoder->setAudioPID(apid, apidtype); + if (!(m_is_pvr || m_timeshift_active)) + m_decoder->setSyncPCR(pcrpid); + else + m_decoder->setSyncPCR(-1); + m_decoder->setTextPID(tpid); + m_decoder->start(); +// how we can do this better? +// update cache pid when the user changed the audio track or video track +// TODO handling of difference audio types.. default audio types.. + + /* don't worry about non-existing services, nor pvr services */ + if (m_dvb_service && !m_is_pvr) + { + if (apidtype == eDVBAudio::aMPEG) + { + m_dvb_service->setCachePID(eDVBService::cAPID, apid); + m_dvb_service->setCachePID(eDVBService::cAC3PID, -1); + } + else + { + m_dvb_service->setCachePID(eDVBService::cAPID, -1); + m_dvb_service->setCachePID(eDVBService::cAC3PID, apid); + } + m_dvb_service->setCachePID(eDVBService::cVPID, vpid); + m_dvb_service->setCachePID(eDVBService::cPCRPID, pcrpid); + m_dvb_service->setCachePID(eDVBService::cTPID, tpid); + } + } +} + +void eDVBServicePlay::loadCuesheet() +{ + std::string filename = m_reference.path + ".cuts"; + + m_cue_entries.clear(); + + FILE *f = fopen(filename.c_str(), "rb"); + + if (f) + { + eDebug("loading cuts.."); + while (1) + { + unsigned long long where; + unsigned int what; + + if (!fread(&where, sizeof(where), 1, f)) + break; + if (!fread(&what, sizeof(what), 1, f)) + break; + +#if BYTE_ORDER == LITTLE_ENDIAN + where = bswap_64(where); +#endif + what = ntohl(what); + + if (what > 2) + break; + + m_cue_entries.insert(cueEntry(where, what)); + } + fclose(f); + eDebug("%d entries", m_cue_entries.size()); + } else + eDebug("cutfile not found!"); + + m_cuesheet_changed = 0; +} + +void eDVBServicePlay::saveCuesheet() +{ + std::string filename = m_reference.path + ".cuts"; + + FILE *f = fopen(filename.c_str(), "wb"); + + if (f) + { + unsigned long long where; + int what; + + for (std::multiset::iterator i(m_cue_entries.begin()); i != m_cue_entries.end(); ++i) + { +#if BYTE_ORDER == BIG_ENDIAN + where = i->where; +#else + where = bswap_64(i->where); +#endif + what = htonl(i->what); + fwrite(&where, sizeof(where), 1, f); + fwrite(&what, sizeof(what), 1, f); + + } + fclose(f); + } + + m_cuesheet_changed = 0; } DEFINE_REF(eDVBServicePlay)