testProgs/testMPEG4VideoStreamer.cpp

Go to the documentation of this file.
00001 /**********
00002 This library is free software; you can redistribute it and/or modify it under
00003 the terms of the GNU Lesser General Public License as published by the
00004 Free Software Foundation; either version 2.1 of the License, or (at your
00005 option) any later version. (See <http://www.gnu.org/copyleft/lesser.html>.)
00006 
00007 This library is distributed in the hope that it will be useful, but WITHOUT
00008 ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
00009 FOR A PARTICULAR PURPOSE.  See the GNU Lesser General Public License for
00010 more details.
00011 
00012 You should have received a copy of the GNU Lesser General Public License
00013 along with this library; if not, write to the Free Software Foundation, Inc.,
00014 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301  USA
00015 **********/
00016 // Copyright (c) 1996-2012, Live Networks, Inc.  All rights reserved
00017 // A test program that reads a MPEG-4 Video Elementary Stream file,
00018 // and streams it using RTP
00019 // main program
00020 
00021 #include "liveMedia.hh"
00022 #include "BasicUsageEnvironment.hh"
00023 #include "GroupsockHelper.hh"
00024 
00025 UsageEnvironment* env;
00026 char const* inputFileName = "test.m4e";
00027 MPEG4VideoStreamFramer* videoSource;
00028 RTPSink* videoSink;
00029 
00030 void play(); // forward
00031 
00032 int main(int argc, char** argv) {
00033   // Begin by setting up our usage environment:
00034   TaskScheduler* scheduler = BasicTaskScheduler::createNew();
00035   env = BasicUsageEnvironment::createNew(*scheduler);
00036 
00037   // Create 'groupsocks' for RTP and RTCP:
00038   struct in_addr destinationAddress;
00039   destinationAddress.s_addr = chooseRandomIPv4SSMAddress(*env);
00040   // Note: This is a multicast address.  If you wish instead to stream
00041   // using unicast, then you should use the "testOnDemandRTSPServer"
00042   // test program - not this test program - as a model.
00043 
00044   const unsigned short rtpPortNum = 18888;
00045   const unsigned short rtcpPortNum = rtpPortNum+1;
00046   const unsigned char ttl = 255;
00047 
00048   const Port rtpPort(rtpPortNum);
00049   const Port rtcpPort(rtcpPortNum);
00050 
00051   Groupsock rtpGroupsock(*env, destinationAddress, rtpPort, ttl);
00052   rtpGroupsock.multicastSendOnly(); // we're a SSM source
00053   Groupsock rtcpGroupsock(*env, destinationAddress, rtcpPort, ttl);
00054   rtcpGroupsock.multicastSendOnly(); // we're a SSM source
00055 
00056   // Create a 'MPEG-4 Video RTP' sink from the RTP 'groupsock':
00057   videoSink = MPEG4ESVideoRTPSink::createNew(*env, &rtpGroupsock, 96);
00058 
00059   // Create (and start) a 'RTCP instance' for this RTP sink:
00060   const unsigned estimatedSessionBandwidth = 500; // in kbps; for RTCP b/w share
00061   const unsigned maxCNAMElen = 100;
00062   unsigned char CNAME[maxCNAMElen+1];
00063   gethostname((char*)CNAME, maxCNAMElen);
00064   CNAME[maxCNAMElen] = '\0'; // just in case
00065   RTCPInstance* rtcp
00066   = RTCPInstance::createNew(*env, &rtcpGroupsock,
00067                             estimatedSessionBandwidth, CNAME,
00068                             videoSink, NULL /* we're a server */,
00069                             True /* we're a SSM source */);
00070   // Note: This starts RTCP running automatically
00071 
00072   RTSPServer* rtspServer = RTSPServer::createNew(*env, 8554);
00073   if (rtspServer == NULL) {
00074     *env << "Failed to create RTSP server: " << env->getResultMsg() << "\n";
00075     exit(1);
00076   }
00077   ServerMediaSession* sms
00078     = ServerMediaSession::createNew(*env, "testStream", inputFileName,
00079                    "Session streamed by \"testMPEG4VideoStreamer\"",
00080                                            True /*SSM*/);
00081   sms->addSubsession(PassiveServerMediaSubsession::createNew(*videoSink, rtcp));
00082   rtspServer->addServerMediaSession(sms);
00083 
00084   char* url = rtspServer->rtspURL(sms);
00085   *env << "Play this stream using the URL \"" << url << "\"\n";
00086   delete[] url;
00087 
00088   // Start the streaming:
00089   *env << "Beginning streaming...\n";
00090   play();
00091 
00092   env->taskScheduler().doEventLoop(); // does not return
00093 
00094   return 0; // only to prevent compiler warning
00095 }
00096 
00097 void afterPlaying(void* /*clientData*/) {
00098   *env << "...done reading from file\n";
00099 
00100   videoSink->stopPlaying();
00101   Medium::close(videoSource);
00102   // Note that this also closes the input file that this source read from.
00103 
00104   // Start playing once again:
00105   play();
00106 }
00107 
00108 void play() {
00109   // Open the input file as a 'byte-stream file source':
00110   ByteStreamFileSource* fileSource
00111     = ByteStreamFileSource::createNew(*env, inputFileName);
00112   if (fileSource == NULL) {
00113     *env << "Unable to open file \"" << inputFileName
00114          << "\" as a byte-stream file source\n";
00115     exit(1);
00116   }
00117 
00118   FramedSource* videoES = fileSource;
00119 
00120   // Create a framer for the Video Elementary Stream:
00121   videoSource = MPEG4VideoStreamFramer::createNew(*env, videoES);
00122 
00123   // Finally, start playing:
00124   *env << "Beginning to read from file...\n";
00125   videoSink->startPlaying(*videoSource, afterPlaying, videoSink);
00126 }

Generated on Thu May 17 07:11:48 2012 for live by  doxygen 1.5.2