"ViE GetSentRTCPStatistics fails on a sending channel if it don't receive rtp video packets. BUG=222 TEST= tested in loopback. No new test added yet. Review URL: http://webrtc-codereview.appspot.com/343003 git-svn-id: http://webrtc.googlecode.com/svn/trunk@1387 4adac7df-926f-26a2-2b94-8c16560cd09d
This commit is contained in:
		| @@ -11,6 +11,8 @@ | ||||
| #ifndef WEBRTC_MODULES_RTP_RTCP_INTERFACE_RTP_RTCP_H_ | ||||
| #define WEBRTC_MODULES_RTP_RTCP_INTERFACE_RTP_RTCP_H_ | ||||
|  | ||||
| #include <vector> | ||||
|  | ||||
| #include "module.h" | ||||
| #include "rtp_rtcp_defines.h" | ||||
|  | ||||
| @@ -740,22 +742,23 @@ public: | ||||
|     * | ||||
|     *   return -1 on failure else 0 | ||||
|     */ | ||||
|     virtual WebRtc_Word32 RemoteRTCPStat( RTCPSenderInfo* senderInfo) = 0; | ||||
|     virtual WebRtc_Word32 RemoteRTCPStat(RTCPSenderInfo* senderInfo) = 0; | ||||
|  | ||||
|     /* | ||||
|     *   Get received RTCP report block | ||||
|     * | ||||
|     *   return -1 on failure else 0 | ||||
|     */ | ||||
|     virtual WebRtc_Word32 RemoteRTCPStat( const WebRtc_UWord32 remoteSSRC, | ||||
|                                         RTCPReportBlock* receiveBlock) = 0; | ||||
|     virtual WebRtc_Word32 RemoteRTCPStat( | ||||
|         std::vector<RTCPReportBlock>* receiveBlocks) const = 0; | ||||
|     /* | ||||
|     *   Set received RTCP report block | ||||
|     * | ||||
|     *   return -1 on failure else 0 | ||||
|     */ | ||||
|     virtual WebRtc_Word32 AddRTCPReportBlock(const WebRtc_UWord32 SSRC, | ||||
|                                            const RTCPReportBlock* receiveBlock) = 0; | ||||
|     virtual WebRtc_Word32 AddRTCPReportBlock( | ||||
|         const WebRtc_UWord32 SSRC, | ||||
|         const RTCPReportBlock* receiveBlock) = 0; | ||||
|  | ||||
|     /* | ||||
|     *   RemoveRTCPReportBlock | ||||
|   | ||||
| @@ -109,6 +109,9 @@ struct RTCPSenderInfo | ||||
|  | ||||
| struct RTCPReportBlock | ||||
| { | ||||
|   // Fields as described by RFC 3550 6.4.2. | ||||
|     WebRtc_UWord32 remoteSSRC;  // SSRC of sender of this report. | ||||
|     WebRtc_UWord32 sourceSSRC;  // SSRC of the RTP packet sender. | ||||
|     WebRtc_UWord8 fractionLost; | ||||
|     WebRtc_UWord32 cumulativeLost;  // 24 bits valid | ||||
|     WebRtc_UWord32 extendedHighSeqNum; | ||||
|   | ||||
| @@ -170,8 +170,8 @@ class MockRtpRtcp : public RtpRtcp { | ||||
|       WebRtc_Word32(WebRtc_UWord32 *bytesSent, WebRtc_UWord32 *packetsSent, WebRtc_UWord32 *bytesReceived, WebRtc_UWord32 *packetsReceived)); | ||||
|   MOCK_METHOD1(RemoteRTCPStat, | ||||
|       WebRtc_Word32(RTCPSenderInfo* senderInfo)); | ||||
|   MOCK_METHOD2(RemoteRTCPStat, | ||||
|       WebRtc_Word32(const WebRtc_UWord32 remoteSSRC, RTCPReportBlock* receiveBlock)); | ||||
|   MOCK_CONST_METHOD1(RemoteRTCPStat, | ||||
|       WebRtc_Word32(std::vector<RTCPReportBlock>* receiveBlocks)); | ||||
|   MOCK_METHOD2(AddRTCPReportBlock, | ||||
|       WebRtc_Word32(const WebRtc_UWord32 SSRC, const RTCPReportBlock* receiveBlock)); | ||||
|   MOCK_METHOD1(RemoveRTCPReportBlock, | ||||
|   | ||||
| @@ -179,56 +179,52 @@ RTCPReceiver::SetSSRC( const WebRtc_UWord32 ssrc) | ||||
|     _SSRC = ssrc; | ||||
| } | ||||
|  | ||||
| WebRtc_Word32 | ||||
| RTCPReceiver::ResetRTT(const WebRtc_UWord32 remoteSSRC) | ||||
| { | ||||
|     CriticalSectionScoped lock(_criticalSectionRTCPReceiver); | ||||
|     RTCPReportBlockInformation* reportBlock = GetReportBlockInformation(remoteSSRC); | ||||
|     if(reportBlock == NULL) | ||||
|     { | ||||
|         WEBRTC_TRACE(kTraceError, kTraceRtpRtcp, _id,  "\tfailed to GetReportBlockInformation(%d)", remoteSSRC); | ||||
|         return -1; | ||||
|     } | ||||
|     reportBlock->RTT = 0; | ||||
|     reportBlock->avgRTT = 0; | ||||
|     reportBlock->minRTT = 0; | ||||
|     reportBlock->maxRTT = 0; | ||||
| WebRtc_Word32 RTCPReceiver::ResetRTT(const WebRtc_UWord32 remoteSSRC) { | ||||
|   CriticalSectionScoped lock(_criticalSectionRTCPReceiver); | ||||
|   RTCPReportBlockInformation* reportBlock = | ||||
|       GetReportBlockInformation(remoteSSRC); | ||||
|   if (reportBlock == NULL) { | ||||
|     WEBRTC_TRACE(kTraceError, kTraceRtpRtcp, _id, | ||||
|                  "\tfailed to GetReportBlockInformation(%u)", remoteSSRC); | ||||
|     return -1; | ||||
|   } | ||||
|   reportBlock->RTT = 0; | ||||
|   reportBlock->avgRTT = 0; | ||||
|   reportBlock->minRTT = 0; | ||||
|   reportBlock->maxRTT = 0; | ||||
|  | ||||
|     return 0; | ||||
|   return 0; | ||||
| } | ||||
|  | ||||
| WebRtc_Word32 | ||||
| RTCPReceiver::RTT(const WebRtc_UWord32 remoteSSRC, | ||||
|                   WebRtc_UWord16* RTT, | ||||
|                   WebRtc_UWord16* avgRTT, | ||||
|                   WebRtc_UWord16* minRTT, | ||||
|                   WebRtc_UWord16* maxRTT) const | ||||
| WebRtc_Word32 RTCPReceiver::RTT(const WebRtc_UWord32 remoteSSRC, | ||||
|                                 WebRtc_UWord16* RTT, | ||||
|                                 WebRtc_UWord16* avgRTT, | ||||
|                                 WebRtc_UWord16* minRTT, | ||||
|                                 WebRtc_UWord16* maxRTT) const { | ||||
|   CriticalSectionScoped lock(_criticalSectionRTCPReceiver); | ||||
|  | ||||
| { | ||||
|     CriticalSectionScoped lock(_criticalSectionRTCPReceiver); | ||||
|     RTCPReportBlockInformation* reportBlock = GetReportBlockInformation(remoteSSRC); | ||||
|     if(reportBlock == NULL) | ||||
|     { | ||||
|         WEBRTC_TRACE(kTraceError, kTraceRtpRtcp, _id,  "\tfailed to GetReportBlockInformation(%d)", remoteSSRC); | ||||
|         return -1; | ||||
|     } | ||||
|     if(RTT) | ||||
|     { | ||||
|         *RTT = reportBlock->RTT; | ||||
|     } | ||||
|     if(avgRTT) | ||||
|     { | ||||
|         *avgRTT = reportBlock->avgRTT; | ||||
|     } | ||||
|     if(minRTT) | ||||
|     { | ||||
|         *minRTT = reportBlock->minRTT; | ||||
|     } | ||||
|     if(maxRTT) | ||||
|     { | ||||
|         *maxRTT = reportBlock->maxRTT; | ||||
|     } | ||||
|     return 0; | ||||
|   RTCPReportBlockInformation* reportBlock = | ||||
|       GetReportBlockInformation(remoteSSRC); | ||||
|  | ||||
|   if (reportBlock == NULL) { | ||||
|     WEBRTC_TRACE(kTraceError, kTraceRtpRtcp, _id, | ||||
|                  "\tfailed to GetReportBlockInformation(%u)", | ||||
|                  remoteSSRC); | ||||
|     return -1; | ||||
|   } | ||||
|   if (RTT) { | ||||
|     *RTT = reportBlock->RTT; | ||||
|   } | ||||
|   if (avgRTT) { | ||||
|     *avgRTT = reportBlock->avgRTT; | ||||
|   } | ||||
|   if (minRTT) { | ||||
|     *minRTT = reportBlock->minRTT; | ||||
|   } | ||||
|   if (maxRTT) { | ||||
|     *maxRTT = reportBlock->maxRTT; | ||||
|   } | ||||
|   return 0; | ||||
| } | ||||
|  | ||||
| void | ||||
| @@ -287,24 +283,25 @@ RTCPReceiver::SenderInfoReceived(RTCPSenderInfo* senderInfo) const | ||||
|  | ||||
| // statistics | ||||
| // we can get multiple receive reports when we receive the report from a CE | ||||
| WebRtc_Word32 | ||||
| RTCPReceiver::StatisticsReceived(const WebRtc_UWord32 remoteSSRC, | ||||
|                                  RTCPReportBlock* receiveBlock) const | ||||
| { | ||||
|     if(receiveBlock == NULL) | ||||
|     { | ||||
|         WEBRTC_TRACE(kTraceError, kTraceRtpRtcp, _id, "%s invalid argument", __FUNCTION__); | ||||
|         return -1; | ||||
|     } | ||||
|     CriticalSectionScoped lock(_criticalSectionRTCPReceiver); | ||||
|     RTCPReportBlockInformation* reportBlockInfo = GetReportBlockInformation(remoteSSRC); | ||||
|     if(reportBlockInfo == NULL) | ||||
|     { | ||||
|         WEBRTC_TRACE(kTraceError, kTraceRtpRtcp, _id,  "\tfailed to GetReportBlockInformation(%d)", remoteSSRC); | ||||
|         return -1; | ||||
|     } | ||||
|     memcpy(receiveBlock, &(reportBlockInfo->remoteReceiveBlock), sizeof(RTCPReportBlock)); | ||||
|     return 0; | ||||
| WebRtc_Word32 RTCPReceiver::StatisticsReceived( | ||||
|     std::vector<RTCPReportBlock>* receiveBlocks) const { | ||||
|   if (receiveBlocks == NULL) { | ||||
|     WEBRTC_TRACE(kTraceError, kTraceRtpRtcp, _id, "%s invalid argument", | ||||
|                  __FUNCTION__); | ||||
|     return -1; | ||||
|   } | ||||
|   CriticalSectionScoped lock(_criticalSectionRTCPReceiver); | ||||
|  | ||||
|   MapItem* ptrReportBlockInfoItem = _receivedReportBlockMap.First(); | ||||
|   while (ptrReportBlockInfoItem != NULL) { | ||||
|     RTCPReportBlockInformation* item = | ||||
|         static_cast<RTCPReportBlockInformation*> ( | ||||
|             ptrReportBlockInfoItem->GetItem()); | ||||
|     receiveBlocks->push_back(item->remoteReceiveBlock); | ||||
|     ptrReportBlockInfoItem = | ||||
|         _receivedReportBlockMap.Next(ptrReportBlockInfoItem); | ||||
|   } | ||||
|   return 0; | ||||
| } | ||||
|  | ||||
| WebRtc_Word32 | ||||
| @@ -459,135 +456,113 @@ void | ||||
| RTCPReceiver::HandleReportBlock(const RTCPUtility::RTCPPacket& rtcpPacket, | ||||
|                                 RTCPPacketInformation& rtcpPacketInformation, | ||||
|                                 const WebRtc_UWord32 remoteSSRC, | ||||
|                                 const WebRtc_UWord8 numberOfReportBlocks) | ||||
| { | ||||
|     // this will be called once per report block in the RTCP packet | ||||
|     // we store all incoming reports | ||||
|     // each packet has max 31 RR blocks | ||||
|     // | ||||
|     // we can calc RTT if we send a send report and get a report block back | ||||
|                                 const WebRtc_UWord8 numberOfReportBlocks) { | ||||
|   // This will be called once per report block in the RTCP packet. | ||||
|   // We filter out all report blocks that are not for us. | ||||
|   // Each packet has max 31 RR blocks. | ||||
|   // | ||||
|   // We can calc RTT if we send a send report and get a report block back. | ||||
|  | ||||
|     /* | ||||
|         rtcpPacket.ReportBlockItem.SSRC | ||||
|         The SSRC identifier of the source to which the information in this | ||||
|         reception report block pertains. | ||||
|     */ | ||||
|   // |rtcpPacket.ReportBlockItem.SSRC| is the SSRC identifier of the source to | ||||
|   // which the information in this reception report block pertains. | ||||
|  | ||||
|     // if we receive a RTCP packet with multiple reportBlocks only store the ones to us | ||||
|     if( _SSRC && | ||||
|         numberOfReportBlocks > 1) | ||||
|     { | ||||
|         // we have more than one reportBlock in the RTCP packet | ||||
|         if(rtcpPacket.ReportBlockItem.SSRC != _SSRC) | ||||
|         { | ||||
|             // this block is not for us ignore it | ||||
|             return; | ||||
|         } | ||||
|   // Filter out all report blocks that are not for us. | ||||
|   if (rtcpPacket.ReportBlockItem.SSRC != _SSRC) { | ||||
|     // This block is not for us ignore it. | ||||
|     return; | ||||
|   } | ||||
|  | ||||
|   // To avoid problem with acquiring _criticalSectionRTCPSender while holding | ||||
|   // _criticalSectionRTCPReceiver. | ||||
|   _criticalSectionRTCPReceiver->Leave(); | ||||
|   WebRtc_UWord32 sendTimeMS = | ||||
|       _rtpRtcp.SendTimeOfSendReport(rtcpPacket.ReportBlockItem.LastSR); | ||||
|   _criticalSectionRTCPReceiver->Enter(); | ||||
|  | ||||
|   RTCPReportBlockInformation* reportBlock = | ||||
|       CreateReportBlockInformation(remoteSSRC); | ||||
|   if (reportBlock == NULL) { | ||||
|     WEBRTC_TRACE(kTraceError, kTraceRtpRtcp, _id, | ||||
|                  "\tfailed to CreateReportBlockInformation(%u)", remoteSSRC); | ||||
|     return; | ||||
|   } | ||||
|   const RTCPPacketReportBlockItem& rb = rtcpPacket.ReportBlockItem; | ||||
|   reportBlock->remoteReceiveBlock.remoteSSRC = remoteSSRC; | ||||
|   reportBlock->remoteReceiveBlock.sourceSSRC = rb.SSRC; | ||||
|   reportBlock->remoteReceiveBlock.fractionLost = rb.FractionLost; | ||||
|   reportBlock->remoteReceiveBlock.cumulativeLost = | ||||
|       rb.CumulativeNumOfPacketsLost; | ||||
|   reportBlock->remoteReceiveBlock.extendedHighSeqNum = | ||||
|       rb.ExtendedHighestSequenceNumber; | ||||
|   reportBlock->remoteReceiveBlock.jitter = rb.Jitter; | ||||
|   reportBlock->remoteReceiveBlock.delaySinceLastSR = rb.DelayLastSR; | ||||
|   reportBlock->remoteReceiveBlock.lastSR = rb.LastSR; | ||||
|  | ||||
|   if (rtcpPacket.ReportBlockItem.Jitter > reportBlock->remoteMaxJitter) { | ||||
|     reportBlock->remoteMaxJitter = rtcpPacket.ReportBlockItem.Jitter; | ||||
|   } | ||||
|  | ||||
|   WebRtc_UWord32 delaySinceLastSendReport = | ||||
|       rtcpPacket.ReportBlockItem.DelayLastSR; | ||||
|  | ||||
|   // local NTP time when we received this | ||||
|   WebRtc_UWord32 lastReceivedRRNTPsecs = 0; | ||||
|   WebRtc_UWord32 lastReceivedRRNTPfrac = 0; | ||||
|  | ||||
|   _clock.CurrentNTP(lastReceivedRRNTPsecs, lastReceivedRRNTPfrac); | ||||
|  | ||||
|   // time when we received this in MS | ||||
|   WebRtc_UWord32 receiveTimeMS = ModuleRTPUtility::ConvertNTPTimeToMS( | ||||
|       lastReceivedRRNTPsecs, lastReceivedRRNTPfrac); | ||||
|  | ||||
|   // Estimate RTT | ||||
|   WebRtc_UWord32 d = (delaySinceLastSendReport & 0x0000ffff) * 1000; | ||||
|   d /= 65536; | ||||
|   d += ((delaySinceLastSendReport & 0xffff0000) >> 16) * 1000; | ||||
|  | ||||
|   WebRtc_Word32 RTT = 0; | ||||
|  | ||||
|   if (sendTimeMS > 0) { | ||||
|     RTT = receiveTimeMS - d - sendTimeMS; | ||||
|     if (RTT <= 0) { | ||||
|       RTT = 1; | ||||
|     } | ||||
|  | ||||
|     _criticalSectionRTCPReceiver->Leave(); | ||||
|      // to avoid problem with accuireing _criticalSectionRTCPSender while holding _criticalSectionRTCPReceiver | ||||
|  | ||||
|     WebRtc_UWord32 sendTimeMS =  | ||||
|         _rtpRtcp.SendTimeOfSendReport(rtcpPacket.ReportBlockItem.LastSR); | ||||
|  | ||||
|     _criticalSectionRTCPReceiver->Enter(); | ||||
|  | ||||
|     // ReportBlockItem.SSRC is who it's to | ||||
|     // we store all incoming reports, used in conference relay | ||||
|  | ||||
|     RTCPReportBlockInformation* reportBlock = CreateReportBlockInformation(remoteSSRC); | ||||
|     if(reportBlock == NULL) | ||||
|     { | ||||
|         return; | ||||
|     if (RTT > reportBlock->maxRTT) { | ||||
|       // store max RTT | ||||
|       reportBlock->maxRTT = (WebRtc_UWord16) RTT; | ||||
|     } | ||||
|  | ||||
|     reportBlock->remoteReceiveBlock.fractionLost      = rtcpPacket.ReportBlockItem.FractionLost; | ||||
|     reportBlock->remoteReceiveBlock.cumulativeLost    = rtcpPacket.ReportBlockItem.CumulativeNumOfPacketsLost; | ||||
|     reportBlock->remoteReceiveBlock.extendedHighSeqNum= rtcpPacket.ReportBlockItem.ExtendedHighestSequenceNumber; | ||||
|     reportBlock->remoteReceiveBlock.jitter            = rtcpPacket.ReportBlockItem.Jitter; | ||||
|     reportBlock->remoteReceiveBlock.delaySinceLastSR  = rtcpPacket.ReportBlockItem.DelayLastSR; | ||||
|     reportBlock->remoteReceiveBlock.lastSR            = rtcpPacket.ReportBlockItem.LastSR; | ||||
|  | ||||
|     if(rtcpPacket.ReportBlockItem.Jitter > reportBlock->remoteMaxJitter) | ||||
|     { | ||||
|         reportBlock->remoteMaxJitter = rtcpPacket.ReportBlockItem.Jitter; | ||||
|     if (reportBlock->minRTT == 0) { | ||||
|       // first RTT | ||||
|       reportBlock->minRTT = (WebRtc_UWord16) RTT; | ||||
|     } else if (RTT < reportBlock->minRTT) { | ||||
|       // Store min RTT | ||||
|       reportBlock->minRTT = (WebRtc_UWord16) RTT; | ||||
|     } | ||||
|     // store last RTT | ||||
|     reportBlock->RTT = (WebRtc_UWord16) RTT; | ||||
|  | ||||
|     WebRtc_UWord32 delaySinceLastSendReport   = rtcpPacket.ReportBlockItem.DelayLastSR; | ||||
|  | ||||
|     // do we have a local SSRC | ||||
|     // keep track of our relayed SSRC too | ||||
|     if(_SSRC) | ||||
|     { | ||||
|         // we filter rtcpPacket.ReportBlockItem.SSRC to our SSRC | ||||
|         // hence only reports to us | ||||
|         if( rtcpPacket.ReportBlockItem.SSRC == _SSRC) | ||||
|         { | ||||
|             // local NTP time when we received this | ||||
|             WebRtc_UWord32 lastReceivedRRNTPsecs = 0; | ||||
|             WebRtc_UWord32 lastReceivedRRNTPfrac = 0; | ||||
|  | ||||
|             _clock.CurrentNTP(lastReceivedRRNTPsecs, lastReceivedRRNTPfrac); | ||||
|  | ||||
|             // time when we received this in MS | ||||
|             WebRtc_UWord32 receiveTimeMS = ModuleRTPUtility::ConvertNTPTimeToMS(lastReceivedRRNTPsecs, lastReceivedRRNTPfrac); | ||||
|  | ||||
|             // Estimate RTT | ||||
|             WebRtc_UWord32 d =(delaySinceLastSendReport&0x0000ffff)*1000; | ||||
|             d /= 65536; | ||||
|             d+=((delaySinceLastSendReport&0xffff0000)>>16)*1000; | ||||
|  | ||||
|             WebRtc_Word32 RTT = 0; | ||||
|  | ||||
|             if(sendTimeMS > 0) | ||||
|             { | ||||
|                 RTT = receiveTimeMS - d - sendTimeMS; | ||||
|                 if( RTT <= 0) | ||||
|                 { | ||||
|                     RTT = 1; | ||||
|                 } | ||||
|                 if (RTT > reportBlock->maxRTT) | ||||
|                 { | ||||
|                     // store max RTT | ||||
|                     reportBlock->maxRTT = (WebRtc_UWord16)RTT; | ||||
|                 } | ||||
|                 if(reportBlock->minRTT == 0) | ||||
|                 { | ||||
|                     // first RTT | ||||
|                     reportBlock->minRTT = (WebRtc_UWord16)RTT; | ||||
|                 }else if (RTT < reportBlock->minRTT) | ||||
|                 { | ||||
|                     // Store min RTT | ||||
|                     reportBlock->minRTT = (WebRtc_UWord16)RTT; | ||||
|                 } | ||||
|                 // store last RTT | ||||
|                 reportBlock->RTT = (WebRtc_UWord16)RTT; | ||||
|  | ||||
|                 // store average RTT | ||||
|                 if(reportBlock->numAverageCalcs != 0) | ||||
|                 { | ||||
|                     float ac = static_cast<float>(reportBlock->numAverageCalcs); | ||||
|                     float newAverage = ((ac / (ac + 1)) * reportBlock->avgRTT) + ((1 / (ac + 1)) * RTT); | ||||
|                     reportBlock->avgRTT = static_cast<int>(newAverage + 0.5f); | ||||
|                 }else | ||||
|                 { | ||||
|                     // first RTT | ||||
|                     reportBlock->avgRTT = (WebRtc_UWord16)RTT; | ||||
|                 } | ||||
|                 reportBlock->numAverageCalcs++; | ||||
|             } | ||||
|  | ||||
|             WEBRTC_TRACE(kTraceDebug, kTraceRtpRtcp, _id, | ||||
|                 " -> Received report block(%d), from SSRC:0x%x, RTT:%d, loss:%d", _id, remoteSSRC, RTT, rtcpPacket.ReportBlockItem.FractionLost); | ||||
|  | ||||
|             // rtcpPacketInformation | ||||
|             rtcpPacketInformation.AddReportInfo(reportBlock->remoteReceiveBlock.fractionLost, | ||||
|                                                 (WebRtc_UWord16)RTT, | ||||
|                                                 reportBlock->remoteReceiveBlock.extendedHighSeqNum, | ||||
|                                                 reportBlock->remoteReceiveBlock.jitter); | ||||
|         } | ||||
|     // store average RTT | ||||
|     if (reportBlock->numAverageCalcs != 0) { | ||||
|       float ac = static_cast<float> (reportBlock->numAverageCalcs); | ||||
|       float newAverage = ((ac / (ac + 1)) * reportBlock->avgRTT) | ||||
|           + ((1 / (ac + 1)) * RTT); | ||||
|       reportBlock->avgRTT = static_cast<int> (newAverage + 0.5f); | ||||
|     } else { | ||||
|       // first RTT | ||||
|       reportBlock->avgRTT = (WebRtc_UWord16) RTT; | ||||
|     } | ||||
|     reportBlock->numAverageCalcs++; | ||||
|   } | ||||
|  | ||||
|   WEBRTC_TRACE(kTraceDebug, kTraceRtpRtcp, _id, | ||||
|                " -> Received report block(%d), from SSRC:0x%x, RTT:%d, loss:%d", | ||||
|                _id, remoteSSRC, RTT, rtcpPacket.ReportBlockItem.FractionLost); | ||||
|  | ||||
|   // rtcpPacketInformation | ||||
|   rtcpPacketInformation.AddReportInfo( | ||||
|       reportBlock->remoteReceiveBlock.fractionLost, (WebRtc_UWord16) RTT, | ||||
|       reportBlock->remoteReceiveBlock.extendedHighSeqNum, | ||||
|       reportBlock->remoteReceiveBlock.jitter); | ||||
| } | ||||
|  | ||||
| RTCPReportBlockInformation* | ||||
|   | ||||
| @@ -11,6 +11,8 @@ | ||||
| #ifndef WEBRTC_MODULES_RTP_RTCP_SOURCE_RTCP_RECEIVER_H_ | ||||
| #define WEBRTC_MODULES_RTP_RTCP_SOURCE_RTCP_RECEIVER_H_ | ||||
|  | ||||
| #include <vector> | ||||
|  | ||||
| #include "typedefs.h" | ||||
| #include "map_wrapper.h" | ||||
| #include "rtp_utility.h" | ||||
| @@ -81,8 +83,9 @@ public: | ||||
|         const WebRtc_UWord64 pitureID) const; | ||||
|  | ||||
|     // get statistics | ||||
|     WebRtc_Word32 StatisticsReceived(const WebRtc_UWord32 remoteSSRC, | ||||
|                                      RTCPReportBlock* receiveBlock) const; | ||||
|     WebRtc_Word32 StatisticsReceived( | ||||
|         std::vector<RTCPReportBlock>* receiveBlocks) const; | ||||
|  | ||||
|     // Get TMMBR | ||||
|     WebRtc_Word32 TMMBRReceived(const WebRtc_UWord32 size, | ||||
|                                 const WebRtc_UWord32 accNumCandidates, | ||||
|   | ||||
| @@ -1738,13 +1738,11 @@ WebRtc_Word32 ModuleRtpRtcpImpl::RemoteRTCPStat( RTCPSenderInfo* senderInfo) | ||||
| } | ||||
|  | ||||
|     // received RTCP report | ||||
| WebRtc_Word32 | ||||
| ModuleRtpRtcpImpl::RemoteRTCPStat(const WebRtc_UWord32 remoteSSRC, | ||||
|                                   RTCPReportBlock* receiveBlock) | ||||
| { | ||||
|     WEBRTC_TRACE(kTraceModuleCall, kTraceRtpRtcp, _id, "RemoteRTCPStat()"); | ||||
| WebRtc_Word32 ModuleRtpRtcpImpl::RemoteRTCPStat( | ||||
|   std::vector<RTCPReportBlock>* receiveBlocks) const { | ||||
|   WEBRTC_TRACE(kTraceModuleCall, kTraceRtpRtcp, _id, "RemoteRTCPStat()"); | ||||
|  | ||||
|     return _rtcpReceiver.StatisticsReceived(remoteSSRC, receiveBlock); | ||||
|   return _rtcpReceiver.StatisticsReceived(receiveBlocks); | ||||
| } | ||||
|  | ||||
| WebRtc_Word32 | ||||
|   | ||||
| @@ -316,8 +316,8 @@ public: | ||||
|     virtual WebRtc_Word32 RemoteRTCPStat( RTCPSenderInfo* senderInfo); | ||||
|  | ||||
|     // Get received RTCP report, report block | ||||
|     virtual WebRtc_Word32 RemoteRTCPStat( const WebRtc_UWord32 remoteSSRC, | ||||
|                                         RTCPReportBlock* receiveBlock); | ||||
|     virtual WebRtc_Word32 RemoteRTCPStat( | ||||
|         std::vector<RTCPReportBlock>* receiveBlocks) const; | ||||
|  | ||||
|     // Set received RTCP report block | ||||
|     virtual WebRtc_Word32 AddRTCPReportBlock(const WebRtc_UWord32 SSRC, | ||||
|   | ||||
| @@ -113,6 +113,41 @@ class RtpRtcpRtcpTest : public ::testing::Test { | ||||
|     delete receiver; | ||||
|   } | ||||
|  | ||||
|   void SetUpCallFromModule1(RtcpCallback* feedback1, RtcpCallback* feedback2 ) { | ||||
|     EXPECT_EQ(0, module1->RegisterIncomingRTCPCallback(feedback1)); | ||||
|     EXPECT_EQ(0, module2->RegisterIncomingRTCPCallback(feedback2)); | ||||
|  | ||||
|     EXPECT_EQ(0, module1->SetRTCPStatus(kRtcpCompound)); | ||||
|     EXPECT_EQ(0, module2->SetRTCPStatus(kRtcpCompound)); | ||||
|  | ||||
|     EXPECT_EQ(0, module2->SetSSRC(test_ssrc + 1)); | ||||
|     EXPECT_EQ(0, module1->SetSSRC(test_ssrc)); | ||||
|     EXPECT_EQ(0, module1->SetSequenceNumber(test_sequence_number)); | ||||
|     EXPECT_EQ(0, module1->SetStartTimestamp(test_timestamp)); | ||||
|     EXPECT_EQ(0, module1->SetCSRCs(test_CSRC, 2)); | ||||
|     EXPECT_EQ(0, module1->SetCNAME("john.doe@test.test")); | ||||
|  | ||||
|     EXPECT_EQ(0, module1->SetSendingStatus(true)); | ||||
|  | ||||
|     CodecInst voiceCodec; | ||||
|     voiceCodec.pltype = 96; | ||||
|     voiceCodec.plfreq = 8000; | ||||
|     voiceCodec.rate = 64000; | ||||
|     memcpy(voiceCodec.plname, "PCMU", 5); | ||||
|  | ||||
|     EXPECT_EQ(0, module1->RegisterSendPayload(voiceCodec)); | ||||
|     EXPECT_EQ(0, module1->RegisterReceivePayload(voiceCodec)); | ||||
|     EXPECT_EQ(0, module2->RegisterSendPayload(voiceCodec)); | ||||
|     EXPECT_EQ(0, module2->RegisterReceivePayload(voiceCodec)); | ||||
|  | ||||
|     // We need to send one RTP packet to get the RTCP packet to be accepted by | ||||
|     // the receiving module. | ||||
|     // send RTP packet with the data "testtest" | ||||
|     const WebRtc_UWord8 test[9] = "testtest"; | ||||
|     EXPECT_EQ(0, module1->SendOutgoingData(webrtc::kAudioFrameSpeech, 96, | ||||
|                                            0, test, 8)); | ||||
|   } | ||||
|  | ||||
|   int test_id; | ||||
|   RtpRtcp* module1; | ||||
|   RtpRtcp* module2; | ||||
| @@ -129,38 +164,8 @@ class RtpRtcpRtcpTest : public ::testing::Test { | ||||
| TEST_F(RtpRtcpRtcpTest, RTCP) { | ||||
|   RtcpCallback* myRTCPFeedback1 = new RtcpCallback(module1); | ||||
|   RtcpCallback* myRTCPFeedback2 = new RtcpCallback(module2); | ||||
|   EXPECT_EQ(0, module1->RegisterIncomingRTCPCallback(myRTCPFeedback1)); | ||||
|   EXPECT_EQ(0, module2->RegisterIncomingRTCPCallback(myRTCPFeedback2)); | ||||
|  | ||||
|   EXPECT_EQ(0, module1->SetRTCPStatus(kRtcpCompound)); | ||||
|   EXPECT_EQ(0, module2->SetRTCPStatus(kRtcpCompound)); | ||||
|  | ||||
|   EXPECT_EQ(0, module2->SetSSRC(test_ssrc + 1)); | ||||
|   EXPECT_EQ(0, module1->SetSSRC(test_ssrc)); | ||||
|   EXPECT_EQ(0, module1->SetSequenceNumber(test_sequence_number)); | ||||
|   EXPECT_EQ(0, module1->SetStartTimestamp(test_timestamp)); | ||||
|   EXPECT_EQ(0, module1->SetCSRCs(test_CSRC, 2)); | ||||
|   EXPECT_EQ(0, module1->SetCNAME("john.doe@test.test")); | ||||
|  | ||||
|   EXPECT_EQ(0, module1->SetSendingStatus(true)); | ||||
|  | ||||
|   CodecInst voiceCodec; | ||||
|   voiceCodec.pltype = 96; | ||||
|   voiceCodec.plfreq = 8000; | ||||
|   voiceCodec.rate = 64000; | ||||
|   memcpy(voiceCodec.plname, "PCMU", 5); | ||||
|  | ||||
|   EXPECT_EQ(0, module1->RegisterSendPayload(voiceCodec)); | ||||
|   EXPECT_EQ(0, module1->RegisterReceivePayload(voiceCodec)); | ||||
|   EXPECT_EQ(0, module2->RegisterSendPayload(voiceCodec)); | ||||
|   EXPECT_EQ(0, module2->RegisterReceivePayload(voiceCodec)); | ||||
|  | ||||
|   // We need to send one RTP packet to get the RTCP packet to be accepted by | ||||
|   // the receiving module. | ||||
|   // send RTP packet with the data "testtest" | ||||
|   const WebRtc_UWord8 test[9] = "testtest"; | ||||
|   EXPECT_EQ(0, module1->SendOutgoingData(webrtc::kAudioFrameSpeech, 96, | ||||
|                                          0, test, 8)); | ||||
|   SetUpCallFromModule1(myRTCPFeedback1, myRTCPFeedback2); | ||||
|  | ||||
|   EXPECT_EQ(0, module1->SendRTCPReferencePictureSelection(kTestPictureId)); | ||||
|   EXPECT_EQ(0, module1->SendRTCPSliceLossIndication(156)); | ||||
| @@ -237,10 +242,12 @@ TEST_F(RtpRtcpRtcpTest, RTCP) { | ||||
|   EXPECT_EQ(0, strncmp(cName, "jane@192.168.0.2", RTCP_CNAME_SIZE)); | ||||
|  | ||||
|   // get all report blocks | ||||
|   RTCPReportBlock reportBlockReceived; | ||||
|   EXPECT_EQ(-1, module1->RemoteRTCPStat(test_ssrc, &reportBlockReceived)); | ||||
|   EXPECT_EQ(-1, module1->RemoteRTCPStat(test_ssrc + 1, NULL)); | ||||
|   EXPECT_EQ(0, module1->RemoteRTCPStat(test_ssrc + 1, &reportBlockReceived)); | ||||
|   std::vector<RTCPReportBlock> report_blocks; | ||||
|   EXPECT_EQ(-1, module1->RemoteRTCPStat(NULL)); | ||||
|   EXPECT_EQ(0, module1->RemoteRTCPStat(&report_blocks)); | ||||
|   EXPECT_EQ(1u, report_blocks.size()); | ||||
|   const RTCPReportBlock& reportBlockReceived = report_blocks[0]; | ||||
|  | ||||
|   float secSinceLastReport = | ||||
|       static_cast<float>(reportBlockReceived.delaySinceLastSR) / 65536.0f; | ||||
|   EXPECT_GE(0.101f, secSinceLastReport); | ||||
| @@ -295,3 +302,32 @@ TEST_F(RtpRtcpRtcpTest, RTCP) { | ||||
|   delete myRTCPFeedback1; | ||||
|   delete myRTCPFeedback2; | ||||
| } | ||||
|  | ||||
| TEST_F(RtpRtcpRtcpTest, RemoteRTCPStatRemote) { | ||||
|   std::vector<RTCPReportBlock> report_blocks; | ||||
|  | ||||
|   RtcpCallback feedback1(module1); | ||||
|   RtcpCallback feedback2(module2); | ||||
|  | ||||
|   SetUpCallFromModule1(&feedback1, &feedback2); | ||||
|   EXPECT_EQ(0, module1->RemoteRTCPStat(&report_blocks)); | ||||
|   EXPECT_EQ(0u, report_blocks.size()); | ||||
|  | ||||
|   // send RTCP packet, triggered by timer | ||||
|   fake_clock.IncrementTime(7500); | ||||
|   module1->Process(); | ||||
|   fake_clock.IncrementTime(100); | ||||
|   module2->Process(); | ||||
|  | ||||
|   EXPECT_EQ(0, module1->RemoteRTCPStat(&report_blocks)); | ||||
|   ASSERT_EQ(1u, report_blocks.size()); | ||||
|  | ||||
|   // |test_ssrc+1| is the SSRC of module2 that send the report. | ||||
|   EXPECT_EQ(test_ssrc+1, report_blocks[0].remoteSSRC); | ||||
|   EXPECT_EQ(test_ssrc, report_blocks[0].sourceSSRC); | ||||
|  | ||||
|   EXPECT_EQ(0u, report_blocks[0].cumulativeLost); | ||||
|   EXPECT_LT(0u, report_blocks[0].delaySinceLastSR); | ||||
|   EXPECT_EQ(test_sequence_number, report_blocks[0].extendedHighSeqNum); | ||||
|   EXPECT_EQ(0u, report_blocks[0].fractionLost); | ||||
| } | ||||
|   | ||||
| @@ -11,6 +11,7 @@ | ||||
| #include "video_engine/vie_channel.h" | ||||
|  | ||||
| #include <algorithm> | ||||
| #include <vector> | ||||
|  | ||||
| #include "modules/rtp_rtcp/interface/rtp_rtcp.h" | ||||
| #include "modules/udp_transport/interface/udp_transport.h" | ||||
| @@ -941,16 +942,36 @@ WebRtc_Word32 ViEChannel::GetSendRtcpStatistics(WebRtc_UWord16& fraction_lost, | ||||
|   //   RtpRtcp* rtp_rtcp = *it; | ||||
|   // } | ||||
|   WebRtc_UWord32 remoteSSRC = rtp_rtcp_.RemoteSSRC(); | ||||
|   RTCPReportBlock remote_stat; | ||||
|   if (rtp_rtcp_.RemoteRTCPStat(remoteSSRC, &remote_stat) != 0) { | ||||
|  | ||||
|   // Get all RTCP receiver report blocks that have been received on this | ||||
|   // channel. If we receive RTP packets from a remote source we know the | ||||
|   // remote SSRC and use the report block from him. | ||||
|   // Otherwise use the first report block. | ||||
|   std::vector<RTCPReportBlock> remote_stats; | ||||
|   if (rtp_rtcp_.RemoteRTCPStat(&remote_stats) != 0 || remote_stats.empty()) { | ||||
|     WEBRTC_TRACE(kTraceWarning, kTraceVideo, ViEId(engine_id_, channel_id_), | ||||
|                  "%s: Could not get remote stats", __FUNCTION__); | ||||
|     return -1; | ||||
|   } | ||||
|   fraction_lost = remote_stat.fractionLost; | ||||
|   cumulative_lost = remote_stat.cumulativeLost; | ||||
|   extended_max = remote_stat.extendedHighSeqNum; | ||||
|   jitter_samples = remote_stat.jitter; | ||||
|   std::vector<RTCPReportBlock>::const_iterator statistics = | ||||
|       remote_stats.begin(); | ||||
|   for (; statistics != remote_stats.end(); ++statistics) { | ||||
|     if (statistics->remoteSSRC == remoteSSRC) | ||||
|       break; | ||||
|   } | ||||
|  | ||||
|   if (statistics == remote_stats.end()) { | ||||
|     // If we have not received any RTCP packets from this SSRC it probably means | ||||
|     // we have not received any RTP packets. | ||||
|     // Use the first received report block instead. | ||||
|     statistics = remote_stats.begin(); | ||||
|     remoteSSRC = statistics->remoteSSRC; | ||||
|   } | ||||
|  | ||||
|   fraction_lost = statistics->fractionLost; | ||||
|   cumulative_lost = statistics->cumulativeLost; | ||||
|   extended_max = statistics->extendedHighSeqNum; | ||||
|   jitter_samples = statistics->jitter; | ||||
|  | ||||
|   WebRtc_UWord16 dummy; | ||||
|   WebRtc_UWord16 rtt = 0; | ||||
|   | ||||
| @@ -5330,39 +5330,45 @@ Channel::GetRemoteRTCPData( | ||||
|  | ||||
|     if (NULL != jitter || NULL != fractionLost) | ||||
|     { | ||||
|         WebRtc_Word32 ret(-1); | ||||
|         RTCPReportBlock reportBlock; | ||||
|         WebRtc_Word32 remoteSSRC = _rtpRtcpModule.RemoteSSRC(); | ||||
|         if (remoteSSRC > 0) | ||||
|         { | ||||
|             // We must feed the module with remote SSRC to get the correct | ||||
|             // report block. | ||||
|             ret = _rtpRtcpModule.RemoteRTCPStat(remoteSSRC, &reportBlock); | ||||
|         // Get all RTCP receiver report blocks that have been received on this | ||||
|         // channel. If we receive RTP packets from a remote source we know the | ||||
|         // remote SSRC and use the report block from him. | ||||
|         // Otherwise use the first report block. | ||||
|         std::vector<RTCPReportBlock> remote_stats; | ||||
|         if (_rtpRtcpModule.RemoteRTCPStat(&remote_stats) != 0 || | ||||
|             remote_stats.empty()) { | ||||
|           WEBRTC_TRACE(kTraceWarning, kTraceVoice, | ||||
|                        VoEId(_instanceId, _channelId), | ||||
|                        "GetRemoteRTCPData() failed to measure statistics due" | ||||
|                        " to lack of received RTP and/or RTCP packets"); | ||||
|           return -1; | ||||
|         } | ||||
|         if ((remoteSSRC < 0) || (ret != 0)) | ||||
|         { | ||||
|             reportBlock.jitter = 0; | ||||
|             reportBlock.fractionLost = 0; | ||||
|             WEBRTC_TRACE(kTraceWarning, kTraceVoice, | ||||
|                          VoEId(_instanceId, _channelId), | ||||
|                          "GetRemoteRTCPData() failed to measure statistics due" | ||||
|                          " to lack of received RTP and/or RTCP packets"); | ||||
|  | ||||
|         WebRtc_UWord32 remoteSSRC = _rtpRtcpModule.RemoteSSRC(); | ||||
|         std::vector<RTCPReportBlock>::const_iterator it = remote_stats.begin(); | ||||
|         for (; it != remote_stats.end(); ++it) { | ||||
|           if (it->remoteSSRC == remoteSSRC) | ||||
|             break; | ||||
|         } | ||||
|         if (NULL != jitter) | ||||
|         { | ||||
|             *jitter = reportBlock.jitter; | ||||
|             WEBRTC_TRACE(kTraceStateInfo, kTraceVoice, | ||||
|                          VoEId(_instanceId, _channelId), | ||||
|                          "GetRemoteRTCPData() => jitter = %lu", *jitter); | ||||
|         } | ||||
|         if (NULL != fractionLost) | ||||
|         { | ||||
|             *fractionLost = reportBlock.fractionLost; | ||||
|             WEBRTC_TRACE(kTraceStateInfo, kTraceVoice, | ||||
|                          VoEId(_instanceId, _channelId), | ||||
|                          "GetRemoteRTCPData() => fractionLost = %lu", | ||||
|                          *fractionLost); | ||||
|  | ||||
|         if (it == remote_stats.end()) { | ||||
|           // If we have not received any RTCP packets from this SSRC it probably | ||||
|           // means that we have not received any RTP packets. | ||||
|           // Use the first received report block instead. | ||||
|           it = remote_stats.begin(); | ||||
|           remoteSSRC = it->remoteSSRC; | ||||
|         } | ||||
|  | ||||
|         *jitter = it->jitter; | ||||
|         WEBRTC_TRACE(kTraceStateInfo, kTraceVoice, | ||||
|                      VoEId(_instanceId, _channelId), | ||||
|                      "GetRemoteRTCPData() => jitter = %lu", *jitter); | ||||
|  | ||||
|         *fractionLost = it->fractionLost; | ||||
|         WEBRTC_TRACE(kTraceStateInfo, kTraceVoice, | ||||
|                      VoEId(_instanceId, _channelId), | ||||
|                      "GetRemoteRTCPData() => fractionLost = %lu", | ||||
|                      *fractionLost); | ||||
|     } | ||||
|     return 0; | ||||
| } | ||||
|   | ||||
		Reference in New Issue
	
	Block a user
	 perkj@webrtc.org
					perkj@webrtc.org