Files
recycled-ni-neutrino/src/zapit/src/zapit.cpp
BPanther f70d5accca more pip changes
Origin commit data
------------------
Commit: 0239f805f3
Author: BPanther <bpanther_ts@hotmail.com>
Date: 2021-10-15 (Fri, 15 Oct 2021)
2021-10-15 14:58:38 +02:00

3048 lines
93 KiB
C++

/*
* $Id: zapit.cpp,v 1.290.2.50 2003/06/13 10:53:15 digi_casi Exp $
*
* zapit - d-box2 linux project
*
* (C) 2001, 2002 by Philipp Leusmann <faralla@berlios.de>
* (C) 2007-2013 Stefan Seyfried
* Copyright (C) 2011 CoolStream International Ltd
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
*
*/
/* system headers */
#include <fcntl.h>
#include <sys/poll.h>
#include <sys/ioctl.h>
#include <sys/socket.h>
#include <sys/time.h>
#include <unistd.h>
#include <syscall.h>
#include <pthread.h>
#ifdef HAVE_CONFIG_H
#include <config.h>
#endif
/* zapit headers */
#include <zapit/capmt.h>
#include <zapit/client/msgtypes.h>
#include <zapit/debug.h>
#include <zapit/getservices.h>
#include <zapit/pat.h>
#include <zapit/scanpmt.h>
#include <zapit/scan.h>
//#include <zapit/fastscan.h>
#include <zapit/scansdt.h>
#include <zapit/settings.h>
#include <zapit/zapit.h>
#include <xmlinterface.h>
#include <zapit/satconfig.h>
#include <zapit/femanager.h>
#if HAVE_CST_HARDWARE
#include <record_cs.h>
#include <playback_cs.h>
#include <pwrmngr.h>
#endif
#include <hardware/audio.h>
#include <hardware/ca.h>
#include <hardware/dmx.h>
#include <hardware/video.h>
#include <driver/abstime.h>
#include <system/set_threadname.h>
#include <libdvbsub/dvbsub.h>
#include <OpenThreads/ScopedLock>
#include <libtuxtxt/teletext.h>
#include <neutrino.h>
#include <gui/osd_helpers.h>
#ifdef PEDANTIC_VALGRIND_SETUP
#define VALGRIND_PARANOIA(x) memset(&x, 0, sizeof(x))
#else
#define VALGRIND_PARANOIA(x) {}
#endif
/* globals */
int sig_delay = 2; // seconds between signal check
/* the bouquet manager */
CBouquetManager *g_bouquetManager = NULL;
//int cam_ci = 2; // CA_INIT_SC 0 or CA_INIT_CI 1 or CA_INIT_BOTH 2
cCA *ca = NULL;
extern cDemux *pmtDemux;
extern cVideo *videoDecoder;
extern cAudio *audioDecoder;
extern cDemux *audioDemux;
extern cDemux *videoDemux;
#ifdef ENABLE_PIP
extern cVideo *pipVideoDecoder[3];
extern cDemux *pipVideoDemux[3];
extern cAudio * pipAudioDecoder[3];
extern cDemux * pipAudioDemux[3];
#endif
cDemux *pcrDemux = NULL;
/* the map which stores the wanted cable/satellites */
scan_list_t scanProviders;
int zapit_debug = 0;
/* transponder scan */
transponder_list_t transponders;
CZapitClient::bouquetMode bouquetMode = CZapitClient::BM_UPDATEBOUQUETS;
//static CZapitClient::scanType scanType = CZapitClient::ST_TVRADIO;
//static TP_params TP;
static bool update_pmt = true;
/******************************************************************************/
CZapit * CZapit::zapit = NULL;
CZapit::CZapit()
: configfile(',', false)
{
started = false;
pmt_update_fd = -1;
//volume_left = 0, volume_right = 0;
audio_mode = 0;
mosd = 0;
aspectratio = 0;
mode43 = 0;
def_audio_mode = 0;
playbackStopForced = false;
standby = true;
event_mode = true;
firstzap = true;
playing = false;
list_changed = false; // flag to indicate, allchans was changed
currentMode = 0;
current_volume = 100;
volume_percent = 0;
for (unsigned int i = 0; i < 3; i++)
pip_channel_id[i] = 0;
lock_channel_id = 0;
for (unsigned int i = 0; i < 3; i++)
pip_fe[i] = NULL;
#if ENABLE_AIT
ait = new CAit();
#endif
}
CZapit::~CZapit()
{
#if ENABLE_AIT
delete ait;
#endif
Stop();
}
CZapit * CZapit::getInstance()
{
if(zapit == NULL)
zapit = new CZapit();
return zapit;
}
void CZapit::SendEvent(const unsigned int eventID, const void* eventbody, const unsigned int eventbodysize)
{
if(event_mode)
eventServer->sendEvent(eventID, CEventServer::INITID_ZAPIT, eventbody, eventbodysize);
}
void CZapit::SaveSettings(bool write)
{
if (current_channel) {
if ((currentMode & RADIO_MODE))
lastChannelRadio = current_channel->getChannelID();
else
lastChannelTV = current_channel->getChannelID();
}
if (write) {
configfile.setBool("saveLastChannel", config.saveLastChannel);
if (config.saveLastChannel) {
configfile.setInt32("lastChannelMode", (currentMode & RADIO_MODE) ? 1 : 0);
configfile.setInt64("lastChannelRadio", lastChannelRadio);
configfile.setInt64("lastChannelTV", lastChannelTV);
configfile.setInt64("lastChannel", live_channel_id);
configfile.setInt64("lastOTAChannel", last_channel_id);
}
configfile.setInt32("writeChannelsNames", config.writeChannelsNames);
configfile.setBool("makeRemainingChannelsBouquet", config.makeRemainingChannelsBouquet);
configfile.setInt32("feTimeout", config.feTimeout);
configfile.setInt32("feRetries", config.feRetries);
configfile.setInt32("rezapTimeout", config.rezapTimeout);
configfile.setBool("scanPids", config.scanPids);
configfile.setInt32("scanSDT", config.scanSDT);
configfile.setInt32("cam_ci", config.cam_ci);
#if 0 // unused
configfile.setBool("fastZap", config.fastZap);
configfile.setBool("sortNames", config.sortNames);
#endif
/* FIXME FE global */
configfile.setInt32("noSameFE", config.noSameFE);
char tempd[12];
sprintf(tempd, "%3.6f", config.gotoXXLatitude);
configfile.setString("gotoXXLatitude", tempd);
sprintf(tempd, "%3.6f", config.gotoXXLongitude);
configfile.setString("gotoXXLongitude", tempd);
configfile.setInt32("gotoXXLaDirection", config.gotoXXLaDirection);
configfile.setInt32("gotoXXLoDirection", config.gotoXXLoDirection);
configfile.setInt32("repeatUsals", config.repeatUsals);
#if 1
/* FIXME FE specific */
configfile.setBool("highVoltage", config.highVoltage);
configfile.setInt32("lastSatellitePosition", live_fe->getRotorSatellitePosition());
configfile.setInt32("diseqcRepeats", live_fe->getDiseqcRepeats());
configfile.setInt32("diseqcType", live_fe->getDiseqcType());
configfile.setInt32("motorRotationSpeed", config.motorRotationSpeed);
#endif
if (configfile.getModifiedFlag())
configfile.saveConfig(ZAPITCONFIGFILE);
}
}
void CZapit::LoadAudioMap()
{
FILE *audio_config_file = fopen(AUDIO_CONFIG_FILE, "r");
audio_map.clear();
if (!audio_config_file) {
perror(AUDIO_CONFIG_FILE);
return;
}
t_channel_id chan;
int apid = 0, subpid = 0, ttxpid = 0, ttxpage = 0;
int mode = 0;
int volume = 0;
char s[1000];
while (fgets(s, sizeof(s), audio_config_file)) {
sscanf(s, "%" SCNx64 " %d %d %d %d %d %d", &chan, &apid, &mode, &volume, &subpid, &ttxpid, &ttxpage);
audio_map[chan].apid = apid;
audio_map[chan].subpid = subpid;
audio_map[chan].mode = mode;
audio_map[chan].volume = volume;
audio_map[chan].ttxpid = ttxpid;
audio_map[chan].ttxpage = ttxpage;
}
fclose(audio_config_file);
}
void CZapit::SaveAudioMap()
{
FILE *audio_config_file = fopen(AUDIO_CONFIG_FILE, "w");
if (!audio_config_file) {
perror(AUDIO_CONFIG_FILE);
return;
}
for (audio_map_iterator_t audio_map_it = audio_map.begin(); audio_map_it != audio_map.end(); audio_map_it++) {
fprintf(audio_config_file, "%" PRIx64 " %d %d %d %d %d %d\n", (uint64_t) audio_map_it->first,
(int) audio_map_it->second.apid, (int) audio_map_it->second.mode, (int) audio_map_it->second.volume,
(int) audio_map_it->second.subpid, (int) audio_map_it->second.ttxpid, (int) audio_map_it->second.ttxpage);
}
fdatasync(fileno(audio_config_file));
fclose(audio_config_file);
}
void CZapit::LoadVolumeMap()
{
OpenThreads::ScopedLock<OpenThreads::Mutex> m_lock(vol_map_mutex);
vol_map.clear();
FILE *volume_config_file = fopen(VOLUME_CONFIG_FILE, "r");
if (!volume_config_file) {
perror(VOLUME_CONFIG_FILE);
return;
}
t_channel_id chan;
int apid = 0;
int volume = 0;
char s[1000];
while (fgets(s, sizeof(s), volume_config_file)) {
if (sscanf(s, "%" SCNx64 " %d %d", &chan, &apid, &volume) == 3)
vol_map.insert(volume_pair_t(chan, pid_pair_t(apid, volume)));
}
fclose(volume_config_file);
}
void CZapit::SaveVolumeMap()
{
OpenThreads::ScopedLock<OpenThreads::Mutex> m_lock(vol_map_mutex);
FILE *volume_config_file = fopen(VOLUME_CONFIG_FILE, "w");
if (!volume_config_file) {
perror(VOLUME_CONFIG_FILE);
return;
}
for (volume_map_iterator_t it = vol_map.begin(); it != vol_map.end(); ++it)
fprintf(volume_config_file, "%" PRIx64 " %d %d\n", (uint64_t) it->first, it->second.first, it->second.second);
fdatasync(fileno(volume_config_file));
fclose(volume_config_file);
}
void CZapit::ClearVolumeMap()
{
OpenThreads::ScopedLock<OpenThreads::Mutex> m_lock(vol_map_mutex);
vol_map.clear();
unlink(VOLUME_CONFIG_FILE);
if (current_channel) {
CZapitAudioChannel *currentAudioChannel = current_channel->getAudioChannel();
if (currentAudioChannel && (currentAudioChannel->audioChannelType == CZapitAudioChannel::AC3 || currentAudioChannel->audioChannelType == CZapitAudioChannel::EAC3)) {
SetVolumePercent(volume_percent_ac3);
return;
}
}
SetVolumePercent(volume_percent_pcm);
}
void CZapit::LoadSettings()
{
if (!configfile.loadConfig(ZAPITCONFIGFILE))
WARN("%s not found", ZAPITCONFIGFILE);
live_channel_id = configfile.getInt64("lastChannel", 0);
lastChannelRadio = configfile.getInt64("lastChannelRadio", 0);
lastChannelTV = configfile.getInt64("lastChannelTV", 0);
last_channel_id = configfile.getInt64("lastOTAChannel", 0);
#if 0 //unused
config.fastZap = configfile.getBool("fastZap", 1);
config.sortNames = configfile.getBool("sortNames", 0);
voltageOff = configfile.getBool("voltageOff", 0);
#endif
config.saveLastChannel = configfile.getBool("saveLastChannel", true);
config.writeChannelsNames = configfile.getInt32("writeChannelsNames", CBouquetManager::BWN_EVER );
/* FIXME Channels renum should be done for all channels atm. TODO*/
//config.makeRemainingChannelsBouquet = configfile.getBool("makeRemainingChannelsBouquet", 1);
config.makeRemainingChannelsBouquet = 1;
config.scanPids = configfile.getBool("scanPids", 0);
config.scanSDT = configfile.getInt32("scanSDT", 0);
config.cam_ci = configfile.getInt32("cam_ci", 2);
config.rezapTimeout = configfile.getInt32("rezapTimeout", 1);
config.feTimeout = configfile.getInt32("feTimeout", 40);
config.feRetries = configfile.getInt32("feRetries", 1);
config.noSameFE = configfile.getInt32("noSameFE", 0);
config.highVoltage = configfile.getBool("highVoltage", 0);
config.gotoXXLatitude = strtod(configfile.getString("gotoXXLatitude", "0.0").c_str(), NULL);
config.gotoXXLongitude = strtod(configfile.getString("gotoXXLongitude", "0.0").c_str(), NULL);
config.gotoXXLaDirection = configfile.getInt32("gotoXXLaDirection", 1);
config.gotoXXLoDirection = configfile.getInt32("gotoXXLoDirection", 0);
config.repeatUsals = configfile.getInt32("repeatUsals", 0);
/* FIXME FE specific, to be removed */
diseqcType = (diseqc_t)configfile.getInt32("diseqcType", NO_DISEQC);
config.motorRotationSpeed = configfile.getInt32("motorRotationSpeed", 18); // default: 1.8 degrees per second
printf("[zapit.cpp] diseqc type = %d\n", diseqcType);
CFEManager::getInstance()->loadSettings();
/**/
LoadAudioMap();
LoadVolumeMap();
}
void CZapit::ConfigFrontend()
{
int count = CFEManager::getInstance()->getFrontendCount();
for(int i = 0; i < count; i++) {
CFrontend * fe = CFEManager::getInstance()->getFE(i);
if(!fe)
continue;
fe->setTimeout(config.feTimeout);
fe->configUsals(config.gotoXXLatitude, config.gotoXXLongitude,
config.gotoXXLaDirection, config.gotoXXLoDirection, config.repeatUsals);
if(!CFEManager::getInstance()->configExist()) {
INFO("New frontend config not exist");
fe->configRotor(config.motorRotationSpeed, config.highVoltage);
fe->setDiseqcType(diseqcType);
fe->setDiseqcRepeats(configfile.getInt32("diseqcRepeats", 0));
fe->setRotorSatellitePosition(configfile.getInt32("lastSatellitePosition", 0));
fe->setSatellites(CServiceManager::getInstance()->SatelliteList());
}
}
}
void CZapit::SendPMT(bool forupdate)
{
if(!current_channel || (!forupdate && playbackStopForced))
return;
CCamManager::getInstance()->Start(current_channel->getChannelID(), CCamManager::PLAY, forupdate);
}
void CZapit::SaveChannelPids(CZapitChannel* channel)
{
if(channel == NULL)
return;
printf("[zapit] saving channel, apid %x sub pid %x mode %d volume %d\n", channel->getAudioPid(), dvbsub_getpid(), audio_mode, current_volume);
audio_map[channel->getChannelID()].apid = channel->getAudioPid();
audio_map[channel->getChannelID()].mode = audio_mode;
audio_map[channel->getChannelID()].volume = current_volume;
audio_map[channel->getChannelID()].subpid = dvbsub_getpid();
tuxtx_subtitle_running(&audio_map[channel->getChannelID()].ttxpid, &audio_map[channel->getChannelID()].ttxpage, NULL);
}
void CZapit::RestoreChannelPids(CZapitChannel * channel)
{
audio_map_set_t * pidmap = GetSavedPids(channel->getChannelID());
if(pidmap) {
printf("[zapit] channel found, audio pid %x, subtitle pid %x mode %d volume %d\n",
pidmap->apid, pidmap->subpid, pidmap->mode, pidmap->volume);
if(channel->getAudioChannelCount() > 1) {
for (int i = 0; i < channel->getAudioChannelCount(); i++) {
if (channel->getAudioChannel(i)->pid == pidmap->apid ) {
DBG("***** Setting audio!\n");
channel->setAudioChannel(i);
}
}
}
audio_mode = pidmap->mode;
dvbsub_setpid(pidmap->subpid);
std::string tmplang;
for (int i = 0 ; i < (int)channel->getSubtitleCount() ; ++i) {
CZapitAbsSub* s = channel->getChannelSub(i);
if(s->pId == pidmap->ttxpid) {
tmplang = s->ISO639_language_code;
break;
}
}
if(tmplang.empty())
tuxtx_set_pid(pidmap->ttxpid, pidmap->ttxpage, (char *) channel->getTeletextLang());
else
tuxtx_set_pid(pidmap->ttxpid, pidmap->ttxpage, (char *) tmplang.c_str());
} else {
audio_mode = def_audio_mode;
tuxtx_set_pid(0, 0, (char *) channel->getTeletextLang());
}
/* restore saved stereo / left / right channel mode */
audioDecoder->setChannel(audio_mode);
}
audio_map_set_t * CZapit::GetSavedPids(const t_channel_id channel_id)
{
audio_map_iterator_t audio_map_it = audio_map.find(channel_id);
if(audio_map_it != audio_map.end())
return &audio_map_it->second;
return NULL;
}
bool CZapit::TuneChannel(CFrontend * frontend, CZapitChannel * channel, bool &transponder_change, bool send_event)
{
if(channel == NULL || frontend == NULL)
return false;
transponder_change = frontend->setInput(channel, current_is_nvod);
if(transponder_change && !current_is_nvod) {
int waitForMotor = frontend->driveToSatellitePosition(channel->getSatellitePosition());
if(waitForMotor > 0) {
printf("[zapit] waiting %d seconds for motor to turn satellite dish.\n", waitForMotor);
if (send_event)
SendEvent(CZapitClient::EVT_ZAP_MOTOR, &waitForMotor, sizeof(waitForMotor));
for(int i = 0; i < waitForMotor; i++) {
sleep(1);
if(abort_zapit) {
abort_zapit = 0;
return false;
}
}
}
}
/* if channel's transponder does not match frontend's tuned transponder ... */
if (transponder_change /* || current_is_nvod*/) {
if (frontend->tuneChannel(channel, current_is_nvod) == false) {
return false;
}
}
return true;
}
bool CZapit::ParsePatPmt(CZapitChannel * channel)
{
if(channel == NULL)
return false;
CPat pat(channel->getRecordDemux());
CPmt pmt(channel->getRecordDemux());
DBG("looking up pids for channel_id " PRINTF_CHANNEL_ID_TYPE "\n", channel->getChannelID());
if(!pat.Parse(channel)) {
printf("[zapit] pat parsing failed\n");
return false;
}
if (!pmt.Parse(channel)) {
printf("[zapit] pmt parsing failed\n");
return false;
}
return true;
}
bool CZapit::ZapIt(const t_channel_id channel_id, bool forupdate, bool startplayback)
{
bool transponder_change = false;
bool failed = false;
CZapitChannel* newchannel;
abort_zapit = 0;
if((newchannel = CServiceManager::getInstance()->FindChannel(channel_id, &current_is_nvod)) == NULL) {
INFO("channel_id " PRINTF_CHANNEL_ID_TYPE " not found", channel_id);
return false;
}
INFO("[zapit] zap to %s (%" PRIx64 " tp %" PRIx64 ")", newchannel->getName().c_str(), newchannel->getChannelID(), newchannel->getTransponderId());
if (!firstzap && current_channel)
SaveChannelPids(current_channel);
/* firstzap right now does nothing but control saving the audio channel */
firstzap = false;
if (IS_WEBCHAN(newchannel->getChannelID()) && !newchannel->getUrl().empty()) {
dvbsub_stop();
if (current_channel && current_channel->getChannelID() == newchannel->getChannelID() && !newchannel->getScriptName().empty()){
INFO("[zapit] stop rezap to channel %s id %" PRIx64 ")", newchannel->getName().c_str(), newchannel->getChannelID());
live_channel_id = newchannel->getChannelID();
SendEvent(CZapitClient::EVT_WEBTV_ZAP_COMPLETE, &live_channel_id, sizeof(t_channel_id));
return true;
}
if (!IS_WEBCHAN(live_channel_id))
CCamManager::getInstance()->Stop(live_channel_id, CCamManager::PLAY);
live_channel_id = newchannel->getChannelID();
lock_channel_id = live_channel_id;
current_channel = newchannel;
if (newchannel->getServiceType() == ST_DIGITAL_RADIO_SOUND_SERVICE)
lastChannelRadio = channel_id;
else
lastChannelTV = channel_id;
SendEvent(CZapitClient::EVT_WEBTV_ZAP_COMPLETE, &live_channel_id, sizeof(t_channel_id));
return true;
}
#ifdef ENABLE_PIP
/* executed async if zap NOWAIT, race possible with record lock/allocate */
CFEManager::getInstance()->Lock();
if (pip_fe[0])
CFEManager::getInstance()->lockFrontend(pip_fe[0]);
CFrontend * fe = CFEManager::getInstance()->allocateFE(newchannel);
if (pip_fe[0])
CFEManager::getInstance()->unlockFrontend(pip_fe[0]);
if (fe == NULL) {
StopPip();
fe = CFEManager::getInstance()->allocateFE(newchannel);
}
CFEManager::getInstance()->Unlock();
#else
CFrontend * fe = CFEManager::getInstance()->allocateFE(newchannel);
#endif
if(fe == NULL) {
ERROR("Cannot get frontend\n");
return false;
}
sig_delay = 2;
pmt_stop_update_filter(&pmt_update_fd);
/* stop playback on the old frontend... */
StopPlayBack(!forupdate);
/* then select the new one... */
live_fe = fe;
CFEManager::getInstance()->setLiveFE(live_fe);
if(!forupdate && current_channel)
current_channel->resetPids();
current_channel = newchannel;
live_channel_id = current_channel->getChannelID();
lock_channel_id = live_channel_id;
last_channel_id = live_channel_id;
SaveSettings(false);
srand(time(NULL));
/* retry tuning, at least twice when using unicable, TODO: EN50494 sect.8 specifies 4 retries... */
int retry = std::max(config.feRetries, (live_fe->getDiseqcType() == DISEQC_UNICABLE) * 2);
again:
if(!TuneChannel(live_fe, newchannel, transponder_change)) {
if (retry < 1) {
t_channel_id chid = 0;
SendEvent(CZapitClient::EVT_TUNE_COMPLETE, &chid, sizeof(t_channel_id));
return false;
}
int rand_us = (rand() * 1000000LL / RAND_MAX); /* max. 1 second delay */
printf("[zapit] %s:1 SCR retry tuning %d after %dms\n", __func__, retry, rand_us / 1000);
/* EN50494 sect.8 specifies an elaborated way of calculating the delay, but I'm
* pretty sure rand() is not much worse :-) */
usleep(rand_us);
live_fe->tuned = false;
retry--;
goto again;
}
SendEvent(CZapitClient::EVT_TUNE_COMPLETE, &live_channel_id, sizeof(t_channel_id));
#ifdef ENABLE_PIP
if (transponder_change && (live_fe == pip_fe[0]))
StopPip();
#endif
#ifdef BOXMODEL_CST_HD2
if (CCamManager::getInstance()->GetCITuner() < 0)
cCA::GetInstance()->SetTS((CA_DVBCI_TS_INPUT)live_fe->getNumber());
#endif
if (current_channel->getServiceType() == ST_NVOD_REFERENCE_SERVICE) {
current_is_nvod = true;
return true;
}
failed = !ParsePatPmt(current_channel);
if (failed && retry > 0) {
int rand_us = (rand() * 1000000LL / RAND_MAX);
printf("[zapit] %s:2 SCR retry tuning %d after %dms\n", __func__, retry, rand_us / 1000);
usleep(rand_us);
live_fe->tuned = false;
retry--;
goto again;
}
if ((!failed) && (current_channel->getAudioPid() == 0) && (current_channel->getVideoPid() == 0)) {
printf("[zapit] neither audio nor video pid found\n");
failed = true;
}
/* start sdt scan even if the service was not found in pat or pmt
* if the frontend did not tune, we don't get here, so this is fine */
if (transponder_change)
SdtMonitor.Wakeup();
if (failed)
return false;
//current_channel->getCaPmt()->ca_pmt_list_management = transponder_change ? 0x03 : 0x04;
RestoreChannelPids(current_channel);
if (startplayback /* && !we_playing*/)
StartPlayBack(current_channel);
//printf("[zapit] sending capmt....\n");
SendPMT(forupdate);
//play:
int caid = 1;
SendEvent(CZapitClient::EVT_ZAP_CA_ID, &caid, sizeof(int));
if (update_pmt)
pmt_set_update_filter(current_channel, &pmt_update_fd);
#if ENABLE_AIT
ait->setDemux(current_channel->getRecordDemux());
ait->Parse(current_channel);
#endif
return true;
}
#ifdef ENABLE_PIP
bool CZapit::StopPip(int pip)
{
if (!g_info.hw_caps->can_pip)
return false;
#if !HAVE_CST_HARDWARE && !HAVE_GENERIC_HARDWARE
if (CNeutrinoApp::getInstance()->avinput_pip) {
CNeutrinoApp::getInstance()->StopAVInputPiP();
}
#endif
if (pip_channel_id[pip]) {
INFO("[pip] stop %llx", pip_channel_id[pip]);
#if !HAVE_CST_HARDWARE
pipVideoDecoder[pip]->ShowPig(0);
#endif
CCamManager::getInstance()->Stop(pip_channel_id[pip], CCamManager::PIP);
pipVideoDemux[pip]->Stop();
pipVideoDecoder[pip]->Stop();
pipAudioDemux[pip]->Stop();
pipAudioDecoder[pip]->Stop();
pip_fe[pip] = NULL;
pip_channel_id[pip] = 0;
return true;
}
return false;
}
bool CZapit::StartPip(const t_channel_id channel_id, int pip)
{
if (!g_info.hw_caps->can_pip)
return false;
CZapitChannel* newchannel;
bool transponder_change;
/* do lock if live is running, or in record mode -
this is for the case temporary timeshift is running, it do not lock its frontend */
bool need_lock = !playbackStopForced || (currentMode & RECORD_MODE);
if((newchannel = CServiceManager::getInstance()->FindChannel(channel_id)) == NULL) {
INFO("channel_id " PRINTF_CHANNEL_ID_TYPE " not found", channel_id);
return false;
}
INFO("[pip] zap to %s (%llx tp %llx)", newchannel->getName().c_str(), newchannel->getChannelID(), newchannel->getTransponderId());
if (need_lock)
CFEManager::getInstance()->lockFrontend(live_fe);
CFrontend * frontend = CFEManager::getInstance()->allocateFE(newchannel);
if (need_lock)
CFEManager::getInstance()->unlockFrontend(live_fe);
if(frontend == NULL) {
ERROR("Cannot get frontend\n");
return false;
}
StopPip(pip);
if (!need_lock && !SAME_TRANSPONDER(newchannel->getChannelID(), live_channel_id))
live_channel_id = newchannel->getChannelID();
if(!TuneChannel(frontend, newchannel, transponder_change))
return false;
if(!ParsePatPmt(newchannel))
return false;
pip_fe[pip] = frontend;
#if HAVE_CST_HARDWARE
INFO("[pip] vpid %X apid %X pcr %X", newchannel->getVideoPid(), newchannel->getAudioPid(), newchannel->getPcrPid());
/* FIXME until proper demux management */
int dnum = newchannel->getPipDemux();
if (pipVideoDemux[0] && (pipVideoDemux[0]->getUnit() != dnum)) {
pipVideoDecoder[0]->SetDemux(NULL);
delete pipVideoDemux[0];
pipVideoDemux[0] = NULL;
}
if (!pipVideoDemux[0]) {
pipVideoDemux[0] = new cDemux(dnum);
pipVideoDemux[0]->Open(DMX_PIP_CHANNEL);
pipVideoDecoder[0]->SetDemux(pipVideoDemux[0]);
}
if (CFEManager::getInstance()->getFrontendCount() > 1)
cDemux::SetSource(dnum, pip_fe[pip]->getNumber());
#else
#ifdef DYNAMIC_DEMUX
int dnum = CFEManager::getInstance()->getDemux(newchannel->getTransponderId(), pip_fe[pip]->getNumber());
INFO("[pip] dyn demux: %d", dnum);
#else
/* FIXME until proper demux management */
int dnum = pip + 1;
INFO("[pip] demux: %d", dnum);
#endif
INFO("[pip] vpid %X apid %X pcr %X", newchannel->getVideoPid(), newchannel->getAudioPid(), newchannel->getPcrPid());
if (!pipVideoDemux[pip]) {
pipVideoDemux[pip] = new cDemux(dnum);
pipVideoDemux[pip]->Open(DMX_VIDEO_CHANNEL);
if (!pipVideoDecoder[pip]) {
pipVideoDecoder[pip] = new cVideo(0, NULL, NULL, dnum);
}
}
pipVideoDemux[pip]->SetSource(dnum, pip_fe[pip]->getNumber());
newchannel->setPipDemux(dnum);
newchannel->setRecordDemux(pip_fe[pip]->getNumber());
#endif
pipVideoDecoder[pip]->SetStreamType((VIDEO_FORMAT) newchannel->type);
pipVideoDemux[pip]->pesFilter(newchannel->getVideoPid());
#if HAVE_CST_HARDWARE
pipVideoDecoder[pip]->Start(0, newchannel->getPcrPid(), newchannel->getVideoPid());
pipVideoDemux[pip]->Start();
pip_channel_id[pip] = channel_id;
#else
pipVideoDemux[pip]->Start();
pipVideoDecoder[pip]->Start(0, newchannel->getPcrPid(), newchannel->getVideoPid());
pip_channel_id[pip] = newchannel->getChannelID();
pipVideoDecoder[pip]->Pig(g_settings.pip_x,g_settings.pip_y,g_settings.pip_width,g_settings.pip_height,g_settings.screen_width,g_settings.screen_height);
pipVideoDecoder[pip]->ShowPig(1);
#endif
if (!pipAudioDemux[pip]) {
pipAudioDemux[pip] = new cDemux(dnum);
pipAudioDemux[pip]->Open(DMX_AUDIO_CHANNEL);
if (!pipAudioDecoder[pip]) {
pipAudioDecoder[pip] = new cAudio(0, NULL, NULL, dnum);
}
}
if (newchannel->getAudioChannel())
pipAudioDecoder[pip]->SetStreamType(newchannel->getAudioChannel()->audioChannelType);
pipAudioDemux[pip]->pesFilter(newchannel->getAudioPid());
pipVideoDecoder[pip]->SetSyncMode((AVSYNC_TYPE) g_settings.avsync);
pipAudioDecoder[pip]->SetSyncMode((AVSYNC_TYPE) g_settings.avsync);
CCamManager::getInstance()->Start(newchannel->getChannelID(), CCamManager::PIP);
return true;
}
#endif
bool CZapit::ZapForRecord(const t_channel_id channel_id)
{
CZapitChannel* newchannel;
bool transponder_change;
/* retry tuning, at least twice when using unicable */
int retry = std::max(config.feRetries, (live_fe->getDiseqcType() == DISEQC_UNICABLE) * 2);
if((newchannel = CServiceManager::getInstance()->FindChannel(channel_id)) == NULL) {
INFO("channel_id " PRINTF_CHANNEL_ID_TYPE " not found", channel_id);
return false;
}
INFO("%s: %s (%" PRIx64 ")", __FUNCTION__, newchannel->getName().c_str(), channel_id);
CFrontend * frontend = CFEManager::getInstance()->allocateFE(newchannel);
if(frontend == NULL) {
ERROR("Cannot get frontend\n");
return false;
}
again:
if(!TuneChannel(frontend, newchannel, transponder_change))
{
if (retry < 1)
return false;
printf("[zapit] %s:1 unicable retry tuning %d\n", __func__, retry);
live_fe->tuned = false;
retry--;
goto again;
}
#ifdef ENABLE_PIP
if (transponder_change && (frontend == pip_fe[0]))
StopPip();
#endif
if(!ParsePatPmt(newchannel))
{
if (retry < 1)
return false;
printf("[zapit] %s:2 unicable retry tuning %d\n", __func__, retry);
live_fe->tuned = false;
retry--;
goto again;
}
return true;
}
bool CZapit::ZapForEpg(const t_channel_id channel_id, bool instandby)
{
CZapitChannel* newchannel;
bool transponder_change;
if((newchannel = CServiceManager::getInstance()->FindChannel(channel_id)) == NULL) {
INFO("channel_id " PRINTF_CHANNEL_ID_TYPE " not found", channel_id);
return false;
}
INFO("%s: %s (%" PRIx64 ")", __FUNCTION__, newchannel->getName().c_str(), channel_id);
/* executed async in zapit thread, race possible with record lock/allocate */
CFEManager::getInstance()->Lock();
/* no need to lock fe in standby mode,
epg scan should care to not call this if recording running */
if (!instandby) {
if (!IS_WEBCHAN(live_channel_id))
CFEManager::getInstance()->lockFrontend(live_fe);
#ifdef ENABLE_PIP
if (pip_fe[0] /* && pip_fe[0] != live_fe */)
CFEManager::getInstance()->lockFrontend(pip_fe[0]);
#endif
}
CFrontend * frontend = CFEManager::getInstance()->allocateFE(newchannel);
if (!instandby) {
if (!IS_WEBCHAN(live_channel_id))
CFEManager::getInstance()->unlockFrontend(live_fe);
#ifdef ENABLE_PIP
if (pip_fe[0] /* && pip_fe[0] != live_fe */)
CFEManager::getInstance()->unlockFrontend(pip_fe[0]);
#endif
}
CFEManager::getInstance()->Unlock();
if(frontend == NULL) {
ERROR("Cannot get frontend\n");
return false;
}
/* sort of hack. epg scan in standby can use any frontend,
and break check in record code ie */
if (instandby)
live_channel_id = newchannel->getChannelID();
if(!TuneChannel(frontend, newchannel, transponder_change, false))
return false;
#if 0
if(!ParsePatPmt(newchannel))
return false;
#endif
ParsePatPmt(newchannel);
return true;
}
/* set channel/pid volume percent, using current channel_id and pid, if those params is 0 */
void CZapit::SetPidVolume(t_channel_id channel_id, int pid, int percent)
{
if (!channel_id)
channel_id = live_channel_id;
if ((pid < 0) && (channel_id == live_channel_id) && current_channel)
pid = current_channel->getAudioPid();
OpenThreads::ScopedLock<OpenThreads::Mutex> m_lock(vol_map_mutex);
INFO("############################### channel %" PRIx64 " pid %x map size %d percent %d", channel_id, pid, (int)vol_map.size(), percent);
volume_map_range_t pids = vol_map.equal_range(channel_id);
for (volume_map_iterator_t it = pids.first; it != pids.second; ++it) {
if (it->second.first == pid) {
it->second.second = percent;
return;
}
}
vol_map.insert(volume_pair_t(channel_id, pid_pair_t(pid, percent)));
}
/* return channel/pid volume percent, using current channel_id and pid, if those params is 0 */
int CZapit::GetPidVolume(t_channel_id channel_id, int pid, bool ac3)
{
int percent = -1;
if (!channel_id)
channel_id = live_channel_id;
if ((pid < 0) && (channel_id == live_channel_id) && current_channel)
pid = current_channel->getAudioPid();
OpenThreads::ScopedLock<OpenThreads::Mutex> m_lock(vol_map_mutex);
volume_map_range_t pids = vol_map.equal_range(channel_id);
for (volume_map_iterator_t it = pids.first; it != pids.second; ++it) {
if (it->second.first == pid) {
percent = it->second.second;
break;
}
}
if (percent < 0) {
percent = ac3 ? volume_percent_ac3 : volume_percent_pcm;
if ((channel_id == live_channel_id) && current_channel) {
for (int i = 0; i < current_channel->getAudioChannelCount(); i++) {
if (pid == current_channel->getAudioPid(i)) {
percent = (current_channel->getAudioChannel(i)->audioChannelType == CZapitAudioChannel::AC3
|| current_channel->getAudioChannel(i)->audioChannelType == CZapitAudioChannel::EAC3) ?
volume_percent_ac3 : volume_percent_pcm;
break;
}
}
}
}
DBG("channel %" PRIx64 " pid %x map size %d percent %d", channel_id, pid, (int)vol_map.size(), percent);
return percent;
}
void CZapit::SetVolume(int vol)
{
current_volume = vol;
if (current_volume < 0)
current_volume = 0;
if (current_volume > 100)
current_volume = 100;
int value = (current_volume * volume_percent) / 100;
DBG("volume %d percent %d -> %d", current_volume, volume_percent, value);
audioDecoder->setVolume(value, value);
//volume_left = volume_right = current_volume;
}
int CZapit::SetVolumePercent(int percent)
{
int ret = volume_percent;
if (volume_percent != percent) {
volume_percent = percent;
if (!audioDecoder->getMuteStatus())
SetVolume(current_volume);
}
return ret;
}
void CZapit::SetVolumePercent(int default_ac3, int default_pcm)
{
volume_percent_ac3 = default_ac3;
volume_percent_pcm = default_pcm;
}
void CZapit::SetAudioStreamType(CZapitAudioChannel::ZapitAudioChannelType audioChannelType)
{
const char *audioStr = "UNKNOWN";
#if HAVE_CST_HARDWARE
//TODO: align libcoolstream audioDecoder->SetStreamType()
switch (audioChannelType) {
case CZapitAudioChannel::AC3:
audioStr = "AC3";
audioDecoder->SetStreamType(AUDIO_FMT_DOLBY_DIGITAL);
break;
case CZapitAudioChannel::MPEG:
audioStr = "MPEG2";
audioDecoder->SetStreamType(AUDIO_FMT_MPEG);
break;
case CZapitAudioChannel::AAC:
audioStr = "AAC";
audioDecoder->SetStreamType(AUDIO_FMT_AAC);
break;
case CZapitAudioChannel::AACPLUS:
audioStr = "AAC-HE";
audioDecoder->SetStreamType(AUDIO_FMT_AAC_PLUS);
break;
case CZapitAudioChannel::DTS:
audioStr = "DTS";
audioDecoder->SetStreamType(AUDIO_FMT_DTS);
break;
case CZapitAudioChannel::EAC3:
audioStr = "EAC3";
audioDecoder->SetStreamType(AUDIO_FMT_DD_PLUS);
break;
default:
printf("[zapit] unknown audio channel type 0x%x\n", audioChannelType);
break;
}
#else
switch (audioChannelType) {
case CZapitAudioChannel::AC3:
audioStr = "AC3";
break;
case CZapitAudioChannel::MPEG:
audioStr = "MPEG2";
break;
case CZapitAudioChannel::AAC:
audioStr = "AAC";
break;
case CZapitAudioChannel::AACPLUS:
audioStr = "AAC-HE";
break;
case CZapitAudioChannel::DTS:
audioStr = "DTS";
break;
case CZapitAudioChannel::DTSHD:
audioStr = "DTSHD";
break;
case CZapitAudioChannel::EAC3:
audioStr = "EAC3";
break;
case CZapitAudioChannel::LPCM:
audioStr = "LPCM";
break;
default:
printf("[zapit] unknown audio channel type 0x%x\n", audioChannelType);
break;
}
audioDecoder->SetStreamType(audioChannelType);
#endif
/* FIXME: bigger percent for AC3 only, what about AAC etc ? */
int newpercent = GetPidVolume(0, 0, audioChannelType == CZapitAudioChannel::AC3 || audioChannelType == CZapitAudioChannel::EAC3);
SetVolumePercent(newpercent);
DBG("[zapit] starting %s audio\n", audioStr);
}
bool CZapit::ChangeAudioPid(uint8_t index)
{
if (!current_channel)
return false;
/* stop demux filter */
if (audioDemux->Stop() == false)
return false;
/* stop audio playback */
if (audioDecoder->Stop() < 0)
return false;
/* update current channel */
current_channel->setAudioChannel(index);
/* set bypass mode */
CZapitAudioChannel *currentAudioChannel = current_channel->getAudioChannel();
if (!currentAudioChannel) {
WARN("No current audio channel");
return false;
}
printf("[zapit] change apid to 0x%x\n", current_channel->getAudioPid());
SetAudioStreamType(currentAudioChannel->audioChannelType);
/* set demux filter */
if (audioDemux->pesFilter(current_channel->getAudioPid()) == false)
return false;
/* start demux filter */
if (audioDemux->Start() == false)
return false;
/* start audio playback */
if (audioDecoder->Start() < 0)
return false;
return true;
}
void CZapit::SetRadioMode(void)
{
currentMode |= RADIO_MODE;
currentMode &= ~TV_MODE;
}
void CZapit::SetTVMode(void)
{
currentMode |= TV_MODE;
currentMode &= ~RADIO_MODE;
}
int CZapit::getMode(void)
{
if (currentMode & TV_MODE)
return CZapitClient::MODE_TV;
if (currentMode & RADIO_MODE)
return CZapitClient::MODE_RADIO;
return 0;
}
void CZapit::SetRecordMode(bool enable)
{
unsigned int event;
bool mode = currentMode & RECORD_MODE;
if(mode == enable)
return;
if(enable) {
currentMode |= RECORD_MODE;
event = CZapitClient::EVT_RECORDMODE_ACTIVATED;
} else {
currentMode &= ~RECORD_MODE;
event = CZapitClient::EVT_RECORDMODE_DEACTIVATED;
}
SendEvent(event);
}
bool CZapit::PrepareChannels()
{
current_channel = 0;
if (!CServiceManager::getInstance()->LoadServices(false))
return false;
INFO("LoadServices: success");
g_bouquetManager->loadBouquets();
/* save if services changed (update from sdt, etc) */
CServiceManager::getInstance()->SaveServices(true, true);
return true;
}
void CZapit::PrepareScan()
{
StopPlayBack(true);
pmt_stop_update_filter(&pmt_update_fd);
current_channel = 0;
}
bool CZapit::StartScan(int scan_mode)
{
PrepareScan();
CServiceScan::getInstance()->Start(CServiceScan::SCAN_PROVIDER, &scan_mode);
return true;
}
bool CZapit::StartScanTP(TP_params * TPparams)
{
PrepareScan();
CServiceScan::getInstance()->Start(CServiceScan::SCAN_TRANSPONDER, (void *) TPparams);
return true;
}
bool CZapit::StartFastScan(int scan_mode, int opid)
{
scant.type = scan_mode;
scant.op = (fs_operator_t) opid;
PrepareScan();
CServiceScan::getInstance()->Start(CServiceScan::SCAN_FAST, (void *) &scant);
return true;
}
void CZapit::SendCmdReady(int connfd)
{
CZapitMessages::responseCmd response;
VALGRIND_PARANOIA(response);
response.cmd = CZapitMessages::CMD_READY;
CBasicServer::send_data(connfd, &response, sizeof(response));
}
void CZapit::lockPlayBack(const bool sendpmt)
{
/* hack. if standby true, dont blank video */
standby = true;
StopPlayBack(sendpmt);
standby = false;
playbackStopForced = true;
lock_channel_id = live_channel_id;
}
void CZapit::unlockPlayBack(const bool /*sendpmt*/)
{
playbackStopForced = false;
if (lock_channel_id == live_channel_id) {
StartPlayBack(current_channel);
SendPMT();
} else {
live_fe->setTsidOnid(0);
if (!ZapIt(lock_channel_id))
SendEvent(CZapitClient::EVT_ZAP_FAILED, &lock_channel_id, sizeof(lock_channel_id));
lock_channel_id = 0;
}
}
void CZapit::Rezap(void)
{
if (currentMode & RECORD_MODE)
return;
if(config.rezapTimeout > 0)
sleep(config.rezapTimeout);
if(current_channel)
ZapIt(current_channel->getChannelID());
}
bool CZapit::ParseCommand(CBasicMessage::Header &rmsg, int connfd)
{
DBG("cmd %d (version %d) received\n", rmsg.cmd, rmsg.version);
if ((standby) && ((rmsg.cmd != CZapitMessages::CMD_SET_VOLUME)
&& (rmsg.cmd != CZapitMessages::CMD_MUTE)
&& (rmsg.cmd != CZapitMessages::CMD_REGISTEREVENTS)
&& (rmsg.cmd != CZapitMessages::CMD_IS_TV_CHANNEL)
&& (rmsg.cmd != CZapitMessages::CMD_SET_STANDBY))) {
WARN("cmd %d in standby mode", rmsg.cmd);
//return true;
}
switch (rmsg.cmd) {
case CZapitMessages::CMD_SHUTDOWN:
return false;
case CZapitMessages::CMD_ZAPTO:
{
CZapitMessages::commandZapto msgZapto;
CBasicServer::receive_data(connfd, &msgZapto, sizeof(msgZapto)); // bouquet & channel number are already starting at 0!
ZapTo(msgZapto.bouquet, msgZapto.channel);
break;
}
case CZapitMessages::CMD_ZAPTO_CHANNELNR: {
CZapitMessages::commandZaptoChannelNr msgZaptoChannelNr;
CBasicServer::receive_data(connfd, &msgZaptoChannelNr, sizeof(msgZaptoChannelNr)); // bouquet & channel number are already starting at 0!
ZapTo(msgZaptoChannelNr.channel);
break;
}
case CZapitMessages::CMD_ZAPTO_SERVICEID:
case CZapitMessages::CMD_ZAPTO_SUBSERVICEID: {
CZapitMessages::commandZaptoServiceID msgZaptoServiceID;
CZapitMessages::responseZapComplete msgResponseZapComplete;
VALGRIND_PARANOIA(msgResponseZapComplete);
CBasicServer::receive_data(connfd, &msgZaptoServiceID, sizeof(msgZaptoServiceID));
if(msgZaptoServiceID.record)
msgResponseZapComplete.zapStatus = ZapForRecord(msgZaptoServiceID.channel_id);
#ifdef ENABLE_PIP
else if(msgZaptoServiceID.pip)
msgResponseZapComplete.zapStatus = StartPip(msgZaptoServiceID.channel_id, msgZaptoServiceID.pip_dev);
#endif
else
msgResponseZapComplete.zapStatus = ZapTo(msgZaptoServiceID.channel_id, (rmsg.cmd == CZapitMessages::CMD_ZAPTO_SUBSERVICEID));
CBasicServer::send_data(connfd, &msgResponseZapComplete, sizeof(msgResponseZapComplete));
break;
}
case CZapitMessages::CMD_ZAPTO_EPG:
{
CZapitMessages::responseZapComplete msgResponseZapComplete;
CZapitMessages::commandZaptoEpg msg;
CBasicServer::receive_data(connfd, &msg, sizeof(msg));
msgResponseZapComplete.zapStatus = 0;
CBasicServer::send_data(connfd, &msgResponseZapComplete, sizeof(msgResponseZapComplete));
bool ret = ZapForEpg(msg.channel_id, msg.standby);
if (!ret)
msg.channel_id = 0;
SendEvent(CZapitClient::EVT_BACK_ZAP_COMPLETE, &msg.channel_id, sizeof(t_channel_id));
break;
}
case CZapitMessages::CMD_ZAPTO_SERVICEID_NOWAIT:
case CZapitMessages::CMD_ZAPTO_SUBSERVICEID_NOWAIT: {
CZapitMessages::commandZaptoServiceID msgZaptoServiceID;
CBasicServer::receive_data(connfd, &msgZaptoServiceID, sizeof(msgZaptoServiceID));
ZapTo(msgZaptoServiceID.channel_id, (rmsg.cmd == CZapitMessages::CMD_ZAPTO_SUBSERVICEID_NOWAIT));
break;
}
#if 0
case CZapitMessages::CMD_GET_LAST_CHANNEL: {
CZapitClient::responseGetLastChannel lastchannel;
lastchannel.channel_id = (currentMode & RADIO_MODE) ? lastChannelRadio : lastChannelTV;
lastchannel.mode = getMode();
CBasicServer::send_data(connfd, &lastchannel, sizeof(lastchannel)); // bouquet & channel number are already starting at 0!
ERROR("CZapitMessages::CMD_GET_LAST_CHANNEL: depreciated command\n");
break;
}
#endif
#if 0
case CZapitMessages::CMD_GET_CURRENT_SATELLITE_POSITION: {
int32_t currentSatellitePosition = current_channel ? current_channel->getSatellitePosition() : live_fe->getCurrentSatellitePosition();
CBasicServer::send_data(connfd, &currentSatellitePosition, sizeof(currentSatellitePosition));
break;
}
#endif
case CZapitMessages::CMD_SET_AUDIOCHAN: {
CZapitMessages::commandSetAudioChannel msgSetAudioChannel;
CBasicServer::receive_data(connfd, &msgSetAudioChannel, sizeof(msgSetAudioChannel));
ChangeAudioPid(msgSetAudioChannel.channel);
break;
}
case CZapitMessages::CMD_SET_MODE: {
CZapitMessages::commandSetMode msgSetMode;
CBasicServer::receive_data(connfd, &msgSetMode, sizeof(msgSetMode));
if (msgSetMode.mode == CZapitClient::MODE_TV)
SetTVMode();
else if (msgSetMode.mode == CZapitClient::MODE_RADIO)
SetRadioMode();
break;
}
case CZapitMessages::CMD_GET_MODE: {
CZapitMessages::responseGetMode msgGetMode;
VALGRIND_PARANOIA(msgGetMode);
msgGetMode.mode = (CZapitClient::channelsMode) getMode();
CBasicServer::send_data(connfd, &msgGetMode, sizeof(msgGetMode));
break;
}
case CZapitMessages::CMD_GET_CURRENT_SERVICEID: {
CZapitMessages::responseGetCurrentServiceID msgCurrentSID;
VALGRIND_PARANOIA(msgCurrentSID);
msgCurrentSID.channel_id = (current_channel != 0) ? current_channel->getChannelID() : 0;
CBasicServer::send_data(connfd, &msgCurrentSID, sizeof(msgCurrentSID));
break;
}
case CZapitMessages::CMD_GET_CURRENT_SERVICEINFO: {
CZapitClient::CCurrentServiceInfo msgCurrentServiceInfo;
VALGRIND_PARANOIA(msgCurrentServiceInfo);
memset(&msgCurrentServiceInfo, 0, sizeof(CZapitClient::CCurrentServiceInfo));
if(current_channel) {
msgCurrentServiceInfo.onid = current_channel->getOriginalNetworkId();
msgCurrentServiceInfo.sid = current_channel->getServiceId();
msgCurrentServiceInfo.tsid = current_channel->getTransportStreamId();
msgCurrentServiceInfo.vpid = current_channel->getVideoPid();
msgCurrentServiceInfo.apid = current_channel->getAudioPid();
msgCurrentServiceInfo.vtxtpid = current_channel->getTeletextPid();
msgCurrentServiceInfo.pmtpid = current_channel->getPmtPid();
//msgCurrentServiceInfo.pmt_version = (current_channel->getCaPmt() != NULL) ? current_channel->getCaPmt()->version_number : 0xff;
msgCurrentServiceInfo.pmt_version = current_channel->getPmtVersion();
msgCurrentServiceInfo.pcrpid = current_channel->getPcrPid();
msgCurrentServiceInfo.tsfrequency = live_fe->getFrequency();
msgCurrentServiceInfo.rate = live_fe->getRate();
msgCurrentServiceInfo.fec = live_fe->getCFEC();
msgCurrentServiceInfo.vtype = current_channel->type;
//msgCurrentServiceInfo.diseqc = current_channel->getDiSEqC();
}
if(!msgCurrentServiceInfo.fec)
msgCurrentServiceInfo.fec = (fe_code_rate)3;
if (CFrontend::isSat(live_fe->getCurrentDeliverySystem()))
msgCurrentServiceInfo.polarisation = live_fe->getPolarization();
else
msgCurrentServiceInfo.polarisation = 2;
CBasicServer::send_data(connfd, &msgCurrentServiceInfo, sizeof(msgCurrentServiceInfo));
break;
}
case CZapitMessages::CMD_GET_DELIVERY_SYSTEM: {
CZapitMessages::responseDeliverySystem response;
VALGRIND_PARANOIA(response);
response.system = live_fe->getCurrentDeliverySystem();
CBasicServer::send_data(connfd, &response, sizeof(response));
break;
}
case CZapitMessages::CMD_GET_BOUQUETS: {
CZapitMessages::commandGetBouquets msgGetBouquets;
CBasicServer::receive_data(connfd, &msgGetBouquets, sizeof(msgGetBouquets));
sendBouquets(connfd, msgGetBouquets.emptyBouquetsToo, msgGetBouquets.mode); // bouquet & channel number are already starting at 0!
break;
}
case CZapitMessages::CMD_GET_BOUQUET_CHANNELS: {
CZapitMessages::commandGetBouquetChannels msgGetBouquetChannels;
CBasicServer::receive_data(connfd, &msgGetBouquetChannels, sizeof(msgGetBouquetChannels));
sendBouquetChannels(connfd, msgGetBouquetChannels.bouquet, msgGetBouquetChannels.mode, false); // bouquet & channel number are already starting at 0!
break;
}
#if 0
case CZapitMessages::CMD_GET_BOUQUET_NCHANNELS: {
CZapitMessages::commandGetBouquetChannels msgGetBouquetChannels;
CBasicServer::receive_data(connfd, &msgGetBouquetChannels, sizeof(msgGetBouquetChannels));
sendBouquetChannels(connfd, msgGetBouquetChannels.bouquet, msgGetBouquetChannels.mode, true); // bouquet & channel number are already starting at 0!
break;
}
#endif
case CZapitMessages::CMD_GET_CHANNELS: {
CZapitMessages::commandGetChannels msgGetChannels;
CBasicServer::receive_data(connfd, &msgGetChannels, sizeof(msgGetChannels));
sendChannels(connfd, msgGetChannels.mode, msgGetChannels.order); // bouquet & channel number are already starting at 0!
break;
}
case CZapitMessages::CMD_GET_CHANNEL_NAME: {
t_channel_id requested_channel_id;
CZapitMessages::responseGetChannelName response;
VALGRIND_PARANOIA(response);
CBasicServer::receive_data(connfd, &requested_channel_id, sizeof(requested_channel_id));
response.name[0] = 0;
CZapitChannel * channel = (requested_channel_id == 0) ? current_channel :
CServiceManager::getInstance()->FindChannel(requested_channel_id);
if(channel) {
strncpy(response.name, channel->getName().c_str(), CHANNEL_NAME_SIZE-1);
response.name[CHANNEL_NAME_SIZE-1] = 0;
}
CBasicServer::send_data(connfd, &response, sizeof(response));
break;
}
#if 0
case CZapitMessages::CMD_IS_TV_CHANNEL: {
t_channel_id requested_channel_id;
CZapitMessages::responseGeneralTrueFalse response;
CBasicServer::receive_data(connfd, &requested_channel_id, sizeof(requested_channel_id));
/* if in doubt (i.e. unknown channel) answer yes */
response.status = true;
CZapitChannel * channel = CServiceManager::getInstance()->FindChannel(requested_channel_id);
if(channel)
/* FIXME: the following check is no even remotely accurate */
response.status = (channel->getServiceType() != ST_DIGITAL_RADIO_SOUND_SERVICE);
CBasicServer::send_data(connfd, &response, sizeof(response));
break;
}
#endif
case CZapitMessages::CMD_BQ_RESTORE: {
//2004.08.02 g_bouquetManager->restoreBouquets();
if(list_changed) {
PrepareChannels();
list_changed = 0;
} else {
g_bouquetManager->clearAll();
g_bouquetManager->loadBouquets();
}
SendCmdReady(connfd);
break;
}
case CZapitMessages::CMD_REINIT_CHANNELS: {
// Houdini: save actual channel to restore it later, old version's channel was set to scans.conf initial channel
t_channel_id cid= current_channel ? current_channel->getChannelID() : 0;
PrepareChannels();
current_channel = CServiceManager::getInstance()->FindChannel(cid);
ParsePatPmt(current_channel);//reinit pids
SendCmdReady(connfd);
SendEvent(CZapitClient::EVT_SERVICES_CHANGED);
break;
}
case CZapitMessages::CMD_RELOAD_CURRENTSERVICES: {
#if 0
SendCmdReady(connfd);
#endif
DBG("[zapit] sending EVT_SERVICES_CHANGED\n");
SendEvent(CZapitClient::EVT_SERVICES_CHANGED);
live_fe->setTsidOnid(0);
//ZapIt(live_channel_id, current_is_nvod);
//SendEvent(CZapitClient::EVT_BOUQUETS_CHANGED);
break;
}
case CZapitMessages::CMD_SCANSTART: {
int scan_mode;
CBasicServer::receive_data(connfd, &scan_mode, sizeof(scan_mode));
if (!StartScan(scan_mode))
SendEvent(CZapitClient::EVT_SCAN_FAILED);
break;
}
case CZapitMessages::CMD_SCANSTOP:
CServiceScan::getInstance()->Abort();
CServiceScan::getInstance()->Stop();
break;
#if 0
case CZapitMessages::CMD_SETCONFIG:
Zapit_config Cfg;
CBasicServer::receive_data(connfd, &Cfg, sizeof(Cfg));
SetConfig(&Cfg);
break;
case CZapitMessages::CMD_GETCONFIG:
SendConfig(connfd);
break;
#endif
case CZapitMessages::CMD_REZAP:
#if 0
if (currentMode & RECORD_MODE)
break;
if(config.rezapTimeout > 0)
sleep(config.rezapTimeout);
if(current_channel)
ZapIt(current_channel->getChannelID());
#endif
Rezap();
break;
case CZapitMessages::CMD_TUNE_TP: {
CBasicServer::receive_data(connfd, &TP, sizeof(TP));
sig_delay = 0;
TP.feparams.inversion = INVERSION_AUTO;
const char *name = scanProviders.empty() ? "unknown" : scanProviders.begin()->second.c_str();
if (CFrontend::isSat(TP.feparams.delsys)) {
//FIXME check scanProviders.size() !
t_satellite_position satellitePosition = scanProviders.begin()->first;
printf("[zapit] tune to sat %s freq %d rate %d fec %d pol %d\n", name, TP.feparams.frequency, TP.feparams.symbol_rate, TP.feparams.fec_inner, TP.feparams.polarization);
live_fe->setInput(satellitePosition, TP.feparams.frequency, TP.feparams.polarization);
live_fe->driveToSatellitePosition(satellitePosition);
} else if (CFrontend::isCable(TP.feparams.delsys)) {
printf("[zapit] tune to cable %s freq %d rate %d fec %d\n", name, TP.feparams.frequency, TP.feparams.symbol_rate, TP.feparams.fec_inner);
} else if (CFrontend::isTerr(TP.feparams.delsys)) {
printf("[zapit] tune to terr %s freq %d bw %d fec %d\n", name, TP.feparams.frequency, TP.feparams.bandwidth, TP.feparams.modulation);
} else {
WARN("Unknown type %d", TP.feparams.delsys);
return false;
}
live_fe->tuneFrequency(&TP.feparams, true);
}
break;
case CZapitMessages::CMD_SCAN_TP: {
CBasicServer::receive_data(connfd, &TP, sizeof(TP));
StartScanTP(&TP);
break;
}
case CZapitMessages::CMD_SCANREADY: {
CZapitMessages::responseIsScanReady msgResponseIsScanReady;
VALGRIND_PARANOIA(msgResponseIsScanReady);
#if 0 //FIXME used only when scanning done using pzapit client, is it really needed ?
msgResponseIsScanReady.satellite = curr_sat;
msgResponseIsScanReady.transponder = found_transponders;
msgResponseIsScanReady.processed_transponder = processed_transponders;
msgResponseIsScanReady.services = found_channels;
msgResponseIsScanReady.scanReady = !CServiceScan::getInstance()->Scanning();
#endif
CBasicServer::send_data(connfd, &msgResponseIsScanReady, sizeof(msgResponseIsScanReady));
break;
}
case CZapitMessages::CMD_SCANGETSATLIST: {
uint32_t satlength;
CZapitClient::responseGetSatelliteList sat;
VALGRIND_PARANOIA(sat);
satlength = sizeof(sat);
satellite_map_t satmap = CServiceManager::getInstance()->SatelliteList();
for(sat_iterator_t sit = satmap.begin(); sit != satmap.end(); sit++) {
strncpy(sat.satName, sit->second.name.c_str(), 49);
sat.satName[49] = 0;
sat.satPosition = sit->first;
sat.motorPosition = sit->second.motor_position;
CBasicServer::send_data(connfd, &satlength, sizeof(satlength));
CBasicServer::send_data(connfd, (char *)&sat, satlength);
}
satlength = SATNAMES_END_MARKER;
CBasicServer::send_data(connfd, &satlength, sizeof(satlength));
break;
}
case CZapitMessages::CMD_SCANSETSCANSATLIST: {
bool setfe = true;
CZapitClient::commandSetScanSatelliteList sat;
scanProviders.clear();
while (CBasicServer::receive_data(connfd, &sat, sizeof(sat))) {
printf("[zapit] adding to scan %s (position %d)\n", sat.satName, sat.position);
scanProviders[sat.position] = sat.satName;
if(setfe) {
/* for test signal */
CServiceScan::getInstance()->SetFrontend(sat.position);
live_fe = CServiceScan::getInstance()->GetFrontend();
setfe = false;
}
}
break;
}
#if 0
case CZapitMessages::CMD_SCANSETSCANMOTORPOSLIST: {
// absolute
ERROR("CZapitMessages::CMD_SCANSETSCANMOTORPOSLIST: depreciated command\n");
break;
}
#endif
case CZapitMessages::CMD_SCANSETDISEQCTYPE: {
//FIXME diseqcType is global
#if 0
CBasicServer::receive_data(connfd, &diseqcType, sizeof(diseqcType));
live_fe->setDiseqcType(diseqcType);
#endif
ERROR("CZapitMessages::CMD_SCANSETDISEQCTYPE: depreciated command\n");
break;
}
case CZapitMessages::CMD_SCANSETDISEQCREPEAT: {
#if 0
uint32_t repeats;
CBasicServer::receive_data(connfd, &repeats, sizeof(repeats));
live_fe->setDiseqcRepeats(repeats);
#endif
ERROR("CZapitMessages::CMD_SCANSETDISEQCREPEAT: depreciated command\n");
break;
}
case CZapitMessages::CMD_SCANSETBOUQUETMODE:
CBasicServer::receive_data(connfd, &bouquetMode, sizeof(bouquetMode));
break;
#if 0
case CZapitMessages::CMD_SCANSETTYPE:
//CBasicServer::receive_data(connfd, &scanType, sizeof(scanType));
ERROR("CZapitMessages::CMD_SCANSETTYPE: depreciated command\n");
break;
#endif
case CZapitMessages::CMD_SET_EVENT_MODE: {
CZapitMessages::commandSetRecordMode msgSetRecordMode;
CBasicServer::receive_data(connfd, &msgSetRecordMode, sizeof(msgSetRecordMode));
//printf("[zapit] event mode: %d\n", msgSetRecordMode.activate);fflush(stdout);
event_mode = msgSetRecordMode.activate;
break;
}
case CZapitMessages::CMD_SET_RECORD_MODE: {
CZapitMessages::commandSetRecordMode msgSetRecordMode;
CBasicServer::receive_data(connfd, &msgSetRecordMode, sizeof(msgSetRecordMode));
printf("[zapit] recording mode: %d\n", msgSetRecordMode.activate);fflush(stdout);
SetRecordMode(msgSetRecordMode.activate);
break;
}
case CZapitMessages::CMD_GET_RECORD_MODE: {
CZapitMessages::responseGetRecordModeState msgGetRecordModeState;
VALGRIND_PARANOIA(msgGetRecordModeState);
msgGetRecordModeState.activated = (currentMode & RECORD_MODE);
CBasicServer::send_data(connfd, &msgGetRecordModeState, sizeof(msgGetRecordModeState));
break;
}
case CZapitMessages::CMD_SB_GET_PLAYBACK_ACTIVE: {
/* FIXME check if needed */
CZapitMessages::responseGetPlaybackState msgGetPlaybackState;
VALGRIND_PARANOIA(msgGetPlaybackState);
msgGetPlaybackState.activated = playing;
msgGetPlaybackState.activated = videoDecoder->getPlayState();
CBasicServer::send_data(connfd, &msgGetPlaybackState, sizeof(msgGetPlaybackState));
break;
}
case CZapitMessages::CMD_BQ_ADD_BOUQUET: {
char * name = CBasicServer::receive_string(connfd);
g_bouquetManager->addBouquet(name, true);
CBasicServer::delete_string(name);
break;
}
case CZapitMessages::CMD_BQ_DELETE_BOUQUET: {
CZapitMessages::commandDeleteBouquet msgDeleteBouquet;
CBasicServer::receive_data(connfd, &msgDeleteBouquet, sizeof(msgDeleteBouquet)); // bouquet & channel number are already starting at 0!
g_bouquetManager->deleteBouquet(msgDeleteBouquet.bouquet);
break;
}
case CZapitMessages::CMD_BQ_RENAME_BOUQUET: {
CZapitMessages::commandRenameBouquet msgRenameBouquet;
CBasicServer::receive_data(connfd, &msgRenameBouquet, sizeof(msgRenameBouquet)); // bouquet & channel number are already starting at 0!
char * name = CBasicServer::receive_string(connfd);
if (msgRenameBouquet.bouquet < g_bouquetManager->Bouquets.size()) {
g_bouquetManager->Bouquets[msgRenameBouquet.bouquet]->Name = name;
g_bouquetManager->Bouquets[msgRenameBouquet.bouquet]->bUser = true;
}
CBasicServer::delete_string(name);
break;
}
case CZapitMessages::CMD_BQ_EXISTS_BOUQUET: {
CZapitMessages::responseGeneralInteger responseInteger;
VALGRIND_PARANOIA(responseInteger);
char * name = CBasicServer::receive_string(connfd);
responseInteger.number = g_bouquetManager->existsBouquet(name);
CBasicServer::delete_string(name);
CBasicServer::send_data(connfd, &responseInteger, sizeof(responseInteger)); // bouquet & channel number are already starting at 0!
break;
}
#if 0
case CZapitMessages::CMD_BQ_EXISTS_CHANNEL_IN_BOUQUET: {
CZapitMessages::commandExistsChannelInBouquet msgExistsChInBq;
CZapitMessages::responseGeneralTrueFalse responseBool;
CBasicServer::receive_data(connfd, &msgExistsChInBq, sizeof(msgExistsChInBq)); // bouquet & channel number are already starting at 0!
responseBool.status = g_bouquetManager->existsChannelInBouquet(msgExistsChInBq.bouquet, msgExistsChInBq.channel_id);
CBasicServer::send_data(connfd, &responseBool, sizeof(responseBool));
break;
}
#endif
case CZapitMessages::CMD_BQ_MOVE_BOUQUET: {
CZapitMessages::commandMoveBouquet msgMoveBouquet;
CBasicServer::receive_data(connfd, &msgMoveBouquet, sizeof(msgMoveBouquet)); // bouquet & channel number are already starting at 0!
g_bouquetManager->moveBouquet(msgMoveBouquet.bouquet, msgMoveBouquet.newPos);
break;
}
case CZapitMessages::CMD_BQ_ADD_CHANNEL_TO_BOUQUET: {
CZapitMessages::commandAddChannelToBouquet msgAddChannelToBouquet;
CBasicServer::receive_data(connfd, &msgAddChannelToBouquet, sizeof(msgAddChannelToBouquet)); // bouquet & channel number are already starting at 0!
addChannelToBouquet(msgAddChannelToBouquet.bouquet, msgAddChannelToBouquet.channel_id);
break;
}
case CZapitMessages::CMD_BQ_REMOVE_CHANNEL_FROM_BOUQUET: {
/* removeChannelFromBouquet, still used in webif */
CZapitMessages::commandRemoveChannelFromBouquet msgRemoveChannelFromBouquet;
CBasicServer::receive_data(connfd, &msgRemoveChannelFromBouquet, sizeof(msgRemoveChannelFromBouquet)); // bouquet & channel number are already starting at 0!
if (msgRemoveChannelFromBouquet.bouquet < g_bouquetManager->Bouquets.size())
g_bouquetManager->Bouquets[msgRemoveChannelFromBouquet.bouquet]->removeService(msgRemoveChannelFromBouquet.channel_id);
#if 1 // REAL_REMOVE
bool status = 0;
for (unsigned int i = 0; i < g_bouquetManager->Bouquets.size(); i++) {
status = g_bouquetManager->existsChannelInBouquet(i, msgRemoveChannelFromBouquet.channel_id);
if(status)
break;
}
if(!status) {
CServiceManager::getInstance()->RemoveChannel(msgRemoveChannelFromBouquet.channel_id);
current_channel = 0;
list_changed = 1;
}
#endif
break;
}
#if 0
case CZapitMessages::CMD_BQ_MOVE_CHANNEL: {
CZapitMessages::commandMoveChannel msgMoveChannel;
CBasicServer::receive_data(connfd, &msgMoveChannel, sizeof(msgMoveChannel)); // bouquet & channel number are already starting at 0!
if (msgMoveChannel.bouquet < g_bouquetManager->Bouquets.size())
g_bouquetManager->Bouquets[msgMoveChannel.bouquet]->moveService(msgMoveChannel.oldPos, msgMoveChannel.newPos,
(((currentMode & RADIO_MODE) && msgMoveChannel.mode == CZapitClient::MODE_CURRENT)
|| (msgMoveChannel.mode==CZapitClient::MODE_RADIO)) ? 2 : 1);
break;
}
#endif
case CZapitMessages::CMD_BQ_SET_LOCKSTATE: {
CZapitMessages::commandBouquetState msgBouquetLockState;
CBasicServer::receive_data(connfd, &msgBouquetLockState, sizeof(msgBouquetLockState)); // bouquet & channel number are already starting at 0!
g_bouquetManager->setBouquetLock(msgBouquetLockState.bouquet, msgBouquetLockState.state);
break;
}
case CZapitMessages::CMD_BQ_SET_HIDDENSTATE: {
CZapitMessages::commandBouquetState msgBouquetHiddenState;
CBasicServer::receive_data(connfd, &msgBouquetHiddenState, sizeof(msgBouquetHiddenState)); // bouquet & channel number are already starting at 0!
if (msgBouquetHiddenState.bouquet < g_bouquetManager->Bouquets.size())
g_bouquetManager->Bouquets[msgBouquetHiddenState.bouquet]->bHidden = msgBouquetHiddenState.state;
break;
}
case CZapitMessages::CMD_BQ_RENUM_CHANNELLIST:
g_bouquetManager->renumServices();
// 2004.08.02 g_bouquetManager->storeBouquets();
break;
case CZapitMessages::CMD_BQ_SAVE_BOUQUETS: {
CZapitMessages::commandBoolean msgBoolean;
CBasicServer::receive_data(connfd, &msgBoolean, sizeof(msgBoolean));
SendCmdReady(connfd);
#if 0
//if (msgBoolean.truefalse)
if(list_changed) {
SendEvent(CZapitClient::EVT_SERVICES_CHANGED);
} else
SendEvent(CZapitClient::EVT_BOUQUETS_CHANGED);
#endif
g_bouquetManager->saveBouquets();
g_bouquetManager->saveUBouquets();
g_bouquetManager->renumServices();
//SendEvent(CZapitClient::EVT_SERVICES_CHANGED);
SendEvent(CZapitClient::EVT_BOUQUETS_CHANGED);
if(list_changed) {
CServiceManager::getInstance()->SaveServices(true);
list_changed = 0;
}
break;
}
case CZapitMessages::CMD_SET_VIDEO_SYSTEM: {
CZapitMessages::commandInt msg;
CBasicServer::receive_data(connfd, &msg, sizeof(msg));
COsdHelpers::getInstance()->setVideoSystem(msg.val);
COsdHelpers::getInstance()->changeOsdResolution(0, true);
CNeutrinoApp::getInstance()->g_settings_video_Mode(msg.val);
break;
}
#if 0
case CZapitMessages::CMD_SET_NTSC: {
videoDecoder->SetVideoSystem(8);
break;
}
#endif
case CZapitMessages::CMD_SB_START_PLAYBACK:
{
CZapitMessages::commandBoolean msgBool;
CBasicServer::receive_data(connfd, &msgBool, sizeof(msgBool));
//playbackStopForced = false;
StartPlayBack(current_channel);
if (msgBool.truefalse)
SendPMT();
break;
}
case CZapitMessages::CMD_SB_STOP_PLAYBACK:
{
CZapitMessages::commandBoolean msgBool;
CBasicServer::receive_data(connfd, &msgBool, sizeof(msgBool));
StopPlayBack(msgBool.truefalse);
SendCmdReady(connfd);
break;
}
#ifdef ENABLE_PIP
case CZapitMessages::CMD_STOP_PIP:
for (unsigned i=0; i < (unsigned int) g_info.hw_caps->pip_devs; i++)
{
StopPip(i);
}
SendCmdReady(connfd);
break;
#endif
case CZapitMessages::CMD_SB_LOCK_PLAYBACK:
{
CZapitMessages::commandBoolean msgBool;
CBasicServer::receive_data(connfd, &msgBool, sizeof(msgBool));
StopPlayBack(msgBool.truefalse, false);
playbackStopForced = true;
lock_channel_id = live_channel_id;
//lockPlayBack();
SendCmdReady(connfd);
break;
}
case CZapitMessages::CMD_SB_UNLOCK_PLAYBACK:
{
CZapitMessages::commandBoolean msgBool;
CBasicServer::receive_data(connfd, &msgBool, sizeof(msgBool));
playbackStopForced = false;
if (lock_channel_id == live_channel_id) {
StartPlayBack(current_channel);
if (msgBool.truefalse)
SendPMT();
} else {
live_fe->setTsidOnid(0);
if (!ZapIt(lock_channel_id))
SendEvent(CZapitClient::EVT_ZAP_FAILED, &lock_channel_id, sizeof(lock_channel_id));
lock_channel_id = 0;
}
//unlockPlayBack();
SendCmdReady(connfd);
break;
}
#if 0
case CZapitMessages::CMD_SET_DISPLAY_FORMAT: {
CZapitMessages::commandInt msg;
CBasicServer::receive_data(connfd, &msg, sizeof(msg));
videoDecoder->setCroppingMode((video_displayformat_t) msg.val);
break;
}
#endif
case CZapitMessages::CMD_SET_AUDIO_MODE: {
CZapitMessages::commandInt msg;
CBasicServer::receive_data(connfd, &msg, sizeof(msg));
audioDecoder->setChannel((int) msg.val);
audio_mode = msg.val;
break;
}
#if 0
case CZapitMessages::CMD_GET_AUDIO_MODE: {
CZapitMessages::commandInt msg;
msg.val = (int) audio_mode;
CBasicServer::send_data(connfd, &msg, sizeof(msg));
break;
}
#endif
case CZapitMessages::CMD_SET_OSD_RES: {
CZapitMessages::commandInt msg;
CBasicServer::receive_data(connfd, &msg, sizeof(msg));
mosd=(int) msg.val;
COsdHelpers::getInstance()->changeOsdResolution(mosd);
break;
}
case CZapitMessages::CMD_GET_OSD_RES: {
CZapitMessages::commandInt msg;
mosd = COsdHelpers::getInstance()->getOsdResolution();
msg.val = mosd;
CBasicServer::send_data(connfd, &msg, sizeof(msg));
break;
}
case CZapitMessages::CMD_SET_ASPECTRATIO: {
CZapitMessages::commandInt msg;
CBasicServer::receive_data(connfd, &msg, sizeof(msg));
aspectratio=(int) msg.val;
videoDecoder->setAspectRatio(aspectratio, -1);
break;
}
case CZapitMessages::CMD_GET_ASPECTRATIO: {
CZapitMessages::commandInt msg;
VALGRIND_PARANOIA(msg);
aspectratio=videoDecoder->getAspectRatio();
msg.val = aspectratio;
CBasicServer::send_data(connfd, &msg, sizeof(msg));
break;
}
case CZapitMessages::CMD_SET_MODE43: {
CZapitMessages::commandInt msg;
CBasicServer::receive_data(connfd, &msg, sizeof(msg));
mode43=(int) msg.val;
videoDecoder->setAspectRatio(-1, mode43);
break;
}
case CZapitMessages::CMD_GET_MODE43: {
CZapitMessages::commandInt msg;
msg.val = mode43;
CBasicServer::send_data(connfd, &msg, sizeof(msg));
break;
}
case CZapitMessages::CMD_GET_VIDEO_FORMAT: {
CZapitMessages::commandInt msg;
msg.val = CNeutrinoApp::getInstance()->getVideoFormat();
CBasicServer::send_data(connfd, &msg, sizeof(msg));
break;
}
case CZapitMessages::CMD_GETPIDS: {
if (current_channel) {
CZapitClient::responseGetOtherPIDs responseGetOtherPIDs;
VALGRIND_PARANOIA(responseGetOtherPIDs);
responseGetOtherPIDs.vpid = current_channel->getVideoPid();
responseGetOtherPIDs.ecmpid = 0; // TODO: remove
responseGetOtherPIDs.vtxtpid = current_channel->getTeletextPid();
responseGetOtherPIDs.pmtpid = current_channel->getPmtPid();
responseGetOtherPIDs.pcrpid = current_channel->getPcrPid();
responseGetOtherPIDs.selected_apid = current_channel->getAudioChannelIndex();
/*responseGetOtherPIDs.privatepid = current_channel->getPrivatePid();*/
CBasicServer::send_data(connfd, &responseGetOtherPIDs, sizeof(responseGetOtherPIDs));
sendAPIDs(connfd);
}
break;
}
#if 0
case CZapitMessages::CMD_GET_FE_SIGNAL: {
CZapitClient::responseFESignal response_FEsig;
response_FEsig.sig = live_fe->getSignalStrength();
response_FEsig.snr = live_fe->getSignalNoiseRatio();
response_FEsig.ber = live_fe->getBitErrorRate();
CBasicServer::send_data(connfd, &response_FEsig, sizeof(CZapitClient::responseFESignal));
ERROR("CZapitMessages::CMD_GET_FE_SIGNAL: depreciated command\n");
break;
}
#endif
case CZapitMessages::CMD_SETSUBSERVICES: {
CZapitClient::commandAddSubServices msgAddSubService;
t_satellite_position satellitePosition = current_channel ? current_channel->getSatellitePosition() : 0;
while (CBasicServer::receive_data(connfd, &msgAddSubService, sizeof(msgAddSubService))) {
#if 0
t_original_network_id original_network_id = msgAddSubService.original_network_id;
t_service_id service_id = msgAddSubService.service_id;
t_channel_id sub_channel_id =
((uint64_t) ( satellitePosition >= 0 ? satellitePosition : (uint64_t)(0xF000+ abs(satellitePosition))) << 48) |
(uint64_t) CREATE_CHANNEL_ID(msgAddSubService.service_id, msgAddSubService.original_network_id, msgAddSubService.transport_stream_id);
DBG("NVOD insert %llx\n", sub_channel_id);
nvodchannels.insert (
std::pair <t_channel_id, CZapitChannel> (
sub_channel_id,
CZapitChannel (
"NVOD",
service_id,
msgAddSubService.transport_stream_id,
original_network_id,
1,
satellitePosition,
0
)
)
);
#endif
CZapitChannel * channel = new CZapitChannel (
"NVOD",
msgAddSubService.service_id,
msgAddSubService.transport_stream_id,
msgAddSubService.original_network_id,
ST_DIGITAL_TELEVISION_SERVICE,
satellitePosition,
0
);
channel->delsys = live_fe->getCurrentDeliverySystem();
CServiceManager::getInstance()->AddNVODChannel(channel);
}
current_is_nvod = true;
break;
}
case CZapitMessages::CMD_REGISTEREVENTS:
eventServer->registerEvent(connfd);
break;
case CZapitMessages::CMD_UNREGISTEREVENTS:
eventServer->unRegisterEvent(connfd);
break;
case CZapitMessages::CMD_MUTE: {
CZapitMessages::commandBoolean msgBoolean;
CBasicServer::receive_data(connfd, &msgBoolean, sizeof(msgBoolean));
//printf("[zapit] mute %d\n", msgBoolean.truefalse);
if (msgBoolean.truefalse)
audioDecoder->mute();
else
audioDecoder->unmute();
break;
}
case CZapitMessages::CMD_LOCKRC: {
CZapitMessages::commandBoolean msgBoolean;
CBasicServer::receive_data(connfd, &msgBoolean, sizeof(msgBoolean));
extern CRCInput *g_RCInput;
if (msgBoolean.truefalse)
g_RCInput->stopInput(true);
else
g_RCInput->restartInput(true);
break;
}
case CZapitMessages::CMD_SET_VOLUME: {
CZapitMessages::commandVolume msgVolume;
CBasicServer::receive_data(connfd, &msgVolume, sizeof(msgVolume));
#if 0
audioDecoder->setVolume(msgVolume.left, msgVolume.right);
volume_left = msgVolume.left;
volume_right = msgVolume.right;
#endif
SetVolume(msgVolume.left);
break;
}
case CZapitMessages::CMD_GET_VOLUME: {
CZapitMessages::commandVolume msgVolume;
VALGRIND_PARANOIA(msgVolume);
#if 0
msgVolume.left = volume_left;
msgVolume.right = volume_right;
#endif
msgVolume.left = msgVolume.right = current_volume;
CBasicServer::send_data(connfd, &msgVolume, sizeof(msgVolume));
break;
}
case CZapitMessages::CMD_GET_MUTE_STATUS: {
CZapitMessages::commandBoolean msgBoolean;
VALGRIND_PARANOIA(msgBoolean);
msgBoolean.truefalse = audioDecoder->getMuteStatus();
CBasicServer::send_data(connfd, &msgBoolean, sizeof(msgBoolean));
break;
}
case CZapitMessages::CMD_SET_STANDBY: {
CZapitMessages::commandBoolean msgBoolean;
CBasicServer::receive_data(connfd, &msgBoolean, sizeof(msgBoolean));
if (msgBoolean.truefalse) {
// if(currentMode & RECORD_MODE) videoDecoder->freeze();
enterStandby();
} else
leaveStandby();
SendCmdReady(connfd);
break;
}
#if 0
case CZapitMessages::CMD_NVOD_SUBSERVICE_NUM: {
CZapitMessages::commandInt msg;
CBasicServer::receive_data(connfd, &msg, sizeof(msg));
}
#endif
case CZapitMessages::CMD_SEND_MOTOR_COMMAND: {
CZapitMessages::commandMotor msgMotor;
CBasicServer::receive_data(connfd, &msgMotor, sizeof(msgMotor));
printf("[zapit] received motor command: %x %x %x %x %x %x\n", msgMotor.cmdtype, msgMotor.address, msgMotor.cmd, msgMotor.num_parameters, msgMotor.param1, msgMotor.param2);
if(msgMotor.cmdtype > 0x20)
live_fe->sendMotorCommand(msgMotor.cmdtype, msgMotor.address, msgMotor.cmd, msgMotor.num_parameters, msgMotor.param1, msgMotor.param2);
// TODO !!
//else if(current_channel)
// live_fe->satFind(msgMotor.cmdtype, current_channel);
break;
}
default:
WARN("unknown command %d (version %d)", rmsg.cmd, CZapitMessages::ACTVERSION);
break;
}
DBG("cmd %d processed\n", rmsg.cmd);
return true;
}
void CZapit::addChannelToBouquet(const unsigned int bouquet, const t_channel_id channel_id)
{
//DBG("addChannelToBouquet(%d, %d)\n", bouquet, channel_id);
CZapitChannel* chan = CServiceManager::getInstance()->FindChannel(channel_id);
if (chan != NULL)
if (bouquet < g_bouquetManager->Bouquets.size())
g_bouquetManager->Bouquets[bouquet]->addService(chan);
else
WARN("bouquet not found");
else
WARN("channel_id not found in channellist");
}
bool CZapit::send_data_count(int connfd, int data_count)
{
CZapitMessages::responseGeneralInteger responseInteger;
VALGRIND_PARANOIA(responseInteger);
responseInteger.number = data_count;
if (CBasicServer::send_data(connfd, &responseInteger, sizeof(responseInteger)) == false) {
ERROR("could not send any return");
return false;
}
return true;
}
void CZapit::sendAPIDs(int connfd)
{
if (!send_data_count(connfd, current_channel->getAudioChannelCount()))
return;
for (uint32_t i = 0; i < current_channel->getAudioChannelCount(); i++) {
CZapitClient::responseGetAPIDs response;
VALGRIND_PARANOIA(response);
response.pid = current_channel->getAudioPid(i);
strncpy(response.desc, current_channel->getAudioChannel(i)->description.c_str(), DESC_MAX_LEN-1);
response.is_ac3 = response.is_aac = response.is_aache = response.is_eac3 = response.is_dts = response.is_dtshd = response.is_lpcm = 0;
if (current_channel->getAudioChannel(i)->audioChannelType == CZapitAudioChannel::AC3) {
response.is_ac3 = 1;
} else if (current_channel->getAudioChannel(i)->audioChannelType == CZapitAudioChannel::AAC) {
response.is_aac = 1;
} else if (current_channel->getAudioChannel(i)->audioChannelType == CZapitAudioChannel::AACPLUS) {
response.is_aache = 1;
} else if (current_channel->getAudioChannel(i)->audioChannelType == CZapitAudioChannel::EAC3) {
response.is_eac3 = 1;
} else if (current_channel->getAudioChannel(i)->audioChannelType == CZapitAudioChannel::DTS) {
response.is_dts = 1;
} else if (current_channel->getAudioChannel(i)->audioChannelType == CZapitAudioChannel::DTSHD) {
response.is_dtshd = 1;
} else if (current_channel->getAudioChannel(i)->audioChannelType == CZapitAudioChannel::LPCM) {
response.is_lpcm = 1;
}
response.component_tag = current_channel->getAudioChannel(i)->componentTag;
if (CBasicServer::send_data(connfd, &response, sizeof(response)) == false) {
ERROR("could not send any return");
return;
}
}
}
void CZapit::internalSendChannels(int connfd, ZapitChannelList* channels, const unsigned int first_channel_nr, bool nonames)
{
int data_count = channels->size();
if (!send_data_count(connfd, data_count))
return;
for (uint32_t i = 0; i < channels->size();i++) {
if(nonames) {
CZapitClient::responseGetBouquetNChannels response;
VALGRIND_PARANOIA(response);
response.nr = first_channel_nr + i;
if (CBasicServer::send_data(connfd, &response, sizeof(response)) == false) {
ERROR("could not send any return");
if (CBasicServer::send_data(connfd, &response, sizeof(response)) == false) {
ERROR("could not send any return, stop");
return;
}
}
} else {
CZapitClient::responseGetBouquetChannels response;
VALGRIND_PARANOIA(response);
strncpy(response.name, ((*channels)[i]->getName()).c_str(), CHANNEL_NAME_SIZE-1);
response.name[CHANNEL_NAME_SIZE-1] = 0;
//printf("internalSendChannels: name %s\n", response.name);
response.satellitePosition = (*channels)[i]->getSatellitePosition();
response.channel_id = (*channels)[i]->getChannelID();
response.nr = first_channel_nr + i;
if (CBasicServer::send_data(connfd, &response, sizeof(response)) == false) {
ERROR("could not send any return");
DBG("current: %d name %s total %d\n", i, response.name, data_count);
if (CBasicServer::send_data(connfd, &response, sizeof(response)) == false) {
ERROR("could not send any return, stop");
return;
}
}
}
}
}
void CZapit::sendBouquets(int connfd, const bool emptyBouquetsToo, CZapitClient::channelsMode mode)
{
CZapitClient::responseGetBouquets msgBouquet;
VALGRIND_PARANOIA(msgBouquet);
int curMode;
switch(mode) {
case CZapitClient::MODE_TV:
curMode = TV_MODE;
break;
case CZapitClient::MODE_RADIO:
curMode = RADIO_MODE;
break;
case CZapitClient::MODE_CURRENT:
default:
curMode = currentMode;
break;
}
for (uint32_t i = 0; i < g_bouquetManager->Bouquets.size(); i++) {
if (emptyBouquetsToo || (!g_bouquetManager->Bouquets[i]->bHidden && g_bouquetManager->Bouquets[i]->bUser)
|| ((!g_bouquetManager->Bouquets[i]->bHidden)
&& (((curMode & RADIO_MODE) && !g_bouquetManager->Bouquets[i]->radioChannels.empty())
|| ((curMode & TV_MODE) && !g_bouquetManager->Bouquets[i]->tvChannels.empty()))))
{
msgBouquet.bouquet_nr = i;
strncpy(msgBouquet.name, g_bouquetManager->Bouquets[i]->Name.c_str(), 29);
msgBouquet.name[29] = 0;
msgBouquet.locked = g_bouquetManager->Bouquets[i]->bLocked;
msgBouquet.hidden = g_bouquetManager->Bouquets[i]->bHidden;
if (CBasicServer::send_data(connfd, &msgBouquet, sizeof(msgBouquet)) == false) {
ERROR("could not send any return");
return;
}
}
}
msgBouquet.bouquet_nr = RESPONSE_GET_BOUQUETS_END_MARKER;
if (CBasicServer::send_data(connfd, &msgBouquet, sizeof(msgBouquet)) == false) {
ERROR("could not send end marker");
return;
}
}
void CZapit::sendBouquetChannels(int connfd, const unsigned int bouquet, const CZapitClient::channelsMode mode, bool nonames)
{
if (bouquet >= g_bouquetManager->Bouquets.size()) {
WARN("invalid bouquet number: %d", bouquet);
return;
}
if (((currentMode & RADIO_MODE) && (mode == CZapitClient::MODE_CURRENT)) || (mode == CZapitClient::MODE_RADIO))
internalSendChannels(connfd, &(g_bouquetManager->Bouquets[bouquet]->radioChannels), g_bouquetManager->radioChannelsBegin().getNrofFirstChannelofBouquet(bouquet), nonames);
else
internalSendChannels(connfd, &(g_bouquetManager->Bouquets[bouquet]->tvChannels), g_bouquetManager->tvChannelsBegin().getNrofFirstChannelofBouquet(bouquet), nonames);
}
void CZapit::sendChannels(int connfd, const CZapitClient::channelsMode mode, const CZapitClient::channelsOrder order)
{
ZapitChannelList channels;
if (order == CZapitClient::SORT_BOUQUET) {
CBouquetManager::ChannelIterator cit = (((currentMode & RADIO_MODE) && (mode == CZapitClient::MODE_CURRENT)) || (mode==CZapitClient::MODE_RADIO)) ? g_bouquetManager->radioChannelsBegin() : g_bouquetManager->tvChannelsBegin();
for (; !(cit.EndOfChannels()); cit++)
channels.push_back(*cit);
}
else if (order == CZapitClient::SORT_ALPHA)
{
// ATTENTION: in this case response.nr does not return the actual number of the channel for zapping!
if (((currentMode & RADIO_MODE) && (mode == CZapitClient::MODE_CURRENT)) || (mode == CZapitClient::MODE_RADIO)) {
CServiceManager::getInstance()->GetAllRadioChannels(channels);
} else {
CServiceManager::getInstance()->GetAllTvChannels(channels);
}
sort(channels.begin(), channels.end(), CmpChannelByChName());
}
internalSendChannels(connfd, &channels, 0, false);
}
bool CZapit::StartPlayBack(CZapitChannel *thisChannel)
{
INFO("standby %d playing %d forced %d", standby, playing, playbackStopForced);
if(!thisChannel)
thisChannel = current_channel;
if (playbackStopForced || !thisChannel || playing)
return false;
if(standby) {
CFEManager::getInstance()->Open();
return true;
}
#if 0
if (IS_WEBCHAN(thisChannel->getChannelID())) {
INFO("WEBTV channel\n");
SendEvent(CZapitClient::EVT_WEBTV_ZAP_COMPLETE, &live_channel_id, sizeof(t_channel_id));
return true;
}
#endif
unsigned short pcr_pid = thisChannel->getPcrPid();
unsigned short audio_pid = thisChannel->getAudioPid();
unsigned short video_pid = (currentMode & TV_MODE) ? thisChannel->getVideoPid() : 0;
unsigned short teletext_pid = thisChannel->getTeletextPid();
printf("[zapit] vpid %X apid %X pcr %X\n", video_pid, audio_pid, pcr_pid);
if (!audio_pid && !video_pid && !teletext_pid)
return false;
#if 1
if(video_pid && (pcr_pid == 0x1FFF)) { //FIXME
thisChannel->setPcrPid(video_pid);
pcr_pid = video_pid;
}
#endif
/* set demux filters */
videoDecoder->SetStreamType((VIDEO_FORMAT)thisChannel->type);
// videoDecoder->SetSync(VIDEO_PLAY_MOTION);
if (pcr_pid)
pcrDemux->pesFilter(pcr_pid);
if (audio_pid)
audioDemux->pesFilter(audio_pid);
if (video_pid)
videoDemux->pesFilter(video_pid);
// audioDecoder->SetSyncMode(AVSYNC_ENABLED);
#if 0 //FIXME hack ?
if(thisChannel->getServiceType() == ST_DIGITAL_RADIO_SOUND_SERVICE) {
audioDecoder->SetSyncMode(AVSYNC_AUDIO_IS_MASTER);
pcr_pid = false;
}
#endif
if (pcr_pid) {
//printf("[zapit] starting PCR 0x%X\n", thisChannel->getPcrPid());
pcrDemux->Start();
}
/* select audio output and start audio */
if (audio_pid) {
SetAudioStreamType(thisChannel->getAudioChannel()->audioChannelType);
audioDemux->Start();
audioDecoder->Start();
}
/* start video */
if (video_pid) {
#if HAVE_CST_HARDWARE
videoDecoder->Start(0, pcr_pid, video_pid);
videoDemux->Start();
#else
videoDemux->Start();
videoDecoder->Start(0, pcr_pid, video_pid);
#endif
}
#ifdef USE_VBI
if(teletext_pid)
videoDecoder->StartVBI(teletext_pid);
#endif
playing = true;
return true;
}
bool CZapit::StopPlayBack(bool send_pmt, bool blank)
{
INFO("standby %d playing %d forced %d send_pmt %d", standby, playing, playbackStopForced, send_pmt);
if(send_pmt)
CCamManager::getInstance()->Stop(live_channel_id, CCamManager::PLAY);
#if 0
if (current_channel && IS_WEBCHAN(current_channel->getChannelID())) {
playing = false;
return true;
}
#endif
if (!playing)
return true;
if (playbackStopForced)
return false;
videoDemux->Stop();
audioDemux->Stop();
pcrDemux->Stop();
audioDecoder->Stop();
/* hack. if standby, dont blank video -> for paused timeshift */
videoDecoder->Stop(standby ? false : blank);
#if ENABLE_AIT
ait->Stop();
#endif
#ifdef USE_VBI
videoDecoder->StopVBI();
#endif
playing = false;
tuxtx_stop_subtitle();
if(standby)
dvbsub_pause();
else
dvbsub_stop();
/* reset volume percent to 100% i.e. for media playback, should be safe
* because StartPlayBack will use defaults or saved value */
SetVolumePercent(100);
return true;
}
void CZapit::enterStandby(void)
{
INFO("standby %d recording %d", standby, (currentMode & RECORD_MODE));
if (standby)
return;
standby = true;
SaveSettings(true);
SaveAudioMap();
SaveVolumeMap();
StopPlayBack(true);
#ifdef ENABLE_PIP
for (unsigned i=0; i < (unsigned int) g_info.hw_caps->pip_devs; i++)
{
StopPip(i);
}
#endif
if(!(currentMode & RECORD_MODE)) {
pmt_stop_update_filter(&pmt_update_fd);
CFEManager::getInstance()->Close();
}
}
void CZapit::leaveStandby(void)
{
INFO("standby %d recording %d", standby, (currentMode & RECORD_MODE));
if(!standby)
return;
if(!(currentMode & RECORD_MODE)) {
CFEManager::getInstance()->Open();
}
standby = false;
if (current_channel) {
/* tune channel, with stopped playback to not bypass the parental PIN check */
ZapIt(live_channel_id, false, false);
if (IS_WEBCHAN(live_channel_id)) {
CZapitChannel* newchannel = CServiceManager::getInstance()->FindChannel(last_channel_id);
CFrontend * fe = newchannel ? CFEManager::getInstance()->allocateFE(newchannel) : NULL;
bool transponder_change;
if (fe)
TuneChannel(fe, newchannel, transponder_change, false);
}
}
}
unsigned CZapit::ZapTo(const unsigned int bouquet, const unsigned int pchannel)
{
if (bouquet >= g_bouquetManager->Bouquets.size()) {
WARN("Invalid bouquet %d", bouquet);
return CZapitClient::ZAP_INVALID_PARAM;
}
ZapitChannelList *channels;
if (currentMode & RADIO_MODE)
channels = &(g_bouquetManager->Bouquets[bouquet]->radioChannels);
else
channels = &(g_bouquetManager->Bouquets[bouquet]->tvChannels);
if (pchannel >= channels->size()) {
WARN("Invalid channel %d in bouquet %d", pchannel, bouquet);
return CZapitClient::ZAP_INVALID_PARAM;
}
return ZapTo((*channels)[pchannel]->getChannelID(), false);
}
unsigned int CZapit::ZapTo(t_channel_id channel_id, bool isSubService)
{
unsigned int result = 0;
if (!ZapIt(channel_id)) {
DBG("[zapit] zapit failed, chid %" PRIx64 "\n", channel_id);
SendEvent((isSubService ? CZapitClient::EVT_ZAP_SUB_FAILED : CZapitClient::EVT_ZAP_FAILED), &channel_id, sizeof(channel_id));
#if HAVE_ARM_HARDWARE || HAVE_MIPS_HARDWARE
//zapit failed, but we have to continue the CA pollthread
if (!ca->getZapitReady())
ca->setZapitReady();
#endif
return result;
}
result |= CZapitClient::ZAP_OK;
DBG("[zapit] zapit OK, chid %" PRIx64 "\n", channel_id);
if (isSubService) {
DBG("[zapit] isSubService chid %" PRIx64 "\n", channel_id);
SendEvent(CZapitClient::EVT_ZAP_SUB_COMPLETE, &channel_id, sizeof(channel_id));
}
else if (current_is_nvod) {
DBG("[zapit] NVOD chid %" PRIx64 "\n", channel_id);
SendEvent(CZapitClient::EVT_ZAP_COMPLETE_IS_NVOD, &channel_id, sizeof(channel_id));
result |= CZapitClient::ZAP_IS_NVOD;
}
else
SendEvent(CZapitClient::EVT_ZAP_COMPLETE, &channel_id, sizeof(channel_id));
return result;
}
unsigned CZapit::ZapTo(const unsigned int pchannel)
{
CBouquetManager::ChannelIterator cit = ((currentMode & RADIO_MODE) ? g_bouquetManager->radioChannelsBegin() : g_bouquetManager->tvChannelsBegin()).FindChannelNr(pchannel);
if (!(cit.EndOfChannels()))
return ZapTo((*cit)->getChannelID(), false);
return 0;
}
bool CZapit::Start(Z_start_arg *ZapStart_arg)
{
if(started)
return false;
CFEManager::getInstance()->Init();
live_fe = CFEManager::getInstance()->getFE(0);
/* load configuration or set defaults if no configuration file exists */
video_mode = ZapStart_arg->video_mode;
current_volume = ZapStart_arg->volume;
webtv_xml = ZapStart_arg->webtv_xml;
webradio_xml = ZapStart_arg->webradio_xml;
videoDemux = new cDemux();
videoDemux->Open(DMX_VIDEO_CHANNEL);
pcrDemux = new cDemux();
pcrDemux->Open(DMX_PCR_ONLY_CHANNEL, videoDemux->getBuffer());
audioDemux = new cDemux();
audioDemux->Open(DMX_AUDIO_CHANNEL);
#ifdef BOXMODEL_CST_HD2
videoDecoder = cVideo::GetDecoder(0);
audioDecoder = cAudio::GetDecoder(0);
videoDecoder->SetDemux(videoDemux);
COsdHelpers::getInstance()->setVideoSystem(video_mode);
uint32_t osd_resolution = ZapStart_arg->osd_resolution;
COsdHelpers::getInstance()->changeOsdResolution(osd_resolution);
videoDecoder->Standby(false);
audioDecoder->SetDemux(audioDemux);
audioDecoder->SetVideo(videoDecoder);
#ifdef ENABLE_PIP
#if HAVE_CST_HARDWARE
int dnum = 1; // FIXME?
#endif
if (g_info.hw_caps->can_pip)
{
pipVideoDemux[0] = new cDemux(dnum);
pipVideoDemux[0]->Open(DMX_PIP_CHANNEL);
pipVideoDecoder[0] = cVideo::GetDecoder(1);
pipVideoDecoder[0]->SetDemux(pipVideoDemux[0]);
}
#endif
#else
#if HAVE_CST_HARDWARE
/* work around broken drivers: when starting up with 720p50 image is pink on hd1 */
videoDecoder = new cVideo(VIDEO_STD_1080I50, videoDemux->getChannel(), videoDemux->getBuffer());
videoDecoder->SetVideoSystem(video_mode);
#else
videoDecoder = new cVideo(video_mode, videoDemux->getChannel(), videoDemux->getBuffer());
#endif
videoDecoder->Standby(false);
audioDecoder = new cAudio(audioDemux->getBuffer(), videoDecoder->GetTVEnc(), NULL /*videoDecoder->GetTVEncSD()*/);
#ifdef ENABLE_PIP
if (g_info.hw_caps->can_pip)
{
#if HAVE_CST_HARDWARE
pipVideoDecoder[0] = new cVideo(video_mode, pipVideoDemux[0]->getChannel(), pipVideoDemux[0]->getBuffer(), 1);
#else
for (unsigned i=0; i < (unsigned int) g_info.hw_caps->pip_devs; i++)
{
pipVideoDecoder[i] = new cVideo(0, NULL, NULL, i+1);
pipVideoDecoder[i]->ShowPig(0);
pipAudioDecoder[i] = new cAudio(0, NULL, NULL, i+1);
}
#endif
}
#endif
#endif
videoDecoder->SetAudioHandle(audioDecoder->GetHandle());
volume_percent_ac3 = 100;
volume_percent_pcm = 100;
SetVolumePercent(100);
#ifdef USE_VBI
videoDecoder->OpenVBI(1);
#endif
ca = cCA::GetInstance();
if (live_fe == NULL) /* no frontend found? */
return false;
//LoadSettings();
//LoadAudioMap();
/* create bouquet manager */
g_bouquetManager = new CBouquetManager();
if (!PrepareChannels())
WARN("error parsing services");
else
INFO("channels have been loaded succesfully");
/* FIXME to transit from old satconfig.conf to new frontend.conf,
* ConfigFrontend called after PrepareChannels, as it copy satellitePositions to every fe */
LoadSettings();
ConfigFrontend();
if(!CFEManager::getInstance()->configExist())
CFEManager::getInstance()->saveSettings(true);
if (configfile.getInt32("lastChannelMode", 0))
SetRadioMode();
else
SetTVMode();
if(ZapStart_arg->uselastchannel == 0){
live_channel_id = (currentMode & RADIO_MODE) ? ZapStart_arg->startchannelradio_id : ZapStart_arg->startchanneltv_id ;
lastChannelRadio = ZapStart_arg->startchannelradio_id;
lastChannelTV = ZapStart_arg->startchanneltv_id;
}
/* CA_INIT_CI or CA_INIT_SC or CA_INIT_BOTH */
switch(config.cam_ci)
{
case 2:
ca->SetInitMask(CA_INIT_BOTH);
break;
case 1:
ca->SetInitMask(CA_INIT_CI);
break;
case 0:
ca->SetInitMask(CA_INIT_SC);
break;
default:
ca->SetInitMask(CA_INIT_BOTH);
break;
}
// set ci clock to ZapStart_arg->ci_clock
for (unsigned int i = 0; i < ca->GetNumberCISlots(); i++) {
#if HAVE_LIBSTB_HAL
ca->SetTSClock(ZapStart_arg->ci_clock[i] * 1000000, i);
#else
ca->SetTSClock(ZapStart_arg->ci_clock[i] * 1000000);
#endif
}
#if BOXMODEL_VUPLUS_ALL
// dvb wait delay for ci response
ca->SetCIDelay(ZapStart_arg->ci_delay);
// relevant pids routing
for (unsigned int i = 0; i < ca->GetNumberCISlots(); i++) {
ca->SetCIRelevantPidsRouting(ZapStart_arg->ci_rpr[i], i);
}
#endif
ca->Start();
eventServer = new CEventServer;
if (!zapit_server.prepare(ZAPIT_UDS_NAME)) {
perror(ZAPIT_UDS_NAME);
return false;
}
current_channel = CServiceManager::getInstance()->FindChannel(live_channel_id);
// some older? hw needs this sleep. e.g. my hd-1c.
// if sleep is not set -> blackscreen after boot.
// sleep(1) is ok here. (striper)
#if 0
sleep(1);
leaveStandby();
#endif
started = true;
int ret = start();
return (ret == 0);
}
bool CZapit::Stop()
{
if(!started)
return false;
pmt_stop_update_filter(&pmt_update_fd);
started = false;
int ret = join();
return (ret == 0);
}
static bool zapit_parse_command(CBasicMessage::Header &rmsg, int connfd)
{
return CZapit::getInstance()->ParseCommand(rmsg, connfd);
}
//NI
bool CZapit::getUseChannelFilter()
{
return CCamManager::getInstance()->getChannelFilter();
}
//NI
void CZapit::setMoviePlayer(bool enable)
{
CCamManager::getInstance()->enableMoviePlayer(enable);
}
void CZapit::run()
{
set_threadname("zap:main");
printf("[zapit] starting... tid %ld\n", syscall(__NR_gettid));
abort_zapit = 0;
#if 0
CBasicServer zapit_server;
if (!zapit_server.prepare(ZAPIT_UDS_NAME)) {
perror(ZAPIT_UDS_NAME);
return;
}
#endif
SdtMonitor.Start();
while (started && zapit_server.run(zapit_parse_command, CZapitMessages::ACTVERSION, true))
{
if (pmt_update_fd != -1) {
unsigned char buf[4096];
int ret = pmtDemux->Read(buf, 4095, 10);
if (ret > 0) {
pmt_stop_update_filter(&pmt_update_fd);
printf("[zapit] pmt updated, sid 0x%x new version 0x%x\n", (buf[3] << 8) + buf[4], (buf[5] >> 1) & 0x1F);
if(current_channel) {
t_channel_id channel_id = current_channel->getChannelID();
int vpid = current_channel->getVideoPid();
int apid = current_channel->getAudioPid();
CPmt pmt;
pmt.Parse(current_channel);
bool apid_found = false;
/* check if selected audio pid still present */
for (int i = 0; i < current_channel->getAudioChannelCount(); i++) {
if (current_channel->getAudioChannel(i)->pid == apid) {
apid_found = true;
break;
}
}
if(!apid_found || vpid != current_channel->getVideoPid()) {
ZapIt(current_channel->getChannelID(), true);
} else {
SendPMT(true);
pmt_set_update_filter(current_channel, &pmt_update_fd);
}
SendEvent(CZapitClient::EVT_PMT_CHANGED, &channel_id, sizeof(channel_id));
}
}
}
/* yuck, don't waste that much cpu time :) */
usleep(0);
#if 0
static time_t stime = time(0);
if(!standby && !CServiceScan::getInstance()->Scanning() && current_channel) {
time_t curtime = time(0);
//FIXME check if sig_delay needed */
if(sig_delay && (curtime - stime) > sig_delay) {
stime = curtime;
fe_status_t status = live_fe->getStatus();
printf("[zapit] frontend status %d\n", status);
if (status != FE_HAS_LOCK) {
live_fe->retuneChannel();
}
}
}
#endif
}
SaveChannelPids(current_channel);
SaveSettings(true);
SaveAudioMap();
SaveVolumeMap();
StopPlayBack(true);
CFEManager::getInstance()->saveSettings(true);
//CServiceManager::getInstance()->SaveMotorPositions();
CServiceManager::getInstance()->SaveServices(true, true);
SdtMonitor.Stop();
INFO("shutdown started");
#ifdef USE_VBI
videoDecoder->CloseVBI();
#endif
delete pcrDemux;
delete pmtDemux;
delete audioDecoder;
delete audioDemux;
#ifdef ENABLE_PIP
for (unsigned i=0; i < (unsigned int) g_info.hw_caps->pip_devs; i++)
{
StopPip(i);
if (pipVideoDecoder[i])
pipVideoDecoder[i] = NULL;
if (pipVideoDemux[i])
pipVideoDemux[i] = NULL;
if (pipAudioDecoder[i])
pipAudioDecoder[i] = NULL;
if (pipAudioDemux[i])
pipAudioDemux[i] = NULL;
}
#endif
INFO("demuxes/decoders deleted");
#ifdef EXIT_CLEANUP
INFO("cleanup...");
delete eventServer;
delete g_bouquetManager;
delete CServiceManager::getInstance();
delete CServiceScan::getInstance();
#endif
delete CFEManager::getInstance();
INFO("frontend(s) deleted");
if (ca) {
INFO("stopping CA");
ca->Stop();
delete ca;
}
INFO("shutdown complete");
return;
}
void CZapit::SetConfig(Zapit_config * Cfg)
{
printf("[zapit] %s...\n", __FUNCTION__);
config = *Cfg;
SaveSettings(true);
ConfigFrontend();
}
#if 0
//never used
void CZapit::SendConfig(int connfd)
{
printf("[zapit] %s...\n", __FUNCTION__);
CBasicServer::send_data(connfd, &config, sizeof(config));
}
#endif
void CZapit::GetConfig(Zapit_config &Cfg)
{
printf("[zapit] %s...\n", __FUNCTION__);
Cfg = config;
}
CZapitSdtMonitor::CZapitSdtMonitor()
{
sdt_wakeup = false;
started = false;
}
CZapitSdtMonitor::~CZapitSdtMonitor()
{
Stop();
}
void CZapitSdtMonitor::Wakeup()
{
sdt_wakeup = true;
}
bool CZapitSdtMonitor::Start()
{
started = true;
int ret = start();
return (ret == 0);
}
bool CZapitSdtMonitor::Stop()
{
if (!started)
return false;
started = false;
int ret = join();
return (ret == 0);
}
void CZapitSdtMonitor::run()
{
time_t /*tstart,*/ tcur = 0, wtime = 0;
t_transport_stream_id transport_stream_id = 0;
t_original_network_id original_network_id = 0;
t_satellite_position satellitePosition = 0;
freq_id_t freq = 0;
transponder_id_t tpid = 0;
set_threadname("zap:sdtmonitor");
//tstart = time(0);
sdt_tp.clear();
printf("[zapit] sdt monitor started\n");
while(started) {
sleep(1);
if(sdt_wakeup) {
sdt_wakeup = 0;
CZapitChannel * channel = CZapit::getInstance()->GetCurrentChannel();
if(channel) {
wtime = time(0);
transport_stream_id = channel->getTransportStreamId();
original_network_id = channel->getOriginalNetworkId();
satellitePosition = channel->getSatellitePosition();
freq = channel->getFreqId();
tpid = channel->getTransponderId();
}
}
if(!CZapit::getInstance()->GetScanSDT())
continue;
tcur = time(0);
if(wtime && ((tcur - wtime) > 2) && !sdt_wakeup) {
printf("[sdt monitor] wakeup...\n");
wtime = 0;
if(CServiceScan::getInstance()->Scanning() || CZapit::getInstance()->Recording())
continue;
transponder_list_t::iterator tI = transponders.find(tpid);
if(tI == transponders.end()) {
printf("[sdt monitor] tp not found ?!\n");
continue;
}
sdt_tp_map_t::iterator stI = sdt_tp.find(tpid);
if ((stI != sdt_tp.end()) && ((time_monotonic() - stI->second) < 3600)) {
printf("[sdt monitor] TP already updated less than an hour ago.\n");
continue;
}
CServiceManager::getInstance()->RemoveCurrentChannels();
#if 0
int ret = parse_current_sdt(transport_stream_id, original_network_id, satellitePosition, freq);
if(ret) {
if(ret == -1)
printf("[sdt monitor] scanSDT broken ?\n");
continue;
}
#endif
CSdt sdt(satellitePosition, freq, true);
if(!sdt.Parse(transport_stream_id, original_network_id)) {
continue;
}
sdt_tp.insert(std::pair <transponder_id_t, time_t> (tpid, time_monotonic()));
bool updated = CServiceManager::getInstance()->SaveCurrentServices(tpid);
CServiceManager::getInstance()->CopyCurrentServices(tpid);
if(updated && (CZapit::getInstance()->GetScanSDT()))
CZapit::getInstance()->SendEvent(CZapitClient::EVT_SDT_CHANGED);
if(!updated)
printf("[sdt monitor] no changes.\n");
else
printf("[sdt monitor] found changes.\n");
#if HAVE_ARM_HARDWARE || HAVE_MIPS_HARDWARE
//now it's time to continue the CA pollthread
if (!ca->getZapitReady())
ca->setZapitReady();
#endif
}
}
return;
}