#DATE 2012/05/09
#2012/08/27由cnblogs遷入android
Android不支持C++異常機制,若是須要用到的話,則須要在編譯的時候加入比較完整的C++庫.
Android支持的C++庫能夠在Android NDK中找到(解壓後找到libsupc++.a放到代碼環境中便可):
http://www.crystax.net/en/android/ndk/7
編譯時加上參數: c++
-fexceptions -lstdc++
還須要將libsupc++.a連接上git
https://github.com/boltonli/ohbee/tree/master/android/streamer/jnigithub
參考: rfc2326, rfc3550, rfc3984shell
RTP Header結構[#0]ubuntu
0 1 2 3 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ |V=2|P|X| CC |M| PT | sequence number | +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ | timestamp | +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ | synchronization source (SSRC) identifier | +=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+ | contributing source (CSRC) identifiers | | .... | +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
參考: rfc3984, 『H.264中的NAL技術』, 『H.264 NAL層解析』session
參考: ISO_IEC_13818-7.pdf架構
void trace_bin(const unsigned char *bytes_ptr, int bytes_num) { #define LOG_LINE_BYTES 16 int i, j; for (i = 0; i <= bytes_num / LOG_LINE_BYTES; i++) { for (j = 0; j < ( (i < (bytes_num / LOG_LINE_BYTES)) ? LOG_LINE_BYTES : (bytes_num % LOG_LINE_BYTES) ); j++) { if (0 == j) printf("%04d ", i * LOG_LINE_BYTES); if (LOG_LINE_BYTES/2 == j) printf(" "); printf(" %02x", bytes_ptr[i * LOG_LINE_BYTES + j]); } printf("\n"); } }
------------------------------------------------------[#1]-- session <---> client requst | subsession <---> audio/video ------------------------------------------------------------
1.2 數據處理流程:
------------------------------------------------------[#2]-- source --> filter(source) ... --> sink | | | +-------+-------------------+ | | v v subsession.createNewRTPSink() subsession.createNewStreamSource() ------------------------------------------------------------
void BasicTaskScheduler::SingleStep(unsigned maxDelayTime) { //1. 處理io任務 ... int selectResult = select(fMaxNumSockets, &readSet, &writeSet, &exceptionSet, &tv_timeToDelay); ... while ((handler = iter.next()) != NULL) { ... (*handler->handlerProc)(handler->clientData, resultConditionSet); break; } ... //2. handle any newly-triggered event ... //3. handle any delayed event fDelayQueue.handleAlarm(); }
void ADTSAudioFileSource::doGetNextFrame() { // 讀取數據並作一些簡單處理 ... int numBytesRead = fread(fTo, 1, numBytesToRead, fFid); ... // 將FramedSource::afterGetting加入fDelayQueue中 // FramedSource::afterGetting會處理讀取到數據,並又會調用 // ADTSAudioFileSource::doGetNextFrame(),這樣實現循環讀取文件。 nextTask() = envir().taskScheduler().scheduleDelayedTask(0, (TaskFunc*)FramedSource::afterGetting, this); }
[0]->[1]->[3]->[2]->...->[1]->NULL ^ | head()
void DelayQueue::handleAlarm() { ... toRemove->handleTimeout(); } void AlarmHandler::handleTimeout() { (*fProc)(fClientData); DelayQueueEntry::handleTimeout(); }
任務在處理完成後便會被刪除。
------------------------------------------------------[#3]-- Medium +ServerMediaSubsession | +OnDemandServerMediaSubsession | +FileServerMediaSubsession | +H264VideoFileServerMediaSubsession //h264 | +ADTSAudioFileServerMediaSubsession //aac | +MediaSink | +RTPSink | +MultiFramedRTPSink | +VideoRTPSink | | +H264VideoRTPSink //h264 | +MPEG4GenericRTPSink //aac | +MediaSource +FramedSource //+doGetNextFrame(); +fAfterGettingFunc; +FramedFilter | +H264FUAFragmenter //h264 | +MPEGVideoStreamFramer | +H264VideoStreamFramer //h264 +FramedFileSource +ByteStreamFileSource //h264 +ADTSAudioFileSource //acc StreamParser +MPEGVideoStreamParser +H264VideoStreamParser //h264 ------------------------------------------------------------
咱們看下FramedFilter和FramedFileSource相對於FramedSource增長了哪些成員:
FramedFilter { FramedSource* fInputSource; } FramedFileSource { FILE* fFid; }
H264VideoStreamFramer.fInputSource = ByteStreamFileSource; H264FUAFragmenter.fInputSource = H264VideoStreamFramer;
將上游source賦值到下游filter的fInputSource便可,對於H264即可以獲得下面的一個處理流程:
ByteStreamFileSource -> H264VideoStreamFramer -> H264FUAFragmenter -> H264VideoRTPSink
在H264VideoStreamFramer的父類MPEGVideoStreamFramer中也有新增成員,
MPEGVideoStreamFramer { MPEGVideoStreamParser* fParser; } MPEGVideoStreamFramer.fParser = H264VideoStreamParser;
------------------------------------------------------[#4]-- RTSPServer::RTSPClientSession::handleCmd_SETUP() OnDemandServerMediaSubsession::getStreamParameters( streamToken: new StreamState( fMediaSource: H264VideoFileServerMediaSubsession::createNewStreamSource() ) ) ********** RTSPServer::RTSPClientSession::handleCmd_DESCRIBE() ServerMediaSession::generateSDPDescription() OnDemandServerMediaSubsession::sdpLines() H264VideoFileServerMediaSubsession::createNewStreamSource() H264VideoStreamFramer::createNew( fInputSource: ByteStreamFileSource::createNew(), fParser: new H264VideoStreamParser( fInputSource: H264VideoStreamFramer.fInputSource) ) ********** RTSPServer::RTSPClientSession::handleCmd_PLAY() H264VideoFileServerMediaSubsession::startStream() [OnDemandServerMediaSubsession::startStream()] StreamState::startPlaying() H264VideoRTPSink::startPlaying() [MediaSink::startPlaying(fMediaSource)]//got in handleCmd_SETUP() H264VideoRTPSink::continuePlaying() fSource, fOurFragmenter: H264FUAFragmenter(fInputSource: fMediaSource) MultiFramedRTPSink::continuePlaying() MultiFramedRTPSink::buildAndSendPacket() MultiFramedRTPSink::packFrame() H264FUAFragmenter::getNextFrame() [FramedSource::getNextFrame()] H264FUAFragmenter::doGetNextFrame() {1} 1)=No NALU= H264VideoStreamFramer::getNextFrame() [FramedSource::getNextFrame()] MPEGVideoStreamFramer::doGetNextFrame() H264VideoStreamParser::registerReadInterest() MPEGVideoStreamFramer::continueReadProcessing() H264VideoStreamParser::parse() H264VideoStreamFramer::afterGetting() [FramedSource::afterGetting()] H264FUAFragmenter::afterGettingFrame() H264FUAFragmenter::afterGettingFrame1() goto {1} //Now we have got NALU 2)=Has NALU= FramedSource::afterGetting() MultiFramedRTPSink::afterGettingFrame() MultiFramedRTPSink::afterGettingFrame1() MultiFramedRTPSink::sendPacketIfNecessary() ------------------------------------------------------------
------------------------------------------------------[#5]-- ServerMediaSession{#1}.fSubsessionsTail = H264VideoFileServerMediaSubsession{2}.fParentSession = {1} fStreamStates[] { .subsession = {2} .streamToken = StreamState { .fMaster = {2} .fRTPSink = H264VideoRTPSink{5}.fSource/fOurFragmenter = H264FUAFragmenter{4} { .fInputSource = H264VideoStreamFramer{3} .fAfterGettingFunc = MultiFramedRTPSink::afterGettingFrame() .fAfterGettingClientData = {5} .fOnCloseFunc = MultiFramedRTPSink::ourHandleClosure() .fOnCloseClientData = {5} } .fMediaSource = {3} { .fParser = H264VideoStreamParser { .fInputSource = ByteStreamFileSource{6} .fTo = [{5}.]fOutBuf->curPtr() } .fInputSource = {6} .fAfterGettingFunc = H264FUAFragmenter::afterGettingFrame() .fAfterGettingClientData = {4} .fOnCloseFunc = FramedSource::handleClosure() .fOnCloseClientData = {4} } } } ------------------------------------------------------------
4.2 AAC對象關係圖
------------------------------------------------------[#6]-- ServerMediaSession{1}.fSubsessionsTail = ADTSAudioFileServerMediaSubsession{2}.fParentSession = {1} fStreamStates[] { .subsession = {2} .streamToken = StreamState { .fMaster = {2} .fRTPSink = MPEG4GenericRTPSink { .fOutBuf = OutPacketBuffer .fSource = ADTSAudioFileSource {3} .fRTPInterface = RTPInterface.fGS = Groupsock } .fMediaSource = {3} } } ------------------------------------------------------------
RTSP命令 live555中處理函數 --------------------------------------------- OPTIONS <---> handleCmd_OPTIONS DESCRIBE <---> handleCmd_DESCRIBE SETUP <---> handleCmd_SETUP PLAY <---> handleCmd_PLAY PAUSE <---> handleCmd_PAUSE TEARDOWN <---> handleCmd_TEARDOWN GET_PARAMETER <---> handleCmd_GET_PARAMETER SET_PARAMETER <---> handleCmd_SET_PARAMETER
5.2 RTSP播放交互示例(openRTSP)
-------------------------------------------------------------------------------- ubuntu$ ./openRTSP rtsp://192.168.43.1/grandma.264 Opening connection to 192.168.43.1, port 554... ...remote connection opened Sending request: OPTIONS rtsp://192.168.43.1/grandma.264 RTSP/1.0 CSeq: 2 User-Agent: ./openRTSP (LIVE555 Streaming Media v2012.02.29) Received 152 new bytes of response data. Received a complete OPTIONS response: RTSP/1.0 200 OK CSeq: 2 Date: Tue, Jan 25 2011 21:02:53 GMT Public: OPTIONS, DESCRIBE, SETUP, TEARDOWN, PLAY, PAUSE, GET_PARAMETER, SET_PARAMETER -------------------------------------------------------------------------------- Sending request: DESCRIBE rtsp://192.168.43.1/grandma.264 RTSP/1.0 CSeq: 3 User-Agent: ./openRTSP (LIVE555 Streaming Media v2012.02.29) Accept: application/sdp Received 682 new bytes of response data. Received a complete DESCRIBE response: RTSP/1.0 200 OK CSeq: 3 Date: Tue, Jan 25 2011 21:02:53 GMT Content-Base: rtsp://192.168.43.1/grandma.264/ Content-Type: application/sdp Content-Length: 517 v=0 o=- 1295989373493698 1 IN IP4 0.0.0.0 s=H.264 Video, streamed by the LIVE555 Media Server i=grandma.264 t=0 0 a=tool:LIVE555 Streaming Media v2012.02.04 a=type:broadcast a=control:* a=range:npt=0- a=x-qt-text-nam:H.264 Video, streamed by the LIVE555 Media Server a=x-qt-text-inf:grandma.264 m=video 0 RTP/AVP 96 c=IN IP4 0.0.0.0 b=AS:500 a=rtpmap:96 H264/90000 a=fmtp:96 packetization-mode=1;profile-level-id=4D4033; sprop-parameter-sets=Z01AM5p0FidCAAADAAIAAAMAZR4wZUA=,aO48gA== a=control:track1 Opened URL "rtsp://192.168.43.1/grandma.264", returning a SDP description: v=0 o=- 1295989373493698 1 IN IP4 0.0.0.0 s=H.264 Video, streamed by the LIVE555 Media Server i=grandma.264 t=0 0 a=tool:LIVE555 Streaming Media v2012.02.04 a=type:broadcast a=control:* a=range:npt=0- a=x-qt-text-nam:H.264 Video, streamed by the LIVE555 Media Server a=x-qt-text-inf:grandma.264 m=video 0 RTP/AVP 96 c=IN IP4 0.0.0.0 b=AS:500 a=rtpmap:96 H264/90000 a=fmtp:96 packetization-mode=1;profile-level-id=4D4033; sprop-parameter-sets=Z01AM5p0FidCAAADAAIAAAMAZR4wZUA=,aO48gA== a=control:track1 Created receiver for "video/H264" subsession (client ports 56488-56489) -------------------------------------------------------------------------------- Sending request: SETUP rtsp://192.168.43.1/grandma.264/track1 RTSP/1.0 CSeq: 4 User-Agent: ./openRTSP (LIVE555 Streaming Media v2012.02.29) Transport: RTP/AVP;unicast;client_port=56488-56489 Received 205 new bytes of response data. Received a complete SETUP response: RTSP/1.0 200 OK CSeq: 4 Date: Tue, Jan 25 2011 21:02:53 GMT Transport: RTP/AVP;unicast;destination=192.168.43.244;source=192.168.43.1; client_port=56488-56489;server_port=6970-6971 Session: 7626020D Setup "video/H264" subsession (client ports 56488-56489) Created output file: "video-H264-1" -------------------------------------------------------------------------------- Sending request: PLAY rtsp://192.168.43.1/grandma.264/ RTSP/1.0 CSeq: 5 User-Agent: ./openRTSP (LIVE555 Streaming Media v2012.02.29) Session: 7626020D Range: npt=0.000- Received 186 new bytes of response data. Received a complete PLAY response: RTSP/1.0 200 OK CSeq: 5 Date: Tue, Jan 25 2011 21:02:53 GMT Range: npt=0.000- Session: 7626020D RTP-Info: url=rtsp://192.168.43.1/grandma.264/track1;seq=26490;rtptime=1809652062 Started playing session Receiving streamed data (signal with "kill -HUP 6297" or "kill -USR1 6297" to terminate)... Received RTCP "BYE" on "video/H264" subsession (after 35 seconds) -------------------------------------------------------------------------------- Sending request: TEARDOWN rtsp://192.168.43.1/grandma.264/ RTSP/1.0 CSeq: 6 User-Agent: ./openRTSP (LIVE555 Streaming Media v2012.02.29) Session: 7626020D Received 65 new bytes of response data. Received a complete TEARDOWN response: RTSP/1.0 200 OK CSeq: 6 Date: Tue, Jan 25 2011 21:03:28 GMT --------------------------------------------------------------------------------*** END ***