120 lines
4.6 KiB
C++
120 lines
4.6 KiB
C++
|
/**********
|
||
|
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-2016 Live Networks, Inc. All rights reserved.
|
||
|
// A 'ServerMediaSubsession' object that creates new, unicast, "RTPSink"s
|
||
|
// on demand, from a H264 video file.
|
||
|
// Implementation
|
||
|
|
||
|
#include "H264VideoFileServerMediaSubsession.hh"
|
||
|
#include "H264VideoRTPSink.hh"
|
||
|
#include "ByteStreamFileSource.hh"
|
||
|
#include "H264VideoStreamFramer.hh"
|
||
|
|
||
|
H264VideoFileServerMediaSubsession*
|
||
|
H264VideoFileServerMediaSubsession::createNew(UsageEnvironment& env,
|
||
|
char const* fileName,
|
||
|
Boolean reuseFirstSource) {
|
||
|
return new H264VideoFileServerMediaSubsession(env, fileName, reuseFirstSource);
|
||
|
}
|
||
|
|
||
|
H264VideoFileServerMediaSubsession::H264VideoFileServerMediaSubsession(UsageEnvironment& env,
|
||
|
char const* fileName, Boolean reuseFirstSource)
|
||
|
: FileServerMediaSubsession(env, fileName, reuseFirstSource),
|
||
|
fAuxSDPLine(NULL), fDoneFlag(0), fDummyRTPSink(NULL) {
|
||
|
}
|
||
|
|
||
|
H264VideoFileServerMediaSubsession::~H264VideoFileServerMediaSubsession() {
|
||
|
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();
|
||
|
}
|
||
|
|
||
|
static void checkForAuxSDPLine(void* clientData) {
|
||
|
H264VideoFileServerMediaSubsession* subsess = (H264VideoFileServerMediaSubsession*)clientData;
|
||
|
subsess->checkForAuxSDPLine1();
|
||
|
}
|
||
|
|
||
|
void H264VideoFileServerMediaSubsession::checkForAuxSDPLine1() {
|
||
|
char const* dasl;
|
||
|
|
||
|
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 if (!fDoneFlag) {
|
||
|
// try again after a brief delay:
|
||
|
int uSecsToDelay = 100000; // 100 ms
|
||
|
nextTask() = envir().taskScheduler().scheduleDelayedTask(uSecsToDelay,
|
||
|
(TaskFunc*)checkForAuxSDPLine, this);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
char const* H264VideoFileServerMediaSubsession::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;
|
||
|
}
|
||
|
|
||
|
FramedSource* H264VideoFileServerMediaSubsession::createNewStreamSource(unsigned /*clientSessionId*/, unsigned& estBitrate) {
|
||
|
estBitrate = 500; // kbps, estimate
|
||
|
|
||
|
// Create the video source:
|
||
|
ByteStreamFileSource* fileSource = ByteStreamFileSource::createNew(envir(), fFileName);
|
||
|
if (fileSource == NULL) return NULL;
|
||
|
fFileSize = fileSource->fileSize();
|
||
|
|
||
|
// Create a framer for the Video Elementary Stream:
|
||
|
return H264VideoStreamFramer::createNew(envir(), fileSource);
|
||
|
}
|
||
|
|
||
|
RTPSink* H264VideoFileServerMediaSubsession
|
||
|
::createNewRTPSink(Groupsock* rtpGroupsock,
|
||
|
unsigned char rtpPayloadTypeIfDynamic,
|
||
|
FramedSource* /*inputSource*/) {
|
||
|
return H264VideoRTPSink::createNew(envir(), rtpGroupsock, rtpPayloadTypeIfDynamic);
|
||
|
}
|