live555源码分析----mpg文件的处理

 

live555源码分析----mpg文件的处理


    live555支持的文件格式多为单流的文件,仅支持*.mpg、*.mkv、*.webm几种音视频混合类型的文件。其实我的目的是扩展其支持的格式,如avi等, 所以来分析一下mpg文件的处理。


    mpg文件处理涉及的类相当多,类间的关系比较复杂,对于RTP打包的过程(在RTPSink中完成)所以有的媒体类型比较相似(细节上有差异,这些都是在特定媒体相关的***RTPSink中完成的),所以主要分析的是从source中获取数据的过程。一个subsession对应一个流,当一个session中拥有多个subsession时,需要对每一个subsession进行单独的控制。我们可以看到,对处理“PLAY”命令时,对session中的每个subsession都调用了一次startStream操作,如下:


[cpp]  view plain copy print ?
  1. void RTSPServer::RTSPClientSession  
  2.   ::handleCmd_PLAY(ServerMediaSubsession* subsession, char const* cseq,  
  3.   char const* fullRequestStr) {  
  4.   
  5.   //现在终于开始媒体数据传输了  
  6.   
  7.   // Now, start streaming:  
  8.   for (i = 0; i < fNumStreamStates; ++i) {  
  9.     if (subsession == NULL /* means: aggregated operation */  
  10. || subsession == fStreamStates[i].subsession) {  
  11.       unsigned short rtpSeqNum = 0;  
  12.       unsigned rtpTimestamp = 0;  
  13.   
  14.       //开始各个subsession上的数据传输, 即开始播放了  
  15.   
  16.       fStreamStates[i].subsession->startStream(fOurSessionId,  
  17.       fStreamStates[i].streamToken,  
  18.       (TaskFunc*)noteClientLiveness, this,  
  19.       rtpSeqNum, rtpTimestamp,  
  20.       handleAlternativeRequestByte, this);  
  21.     ...  
  22.     }  
  23.   }  
  24.     
  25. ...  
  26. }  


    
    PLAY命令的处理过程前面的文章已经分析过了,通过subsession->startStream调用启动每个流上的播放,其从source获取源数据是将在MultiFramedRTPSink::packFrame()中进行的。


[cpp]  view plain copy print ?
  1. void MultiFramedRTPSink::packFrame() {  
  2.   if (fOutBuf->haveOverflowData()) {  
  3. ...  
  4.   } else {  
  5. ...  
  6.   
  7.     //  
  8.     //从source中获取下一个frame  
  9.     //  
  10.     fSource->getNextFrame(fOutBuf->curPtr(), fOutBuf->totalBytesAvailable(),  
  11.               afterGettingFrame, this, ourHandleClosure, this);  
  12.   }  
  13. }  


    对于mpeg来讲,fSource是一个MPEG1or2VideoStreamFramer或者MPEG1or2AudioStreamFramer实例(根据live555中的源码,还有一种AAC格式音频,这里为了简便不作分析)。它们的继承关系如下:
    MPEG1or2VideoStreamFramer->MPEGVideoStreamFramer->FramedFilter->FramedSource->MediaSource
    MPEG1or2AudioStreamFramer->FramedSource->MediaSource


    先来分析mpeg video的处理。
    MPEG1or2VideoStreamFramer类的实现简单,主要是创建对应的语法分析器MPEG1or2VideoStreamParser实例。显然MPEG1or2VideoStreamFramer(是FrameSource的直接子类)是一个Filter,跟它的创建过程可以发现,它的输入source是一个MPEG1or2DemuxedElementaryStream类实例。对于单流的文件来讲, 一般包装的是ByteStreamFileSource类实例,从后面我们可以发现,最终直接读取文件的还是ByteStreamFileSource实例。对于语法分析部分,不作分析,我们只关心如何从文件中解析出音视频数据,所以直接跟踪Filter所包装的MPEG1or2DemuxedElementaryStream类。在语法分析器中,将会调用MPEG1or2DemuxedElementaryStream的getNextFrame函数。
    

    getNextFrame是定义在FramedSource中的非虚函数,实现如下

[cpp]  view plain copy print ?
  1. void FramedSource::getNextFrame(unsigned char* to, unsigned maxSize,  
  2.                 afterGettingFunc* afterGettingFunc,  
  3.                 void* afterGettingClientData,  
  4.                 onCloseFunc* onCloseFunc,  
  5.                 void* onCloseClientData) {  
  6.   // Make sure we're not already being read:  
  7.   if (fIsCurrentlyAwaitingData) {  
  8.     envir() << "FramedSource[" << this << "]::getNextFrame(): attempting to read more than once at the same time!\n";  
  9.     envir().internalError();  
  10.   }  
  11.   
  12.   
  13.   fTo = to;  
  14.   fMaxSize = maxSize;  
  15.   fNumTruncatedBytes = 0; // by default; could be changed by doGetNextFrame()  
  16.   fDurationInMicroseconds = 0; // by default; could be changed by doGetNextFrame()  
  17.   fAfterGettingFunc = afterGettingFunc;  
  18.   fAfterGettingClientData = afterGettingClientData;  
  19.   fOnCloseFunc = onCloseFunc;  
  20.   fOnCloseClientData = onCloseClientData;  
  21.   fIsCurrentlyAwaitingData = True;  
  22.   
  23.   
  24.   doGetNextFrame();     //获取下一个frame  
  25. }  



    FramedSource::getNextFrame完成了一些员变量的初始化工作,其它工作交给dogetNextFrame函数,它是FrameSource上的一个纯虚函数,在子类MPEG1or2DemuxedElementaryStream中重新实现。
    
[cpp]  view plain copy print ?
  1. void MPEG1or2DemuxedElementaryStream::doGetNextFrame() {  
  2.   fOurSourceDemux.getNextFrame(fOurStreamIdTag, fTo, fMaxSize,  
  3.                    afterGettingFrame, this,  
  4.                    handleClosure, this);  
  5. }  



    fOurSourceDemux,被定义为MPEG1or2Demux(直接继承自Medium)对像的引用。为什么是引用而不是指针?我们想想,前面涉及到的类MPEG1or2DemuxedServerMediaSubsession、MPEG1or2VideoRTPSink、MPEG1or2VideoStreamFramer、MPEG1or2VideoStreamParser,这些对于文件中的每一个流(当然这里只是列出了视频相关的类)都需要创建一个实例,但是所有的流对应的文件却只有一个,最后读取文件的过程必然只有一份实现。文件中的每个流会对应一个MPEG1or2DemuxedElementaryStream实例,但是却对应同一个MPEG1or2Demux实例。这里用引用而不是指针,就是为了说明fOurSourceDemux并属于某个MPEG1or2DemuxedElementaryStream实例。


    在创建MPEG1or2Demux实例之前,会创建一个ByteStreamFileSource,来看MPEG1or2Demux::getNextFrame函数的定义

[cpp]  view plain copy print ?
  1. void MPEG1or2Demux::getNextFrame(u_int8_t streamIdTag,  
  2.                  unsigned char* to, unsigned maxSize,  
  3.                  FramedSource::afterGettingFunc* afterGettingFunc,  
  4.                  void* afterGettingClientData,  
  5.                  FramedSource::onCloseFunc* onCloseFunc,  
  6.                  void* onCloseClientData) {  
  7.   // First, check whether we have saved data for this stream id:  
  8.   //  
  9.   // 检查缓存中是否已经存在streamIdTag流的数据  
  10.   //  
  11.   if (useSavedData(streamIdTag, to, maxSize,  
  12.            afterGettingFunc, afterGettingClientData)) {  
  13.     return;  
  14.   }  
  15.       
  16.   //注意,这里设置了回调用函数  
  17.   // Then save the parameters of the specified stream id:  
  18.   registerReadInterest(streamIdTag, to, maxSize,  
  19.                afterGettingFunc, afterGettingClientData,  
  20.                onCloseFunc, onCloseClientData);  
  21.   
  22.   
  23.   // Next, if we're the only currently pending read, continue looking for data:  
  24.   if (fNumPendingReads == 1 || fHaveUndeliveredData) {  
  25.     fHaveUndeliveredData = 0;  
  26.     continueReadProcessing();       //继续读取数据  
  27.   } // otherwise the continued read processing has already been taken care of  
  28. }  


    
    上面的代码中首先调用useSavedData函数,检查缓存中是否有相应流的数据存在,若是不存在,就只有再从文件中读取了。现在可以做一个猜测,缓存是必需的,文件中的不同流数据交错排列,而读取时一般会按顺序来读。现在要读取一个视频frame,但是当前文件指针处读取到的刚好是音视频frame,我们就需要将这个音频frame保存到缓存中,这样一直到读取视频数据为止。实际情况是不是这样呢?
    
    先来看MPEG1or2Demux::useSavedData函数的实现
[cpp]  view plain copy print ?
  1. Boolean MPEG1or2Demux::useSavedData(u_int8_t streamIdTag,  
  2.                     unsigned char* to, unsigned maxSize,  
  3.                     FramedSource::afterGettingFunc* afterGettingFunc,  
  4.                     void* afterGettingClientData) {  
  5.   struct OutputDescriptor& out = fOutput[streamIdTag];      //fOutput是一个缓存数组  
  6.   //正常情况,缓存中没有数据,直接返回了  
  7.   if (out.savedDataHead == NULL) return False; // common case  
  8.   
  9.   
  10.   unsigned totNumBytesCopied = 0;  
  11.   
  12.   
  13.   //  
  14.   //从OutputDescriptor类型的缓存中读取全部数据  
  15.   //  
  16.   while (maxSize > 0 && out.savedDataHead != NULL) {  
  17.     OutputDescriptor::SavedData& savedData = *(out.savedDataHead);  
  18.     unsigned char* from = &savedData.data[savedData.numBytesUsed];  
  19.     unsigned numBytesToCopy = savedData.dataSize - savedData.numBytesUsed;  
  20.     if (numBytesToCopy > maxSize) numBytesToCopy = maxSize;   
  21.     memmove(to, from, numBytesToCopy);  
  22.     to += numBytesToCopy;  
  23.     maxSize -= numBytesToCopy;  
  24.     out.savedDataTotalSize -= numBytesToCopy;  
  25.     totNumBytesCopied += numBytesToCopy;  
  26.     savedData.numBytesUsed += numBytesToCopy;  
  27.     if (savedData.numBytesUsed == savedData.dataSize) {  
  28.       out.savedDataHead = savedData.next;  
  29.       if (out.savedDataHead == NULL) out.savedDataTail = NULL;  
  30.       savedData.next = NULL;  
  31.       delete &savedData;  
  32.     }  
  33.   }  
  34.   
  35.   
  36.   out.isCurrentlyActive = True;  
  37.   if (afterGettingFunc != NULL) {  
  38.     struct timeval presentationTime;  
  39.     presentationTime.tv_sec = 0; presentationTime.tv_usec = 0; // should fix #####  
  40.     (*afterGettingFunc)(afterGettingClientData, totNumBytesCopied,    
  41.             0 /* numTruncatedBytes */, presentationTime,      
  42.             0 /* durationInMicroseconds ?????#####*/);  
  43.   }  
  44.   return True;  
  45. }  


    来分析一下上面的代码。缓存被定义为一个OutputDescriptor类型数组,看其定义
      OutputDescriptor_t fOutput[256];
    竟然定义了一个256个实例,不过也只有这个样才能使用流索引streamIdTag直接访问数组元素,streamIdTag应该小于256。每一个流将对应一个OutputDescriptor实例。实际数据保存在 OutputDescriptor::SavedData类型的链表中。


    现在继续来看MPEG1or2Demux::getNextFrame函数,我们需要知道是怎么分离出不同媒体流的,来看continueReadProcessing函数
[cpp]  view plain copy print ?
  1. void MPEG1or2Demux::continueReadProcessing() {  
  2.   while (fNumPendingReads > 0) {  
  3.     unsigned char acquiredStreamIdTag = fParser->parse();   //文件语法分析  
  4.       
  5.     if (acquiredStreamIdTag != 0) { //若未读取到所需要的数据,这个值为0  
  6.         //我们从输入源获取到一个frame  
  7.       // We were able to acquire a frame from the input.  
  8.       struct OutputDescriptor& newOut = fOutput[acquiredStreamIdTag];  
  9.       newOut.isCurrentlyAwaitingData = False;   //指示我们可以读下一个frame了,parse中会根据这个值判断是否正在处理的流  
  10.       // indicates that we can be read again  
  11.         // (This needs to be set before the 'after getting' call below,  
  12.         //  in case it tries to read another frame)  
  13.   
  14.   
  15.       //  
  16.       //调用“after getting”函数  
  17.       //  
  18.       // Call our own 'after getting' function.  Because we're not a 'leaf'  
  19.       // source, we can call this directly, without risking infinite recursion.  
  20.       if (newOut.fAfterGettingFunc != NULL) {  
  21.     (*newOut.fAfterGettingFunc)(newOut.afterGettingClientData,  
  22.                     newOut.frameSize, 0 /* numTruncatedBytes */,  
  23.                     newOut.presentationTime,  
  24.                     0 /* durationInMicroseconds ?????#####*/);  
  25.       --fNumPendingReads;  
  26.       }  
  27.     } else {  
  28.       // We were unable to parse a complete frame from the input, because:  
  29.       // - we had to read more data from the source stream, or  
  30.       // - we found a frame for a stream that was being read, but whose  
  31.       //   reader is not ready to get the frame right now, or  
  32.       // - the source stream has ended.  
  33.       break;  
  34.     }  
  35.   }  
  36. }  


    MPEG1or2Demux::continueReadProcessing的实现似乎很眼熟悉啊!是的,其与MPEGVideoStreamFramer::continueReadProcessing()有些相似。两个函数中均调用了语法分析器进行语法分析,不过应注意前者的语法分析器是MPEGProgramStreamParser类实例,完成了对复合文件的解析,主要目的是从中分离出音视频流数据(其实就是demux过程),而后者是一个MPEG1or2VideoStreamParser,对指定的流数据进一步分析。MPEGProgramStreamParser的数据源,是一个ByteStreamFileSource实例,它是MPEG1or2Demux构造函数中传递下来的参数。若未读取到所需要的数据,fParser->parse()将返回0。


    来看MPEGProgramStreamParser::parse函数实现
[cpp]  view plain copy print ?
  1. unsigned char MPEGProgramStreamParser::parse() {  
  2.   unsigned char acquiredStreamTagId = 0;  
  3.   
  4.   
  5.   try {  
  6.     do {  
  7.       switch (fCurrentParseState) {  
  8.       case PARSING_PACK_HEADER: {  
  9.     parsePackHeader();      //分析包头  
  10.     break;  
  11.       }  
  12.       case PARSING_SYSTEM_HEADER: {  
  13.     parseSystemHeader();    //分析系统头  
  14.     break;  
  15.       }  
  16.       case PARSING_PES_PACKET: {  
  17.     acquiredStreamTagId = parsePESPacket(); //分析流数据  
  18.     break;  
  19.       }  
  20.       }  
  21.     } while(acquiredStreamTagId == 0);  
  22.   
  23.   
  24.     return acquiredStreamTagId;  
  25.   } catch (int /*e*/) {  
  26. #ifdef DEBUG  
  27.     fprintf(stderr, "MPEGProgramStreamParser::parse() EXCEPTION (This is normal behavior - *not* an error)\n");  
  28.     fflush(stderr);  
  29. #endif  
  30.     return 0;  // the parsing got interrupted  
  31.   }  
  32. }  



对于mpeg文件格式不熟悉,就不看了,只看获取流数据包的函数MPEGProgramStreamParser::parsePESPacket

[cpp]  view plain copy print ?
  1. unsigned char MPEGProgramStreamParser::parsePESPacket() {  
  2. ...  
  3.     //  
  4.     //检查source是否正在等待这个流类型,如果是就把数据交给它  
  5.     //  
  6.     // Check whether our using source is interested in this stream type.  
  7.     // If so, deliver the frame to him:  
  8.     MPEG1or2Demux::OutputDescriptor_t& out = fUsingDemux->fOutput[stream_id];  
  9.     if (out.isCurrentlyAwaitingData) {  
  10.       unsigned numBytesToCopy;  
  11.       if (PES_packet_length > out.maxSize) {  
  12.         numBytesToCopy = out.maxSize;  
  13.       } else {  
  14.     numBytesToCopy = PES_packet_length;  
  15.       }  
  16.   
  17.   
  18.       getBytes(out.to, numBytesToCopy); //拷贝数据  
  19.       out.frameSize = numBytesToCopy;  
  20.   
  21.   
  22.       // set out.presentationTime later #####  
  23.       acquiredStreamIdTag = stream_id;  
  24.       PES_packet_length -= numBytesToCopy;0  
  25.     } else if (out.isCurrentlyActive) {  
  26.     //  
  27.     //这个流是需要用到的,但是不是现在,当需要的时候我们才能把数据交出去  
  28.     //  
  29.         // Someone has been reading this stream, but isn't right now.  
  30.       // We can't deliver this frame until he asks for it, so punt for now.  
  31.       // The next time he asks for a frame, he'll get it.  
  32.   
  33.   
  34.       restoreSavedParserState(); // so we read from the beginning next time  
  35.       fUsingDemux->fHaveUndeliveredData = True;  
  36.       throw READER_NOT_READY;   //抛出异常了  
  37.     } else if (out.isPotentiallyReadable &&  
  38.            out.savedDataTotalSize + PES_packet_length < 1000000 /*limit*/) {  
  39.         //  
  40.         //这个流也是需要用到的,只是当前没有读取而已,因此将其保存到缓存中(OutputDescriptor)  
  41.         //  
  42.       // Someone is interested in this stream, but hasn't begun reading it yet.  
  43.       // Save this data, so that the reader will get it when he later asks for it.  
  44.       unsigned char* buf = new unsigned char[PES_packet_length];  
  45.       getBytes(buf, PES_packet_length);  
  46.       MPEG1or2Demux::OutputDescriptor::SavedData* savedData  
  47.     = new MPEG1or2Demux::OutputDescriptor::SavedData(buf, PES_packet_length);   //新建一个SavedData实例  
  48.       if (out.savedDataHead == NULL) {  
  49.     out.savedDataHead = out.savedDataTail = savedData;  
  50.       } else {  
  51.     out.savedDataTail->next = savedData;  
  52.     out.savedDataTail = savedData;  
  53.       }  
  54.       out.savedDataTotalSize += PES_packet_length;  
  55.       PES_packet_length = 0;  
  56.     }  
  57.     skipBytes(PES_packet_length);  
  58.   }  
  59.   
  60.   
  61.   // Check for another PES Packet next:  
  62.   setParseState(PARSING_PES_PACKET);  
  63.   
  64.   
  65.  return acquiredStreamIdTag;  
  66. }  


    MPEGProgramStreamParser::parsePESPacket函数会读取流中的数据,若读取到的流数据不是当前需要的就保存到缓存中,也就是其对应的OutputDescriptor实例。需要注意局部变量acquiredStreamIdTag,它的值被初始化为0, 只有当读取到当前所需要的流数据时才会被赋值。所以若未读取到所需要的数据,这个函数将返回0值。










 

live555源码分析----mpg文件的处理(续)


    前一篇文章对mpg文件处理的分析中,有个一个比较严重的错误,因为有些重要的细节没有注意到。mpg文件是音视频交错排列的,如果需要读取的是视频数据,但当前文件位置却是音频数据该怎么办?前面分析时说,将把遇到的音频数据保存到缓存中,直到读取到视频数据,live555中并非如此处理的。


先简单的说明一下mpg处理过程涉及的几个类的作用
MPEG1or2Demux, 对应一个session,完成文件的解复用操作。其读取文件的过程是通过ByteStreamFileSource实现的
MPEG1or2DemuxedElementaryStream,作为source,对应一个流,一个或多个实例对应一个MPEG1or2Demux
MPEG1or2DemuxedServerMediaSubsession,作为subsession
MPEG1or2FileServerDemux, 一个服务类,整个程序只有一个实例,MPEG1or2Demux、MPEG1or2DemuxedElementaryStream、MPEG1or2DemuxedServerMediaSubsession三个类将通过本类进行联系。完成的工作如下:
1)创建subsession实例(MPEG1or2DemuxedServerMediaSubsession);
2)为每一个session创建MPEG1or2Demux实例,用于解复用。
3)创建source实例(MPEG1or2DemuxedElementaryStream), subsession中


再来看一个重要的结构体OutputDescriptor,它被定义在MPEG1or2Demux类中,文件中的每一个流对应一个OutputDescriptor实例

[cpp]  view plain copy print ?
  1. // A descriptor for each possible stream id tag:  
  2. typedef struct OutputDescriptor {  
  3.   // input parameters  
  4.   unsigned char* to; unsigned maxSize;  
  5.   FramedSource::afterGettingFunc* fAfterGettingFunc;  
  6.   void* afterGettingClientData;  
  7.   FramedSource::onCloseFunc* fOnCloseFunc;  
  8.   void* onCloseClientData;  
  9.   
  10.   
  11.   // output parameters  
  12.   unsigned frameSize; struct timeval presentationTime;  
  13.   class SavedData; // forward  
  14.   SavedData* savedDataHead;  
  15.   SavedData* savedDataTail;  
  16.   unsigned savedDataTotalSize;  
  17.   
  18.   
  19.   // status parameters  
  20.   Boolean isPotentiallyReadable;  
  21.   Boolean isCurrentlyActive;  
  22.   Boolean isCurrentlyAwaitingData;  
  23. } OutputDescriptor_t;  
  24. OutputDescriptor_t fOutput[256];  


  
需要注意以下3个成员:
isPotentiallyReadable,对应的流存在,即存在对应流的MPEG1or2DemuxedElementaryStream实例
isCurrentlyActive,对应的流是活动的,当从第一次从这个流中读取数据时就置为True, 停止读取后置为False
isCurrentlyAwaitingData,当前正需要从对应的流中读取数据。从文件中读取数据并分析后,通过此标志判断读取到的数据是否是当前需要的流数据,如果不是则需要数据存在到缓存中


我们来看MPEG1or2Demux::getNextFrame函数

[cpp]  view plain copy print ?
  1. void MPEG1or2Demux::getNextFrame(u_int8_t streamIdTag,  
  2.                  unsigned char* to, unsigned maxSize,  
  3.                  FramedSource::afterGettingFunc* afterGettingFunc,  
  4.                  void* afterGettingClientData,  
  5.                  FramedSource::onCloseFunc* onCloseFunc,  
  6.                  void* onCloseClientData) {  
  7.   // First, check whether we have saved data for this stream id:  
  8.   //从缓存中读取数据  
  9.   if (useSavedData(streamIdTag, to, maxSize,  
  10.            afterGettingFunc, afterGettingClientData)) {  
  11.     return;  
  12.   }  
  13.   
  14.   
  15.   // Then save the parameters of the specified stream id:  
  16.   // 将信息保存到流对应的OutputDescriptor_t实例中  
  17.   registerReadInterest(streamIdTag, to, maxSize,  
  18.                afterGettingFunc, afterGettingClientData,  
  19.                onCloseFunc, onCloseClientData);  
  20.   
  21.   
  22.   // Next, if we're the only currently pending read, continue looking for data:  
  23.   // 进一步处理,将从文件中读取数据  
  24.   if (fNumPendingReads == 1 || fHaveUndeliveredData) {  
  25.     fHaveUndeliveredData = 0;  
  26.     continueReadProcessing();  
  27.   } // otherwise the continued read processing has already been taken care of  
  28. }  



useSavedData函数从缓存中读取数据,没什么好说的。简单看一下registerReadInterest函数

[cpp]  view plain copy print ?
  1. void MPEG1or2Demux::registerReadInterest(u_int8_t streamIdTag,  
  2.                      unsigned char* to, unsigned maxSize,  
  3.                      FramedSource::afterGettingFunc* afterGettingFunc,  
  4.                      void* afterGettingClientData,  
  5.                      FramedSource::onCloseFunc* onCloseFunc,  
  6.                      void* onCloseClientData) {  
  7.   struct OutputDescriptor& out = fOutput[streamIdTag];  
  8.   
  9.   
  10.   // Make sure this stream is not already being read:  
  11.   if (out.isCurrentlyAwaitingData) {  
  12.     envir() << "MPEG1or2Demux::registerReadInterest(): attempt to read stream id "  
  13.         << (void*)streamIdTag << " more than once!\n";  
  14.     envir().internalError();  
  15.   }  
  16.   
  17.   
  18.   out.to = to; out.maxSize = maxSize;  
  19.   out.fAfterGettingFunc = afterGettingFunc;  
  20.   out.afterGettingClientData = afterGettingClientData;  
  21.   out.fOnCloseFunc = onCloseFunc;  
  22.   out.onCloseClientData = onCloseClientData;  
  23.   out.isCurrentlyActive = True;  //当前流处理活动中  
  24.   out.isCurrentlyAwaitingData = True;  //当前流需要读取数据  
  25.   // out.frameSize and out.presentationTime will be set when a frame's read  
  26.   
  27.   
  28.   ++fNumPendingReads;  //等待读取数据的流计数  
  29. }  




registerReadInterest函数中,最后三条语句应注意。isCurrentlyAwaitingData为True表示对应流需要读取数据,fNumPendingReads表示等读取数据的流计数。


现在再来看MPEG1or2Demux::getNextFrame函数中后面的if条件语句
[cpp]  view plain copy print ?
  1. if (fNumPendingReads == 1 || fHaveUndeliveredData) {  
  2.   fHaveUndeliveredData = 0;  
  3.   continueReadProcessing();  
  4. }  


   条件一fNumPendingReads == 1,fNumPendingReads表示未读取到数据的流的个数。为什么同时读取数据的流会超过一个呢?对于mpg文件来说,需要读取音频时当前文件位置可能正好是一个视频包,这时无法读取到所需要的数据,就会递增fNumPendingReads变量。条件二fHaveUndeliveredData,表示前面的处理过程中,某一个流未能如愿读取到需要的数据。


   上面的两个条件,我觉得有点问题,fNumPendingReads = 0是不可能的, fNumPendingReads>1时,表示前面的处理过程中,某一个流未能如愿读取到需要的数据,这时fHaveUndeliveredData就应该为True,这样的话if语句是不是永远为真呢?这个疑问有待确定。


继续来看MPEG1or2Demux::continueReadProcessing函数的实现

[cpp]  view plain copy print ?
  1. void MPEG1or2Demux::continueReadProcessing() {  
  2.   while (fNumPendingReads > 0) {  
  3.     unsigned char acquiredStreamIdTag = fParser->parse();  
  4.   
  5.   
  6.     if (acquiredStreamIdTag != 0) {  
  7.       // We were able to acquire a frame from the input.  
  8.       struct OutputDescriptor& newOut = fOutput[acquiredStreamIdTag];  
  9.       newOut.isCurrentlyAwaitingData = False;  
  10.         // indicates that we can be read again  
  11.         // (This needs to be set before the 'after getting' call below,  
  12.         //  in case it tries to read another frame)  
  13.   
  14.   
  15.       // Call our own 'after getting' function.  Because we're not a 'leaf'  
  16.       // source, we can call this directly, without risking infinite recursion.  
  17.       if (newOut.fAfterGettingFunc != NULL) {  
  18.     (*newOut.fAfterGettingFunc)(newOut.afterGettingClientData,  
  19.                     newOut.frameSize, 0 /* numTruncatedBytes */,  
  20.                     newOut.presentationTime,  
  21.                     0 /* durationInMicroseconds ?????#####*/);  
  22.       --fNumPendingReads;  
  23.       }  
  24.     } else {  
  25.       // We were unable to parse a complete frame from the input, because:  
  26.       // - we had to read more data from the source stream, or  
  27.       // - we found a frame for a stream that was being read, but whose  
  28.       //   reader is not ready to get the frame right now, or  
  29.       // - the source stream has ended.  
  30.       break;  
  31.     }  
  32.   }  
  33. }  




    这个函数中有一个while循环,fParser->parse()返回当前读取到的数据的流标号,第一次循环过程的返回值必定是当前所需要的流数据,第二次循环返回值则是先前未读取到的流数据,当所有需要读取数据的流均已经读取数据后,fNumPendingReads值递减为0,循环结束。


再来看MPEGProgramStreamParser::parse函数
[cpp]  view plain copy print ?
  1. unsigned char MPEGProgramStreamParser::parse() {  
  2.   unsigned char acquiredStreamTagId = 0;  
  3.   
  4.   
  5.   try {  
  6.     do {  
  7.       switch (fCurrentParseState) {  
  8.       case PARSING_PACK_HEADER: {  
  9.     parsePackHeader();  
  10.     break;  
  11.       }  
  12.       case PARSING_SYSTEM_HEADER: {  
  13.     parseSystemHeader();  
  14.     break;  
  15.       }  
  16.       case PARSING_PES_PACKET: {  
  17.     acquiredStreamTagId = parsePESPacket();  
  18.     break;  
  19.       }  
  20.       }  
  21.     } while(acquiredStreamTagId == 0);  
  22.   
  23.   
  24.     return acquiredStreamTagId;  
  25.   } catch (int /*e*/) {  
  26. #ifdef DEBUG  
  27.     fprintf(stderr, "MPEGProgramStreamParser::parse() EXCEPTION (This is normal behavior - *not* an error)\n");  
  28.     fflush(stderr);  
  29. #endif  
  30.     return 0;  // the parsing got interrupted  
  31.   }  
  32. }  




上面的函数中,最有catch语句,所以前面的函数调用必定会抛出异常。parsePESPacket函数返回流标识作为循环条件,所以这个函数最重要
[cpp]  view plain copy print ?
  1. unsigned char MPEGProgramStreamParser::parsePESPacket() {  
  2.   
  3.   
  4.     ...  
  5.   
  6.   
  7.     // Check whether our using source is interested in this stream type.  
  8.     // If so, deliver the frame to him:  
  9.     MPEG1or2Demux::OutputDescriptor_t& out = fUsingDemux->fOutput[stream_id];  
  10.     if (out.isCurrentlyAwaitingData) {      //当前流正在等待数据  
  11.       unsigned numBytesToCopy;  
  12.       if (PES_packet_length > out.maxSize) {  
  13.     fUsingDemux->envir() << "MPEGProgramStreamParser::parsePESPacket() error: PES_packet_length ("  
  14.                   << PES_packet_length  
  15.                   << ") exceeds max frame size asked for ("  
  16.                   << out.maxSize << ")\n";  
  17.     numBytesToCopy = out.maxSize;  
  18.       } else {  
  19.     numBytesToCopy = PES_packet_length;  
  20.       }  
  21.   
  22.   
  23.       getBytes(out.to, numBytesToCopy);  
  24.       out.frameSize = numBytesToCopy;  
  25.   
  26.   
  27.       // set out.presentationTime later #####  
  28.       acquiredStreamIdTag = stream_id;  
  29.       PES_packet_length -= numBytesToCopy;  
  30.     } else if (out.isCurrentlyActive) {     //当前流是活动的  
  31.       // Someone has been reading this stream, but isn't right now.  
  32.       // We can't deliver this frame until he asks for it, so punt for now.  
  33.       // The next time he asks for a frame, he'll get it.  
  34.   
  35.   
  36.       restoreSavedParserState(); // so we read from the beginning next time  
  37.       fUsingDemux->fHaveUndeliveredData = True;     //有未投递的数据  
  38.       throw READER_NOT_READY;  //直接抛出异常,  
  39.     } else if (out.isPotentiallyReadable &&  //当前流需要读取数据  
  40.            out.savedDataTotalSize + PES_packet_length < 1000000 /*limit*/) {      
  41.       // Someone is interested in this stream, but hasn't begun reading it yet.  
  42.       // Save this data, so that the reader will get it when he later asks for it.  
  43.       //  
  44.       // 从文件中读取的数据,当前并不需要,所以就把它保存到存在中  
  45.       //  
  46.       unsigned char* buf = new unsigned char[PES_packet_length];  
  47.       getBytes(buf, PES_packet_length);  
  48.       MPEG1or2Demux::OutputDescriptor::SavedData* savedData  
  49.     = new MPEG1or2Demux::OutputDescriptor::SavedData(buf, PES_packet_length);  
  50.       if (out.savedDataHead == NULL) {  
  51.     out.savedDataHead = out.savedDataTail = savedData;  
  52.       } else {  
  53.     out.savedDataTail->next = savedData;  
  54.     out.savedDataTail = savedData;  
  55.       }  
  56.       out.savedDataTotalSize += PES_packet_length;  
  57.       PES_packet_length = 0;  
  58.     }  
  59.     skipBytes(PES_packet_length);  
  60.   }  
  61.   
  62.   
  63.   // Check for another PES Packet next:  
  64.   setParseState(PARSING_PES_PACKET);  
  65.   
  66.   
  67.  return acquiredStreamIdTag;  
  68. }  

我来看上面有中文注释的if结构,
第一个条件out.isCurrentlyAwaitingData为True时,表明读取到的数据正是需要读取的。这应该是正常的情况
第二个条件out.isCurrentlyActive为True时,这个条件是在表明对应的流是活动的,但当前并没有要求读取数据。这个条件只要流已经开始播放,在其停止之前都为True。这个条件处理中,将对应流的fHaveUndeliveredData标记为True,最后抛出了异常,这个异常在MPEGProgramStreamParser::parse中处理的
第三个条件out.isPotentiallyReadable为True时,保存数据到缓存中。isPotentiallyReadable是在MPEG1or2DemuxedElementaryStream类构造函数中就置为True的。


根据上面的分析,当流正在播放时,无论如何都不会将数据保存到缓冲中,只有在流的播放停止的情况下才会进行。


从上面的分析中,我们还可以发现其它一些东西,这里指明一下。当未能读取到所以的流数据时,就不会调用after函数进行处理,但是after函数中完成了一个重要的功能,每一个after调用最后都会将下一次数据处理的任务注册到任务管理器中。那不调用after函数流的处理是不是会中断呢?当然不会。例如,当前处理过程没能读取到需要的视频数据,但在下一次处理音频数据时,会把视频数据也读取了,并调用视频处理的after函数,这就是MPEG1or2Demux::continueReadProcessing() 函数中循环的作用。


一个需要进一步思考的问题:
1).按上述方式,当前需要处理的数据会推迟(直到其它流读取完数据),这样是不是会导致延时的时间比正常情况下要久,从而使得发送变慢呢?
2).out.isCurrentlyActive为True时,保存数据到缓存中,这样是不是更好些呢?


你可能感兴趣的:(live555源码分析----mpg文件的处理)