summaryrefslogtreecommitdiff
path: root/server/streamer.c
blob: 465eb889d997e4e00e1c3a54f242f3d215504e96 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
/*
 *  $Id: streamer.c,v 1.3 2005/02/08 17:22:35 lordjaxom Exp $
 */
 
#include <vdr/ringbuffer.h>
#include <vdr/device.h>
#include <sys/types.h>
#include <unistd.h>

#include "server/streamer.h"
#include "server/suspend.h"
#include "server/setup.h"
#include "tools/socket.h"
#include "common.h"

#define VIDEOBUFSIZE MEGABYTE(4)
#define MAXBLOCKSIZE TS_SIZE*10

cStreamdevStreamer::cStreamdevStreamer(const char *Name):
		cThread(((std::string)"Streamdev: " + Name).c_str())
{
	m_Active     = false;
	m_Receivers  = 0;
	m_Buffer     = NULL;
	m_Name       = Name;
	m_Socket     = NULL;
	m_RingBuffer = new cRingBufferLinear(VIDEOBUFSIZE, TS_SIZE * 2, true);
}

cStreamdevStreamer::~cStreamdevStreamer() {
	Stop();
	if (m_Buffer != NULL) delete[] m_Buffer;
	delete m_RingBuffer;
}

void cStreamdevStreamer::Start(cTBSocket *Socket) {
	m_Socket = Socket;
	Attach();
	if (!m_Active)
		cThread::Start();
}

void cStreamdevStreamer::Stop(void) {
	if (m_Active) {
		Dprintf("stopping live streamer\n");
		m_Active = false;
		Cancel(3);
	}
}

uchar *cStreamdevStreamer::Process(const uchar *Data, int &Count, int &Result) {
	if (m_Buffer == NULL)
		m_Buffer = new uchar[MAXBLOCKSIZE];

	if (Count > MAXBLOCKSIZE)
		Count = MAXBLOCKSIZE;
	memcpy(m_Buffer, Data, Count);
	Result = Count;
	return m_Buffer;
}

void cStreamdevStreamer::Action(void) {
	int max = 0;

#if VDRVERSNUM < 10300
	isyslog("Streamdev: %s thread started (pid=%d)", m_Name, getpid());
#endif

	m_Active = true;
	while (m_Active) {
		int recvd;
		const uchar *block = m_RingBuffer->Get(recvd);

		if (block && recvd > 0) {
			int result = 0;
			uchar *sendBlock = Process(block, recvd, result);

			m_RingBuffer->Del(recvd);
			if (result > max) max = result;

			if (!m_Socket->TimedWrite(sendBlock, result, 150)) {
				if (errno != ETIMEDOUT) {
					esyslog("ERROR: Streamdev: Couldn't write data: %s", strerror(errno));
					m_Active = false;
				}
			}
		} else
			usleep(1); // this keeps the CPU load low (XXX: waiting buffers)
	}

	Dprintf("Max. Transmit Blocksize was: %d\n", max);

#if VDRVERSNUM < 10300
	isyslog("Streamdev: %s thread stopped", m_Name);
#endif
}