testProgs/testDVVideoStreamer.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 DV 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.dv";
00027 DVVideoStreamFramer* 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 'DV Video RTP' sink from the RTP 'groupsock':
00057   // (But first, make sure that its buffers will be large enough to handle the huge size of DV frames (as big as 288000).)
00058   OutPacketBuffer::maxSize = 300000;
00059   videoSink = DVVideoRTPSink::createNew(*env, &rtpGroupsock, 96);
00060 
00061   // Create (and start) a 'RTCP instance' for this RTP sink:
00062   const unsigned estimatedSessionBandwidth = 50000; // in kbps; for RTCP b/w share
00063   const unsigned maxCNAMElen = 100;
00064   unsigned char CNAME[maxCNAMElen+1];
00065   gethostname((char*)CNAME, maxCNAMElen);
00066   CNAME[maxCNAMElen] = '\0'; // just in case
00067   RTCPInstance* rtcp
00068   = RTCPInstance::createNew(*env, &rtcpGroupsock,
00069                             estimatedSessionBandwidth, CNAME,
00070                             videoSink, NULL /* we're a server */,
00071                             True /* we're a SSM source */);
00072   // Note: This starts RTCP running automatically
00073 
00074   RTSPServer* rtspServer = RTSPServer::createNew(*env, 8554);
00075   if (rtspServer == NULL) {
00076     *env << "Failed to create RTSP server: " << env->getResultMsg() << "\n";
00077     exit(1);
00078   }
00079   ServerMediaSession* sms
00080     = ServerMediaSession::createNew(*env, "testStream", inputFileName,
00081                    "Session streamed by \"testDVVideoStreamer\"",
00082                                            True /*SSM*/);
00083   sms->addSubsession(PassiveServerMediaSubsession::createNew(*videoSink, rtcp));
00084   rtspServer->addServerMediaSession(sms);
00085 
00086   char* url = rtspServer->rtspURL(sms);
00087   *env << "Play this stream using the URL \"" << url << "\"\n";
00088   delete[] url;
00089 
00090   // Start the streaming:
00091   *env << "Beginning streaming...\n";
00092   play();
00093 
00094   env->taskScheduler().doEventLoop(); // does not return
00095 
00096   return 0; // only to prevent compiler warning
00097 }
00098 
00099 void afterPlaying(void* /*clientData*/) {
00100   *env << "...done reading from file\n";
00101 
00102   videoSink->stopPlaying();
00103   Medium::close(videoSource);
00104   // Note that this also closes the input file that this source read from.
00105 
00106   // Start playing once again:
00107   play();
00108 }
00109 
00110 void play() {
00111   // Open the input file as a 'byte-stream file source':
00112   ByteStreamFileSource* fileSource
00113     = ByteStreamFileSource::createNew(*env, inputFileName);
00114   if (fileSource == NULL) {
00115     *env << "Unable to open file \"" << inputFileName
00116          << "\" as a byte-stream file source\n";
00117     exit(1);
00118   }
00119 
00120   FramedSource* videoES = fileSource;
00121 
00122   // Create a framer for the Video Elementary Stream:
00123   videoSource = DVVideoStreamFramer::createNew(*env, videoES);
00124 
00125   // Finally, start playing:
00126   *env << "Beginning to read from file...\n";
00127   videoSink->startPlaying(*videoSource, afterPlaying, videoSink);
00128 }

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