--- /dev/null
+/*
+ Copyright 2007 Chris Tallon
+
+ This file is part of VOMP.
+
+ VOMP 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.
+
+ VOMP 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 VOMP; if not, write to the Free Software
+ Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
+*/
+
+#include "playerlivetv.h"
+
+#include "log.h"
+#include "audio.h"
+#include "video.h"
+#include "demuxerts.h"
+#include "vdr.h"
+#include "messagequeue.h"
+#include "remote.h"
+#include "message.h"
+#include "channel.h"
+
+// ----------------------------------- Called from outside, one offs or info funcs
+
+PlayerLiveTV::PlayerLiveTV(MessageQueue* tmessageQueue, void* tmessageReceiver, ChannelList* tchanList)
+: vfeed(this), afeed(this)
+{
+ messageQueue = tmessageQueue;
+ messageReceiver = tmessageReceiver;
+ chanList = tchanList;
+
+ audio = Audio::getInstance();
+ video = Video::getInstance();
+ logger = Log::getInstance();
+ vdr = VDR::getInstance();
+ initted = false;
+
+// videoStartup = false;
+// preBuffering = false;
+
+ stopNow = false;
+ state = 1;
+
+ video->turnVideoOn();
+}
+
+PlayerLiveTV::~PlayerLiveTV()
+{
+ if (initted) shutdown();
+}
+
+int PlayerLiveTV::init()
+{
+ if (initted) return 0;
+
+ demuxer = new DemuxerTS();
+ if (!demuxer) return 0;
+
+ if (!demuxer->init(this, audio, video, 2097152, 524288))
+ {
+ logger->log("PlayerLiveTV", Log::ERR, "Demuxer failed to init");
+ shutdown();
+ return 0;
+ }
+
+ vfeed.init();
+ afeed.init();
+
+ video->stop();
+ video->blank();
+ audio->stop();
+
+ initted = true;
+ return 1;
+}
+
+int PlayerLiveTV::shutdown()
+{
+ if (!initted) return 0;
+ stop();
+ initted = false;
+
+ delete demuxer;
+
+#ifdef WIN32
+ CloseHandle(mutex);
+#endif
+
+ return 1;
+}
+
+bool* PlayerLiveTV::getDemuxerMpegAudioChannels()
+{
+ return demuxer->getmpAudioChannels();
+}
+
+bool* PlayerLiveTV::getDemuxerAc3AudioChannels()
+{
+ return demuxer->getac3AudioChannels();
+}
+
+int PlayerLiveTV::getCurrentAudioChannel()
+{
+ return demuxer->getAID();
+}
+
+void PlayerLiveTV::setAudioChannel(int newChannel)
+{
+ return demuxer->setAID(newChannel);
+}
+
+// ----------------------------------- Externally called events
+
+void PlayerLiveTV::go(ULONG index)
+{
+ struct PLTVInstruction i;
+ i.instruction = 1;
+ i.channelIndex = index;
+ instructions.push(i);
+
+ threadStart();
+}
+
+void PlayerLiveTV::setChannel(ULONG index)
+{
+// demuxer->setVID(Vpid);
+// demuxer->setAID(Apid);
+// play();
+ logger->log("PlayerLiveTV", Log::DEBUG, "setChannel");
+
+ struct PLTVInstruction i;
+ i.instruction = 1;
+ i.channelIndex = index;
+ instructions.push(i);
+
+ threadSignalNoLock();
+}
+
+void PlayerLiveTV::stop()
+{
+ logger->log("PlayerLiveTV", Log::DEBUG, "stop");
+
+ struct PLTVInstruction i;
+ i.instruction = 2;
+ instructions.push(i);
+
+ logger->log("PlayerLiveTV", Log::DEBUG, "Pushed instruction");
+
+ threadSignal();
+
+ logger->log("PlayerLiveTV", Log::DEBUG, "Signal sent");
+
+ threadStop();
+
+ logger->log("PlayerLiveTV", Log::DEBUG, "Stop returning");
+}
+
+// ----------------------------------- Callback
+
+void PlayerLiveTV::call(void* caller)
+{
+ if (caller == demuxer)
+ {
+ logger->log("PlayerLiveTV", Log::DEBUG, "Callback from demuxer");
+
+ if (video->getTVsize() == Video::ASPECT4X3)
+ {
+ logger->log("PlayerLiveTV", Log::DEBUG, "TV is 4:3, ignoring aspect switching");
+ return;
+ }
+
+ int dxCurrentAspect = demuxer->getAspectRatio();
+ if (dxCurrentAspect == Demuxer::ASPECT_4_3)
+ {
+ logger->log("PlayerLiveTV", Log::DEBUG, "Demuxer said video is 4:3 aspect, switching TV");
+ video->setAspectRatio(Video::ASPECT4X3);
+
+ Message* m = new Message();
+ m->from = this;
+ m->to = messageReceiver;
+ m->message = Message::PLAYER_EVENT;
+ m->parameter = PlayerLiveTV::ASPECT43;
+ messageQueue->postMessageFromOuterSpace(m);
+ }
+ else if (dxCurrentAspect == Demuxer::ASPECT_16_9)
+ {
+ logger->log("PlayerLiveTV", Log::DEBUG, "Demuxer said video is 16:9 aspect, switching TV");
+ video->setAspectRatio(Video::ASPECT16X9);
+
+ Message* m = new Message();
+ m->from = this;
+ m->to = messageReceiver;
+ m->message = Message::PLAYER_EVENT;
+ m->parameter = PlayerLiveTV::ASPECT169;
+ messageQueue->postMessageFromOuterSpace(m);
+ }
+ else
+ {
+ logger->log("PlayerLiveTV", Log::DEBUG, "Demuxer said video is something else... ignoring");
+ }
+
+ }
+ else
+ {
+ if (videoStartup)
+ {
+ logger->log("PlayerLiveTV", Log::DEBUG, "Video startup");
+ videoStartup = false;
+ video->reset();
+ video->play();
+ video->sync();
+ vfeed.release();
+ }
+
+ threadSignalNoLock();
+ }
+}
+
+// -----------------------------------
+
+void PlayerLiveTV::streamReceive(void* data, ULONG len)
+{
+ logger->log("PlayerLiveTV", Log::DEBUG, "Got data, %p, %lu", data, len);
+
+
+ if (streamChunks.size() < 30)
+ {
+ StreamChunk s;
+ s.data = data;
+ s.len = len;
+ streamChunks.push(s);
+ threadSignalNoLock();
+ }
+ else
+
+ {
+ // Too many chunks in streamChunks, drop this chunk
+ free(data);
+ logger->log("PlayerLiveTV", Log::DEBUG, "Dropped chunk");
+ }
+}
+
+void PlayerLiveTV::clearStreamChunks()
+{
+ while(streamChunks.size())
+ {
+ logger->log("PlayerLiveTV", Log::DEBUG, "Dropping chunk from old stream");
+ struct StreamChunk s = streamChunks.front();
+ streamChunks.pop();
+ free(s.data);
+ }
+}
+
+void PlayerLiveTV::chunkToDemuxer()
+{
+ StreamChunk s = streamChunks.front();
+ streamChunks.pop();
+ logger->log("PlayerLiveTV", Log::DEBUG, "About to call demuxer with %p %lu", s.data, s.len);
+ demuxer->put((UCHAR*)s.data, s.len);
+
+/*
+ FILE* fp = fopen("/data.ts", "a");
+ fwrite(s.data, s.len, 1, fp);
+ fclose(fp);
+*/
+
+ /* logger->log("PlayerLiveTV", Log::DEBUG, "Back from demuxer %i", a);
+ if (0)//(a != 50000)
+ {
+ logger->log("PlayerLiveTV", Log::ERR, "--------------------------------------- Back from demuxer %i", a);
+ logger->log("PlayerLiveTV", Log::ERR, "--------------------------------------- Back from demuxer %i", a);
+ logger->log("PlayerLiveTV", Log::ERR, "--------------------------------------- Back from demuxer %i", a);
+ logger->log("PlayerLiveTV", Log::ERR, "--------------------------------------- Back from demuxer %i", a);
+ logger->log("PlayerLiveTV", Log::ERR, "--------------------------------------- Back from demuxer %i", a);
+ logger->log("PlayerLiveTV", Log::ERR, "--------------------------------------- Back from demuxer %i", a);
+ logger->log("PlayerLiveTV", Log::ERR, "--------------------------------------- Back from demuxer %i", a);
+ logger->log("PlayerLiveTV", Log::ERR, "--------------------------------------- Back from demuxer %i", a);
+ logger->log("PlayerLiveTV", Log::ERR, "--------------------------------------- Back from demuxer %i", a);
+ logger->log("PlayerLiveTV", Log::ERR, "--------------------------------------- Back from demuxer %i", a);
+ logger->log("PlayerLiveTV", Log::ERR, "--------------------------------------- Back from demuxer %i", a);
+ logger->log("PlayerLiveTV", Log::ERR, "--------------------------------------- Back from demuxer %i", a);
+ logger->log("PlayerLiveTV", Log::ERR, "--------------------------------------- Back from demuxer %i", a);
+ logger->log("PlayerLiveTV", Log::ERR, "--------------------------------------- Back from demuxer %i", a);
+ logger->log("PlayerLiveTV", Log::ERR, "--------------------------------------- Back from demuxer %i", a);
+ logger->log("PlayerLiveTV", Log::ERR, "--------------------------------------- Back from demuxer %i", a);
+ logger->log("PlayerLiveTV", Log::ERR, "--------------------------------------- Back from demuxer %i", a);
+ logger->log("PlayerLiveTV", Log::ERR, "--------------------------------------- Back from demuxer %i", a);
+ logger->log("PlayerLiveTV", Log::ERR, "--------------------------------------- Back from demuxer %i", a);
+ logger->log("PlayerLiveTV", Log::ERR, "--------------------------------------- Back from demuxer %i", a);
+ logger->log("PlayerLiveTV", Log::ERR, "--------------------------------------- Back from demuxer %i", a);
+ abort();
+
+ }
+ */
+ free(s.data);
+}
+
+void PlayerLiveTV::switchState(UCHAR newState)
+{
+ logger->log("PlayerLiveTV", Log::DEBUG, "Switch from state %u to state %u", state, newState);
+
+ switch(state)
+ {
+ case S_STOP: // FROM S_STOP
+ {
+ switch(newState)
+ {
+ case S_STOP:
+ {
+ abort();
+ }
+ case S_PREBUFFERING:
+ {
+ break;
+ }
+ case S_PLAY:
+ {
+ abort();
+ }
+ }
+ break;
+ }
+
+ case S_PREBUFFERING: // FROM S_PREBUFFERING
+ {
+ switch(newState)
+ {
+ case S_STOP:
+ {
+ vdr->stopStreaming();
+ clearStreamChunks();
+ break;
+ }
+ case S_PREBUFFERING:
+ {
+ vdr->stopStreaming();
+ clearStreamChunks();
+ break;
+ }
+ case S_PLAY:
+ {
+ videoStartup = true;
+
+ audio->reset();
+ audio->setStreamType(Audio::MPEG2_PES);
+ audio->systemMuteOff();
+ video->reset();
+ demuxer->reset();
+ demuxer->seek();
+
+ audio->sync();
+ audio->play();
+ video->sync();
+ video->pause();
+ afeed.start();
+ vfeed.start();
+ break;
+ }
+ }
+ break;
+ }
+
+ case S_PLAY: // FROM S_PLAY
+ {
+ switch(newState)
+ {
+ case S_STOP:
+ {
+ vdr->stopStreaming();
+ clearStreamChunks();
+ vfeed.stop();
+ afeed.stop();
+ video->stop();
+ video->blank();
+ audio->stop();
+ audio->unPause();
+ video->reset();
+ break;
+ }
+ case S_PREBUFFERING:
+ {
+ vdr->stopStreaming();
+ clearStreamChunks();
+ vfeed.stop();
+ afeed.stop();
+ video->stop();
+ video->blank();
+ audio->stop();
+ audio->unPause();
+ video->reset();
+ demuxer->reset();
+ break;
+ }
+ case S_PLAY:
+ {
+ abort();
+ }
+ }
+ break;
+ }
+ }
+
+ state = newState;
+}
+
+void PlayerLiveTV::threadMethod()
+{
+ logger->log("PlayerLiveTV", Log::DEBUG, "Thread started");
+
+ while(1)
+ {
+ while(!instructions.empty())
+ {
+ struct PLTVInstruction i = instructions.front();
+ instructions.pop();
+
+ logger->log("PlayerLiveTV", Log::DEBUG, "%u %lu", i.instruction, i.channelIndex);
+
+
+ if (i.instruction == 1)
+ {
+ logger->log("PlayerLiveTV", Log::DEBUG, "start new stream");
+
+ switchState(S_PREBUFFERING);
+
+ Channel* chan = (*chanList)[i.channelIndex];
+ chan->loadPids();
+ demuxer->setVID(chan->vpid);
+ demuxer->setAID(chan->apids[0].pid);
+ logger->log("PlayerLiveTV", Log::DEBUG, "Demuxer pids: %u %u", chan->vpid, chan->apids[0].pid);
+ vdr->streamChannel(chan->number, this);
+
+ }
+ else if (i.instruction == 2)
+ {
+ logger->log("PlayerLiveTV", Log::DEBUG, "Stopping");
+ switchState(S_STOP);
+
+ stopNow = true;
+ break;
+ }
+ }
+
+ if (stopNow) break;
+
+ if (streamChunks.size())
+ {
+ if (state == S_PREBUFFERING)
+ {
+ if (streamChunks.size() > 0)
+ {
+ while(streamChunks.size()) chunkToDemuxer();
+ switchState(S_PLAY);
+ }
+ }
+ else if (state == S_PLAY)
+ {
+ while(streamChunks.size()) chunkToDemuxer();
+ }
+ }
+
+ threadLock();
+ threadWaitForSignal(); // unlocks and waits for signal
+
+ threadUnlock();
+ logger->log("PlayerLiveTV", Log::DEBUG, "Woken");
+ }
+
+ logger->log("PlayerLiveTV", Log::DEBUG, "End of thread");
+}
+
+void PlayerLiveTV::threadPostStopCleanup()
+{
+ logger->log("PlayerLiveTV", Log::DEBUG, "Post stop cleanup");
+}
+
+// ----------------------------------- Dev
+
+#ifdef DEV
+void PlayerLiveTV::test1()
+{
+ logger->log("PlayerLiveTV", Log::DEBUG, "PLAYER TEST 1");
+}
+
+void PlayerLiveTV::test2()
+{
+ logger->log("PlayerLiveTV", Log::DEBUG, "PLAYER TEST 2");
+}
+#endif
+
--- /dev/null
+/*
+ Copyright 2004-2005 Chris Tallon
+
+ This file is part of VOMP.
+
+ VOMP 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.
+
+ VOMP 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 VOMP; if not, write to the Free Software
+ Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
+*/
+
+#ifndef PLAYER_H
+#define PLAYER_H
+
+#include <stdio.h>
+#include <stdlib.h>
+#ifndef WIN32
+#include <sys/time.h>
+#endif
+#include <time.h>
+
+#include <queue>
+
+#ifdef WIN32
+#include "threadwin.h"
+#else
+#include "threadp.h"
+#endif
+
+#include "callback.h"
+#include "defines.h"
+#include "vfeed.h"
+#include "afeed.h"
+#include "vdr.h"
+
+class MessageQueue;
+class Audio;
+class Video;
+class Log;
+class DemuxerTS;
+
+struct PLTVInstruction
+{
+ UCHAR instruction; // 1 = setChannel, 2 = stop
+ ULONG channelIndex;
+};
+
+struct StreamChunk
+{
+ void* data;
+ ULONG len;
+};
+
+class PlayerLiveTV : public Thread_TYPE, public Callback, public StreamReceiver
+{
+ public:
+ PlayerLiveTV(MessageQueue* messageQueue, void* messageReceiver, ChannelList* chanList);
+ virtual ~PlayerLiveTV();
+
+ int init();
+ int shutdown();
+
+ void go(ULONG index);
+ void setChannel(ULONG index);
+ void stop();
+ void setAudioChannel(int newChannel);
+
+ bool* getDemuxerMpegAudioChannels();
+ bool* getDemuxerAc3AudioChannels();
+ int getCurrentAudioChannel();
+
+ void call(void*); // for callback interface
+
+ virtual void streamReceive(void*, ULONG); // stream receiver interface
+
+ // Player events
+
+ // FIXME so far this just duplicates the old system + the wss
+
+ const static UCHAR CONNECTION_LOST = 1;
+ const static UCHAR STOP_PLAYBACK = 2;
+ const static UCHAR STREAM_END = 3;
+ const static UCHAR ASPECT43 = 4;
+ const static UCHAR ASPECT169 = 5;
+
+#ifdef DEV
+ void test1();
+ void test2();
+#endif
+
+ protected:
+ void threadMethod();
+ void threadPostStopCleanup();
+
+ private:
+
+ MessageQueue* messageQueue;
+ void* messageReceiver;
+ Log* logger;
+ Audio* audio;
+ Video* video;
+ DemuxerTS* demuxer;
+ VDR* vdr;
+ VFeed vfeed;
+ AFeed afeed;
+ ChannelList* chanList;
+
+ queue<PLTVInstruction> instructions;
+ queue<StreamChunk> streamChunks;
+
+ bool initted;
+
+ UCHAR state;
+ const static UCHAR S_STOP = 1;
+ const static UCHAR S_PREBUFFERING = 2;
+ const static UCHAR S_PLAY = 3;
+ void switchState(UCHAR newState);
+
+ bool videoStartup;
+ bool stopNow;
+
+ void clearStreamChunks();
+ void chunkToDemuxer();
+
+};
+
+#endif
+