liveMedia/VorbisAudioMatroskaFileServerMediaSubsession.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 // "liveMedia"
00017 // Copyright (c) 1996-2012 Live Networks, Inc.  All rights reserved.
00018 // A 'ServerMediaSubsession' object that creates new, unicast, "RTPSink"s
00019 // on demand, from a Vorbis audio track within a Matroska file.
00020 // Implementation
00021 
00022 #include "VorbisAudioMatroskaFileServerMediaSubsession.hh"
00023 #include "VorbisAudioRTPSink.hh"
00024 #include "MatroskaDemuxedTrack.hh"
00025 
00026 VorbisAudioMatroskaFileServerMediaSubsession* VorbisAudioMatroskaFileServerMediaSubsession
00027 ::createNew(MatroskaFileServerDemux& demux, unsigned trackNumber) {
00028   return new VorbisAudioMatroskaFileServerMediaSubsession(demux, trackNumber);
00029 }
00030 
00031 #define getPrivByte(b) if (n == 0) break; else do {b = *p++; --n;} while (0)
00032 
00033 VorbisAudioMatroskaFileServerMediaSubsession
00034 ::VorbisAudioMatroskaFileServerMediaSubsession(MatroskaFileServerDemux& demux, unsigned trackNumber)
00035   : FileServerMediaSubsession(demux.envir(), demux.fileName(), False),
00036     fOurDemux(demux), fTrackNumber(trackNumber),
00037     fIdentificationHeader(NULL), fIdentificationHeaderSize(0),
00038     fCommentHeader(NULL), fCommentHeaderSize(0),
00039     fSetupHeader(NULL), fSetupHeaderSize(0),
00040     fEstBitrate(96/* kbps, default guess */) {
00041   MatroskaTrack* track = fOurDemux.lookup(fTrackNumber);
00042 
00043   // The Matroska file's 'Codec Private' data is assumed to be the Vorbis configuration information,
00044   // containing the "Identification", "Comment", and "Setup" headers.  Extract these headers now:
00045   do {
00046     u_int8_t* p = track->codecPrivate;
00047     unsigned n = track->codecPrivateSize;
00048     if (n == 0 || p == NULL) break; // we have no 'Codec Private' data
00049 
00050     u_int8_t numHeaders;
00051     getPrivByte(numHeaders);
00052     unsigned headerSize[3]; // we don't handle any more than 2+1 headers
00053 
00054     // Extract the sizes of each of these headers:
00055     unsigned sizesSum = 0;
00056     Boolean success = True;
00057     unsigned i;
00058     for (i = 0; i < numHeaders && i < 3; ++i) {
00059       unsigned len = 0;
00060       u_int8_t c;
00061 
00062       do {
00063         success = False;
00064         getPrivByte(c);
00065         success = True;
00066 
00067         len += c;
00068       } while (c == 255);
00069       if (!success || len == 0) break;
00070 
00071       headerSize[i] = len;
00072       sizesSum += len;
00073     }
00074     if (!success) break;
00075 
00076     // Compute the implicit size of the final header:
00077     if (numHeaders < 3) {
00078       int finalHeaderSize = n - sizesSum;
00079       if (finalHeaderSize <= 0) break; // error in data; give up
00080 
00081       headerSize[numHeaders] = (unsigned)finalHeaderSize;
00082       ++numHeaders; // include the final header now
00083     } else {
00084       numHeaders = 3; // The maximum number of headers that we handle
00085     }
00086 
00087     // Then, extract and classify each header:
00088     for (i = 0; i < numHeaders; ++i) {
00089       success = False;
00090       unsigned newHeaderSize = headerSize[i];
00091       u_int8_t* newHeader = new u_int8_t[newHeaderSize];
00092       if (newHeader == NULL) break;
00093       
00094       u_int8_t* hdr = newHeader;
00095       while (newHeaderSize-- > 0) {
00096         success = False;
00097         getPrivByte(*hdr++);
00098         success = True;
00099       }
00100       if (!success) {
00101         delete[] newHeader;
00102         break;
00103       }
00104 
00105       u_int8_t headerType = newHeader[0];
00106       if (headerType == 1) {
00107         delete[] fIdentificationHeader; fIdentificationHeader = newHeader;
00108         fIdentificationHeaderSize = headerSize[i];
00109 
00110         if (fIdentificationHeaderSize >= 28) {
00111           // Get the 'bitrate' values from this header, and use them to set "fEstBitrate":
00112           u_int32_t val;
00113           u_int8_t* p;
00114 
00115           p = &fIdentificationHeader[16];
00116           val = ((p[3]*256 + p[2])*256 + p[1])*256 + p[0]; // i.e., little-endian
00117           int bitrate_maximum = (int)val;
00118           if (bitrate_maximum < 0) bitrate_maximum = 0;
00119 
00120           p = &fIdentificationHeader[20];
00121           val = ((p[3]*256 + p[2])*256 + p[1])*256 + p[0]; // i.e., little-endian
00122           int bitrate_nominal = (int)val;
00123           if (bitrate_nominal < 0) bitrate_nominal = 0;
00124 
00125           p = &fIdentificationHeader[24];
00126           val = ((p[3]*256 + p[2])*256 + p[1])*256 + p[0]; // i.e., little-endian
00127           int bitrate_minimum = (int)val;
00128           if (bitrate_minimum < 0) bitrate_minimum = 0;
00129 
00130           int bitrate
00131             = bitrate_nominal>0 ? bitrate_nominal : bitrate_maximum>0 ? bitrate_maximum : bitrate_minimum>0 ? bitrate_minimum : 0;
00132           if (bitrate > 0) fEstBitrate = ((unsigned)bitrate)/1000;
00133         }
00134       } else if (headerType == 3) {
00135         delete[] fCommentHeader; fCommentHeader = newHeader;
00136         fCommentHeaderSize = headerSize[i];
00137       } else if (headerType == 5) {
00138         delete[] fSetupHeader; fSetupHeader = newHeader;
00139         fSetupHeaderSize = headerSize[i];
00140       } else {
00141         delete[] newHeader; // because it was a header type that we don't understand
00142       }
00143     }
00144     if (!success) break;
00145   } while (0);
00146 }
00147 
00148 VorbisAudioMatroskaFileServerMediaSubsession
00149 ::~VorbisAudioMatroskaFileServerMediaSubsession() {
00150   delete[] fIdentificationHeader;
00151   delete[] fCommentHeader;
00152   delete[] fSetupHeader;
00153 }
00154 
00155 float VorbisAudioMatroskaFileServerMediaSubsession::duration() const { return fOurDemux.fileDuration(); }
00156 
00157 void VorbisAudioMatroskaFileServerMediaSubsession
00158 ::seekStreamSource(FramedSource* inputSource, double& seekNPT, double /*streamDuration*/, u_int64_t& /*numBytes*/) {
00159   ((MatroskaDemuxedTrack*)inputSource)->seekToTime(seekNPT);
00160 }
00161 
00162 FramedSource* VorbisAudioMatroskaFileServerMediaSubsession
00163 ::createNewStreamSource(unsigned clientSessionId, unsigned& estBitrate) {
00164   estBitrate = fEstBitrate; // kbps, estimate
00165 
00166   return fOurDemux.newDemuxedTrack(clientSessionId, fTrackNumber);
00167 }
00168 
00169 RTPSink* VorbisAudioMatroskaFileServerMediaSubsession
00170 ::createNewRTPSink(Groupsock* rtpGroupsock, unsigned char rtpPayloadTypeIfDynamic, FramedSource* /*inputSource*/) {
00171   MatroskaTrack* track = fOurDemux.lookup(fTrackNumber);
00172   return VorbisAudioRTPSink::createNew(envir(), rtpGroupsock,
00173                                        rtpPayloadTypeIfDynamic, track->samplingFrequency, track->numChannels,
00174                                        fIdentificationHeader, fIdentificationHeaderSize,
00175                                        fCommentHeader, fCommentHeaderSize,
00176                                        fSetupHeader, fSetupHeaderSize);
00177 }

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