-
Notifications
You must be signed in to change notification settings - Fork 0
/
H264VideoOnDemandServerMediaSubsession.cpp
143 lines (126 loc) · 5.52 KB
/
H264VideoOnDemandServerMediaSubsession.cpp
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
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
/**********
This library is free software; you can redistribute it and/or modify it under
the terms of the GNU Lesser General Public License as published by the
Free Software Foundation; either version 2.1 of the License, or (at your
option) any later version. (See <http://www.gnu.org/copyleft/lesser.html>.)
This library 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 Lesser General Public License for
more details.
You should have received a copy of the GNU Lesser General Public License
along with this library; if not, write to the Free Software Foundation, Inc.,
51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
**********/
// "liveMedia"
// Copyright (c) 1996-2013 Live Networks, Inc. All rights reserved.
// A 'ServerMediaSubsession' object that creates new, unicast, "RTPSink"s
// on demand, from a H264 video file.
// Implementation
#ifndef INT64_C
#define INT64_C(c) (c ## LL)
#define UINT64_C(c) (c ## ULL)
#endif
#include "H264VideoOnDemandServerMediaSubsession.hh"
#include "H264VideoRTPSink.hh"
#include "MyDeviceSource.hh"
#include "H264VideoStreamFramer.hh"
UsageEnvironment* envi;
H264VideoOnDemandServerMediaSubsession*
H264VideoOnDemandServerMediaSubsession::createNew(UsageEnvironment& env,
char const* fileName,
Boolean reuseFirstSource) {
return new H264VideoOnDemandServerMediaSubsession(env, fileName, reuseFirstSource);
}
H264VideoOnDemandServerMediaSubsession::H264VideoOnDemandServerMediaSubsession(UsageEnvironment& env, char const* fileName, Boolean reuseFirstSource)
: OnDemandServerMediaSubsession(env, reuseFirstSource),
fAuxSDPLine(NULL), fDoneFlag(0), fDummyRTPSink(NULL) {
envi = &env;
}
H264VideoOnDemandServerMediaSubsession::~H264VideoOnDemandServerMediaSubsession() {
delete[] fAuxSDPLine;
}
/*
static void afterPlayingDummy(void* clientData) {
H264VideoFileServerMediaSubsession* subsess = (H264VideoFileServerMediaSubsession*)clientData;
subsess->afterPlayingDummy1();
}
void H264VideoFileServerMediaSubsession::afterPlayingDummy1() {
// Unschedule any pending 'checking' task:
envir().taskScheduler().unscheduleDelayedTask(nextTask());
// Signal the event loop that we're done:
setDoneFlag();
}
*/
FramedSource* H264VideoOnDemandServerMediaSubsession::createNewStreamSource(unsigned /*clientSessionId*/, unsigned& estBitrate)
{
estBitrate = 4000; // kbps, estimate
// Create the video source:
//ByteStreamFileSource* fileSource = ByteStreamFileSource::createNew(envir(), fFileName);
MyDeviceSource *fileSource = MyDeviceSource::createNew(*envi);
//FramedSource wrapper ?
if (fileSource == NULL) return NULL;
//fFileSize = fileSource->fileSize();
// Create a framer for the Video Elementary Stream:
return H264VideoStreamFramer::createNew(envir(), fileSource);
}
RTPSink* H264VideoOnDemandServerMediaSubsession
::createNewRTPSink(Groupsock* rtpGroupsock,
unsigned char rtpPayloadTypeIfDynamic,
FramedSource* /*inputSource*/)
{
return H264VideoRTPSink::createNew(envir(), rtpGroupsock, rtpPayloadTypeIfDynamic);
}
static void afterPlayingDummy(void* clientData) {
H264VideoOnDemandServerMediaSubsession* subsess = (H264VideoOnDemandServerMediaSubsession*)clientData;
subsess->afterPlayingDummy1();
}
void H264VideoOnDemandServerMediaSubsession::afterPlayingDummy1() {
// Unschedule any pending 'checking' task:
envir().taskScheduler().unscheduleDelayedTask(nextTask());
// Signal the event loop that we're done:
setDoneFlag();
}
//static void checkForAuxSDPLine(void* clientData) {
// H264VideoOnDemandServerMediaSubsession* subsess = (H264VideoOnDemandServerMediaSubsession*)clientData;
// subsess->checkForAuxSDPLine1();
//}
//
//void H264VideoOnDemandServerMediaSubsession::checkForAuxSDPLine1() {
// char const* dasl = NULL;
//
// if (fAuxSDPLine != NULL) {
// // Signal the event loop that we're done:
// setDoneFlag();
// } else if (fDummyRTPSink != NULL && (dasl = fDummyRTPSink->auxSDPLine()) != NULL) {
// fAuxSDPLine = strDup(dasl);
// fDummyRTPSink = NULL;
//
// // Signal the event loop that we're done:
// setDoneFlag();
// } else {
// // try again after a brief delay:
// int uSecsToDelay = 100000; // 100 ms
// nextTask() = envir().taskScheduler().scheduleDelayedTask(uSecsToDelay,
// (TaskFunc*)checkForAuxSDPLine, this);
// }
//}
//char const* H264VideoOnDemandServerMediaSubsession::getAuxSDPLine(RTPSink* rtpSink, FramedSource* inputSource) {
// if (fAuxSDPLine != NULL) return fAuxSDPLine; // it's already been set up (for a previous client)
//
// if (fDummyRTPSink == NULL) { // we're not already setting it up for another, concurrent stream
// // Note: For H264 video files, the 'config' information ("profile-level-id" and "sprop-parameter-sets") isn't known
// // until we start reading the file. This means that "rtpSink"s "auxSDPLine()" will be NULL initially,
// // and we need to start reading data from our file until this changes.
// fDummyRTPSink = rtpSink;
//
// // Start reading the file:
// fDummyRTPSink->startPlaying(*inputSource, afterPlayingDummy, this);
//
// // Check whether the sink's 'auxSDPLine()' is ready:
// checkForAuxSDPLine(this);
// }
//
// envir().taskScheduler().doEventLoop(&fDoneFlag);
//
// return fAuxSDPLine;
//}