1209 lines
33 KiB
C++
1209 lines
33 KiB
C++
|
/*----------------------------------------------------------------------------
|
|||
|
* File: RTCPRECV.C
|
|||
|
* Product: RTP/RTCP implementation
|
|||
|
* Description: Provides the RTCP receive network I/O.
|
|||
|
*
|
|||
|
* INTEL Corporation Proprietary Information
|
|||
|
* This listing is supplied under the terms of a license agreement with
|
|||
|
* Intel Corporation and may not be copied nor disclosed except in
|
|||
|
* accordance with the terms of that agreement.
|
|||
|
* Copyright (c) 1995 Intel Corporation.
|
|||
|
*--------------------------------------------------------------------------*/
|
|||
|
|
|||
|
|
|||
|
#include "rrcm.h"
|
|||
|
|
|||
|
|
|||
|
#define MIN(a, b) ((a < b) ? a : b)
|
|||
|
|
|||
|
|
|||
|
/*---------------------------------------------------------------------------
|
|||
|
/ Global Variables
|
|||
|
/--------------------------------------------------------------------------*/
|
|||
|
|
|||
|
|
|||
|
/*---------------------------------------------------------------------------
|
|||
|
/ External Variables
|
|||
|
/--------------------------------------------------------------------------*/
|
|||
|
extern PRTCP_CONTEXT pRTCPContext;
|
|||
|
extern PRTP_CONTEXT pRTPContext;
|
|||
|
extern RRCM_WS RRCMws;
|
|||
|
|
|||
|
#ifdef ENABLE_ISDM2
|
|||
|
extern ISDM2 Isdm2;
|
|||
|
#endif
|
|||
|
|
|||
|
#ifdef _DEBUG
|
|||
|
extern char debug_string[];
|
|||
|
#endif
|
|||
|
|
|||
|
#if (defined(_DEBUG) || defined(PCS_COMPLIANCE))
|
|||
|
//INTEROP
|
|||
|
extern LPInteropLogger RTPLogger;
|
|||
|
#endif
|
|||
|
|
|||
|
|
|||
|
|
|||
|
/*----------------------------------------------------------------------------
|
|||
|
* Function : RTCPrcvInit
|
|||
|
* Description: RTCP receive initialisation.
|
|||
|
*
|
|||
|
* Input : pRTCP : Pointer to the RTCP session information
|
|||
|
*
|
|||
|
* Return: TRUE/FALSE
|
|||
|
---------------------------------------------------------------------------*/
|
|||
|
DWORD RTCPrcvInit (PSSRC_ENTRY pSSRC)
|
|||
|
{
|
|||
|
PRTCP_BFR_LIST pRcvStruct;
|
|||
|
PRTCP_SESSION pRTCP;
|
|||
|
int dwStatus;
|
|||
|
int dwError;
|
|||
|
int errorCnt = 0;
|
|||
|
int bfrErrorCnt = 0;
|
|||
|
DWORD idx;
|
|||
|
int wsockSuccess = FALSE;
|
|||
|
|
|||
|
// save a pointer to the corresponding RTCP session
|
|||
|
pRTCP = pSSRC->pRTCPses;
|
|||
|
|
|||
|
// Post receive buffers for WS-2. As these buffers are posted per receive
|
|||
|
// thread, few of them should be plenty enough for RTCP.
|
|||
|
for (idx = 0; idx < pRTPContext->registry.NumRTCPPostedBfr; idx++)
|
|||
|
{
|
|||
|
// get a free RTCP buffer for a receive operation
|
|||
|
pRcvStruct =
|
|||
|
(PRTCP_BFR_LIST)removePcktFromTail(&pRTCP->RTCPrcvBfrList,
|
|||
|
&pRTCP->critSect);
|
|||
|
|
|||
|
// check buffer
|
|||
|
if (pRcvStruct == NULL)
|
|||
|
{
|
|||
|
RRCM_DBG_MSG ("RTCP: ERROR - Rcv Bfr Allocation Error", 0,
|
|||
|
__FILE__, __LINE__, DBG_ERROR);
|
|||
|
|
|||
|
// make sure we have at least one buffer
|
|||
|
ASSERT (pRcvStruct);
|
|||
|
break;
|
|||
|
}
|
|||
|
|
|||
|
// SSRC entry address of our own session
|
|||
|
pRcvStruct->pSSRC = pSSRC;
|
|||
|
|
|||
|
// received address length reset by the receive routine
|
|||
|
pRcvStruct->addrLen = sizeof(SOCKADDR);
|
|||
|
|
|||
|
// use hEvent to recover the buffer's address
|
|||
|
pRcvStruct->overlapped.hEvent = (WSAEVENT)pRcvStruct;
|
|||
|
|
|||
|
// post the receive buffer for this thread
|
|||
|
dwStatus = RRCMws.recvFrom (pSSRC->RTCPsd,
|
|||
|
&pRcvStruct->bfr,
|
|||
|
pRcvStruct->dwBufferCount,
|
|||
|
&pRcvStruct->dwNumBytesXfr,
|
|||
|
&pRcvStruct->dwFlags,
|
|||
|
(PSOCKADDR)pRcvStruct->addr,
|
|||
|
&pRcvStruct->addrLen,
|
|||
|
(LPWSAOVERLAPPED)&pRcvStruct->overlapped,
|
|||
|
RTCPrcvCallback);
|
|||
|
|
|||
|
// Check Winsock status
|
|||
|
if (dwStatus != 0)
|
|||
|
{
|
|||
|
// error, the receive request won't proceed
|
|||
|
dwError = GetLastError();
|
|||
|
if ((dwError != WSA_IO_PENDING) && (dwError != WSAEMSGSIZE))
|
|||
|
{
|
|||
|
RRCM_DBG_MSG ("RTCP: ERROR WSARecvFrom()", GetLastError(),
|
|||
|
__FILE__, __LINE__, DBG_ERROR);
|
|||
|
|
|||
|
// notify application if interested
|
|||
|
RRCMnotification (RRCM_RTCP_WS_RCV_ERROR, pSSRC,
|
|||
|
pSSRC->SSRC, dwError);
|
|||
|
|
|||
|
// Return the buffer to the free queue
|
|||
|
addToHeadOfList (&pRTCP->RTCPrcvBfrList,
|
|||
|
(PLINK_LIST)pRcvStruct,
|
|||
|
&pRTCP->critSect);
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
wsockSuccess = TRUE;
|
|||
|
|
|||
|
// increment number of I/O pending
|
|||
|
InterlockedIncrement ((long *)&pRTCP->dwNumRcvIoPending);
|
|||
|
}
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
wsockSuccess = TRUE;
|
|||
|
|
|||
|
// increment number of I/O pending
|
|||
|
InterlockedIncrement ((long *)&pRTCP->dwNumRcvIoPending);
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
// make sure we posted at least some buffers
|
|||
|
if (wsockSuccess == FALSE)
|
|||
|
{
|
|||
|
// release all resources and kill the receive thread
|
|||
|
#ifdef _DEBUG
|
|||
|
wsprintf(debug_string,
|
|||
|
"RTCP: ERROR - Exit RCV init %s: Line:%d", __FILE__, __LINE__);
|
|||
|
RRCM_DBG_MSG (debug_string, 0, NULL, 0, DBG_TRACE);
|
|||
|
#endif
|
|||
|
return(FALSE);
|
|||
|
}
|
|||
|
return (TRUE);
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
|
|||
|
/*----------------------------------------------------------------------------
|
|||
|
* Function : RTCPrcvCallback
|
|||
|
* Description: Receive callback routine from Winsock2.
|
|||
|
*
|
|||
|
* Input : dwError: I/O completion status
|
|||
|
* cbTransferred: Number of bytes received
|
|||
|
* lpOverlapped: -> to overlapped structure
|
|||
|
* dwFlags: Flags
|
|||
|
*
|
|||
|
*
|
|||
|
* Return: None
|
|||
|
---------------------------------------------------------------------------*/
|
|||
|
void CALLBACK RTCPrcvCallback (DWORD dwError,
|
|||
|
DWORD cbTransferred,
|
|||
|
LPWSAOVERLAPPED lpOverlapped,
|
|||
|
DWORD dwFlags)
|
|||
|
{
|
|||
|
PRTCP_BFR_LIST pRcvStruct;
|
|||
|
RTCP_T *pRTCPpckt;
|
|||
|
PRTCP_SESSION pRTCPses;
|
|||
|
PSSRC_ENTRY pSSRC;
|
|||
|
PAPP_RTCP_BFR pAppBfr;
|
|||
|
DWORD dwStatus = 0;
|
|||
|
DWORD i;
|
|||
|
DWORD pcktLen;
|
|||
|
DWORD dwSSRC;
|
|||
|
USHORT wHost;
|
|||
|
SOCKET RTCPsd;
|
|||
|
unsigned char *pEndPckt;
|
|||
|
unsigned char *pEndBlock;
|
|||
|
int tmpSize;
|
|||
|
#if IO_CHECK
|
|||
|
DWORD initTime = timeGetTime();
|
|||
|
#endif
|
|||
|
|
|||
|
IN_OUT_STR ("RTCP: Enter RTCPrcvCallback\n");
|
|||
|
|
|||
|
// hEvent in the WSAOVERLAPPED struct points to our buffer's information
|
|||
|
pRcvStruct = (PRTCP_BFR_LIST)lpOverlapped->hEvent;
|
|||
|
|
|||
|
// SSRC entry pointer
|
|||
|
pSSRC = pRcvStruct->pSSRC;
|
|||
|
|
|||
|
// check Winsock callback error status
|
|||
|
if (dwError)
|
|||
|
{
|
|||
|
// 65534 is probably a temporary bug in WS2
|
|||
|
if ((dwError == 65534) || (dwError == WSA_OPERATION_ABORTED))
|
|||
|
{
|
|||
|
RRCM_DBG_MSG ("RTCP: I/O Aborted", dwError,
|
|||
|
__FILE__, __LINE__, DBG_NOTIFY);
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
RRCM_DBG_MSG ("RTCP: ERROR - Rcv Callback", dwError,
|
|||
|
__FILE__, __LINE__, DBG_ERROR);
|
|||
|
}
|
|||
|
|
|||
|
// invalid RTCP packet header, re-queue the buffer
|
|||
|
RTCPpostRecvBfr (pSSRC, pRcvStruct);
|
|||
|
|
|||
|
IN_OUT_STR ("RTCP: Exit RTCPrcvCallback\n");
|
|||
|
return;
|
|||
|
}
|
|||
|
|
|||
|
// read the RTCP packet
|
|||
|
pRTCPpckt = (RTCP_T *)pRcvStruct->bfr.buf;
|
|||
|
|
|||
|
#if (defined(_DEBUG) || defined(PCS_COMPLIANCE))
|
|||
|
//INTEROP
|
|||
|
if (RTPLogger)
|
|||
|
{
|
|||
|
InteropOutput (RTPLogger,
|
|||
|
(BYTE FAR*)(pRcvStruct->bfr.buf),
|
|||
|
cbTransferred,
|
|||
|
RTPLOG_RECEIVED_PDU | RTCP_PDU);
|
|||
|
}
|
|||
|
#endif
|
|||
|
|
|||
|
// get the RTCP session ptr
|
|||
|
pRTCPses = pSSRC->pRTCPses;
|
|||
|
|
|||
|
// Check RTCP header validity of first packet in report.
|
|||
|
// Filter out junk. First thing in RTCP packet must be
|
|||
|
// either SR, RR or BYE
|
|||
|
if ((pRTCPpckt->common.type != RTP_TYPE) ||
|
|||
|
((pRTCPpckt->common.pt != RTCP_SR) &&
|
|||
|
(pRTCPpckt->common.pt != RTCP_RR) &&
|
|||
|
(pRTCPpckt->common.pt != RTCP_BYE)))
|
|||
|
{
|
|||
|
#ifdef MONITOR_STATS
|
|||
|
pRTCPses->dwNumRTCPhdrErr++;
|
|||
|
#endif
|
|||
|
|
|||
|
// invalid RTCP packet header, re-queue the buffer
|
|||
|
RTCPpostRecvBfr (pSSRC, pRcvStruct);
|
|||
|
|
|||
|
#if 0 // we could have shutdown so this code can fault
|
|||
|
if (pRTCPpckt->common.pt == FLUSH_RTP_PAYLOAD_TYPE)
|
|||
|
{
|
|||
|
RRCM_DBG_MSG ("RTCP: Flushing RCV I/O", 0, NULL, 0, DBG_NOTIFY);
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
wsprintf(debug_string,
|
|||
|
"RTCP: ERROR - Pckt Header Error. Type:%d / Payload:%d",
|
|||
|
pRTCPpckt->common.type, pRTCPpckt->common.pt);
|
|||
|
RRCM_DBG_MSG (debug_string, 0, __FILE__, __LINE__, DBG_TRACE);
|
|||
|
}
|
|||
|
#endif
|
|||
|
|
|||
|
IN_OUT_STR ("RTCP: Exit RTCPrcvCallback\n");
|
|||
|
return;
|
|||
|
}
|
|||
|
|
|||
|
// get the socket descriptor
|
|||
|
RTCPsd = pSSRC->RTCPsd;
|
|||
|
|
|||
|
// get the sender's SSRC
|
|||
|
RRCMws.ntohl (RTCPsd, pRTCPpckt->r.sr.ssrc, &dwSSRC);
|
|||
|
|
|||
|
// skip our own loopback if we receive it
|
|||
|
if (ownLoopback (RTCPsd, dwSSRC, pRTCPses))
|
|||
|
{
|
|||
|
RTCPpostRecvBfr (pSSRC, pRcvStruct);
|
|||
|
return;
|
|||
|
}
|
|||
|
|
|||
|
// at this point we think the RTCP packet's valid. Get the sender's
|
|||
|
// address, if not already known
|
|||
|
if (!(pRTCPses->dwSessionStatus & RTCP_DEST_LEARNED))
|
|||
|
{
|
|||
|
pRTCPses->dwSessionStatus |= RTCP_DEST_LEARNED;
|
|||
|
pRTCPses->toLen = pRcvStruct->addrLen;
|
|||
|
memcpy (&pRTCPses->toBfr, &pRcvStruct->addr, pRcvStruct->addrLen);
|
|||
|
|
|||
|
#ifdef ENABLE_ISDM2
|
|||
|
// register our Xmt SSRC - Rcvd one will be found later
|
|||
|
if (Isdm2.hISDMdll)
|
|||
|
registerSessionToISDM (pSSRC, pRTCPses, &Isdm2);
|
|||
|
#endif
|
|||
|
}
|
|||
|
|
|||
|
// Update our RTCP average packet size estimator
|
|||
|
EnterCriticalSection (&pRTCPses->critSect);
|
|||
|
tmpSize = (cbTransferred + NTWRK_HDR_SIZE) - pRTCPses->avgRTCPpktSizeRcvd;
|
|||
|
|
|||
|
#ifdef ENABLE_FLOATING_POINT
|
|||
|
// As per RFC
|
|||
|
tmpSize = (int)(tmpSize * RTCP_SIZE_GAIN);
|
|||
|
#else
|
|||
|
// Need to remove floating point operation
|
|||
|
tmpSize = tmpSize / 16;
|
|||
|
#endif
|
|||
|
|
|||
|
pRTCPses->avgRTCPpktSizeRcvd += tmpSize;
|
|||
|
LeaveCriticalSection (&pRTCPses->critSect);
|
|||
|
|
|||
|
// check if the raw RTCP packet needs to be copied into an application
|
|||
|
// buffer - Mainly used by ActiveMovieRTP to propagate the reports up
|
|||
|
// the filter graph to the Receive Payload Handler
|
|||
|
pAppBfr = (PAPP_RTCP_BFR)removePcktFromHead (&(pRTCPses->appRtcpBfrList),
|
|||
|
&pRTCPses->critSect);
|
|||
|
if (pAppBfr && !(pAppBfr->dwBfrStatus & RTCP_SR_ONLY))
|
|||
|
{
|
|||
|
// copy the full RTCP packet
|
|||
|
memcpy (pAppBfr->bfr,
|
|||
|
pRTCPpckt,
|
|||
|
MIN(pAppBfr->dwBfrLen, cbTransferred));
|
|||
|
|
|||
|
// number of bytes received
|
|||
|
pAppBfr->dwBytesRcvd = MIN(pAppBfr->dwBfrLen, cbTransferred);
|
|||
|
|
|||
|
// set the event associated with this buffer
|
|||
|
SetEvent (pAppBfr->hBfrEvent);
|
|||
|
}
|
|||
|
|
|||
|
// end of the received packet
|
|||
|
pEndPckt = (unsigned char *)pRTCPpckt + cbTransferred;
|
|||
|
|
|||
|
while ((unsigned char *)pRTCPpckt < pEndPckt)
|
|||
|
{
|
|||
|
// get the length
|
|||
|
dwStatus = RRCMws.ntohs (RTCPsd, pRTCPpckt->common.length, &wHost);
|
|||
|
if (dwStatus)
|
|||
|
{
|
|||
|
RRCM_DBG_MSG ("RTCP: ERROR - WSANtohs()", GetLastError(),
|
|||
|
__FILE__, __LINE__, DBG_ERROR);
|
|||
|
}
|
|||
|
|
|||
|
// get this report block length
|
|||
|
pcktLen = (wHost + 1) << 2;
|
|||
|
pEndBlock = (unsigned char *)pRTCPpckt + pcktLen;
|
|||
|
|
|||
|
// sanity check
|
|||
|
if (pEndBlock > pEndPckt)
|
|||
|
{
|
|||
|
RRCM_DBG_MSG ("RTCP: ERROR - Rcv packet length error", 0,
|
|||
|
__FILE__, __LINE__, DBG_ERROR);
|
|||
|
|
|||
|
#ifdef MONITOR_STATS
|
|||
|
pRTCPses->dwNumRTCPlenErr++;
|
|||
|
#endif
|
|||
|
break;
|
|||
|
}
|
|||
|
|
|||
|
// make sure the version is correct for all packets
|
|||
|
if (pRTCPpckt->common.type != RTP_TYPE)
|
|||
|
{
|
|||
|
#ifdef MONITOR_STATS
|
|||
|
pRTCPses->dwNumRTCPhdrErr++;
|
|||
|
#endif
|
|||
|
// invalid RTCP packet header, packet will be re-queued
|
|||
|
break;
|
|||
|
}
|
|||
|
|
|||
|
switch (pRTCPpckt->common.pt)
|
|||
|
{
|
|||
|
case RTCP_SR:
|
|||
|
// check if only the SR needs to be propagated up to the app
|
|||
|
if (pAppBfr && (pAppBfr->dwBfrStatus & RTCP_SR_ONLY))
|
|||
|
{
|
|||
|
// copy the RTCP SR
|
|||
|
memcpy (pAppBfr->bfr,
|
|||
|
pRTCPpckt,
|
|||
|
MIN(pAppBfr->dwBfrLen, 24));
|
|||
|
|
|||
|
// number of bytes received
|
|||
|
pAppBfr->dwBytesRcvd = MIN(pAppBfr->dwBfrLen, 24);
|
|||
|
|
|||
|
// set the event associated with this buffer
|
|||
|
SetEvent (pAppBfr->hBfrEvent);
|
|||
|
}
|
|||
|
|
|||
|
// get the sender's SSRC
|
|||
|
RRCMws.ntohl (RTCPsd, pRTCPpckt->r.sr.ssrc, &dwSSRC);
|
|||
|
|
|||
|
// parse the sender report
|
|||
|
parseRTCPsr (RTCPsd, pRTCPpckt, pRTCPses, pRcvStruct);
|
|||
|
|
|||
|
// parse additional receiver reports if any
|
|||
|
for (i = 0; i < pRTCPpckt->common.count; i++)
|
|||
|
{
|
|||
|
parseRTCPrr (RTCPsd, &pRTCPpckt->r.sr.rr[i],
|
|||
|
pRTCPses, pRcvStruct,
|
|||
|
dwSSRC);
|
|||
|
}
|
|||
|
|
|||
|
// notify application if interested
|
|||
|
RRCMnotification (RRCM_RECV_RTCP_SNDR_REPORT_EVENT, pSSRC,
|
|||
|
dwSSRC, 0);
|
|||
|
break;
|
|||
|
|
|||
|
case RTCP_RR:
|
|||
|
// get the sender's SSRC
|
|||
|
RRCMws.ntohl (RTCPsd, pRTCPpckt->r.rr.ssrc, &dwSSRC);
|
|||
|
|
|||
|
// parse receiver reports
|
|||
|
for (i = 0; i < pRTCPpckt->common.count; i++)
|
|||
|
{
|
|||
|
parseRTCPrr (RTCPsd, &pRTCPpckt->r.rr.rr[i],
|
|||
|
pRTCPses, pRcvStruct,
|
|||
|
dwSSRC);
|
|||
|
}
|
|||
|
|
|||
|
// notify application if interested
|
|||
|
RRCMnotification (RRCM_RECV_RTCP_RECV_REPORT_EVENT, pSSRC,
|
|||
|
dwSSRC, 0);
|
|||
|
break;
|
|||
|
|
|||
|
case RTCP_SDES:
|
|||
|
{
|
|||
|
PCHAR buf;
|
|||
|
|
|||
|
buf = (PCHAR)&pRTCPpckt->r.sdes;
|
|||
|
|
|||
|
for (i = 0; i < pRTCPpckt->common.count; i++)
|
|||
|
{
|
|||
|
buf = parseRTCPsdes (RTCPsd, buf, pRTCPses, pRcvStruct);
|
|||
|
if (buf == NULL)
|
|||
|
break;
|
|||
|
}
|
|||
|
}
|
|||
|
break;
|
|||
|
|
|||
|
case RTCP_BYE:
|
|||
|
for (i = 0; i < pRTCPpckt->common.count; i++)
|
|||
|
parseRTCPbye (RTCPsd, pRTCPpckt->r.bye.src[i],
|
|||
|
pRTCPses, pRcvStruct);
|
|||
|
break;
|
|||
|
|
|||
|
default:
|
|||
|
break;
|
|||
|
}
|
|||
|
|
|||
|
// go to next report block
|
|||
|
pRTCPpckt = (RTCP_T *)(pEndBlock);
|
|||
|
}
|
|||
|
|
|||
|
// post back the buffer to WS-2
|
|||
|
RTCPpostRecvBfr (pSSRC, pRcvStruct);
|
|||
|
|
|||
|
#if IO_CHECK
|
|||
|
wsprintf(debug_string,
|
|||
|
"RTCP: Leaving Rcv Callback after: %ld msec\n",
|
|||
|
timeGetTime() - initTime);
|
|||
|
RRCM_DBG_MSG (debug_string, 0, NULL, 0, DBG_TRACE);
|
|||
|
#endif
|
|||
|
|
|||
|
IN_OUT_STR ("RTCP: Exit RTCPrcvCallback\n");
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
|
|||
|
/*----------------------------------------------------------------------------
|
|||
|
* Function : parseRTCPsr
|
|||
|
* Description: Parse an RTCP sender reports and update the corresponding
|
|||
|
* statistics.
|
|||
|
*
|
|||
|
* Input : sd: RTCP Socket descriptor
|
|||
|
* pRTCPpckt: -> to the RTCP packet
|
|||
|
* pRTCPses: -> to the RTCP session information
|
|||
|
* pRcvStruct: -> to the receive structure information
|
|||
|
*
|
|||
|
* Return: OK: RRCM_NoError
|
|||
|
* !0: Error code (see RRCM.H)
|
|||
|
---------------------------------------------------------------------------*/
|
|||
|
DWORD parseRTCPsr (SOCKET sd,
|
|||
|
RTCP_T *pRTCPpckt,
|
|||
|
PRTCP_SESSION pRTCPses,
|
|||
|
PRTCP_BFR_LIST pRcvStruct)
|
|||
|
{
|
|||
|
PSSRC_ENTRY pSSRC;
|
|||
|
DWORD dwSSRC;
|
|||
|
|
|||
|
IN_OUT_STR ("RTCP: Enter parseRTCPsr\n");
|
|||
|
|
|||
|
// get the sender's SSRC
|
|||
|
RRCMws.ntohl (sd, pRTCPpckt->r.sr.ssrc, &dwSSRC);
|
|||
|
|
|||
|
#ifdef _DEBUG
|
|||
|
wsprintf(debug_string, "RTCP: Receive SR from SSRC:x%lX", dwSSRC);
|
|||
|
RRCM_DBG_MSG (debug_string, 0, NULL, 0, DBG_TRACE);
|
|||
|
#endif
|
|||
|
|
|||
|
// look for the SSRC entry in the list for this RTCP session
|
|||
|
pSSRC = searchforSSRCatTail((PSSRC_ENTRY)pRTCPses->RcvSSRCList.prev,
|
|||
|
dwSSRC);
|
|||
|
if (pSSRC == NULL)
|
|||
|
{
|
|||
|
// new SSRC, create an entry in this RTCP session
|
|||
|
pSSRC = createSSRCEntry(dwSSRC,
|
|||
|
pRTCPses,
|
|||
|
(PSOCKADDR)pRcvStruct->addr,
|
|||
|
(DWORD)pRcvStruct->addrLen,
|
|||
|
FALSE);
|
|||
|
|
|||
|
if (pSSRC == NULL)
|
|||
|
{
|
|||
|
// cannot create a new entry, out of resources
|
|||
|
RRCM_DBG_MSG ("RTCP: ERROR - Resource allocation", 0,
|
|||
|
__FILE__, __LINE__, DBG_ERROR);
|
|||
|
|
|||
|
IN_OUT_STR ("RTCP: Exit parseRTCPsr\n");
|
|||
|
|
|||
|
return (RRCMError_RTCPResources);
|
|||
|
}
|
|||
|
|
|||
|
// notify application if it desired so
|
|||
|
RRCMnotification (RRCM_NEW_SOURCE_EVENT, pSSRC, dwSSRC,
|
|||
|
UNKNOWN_PAYLOAD_TYPE);
|
|||
|
}
|
|||
|
|
|||
|
// get the RTP timestamp
|
|||
|
RRCMws.ntohl (sd, pRTCPpckt->r.sr.rtp_ts, &pSSRC->xmtInfo.dwRTPts);
|
|||
|
|
|||
|
// number of packets send
|
|||
|
RRCMws.ntohl (sd, pRTCPpckt->r.sr.psent, &pSSRC->xmtInfo.dwNumPcktSent);
|
|||
|
|
|||
|
// number of bytes sent
|
|||
|
RRCMws.ntohl (sd, pRTCPpckt->r.sr.osent, &pSSRC->xmtInfo.dwNumBytesSent);
|
|||
|
|
|||
|
// get the NTP most significant word
|
|||
|
RRCMws.ntohl (sd, pRTCPpckt->r.sr.ntp_sec, &pSSRC->xmtInfo.dwNTPmsw);
|
|||
|
|
|||
|
// get the NTP least significant word
|
|||
|
RRCMws.ntohl (sd, pRTCPpckt->r.sr.ntp_frac, &pSSRC->xmtInfo.dwNTPlsw);
|
|||
|
|
|||
|
// last SR timestamp (middle 32 bits of the NTP timestamp)
|
|||
|
pSSRC->xmtInfo.dwLastSR = ((pSSRC->xmtInfo.dwNTPmsw & 0x0000FFFF) << 16);
|
|||
|
pSSRC->xmtInfo.dwLastSR |= ((pSSRC->xmtInfo.dwNTPlsw & 0xFFFF0000) >> 16);
|
|||
|
|
|||
|
// last time this SSRC's heard
|
|||
|
pSSRC->dwLastReportRcvdTime = pSSRC->xmtInfo.dwLastSRLocalTime =
|
|||
|
timeGetTime();
|
|||
|
|
|||
|
// get the source address information
|
|||
|
if (!(pSSRC->dwSSRCStatus & NETWK_ADDR_UPDATED))
|
|||
|
{
|
|||
|
saveNetworkAddress(pSSRC,
|
|||
|
(PSOCKADDR)pRcvStruct->addr,
|
|||
|
pRcvStruct->addrLen);
|
|||
|
}
|
|||
|
|
|||
|
// increment the number of report received from this SSRC
|
|||
|
InterlockedIncrement ((long *)&pSSRC->dwNumRptRcvd);
|
|||
|
|
|||
|
#ifdef ENABLE_ISDM2
|
|||
|
// update ISDM
|
|||
|
if (Isdm2.hISDMdll && pRTCPses->hSessKey)
|
|||
|
{
|
|||
|
if (pSSRC->hISDM)
|
|||
|
updateISDMstat (pSSRC, &Isdm2, RECVR, FALSE);
|
|||
|
else
|
|||
|
registerSessionToISDM (pSSRC, pRTCPses, &Isdm2);
|
|||
|
}
|
|||
|
#endif
|
|||
|
|
|||
|
IN_OUT_STR ("RTCP: Exit parseRTCPsr\n");
|
|||
|
|
|||
|
return (RRCM_NoError);
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
|
|||
|
/*----------------------------------------------------------------------------
|
|||
|
* Function : parseRTCPrr
|
|||
|
* Description: Parse an RTCP receiver reports and update the corresponding
|
|||
|
* statistics.
|
|||
|
*
|
|||
|
* Input : sd: RTCP socket descriptor
|
|||
|
* pRR: -> to receiver report buffer
|
|||
|
* pRTCPses: -> to the RTCP session information
|
|||
|
* pRcvStruct: -> to the receive structure information
|
|||
|
* senderSSRC: Sender's SSRC
|
|||
|
*
|
|||
|
* Return: OK: RRCM_NoError
|
|||
|
* !0: Error code (see RRCM.H)
|
|||
|
---------------------------------------------------------------------------*/
|
|||
|
DWORD parseRTCPrr (SOCKET sd,
|
|||
|
RTCP_RR_T *pRR,
|
|||
|
PRTCP_SESSION pRTCPses,
|
|||
|
PRTCP_BFR_LIST pRcvStruct,
|
|||
|
DWORD senderSSRC)
|
|||
|
{
|
|||
|
PSSRC_ENTRY pSSRC;
|
|||
|
DWORD dwSSRC;
|
|||
|
DWORD dwGetFeedback = FALSE;
|
|||
|
|
|||
|
IN_OUT_STR ("RTCP: Enter parseRTCPrr\n");
|
|||
|
|
|||
|
#ifdef _DEBUG
|
|||
|
wsprintf(debug_string,
|
|||
|
"RTCP: Receive RR from sender SSRC:x%lX", senderSSRC);
|
|||
|
RRCM_DBG_MSG (debug_string, 0, NULL, 0, DBG_TRACE);
|
|||
|
#endif
|
|||
|
|
|||
|
// get the receiver report SSRC
|
|||
|
RRCMws.ntohl (sd, pRR->ssrc, &dwSSRC);
|
|||
|
|
|||
|
#ifdef _DEBUG
|
|||
|
wsprintf(debug_string, "RTCP: RR for SSRC:x%lX", dwSSRC);
|
|||
|
RRCM_DBG_MSG (debug_string, 0, NULL, 0, DBG_TRACE);
|
|||
|
#endif
|
|||
|
|
|||
|
//
|
|||
|
// NOTE:
|
|||
|
// For now we just keep track of feedback information about ourselve. Later
|
|||
|
// the link list can be used to keep track about everybody feedback
|
|||
|
// information.
|
|||
|
//
|
|||
|
// Check to see if we're interested in this report, ie, does this SSRC report
|
|||
|
// information about one of our active sender.
|
|||
|
dwGetFeedback =
|
|||
|
searchforSSRCatTail((PSSRC_ENTRY)pRTCPses->XmtSSRCList.prev,
|
|||
|
dwSSRC) != NULL;
|
|||
|
|
|||
|
// look for the sender SSRC entry in the list for this RTCP session
|
|||
|
pSSRC =
|
|||
|
searchforSSRCatTail((PSSRC_ENTRY)pRTCPses->RcvSSRCList.prev,
|
|||
|
senderSSRC);
|
|||
|
if (pSSRC == NULL)
|
|||
|
{
|
|||
|
// new SSRC, create an entry in this RTCP session
|
|||
|
pSSRC = createSSRCEntry(senderSSRC,
|
|||
|
pRTCPses,
|
|||
|
(PSOCKADDR)pRcvStruct->addr,
|
|||
|
(DWORD)pRcvStruct->addrLen,
|
|||
|
FALSE);
|
|||
|
|
|||
|
if (pSSRC == NULL)
|
|||
|
{
|
|||
|
// cannot create a new entry, out of resources
|
|||
|
RRCM_DBG_MSG ("RTCP: ERROR - Resource allocation", 0,
|
|||
|
__FILE__, __LINE__, DBG_ERROR);
|
|||
|
|
|||
|
IN_OUT_STR ("RTCP: Exit parseRTCPrr\n");
|
|||
|
return (RRCMError_RTCPResources);
|
|||
|
}
|
|||
|
|
|||
|
// notify application if it desired so
|
|||
|
RRCMnotification (RRCM_NEW_SOURCE_EVENT, pSSRC, senderSSRC,
|
|||
|
UNKNOWN_PAYLOAD_TYPE);
|
|||
|
}
|
|||
|
|
|||
|
// update RR feedback information
|
|||
|
if (dwGetFeedback)
|
|||
|
updateRRfeedback (sd, senderSSRC, dwSSRC, pRR, pSSRC);
|
|||
|
|
|||
|
// last time this SSRC's heard
|
|||
|
pSSRC->dwLastReportRcvdTime = timeGetTime();
|
|||
|
|
|||
|
// get the source address information
|
|||
|
if (!(pSSRC->dwSSRCStatus & NETWK_ADDR_UPDATED))
|
|||
|
{
|
|||
|
saveNetworkAddress(pSSRC,
|
|||
|
(PSOCKADDR)pRcvStruct->addr,
|
|||
|
pRcvStruct->addrLen);
|
|||
|
}
|
|||
|
|
|||
|
// increment the number of report received from this SSRC
|
|||
|
InterlockedIncrement ((long *)&pSSRC->dwNumRptRcvd);
|
|||
|
|
|||
|
#ifdef ENABLE_ISDM2
|
|||
|
// update ISDM
|
|||
|
if (Isdm2.hISDMdll && pRTCPses->hSessKey)
|
|||
|
{
|
|||
|
if (pSSRC->hISDM)
|
|||
|
updateISDMstat (pSSRC, &Isdm2, RECVR, TRUE);
|
|||
|
else
|
|||
|
registerSessionToISDM (pSSRC, pRTCPses, &Isdm2);
|
|||
|
}
|
|||
|
#endif
|
|||
|
|
|||
|
IN_OUT_STR ("RTCP: Exit parseRTCPrr\n");
|
|||
|
|
|||
|
return (RRCM_NoError);
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
|
|||
|
/*----------------------------------------------------------------------------
|
|||
|
* Function : parseRTCPsdes
|
|||
|
* Description: Parse an RTCP SDES packet
|
|||
|
*
|
|||
|
* Input : sd: RTCP socket descriptor
|
|||
|
* bfr: -> to SDES buffer
|
|||
|
* pRTCPses: -> to the RTCP session information
|
|||
|
* pRcvStruct: -> to the receive structure information
|
|||
|
*
|
|||
|
* Return: OK: RRCM_NoError
|
|||
|
* !0: Error code (see RRCM.H)
|
|||
|
---------------------------------------------------------------------------*/
|
|||
|
PCHAR parseRTCPsdes (SOCKET sd,
|
|||
|
PCHAR bfr,
|
|||
|
PRTCP_SESSION pRTCPses,
|
|||
|
PRTCP_BFR_LIST pRcvStruct)
|
|||
|
{
|
|||
|
DWORD dwHost;
|
|||
|
DWORD ssrc = *(DWORD *)bfr;
|
|||
|
RTCP_SDES_ITEM_T *pSdes;
|
|||
|
PSSRC_ENTRY pSSRC;
|
|||
|
|
|||
|
IN_OUT_STR ("RTCP: Enter parseRTCPsdes\n");
|
|||
|
|
|||
|
// get the SSRC
|
|||
|
RRCMws.ntohl (sd, ssrc, &dwHost);
|
|||
|
|
|||
|
#ifdef _DEBUG
|
|||
|
wsprintf(debug_string, "RTCP: Receive SDES from SSRC: x%lX", dwHost);
|
|||
|
RRCM_DBG_MSG (debug_string, 0, NULL, 0, DBG_TRACE);
|
|||
|
#endif
|
|||
|
|
|||
|
// look for the SSRC entry in the list for this RTCP session
|
|||
|
pSSRC = searchforSSRCatTail ((PSSRC_ENTRY)pRTCPses->RcvSSRCList.prev,
|
|||
|
dwHost);
|
|||
|
if (pSSRC == NULL)
|
|||
|
{
|
|||
|
#ifdef _DEBUG
|
|||
|
wsprintf(debug_string,
|
|||
|
"RTCP: SDES and SSRC (x%lX) not found for session (Addr x%p)",
|
|||
|
dwHost, pRTCPses);
|
|||
|
RRCM_DBG_MSG (debug_string, 0, NULL, 0, DBG_TRACE);
|
|||
|
#endif
|
|||
|
|
|||
|
// new SSRC, create an entry in this RTCP session
|
|||
|
pSSRC = createSSRCEntry(dwHost,
|
|||
|
pRTCPses,
|
|||
|
(PSOCKADDR)pRcvStruct->addr,
|
|||
|
(DWORD)pRcvStruct->addrLen,
|
|||
|
FALSE);
|
|||
|
|
|||
|
if (pSSRC == NULL)
|
|||
|
{
|
|||
|
// cannot create a new entry, out of resources
|
|||
|
RRCM_DBG_MSG ("RTCP: ERROR - Resource allocation", 0,
|
|||
|
__FILE__, __LINE__, DBG_ERROR);
|
|||
|
|
|||
|
IN_OUT_STR ("RTCP: Exit parseRTCPsdes\n");
|
|||
|
|
|||
|
return (NULL);
|
|||
|
}
|
|||
|
|
|||
|
// notify application if it desired so
|
|||
|
RRCMnotification (RRCM_NEW_SOURCE_EVENT, pSSRC, dwHost,
|
|||
|
UNKNOWN_PAYLOAD_TYPE);
|
|||
|
}
|
|||
|
|
|||
|
// read the SDES chunk
|
|||
|
pSdes = (RTCP_SDES_ITEM_T *)(bfr + sizeof(DWORD));
|
|||
|
|
|||
|
// go through until a 'type = 0' is found
|
|||
|
for (; pSdes->dwSdesType;
|
|||
|
pSdes = (RTCP_SDES_ITEM_T *)((char *)pSdes + pSdes->dwSdesLength + 2))
|
|||
|
{
|
|||
|
switch (pSdes->dwSdesType)
|
|||
|
{
|
|||
|
case RTCP_SDES_CNAME:
|
|||
|
if (pSSRC->cnameInfo.dwSdesLength == 0)
|
|||
|
{
|
|||
|
pSSRC->cnameInfo.dwSdesLength = pSdes->dwSdesLength;
|
|||
|
|
|||
|
// get the Cname
|
|||
|
memcpy (pSSRC->cnameInfo.sdesBfr, pSdes->sdesData,
|
|||
|
min (pSdes->dwSdesLength, MAX_SDES_LEN-1));
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
// check to see for a loop/collision of the SSRC
|
|||
|
if (memcmp (pSdes->sdesData, pSSRC->cnameInfo.sdesBfr,
|
|||
|
min (pSdes->dwSdesLength, MAX_SDES_LEN-1)) != 0)
|
|||
|
{
|
|||
|
// loop/collision of a third-party detected
|
|||
|
pSSRC->dwSSRCStatus |= THIRD_PARTY_COLLISION;
|
|||
|
|
|||
|
// notify application if interested
|
|||
|
RRCMnotification (RRCM_REMOTE_COLLISION_EVENT, pSSRC,
|
|||
|
pSSRC->SSRC, 0);
|
|||
|
|
|||
|
// RTP & RTCP packet from this SSRC will be rejected
|
|||
|
// until the senders resolve the collision
|
|||
|
|
|||
|
IN_OUT_STR ("RTCP: Exit parseRTCPsdes\n");
|
|||
|
|
|||
|
return NULL;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
break;
|
|||
|
|
|||
|
case RTCP_SDES_NAME:
|
|||
|
// the name can change, not like the Cname, so update it
|
|||
|
// every time.
|
|||
|
pSSRC->nameInfo.dwSdesLength = pSdes->dwSdesLength;
|
|||
|
|
|||
|
// get the name
|
|||
|
memcpy (pSSRC->nameInfo.sdesBfr, pSdes->sdesData,
|
|||
|
min (pSdes->dwSdesLength, MAX_SDES_LEN-1));
|
|||
|
break;
|
|||
|
|
|||
|
default:
|
|||
|
break;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
// last time this SSRC's heard
|
|||
|
pSSRC->dwLastReportRcvdTime = timeGetTime();
|
|||
|
|
|||
|
// get the source address information
|
|||
|
if (!(pSSRC->dwSSRCStatus & NETWK_ADDR_UPDATED))
|
|||
|
{
|
|||
|
saveNetworkAddress(pSSRC,
|
|||
|
(PSOCKADDR)pRcvStruct->addr,
|
|||
|
pRcvStruct->addrLen);
|
|||
|
}
|
|||
|
|
|||
|
// adjust pointer
|
|||
|
bfr = (char *)pSdes;
|
|||
|
|
|||
|
IN_OUT_STR ("RTCP: Exit parseRTCPsdes\n");
|
|||
|
|
|||
|
// go the next 32 bits boundary
|
|||
|
return bfr + ((4 - ((LONG_PTR)bfr & 0x3)) & 0x3);
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
|
|||
|
|
|||
|
/*----------------------------------------------------------------------------
|
|||
|
* Function : parseRTCPbye
|
|||
|
* Description: Parse an RTCP BYE packet
|
|||
|
*
|
|||
|
* Input : sd: RTCP socket descriptor
|
|||
|
* ssrc: SSRC
|
|||
|
* pRTCPses: -> to the RTCP session information
|
|||
|
* pRcvStruct: -> to the receive structure
|
|||
|
*
|
|||
|
* Return: OK: RRCM_NoError
|
|||
|
* !0: Error code (see RRCM.H)
|
|||
|
---------------------------------------------------------------------------*/
|
|||
|
DWORD parseRTCPbye (SOCKET sd,
|
|||
|
DWORD ssrc,
|
|||
|
PRTCP_SESSION pRTCPses,
|
|||
|
PRTCP_BFR_LIST pRcvStruct)
|
|||
|
{
|
|||
|
DWORD dwStatus;
|
|||
|
DWORD dwHost;
|
|||
|
PSSRC_ENTRY pSSRC;
|
|||
|
|
|||
|
IN_OUT_STR ("RTCP: Enter parseRTCPbye\n");
|
|||
|
|
|||
|
RRCMws.ntohl (sd, ssrc, &dwHost);
|
|||
|
|
|||
|
#ifdef _DEBUG
|
|||
|
wsprintf(debug_string, "RTCP: BYE from SSRC: x%lX", dwHost);
|
|||
|
RRCM_DBG_MSG (debug_string, 0, NULL, 0, DBG_TRACE);
|
|||
|
#endif
|
|||
|
|
|||
|
// find the SSRC entry
|
|||
|
pSSRC = searchforSSRCatTail((PSSRC_ENTRY)pRTCPses->RcvSSRCList.prev,
|
|||
|
dwHost);
|
|||
|
if (pSSRC == NULL)
|
|||
|
{
|
|||
|
#ifdef _DEBUG
|
|||
|
wsprintf(debug_string,
|
|||
|
"RTCP: SSRC: x%lX not found in session: x%p",
|
|||
|
dwHost, pRTCPses);
|
|||
|
RRCM_DBG_MSG (debug_string, 0, __FILE__, __LINE__, DBG_TRACE);
|
|||
|
|
|||
|
IN_OUT_STR ("RTCP: Exit parseRTCPbye\n");
|
|||
|
#endif
|
|||
|
return (RRCM_NoError);
|
|||
|
}
|
|||
|
|
|||
|
// make sure the BYE is coming from the expected source and not intruder
|
|||
|
if ((pRcvStruct->addrLen != pSSRC->fromLen) ||
|
|||
|
#if 0
|
|||
|
// There is a bug NT's Winsock2 implememtation. The unused bytes of
|
|||
|
// SOCKADDR are not reset to 0 as they should be. Work fine on W95
|
|||
|
// Temporarily just check the first 8 bytes, i.e. address family, port
|
|||
|
// and IP address.
|
|||
|
(memcmp (&pRcvStruct->addr, &pSSRC->from, pSSRC->fromLen)))
|
|||
|
#else
|
|||
|
(memcmp (&pRcvStruct->addr, &pSSRC->from, 8)))
|
|||
|
#endif
|
|||
|
return (RRCM_NoError);
|
|||
|
|
|||
|
// notify application if interested
|
|||
|
RRCMnotification (RRCM_BYE_EVENT, pSSRC, dwHost, 0);
|
|||
|
|
|||
|
// delete this SSRC from the list
|
|||
|
dwStatus = deleteSSRCEntry (dwHost, pRTCPses);
|
|||
|
#ifdef _DEBUG
|
|||
|
if (dwStatus == FALSE)
|
|||
|
{
|
|||
|
wsprintf(debug_string,
|
|||
|
"RTCP: SSRC: x%lX not found in session: x%p",
|
|||
|
dwHost, pRTCPses);
|
|||
|
RRCM_DBG_MSG (debug_string, 0, __FILE__, __LINE__, DBG_TRACE);
|
|||
|
}
|
|||
|
#endif
|
|||
|
|
|||
|
IN_OUT_STR ("RTCP: Exit parseRTCPbye\n");
|
|||
|
return (RRCM_NoError);
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
|
|||
|
/*----------------------------------------------------------------------------
|
|||
|
* Function : ownLoopback
|
|||
|
* Description: Determine if we receive our own loopback. We don't want to
|
|||
|
* create an entry for ourselve, as we're already in the list.
|
|||
|
*
|
|||
|
* Input : sd: RTCP socket descriptor
|
|||
|
* ssrc: SSRC
|
|||
|
* pRTCPses: -> to the RTCP session's information
|
|||
|
*
|
|||
|
* Return: TRUE: Our loopback
|
|||
|
* FALSE: No loopback
|
|||
|
---------------------------------------------------------------------------*/
|
|||
|
DWORD ownLoopback (SOCKET sd,
|
|||
|
DWORD ssrc,
|
|||
|
PRTCP_SESSION pRTCPses)
|
|||
|
{
|
|||
|
PSSRC_ENTRY pSSRC;
|
|||
|
|
|||
|
IN_OUT_STR ("RTCP: Enter ownLoopback\n");
|
|||
|
|
|||
|
// don't create an entry if received our own xmit back
|
|||
|
pSSRC = searchforSSRCatTail((PSSRC_ENTRY)pRTCPses->XmtSSRCList.prev,
|
|||
|
ssrc);
|
|||
|
|
|||
|
IN_OUT_STR ("RTCP: Exit ownLoopback\n");
|
|||
|
|
|||
|
if (pSSRC)
|
|||
|
return TRUE;
|
|||
|
else
|
|||
|
return FALSE;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
|
|||
|
/*----------------------------------------------------------------------------
|
|||
|
* Function : updateRRfeedback
|
|||
|
* Description: Update the Receiver Report feedback for an active source
|
|||
|
*
|
|||
|
* Input : sd: RTCP socket descriptor
|
|||
|
* dwSndSSRC: Sender's SSRC
|
|||
|
* pRR: -> to receiver report entry
|
|||
|
* pSSRC: -> to the SSRC entry
|
|||
|
*
|
|||
|
* Return: TRUE
|
|||
|
---------------------------------------------------------------------------*/
|
|||
|
DWORD updateRRfeedback (SOCKET sd,
|
|||
|
DWORD dwSndSSRC,
|
|||
|
DWORD dwSSRCfedback,
|
|||
|
RTCP_RR_T *pRR,
|
|||
|
PSSRC_ENTRY pSSRC)
|
|||
|
{
|
|||
|
DWORD dwHost;
|
|||
|
|
|||
|
IN_OUT_STR ("RTCP: Enter updateRRfeedback\n");
|
|||
|
|
|||
|
// Note when we last heard from the receiver
|
|||
|
pSSRC->rrFeedback.dwLastRcvRpt = timeGetTime();
|
|||
|
|
|||
|
// SSRC who's feedback is for (ourselve for now)
|
|||
|
pSSRC->rrFeedback.SSRC = dwSSRCfedback;
|
|||
|
|
|||
|
// get delay since last SR
|
|||
|
RRCMws.ntohl (sd, pRR->dlsr, &pSSRC->rrFeedback.dwDelaySinceLastSR);
|
|||
|
|
|||
|
// get last SR
|
|||
|
RRCMws.ntohl (sd, pRR->lsr, &pSSRC->rrFeedback.dwLastSR);
|
|||
|
|
|||
|
// get the jitter
|
|||
|
RRCMws.ntohl (sd, pRR->jitter, &pSSRC->rrFeedback.dwInterJitter);
|
|||
|
|
|||
|
// highest sequence number received
|
|||
|
RRCMws.ntohl (sd, pRR->expected,
|
|||
|
&pSSRC->rrFeedback.XtendedSeqNum.seq_union.dwXtndedHighSeqNumRcvd);
|
|||
|
|
|||
|
// fraction lost
|
|||
|
pSSRC->rrFeedback.fractionLost = (pRR->received & 0xFF);
|
|||
|
|
|||
|
// cumulative number of packet lost
|
|||
|
RRCMws.ntohl (sd, pRR->received, &dwHost);
|
|||
|
dwHost &= 0x00FFFFFF;
|
|||
|
pSSRC->rrFeedback.cumNumPcktLost = dwHost;
|
|||
|
|
|||
|
IN_OUT_STR ("RTCP: Exit updateRRfeedback\n");
|
|||
|
|
|||
|
return TRUE;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
|
|||
|
/*----------------------------------------------------------------------------
|
|||
|
* Function : RTCPpostRecvBfr
|
|||
|
* Description: RTCP post a receive buffer to Winsock-2
|
|||
|
*
|
|||
|
* Input : sd: RTCP socket descriptor
|
|||
|
* pSSRC: -> to the SSRC entry
|
|||
|
*
|
|||
|
* Return: TRUE
|
|||
|
---------------------------------------------------------------------------*/
|
|||
|
void RTCPpostRecvBfr (PSSRC_ENTRY pSSRC,
|
|||
|
PRTCP_BFR_LIST pRcvStruct)
|
|||
|
{
|
|||
|
DWORD dwStatus;
|
|||
|
DWORD dwError;
|
|||
|
|
|||
|
IN_OUT_STR ("RTCP: Enter RTCPpostRecvBfr\n");
|
|||
|
|
|||
|
// decrement number of I/O pending
|
|||
|
InterlockedDecrement ((long *)&pSSRC->pRTCPses->dwNumRcvIoPending);
|
|||
|
|
|||
|
// don't repost any buffer if within the shutdown procedure
|
|||
|
if ((pSSRC->pRTCPses->dwSessionStatus & SHUTDOWN_IN_PROGRESS) &&
|
|||
|
(pSSRC->pRTCPses->dwNumRcvIoPending == 0))
|
|||
|
{
|
|||
|
// shutdown done - set event
|
|||
|
if (SetEvent (pSSRC->pRTCPses->hShutdownDone) == FALSE)
|
|||
|
{
|
|||
|
RRCM_DBG_MSG ("RTCP: SetEvent() Error\n", GetLastError(),
|
|||
|
__FILE__, __LINE__, DBG_ERROR);
|
|||
|
}
|
|||
|
|
|||
|
IN_OUT_STR ("RTCP: Exit RTCPpostRecvBfr\n");
|
|||
|
return;
|
|||
|
}
|
|||
|
else if (pSSRC->pRTCPses->dwSessionStatus & SHUTDOWN_IN_PROGRESS)
|
|||
|
{
|
|||
|
IN_OUT_STR ("RTCP: Exit RTCPpostRecvBfr\n");
|
|||
|
return;
|
|||
|
}
|
|||
|
|
|||
|
// clear number of bytes transferred
|
|||
|
pRcvStruct->dwNumBytesXfr = 0;
|
|||
|
|
|||
|
dwStatus = RRCMws.recvFrom (pSSRC->RTCPsd,
|
|||
|
&pRcvStruct->bfr,
|
|||
|
pRcvStruct->dwBufferCount,
|
|||
|
&pRcvStruct->dwNumBytesXfr,
|
|||
|
&pRcvStruct->dwFlags,
|
|||
|
(PSOCKADDR)pRcvStruct->addr,
|
|||
|
&pRcvStruct->addrLen,
|
|||
|
(LPWSAOVERLAPPED)&pRcvStruct->overlapped,
|
|||
|
RTCPrcvCallback);
|
|||
|
|
|||
|
// Check Winsock status
|
|||
|
if (dwStatus != 0)
|
|||
|
{
|
|||
|
// error, the receive request won't proceed
|
|||
|
dwError = GetLastError();
|
|||
|
if ((dwError != WSA_IO_PENDING) && (dwError != WSAEMSGSIZE))
|
|||
|
{
|
|||
|
RRCM_DBG_MSG ("RTCP: ERROR - WSARecvFrom()", dwError,
|
|||
|
__FILE__, __LINE__, DBG_ERROR);
|
|||
|
|
|||
|
// notify application if interested
|
|||
|
RRCMnotification (RRCM_RTCP_WS_RCV_ERROR, pSSRC,
|
|||
|
pSSRC->SSRC, dwError);
|
|||
|
|
|||
|
// Return the buffer to the free queue
|
|||
|
addToHeadOfList (&pSSRC->pRTCPses->RTCPrcvBfrList,
|
|||
|
(PLINK_LIST)pRcvStruct,
|
|||
|
&pSSRC->pRTCPses->critSect);
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
// increment number of I/O pending
|
|||
|
InterlockedIncrement ((long *)&pSSRC->pRTCPses->dwNumRcvIoPending);
|
|||
|
}
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
// synchronous completion - callback has been scheduled
|
|||
|
// increment number of I/O pending
|
|||
|
InterlockedIncrement ((long *)&pSSRC->pRTCPses->dwNumRcvIoPending);
|
|||
|
}
|
|||
|
|
|||
|
IN_OUT_STR ("RTCP: Exit RTCPpostRecvBfr\n");
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
/*----------------------------------------------------------------------------
|
|||
|
* Function : addApplicationRtcpBfr
|
|||
|
* Description: Add an application provided buffer for RTCP to copy the
|
|||
|
* raw received reports to be used by the application if it
|
|||
|
* desired so.
|
|||
|
*
|
|||
|
* Input : RTPsession: Handle to the RTP session
|
|||
|
* pAppBfr: -> an application buffer data structure
|
|||
|
*
|
|||
|
* Return: TRUE
|
|||
|
---------------------------------------------------------------------------*/
|
|||
|
HRESULT WINAPI addApplicationRtcpBfr (DWORD_PTR RTPsession,
|
|||
|
PAPP_RTCP_BFR pAppBfr)
|
|||
|
{
|
|||
|
IN_OUT_STR ("RTCP : Enter addApplicationRtcpBfr()\n");
|
|||
|
|
|||
|
PRTP_SESSION pSession = (PRTP_SESSION)RTPsession;
|
|||
|
PRTCP_SESSION pRTCPSess;
|
|||
|
|
|||
|
if (pSession == NULL)
|
|||
|
{
|
|||
|
RRCM_DBG_MSG ("RTCP : ERROR - Invalid RTP session", 0,
|
|||
|
__FILE__, __LINE__, DBG_ERROR);
|
|||
|
|
|||
|
IN_OUT_STR ("RTCP : Exit addApplicationRtcpBfr()\n");
|
|||
|
|
|||
|
return (MAKE_RRCM_ERROR(RRCMError_RTPSessResources));
|
|||
|
}
|
|||
|
|
|||
|
pRTCPSess = (PRTCP_SESSION)pSession->pRTCPSession;
|
|||
|
if (pRTCPSess == NULL)
|
|||
|
{
|
|||
|
RRCM_DBG_MSG ("RTCP : ERROR - Invalid RTCP session", 0,
|
|||
|
__FILE__, __LINE__, DBG_ERROR);
|
|||
|
|
|||
|
IN_OUT_STR ("RTCP : Exit addApplicationRtcpBfr()\n");
|
|||
|
|
|||
|
return (MAKE_RRCM_ERROR(RRCMError_RTCPInvalidSession));
|
|||
|
}
|
|||
|
|
|||
|
// Let's add this buffer to our list
|
|||
|
addToTailOfList(&(pRTCPSess->appRtcpBfrList),
|
|||
|
(PLINK_LIST)pAppBfr,
|
|||
|
&pRTCPSess->critSect);
|
|||
|
|
|||
|
IN_OUT_STR ("RTCP : Exit addApplicationRtcpBfr()\n");
|
|||
|
|
|||
|
return NOERROR;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
/*----------------------------------------------------------------------------
|
|||
|
* Function : removeApplicationRtcpBfr
|
|||
|
* Description: Remove an application provided buffer to this RTCP session.
|
|||
|
*
|
|||
|
* Input : RTPsession: RTP session handle
|
|||
|
*
|
|||
|
* Return: Application buffer address / NULL
|
|||
|
---------------------------------------------------------------------------*/
|
|||
|
PAPP_RTCP_BFR WINAPI removeApplicationRtcpBfr (DWORD_PTR RTPsession)
|
|||
|
{
|
|||
|
PRTP_SESSION pSession = (PRTP_SESSION)RTPsession;
|
|||
|
PRTCP_SESSION pRTCPSess;
|
|||
|
PAPP_RTCP_BFR pAppBfr;
|
|||
|
|
|||
|
IN_OUT_STR ("RTCP : Enter removeApplicationRtcpBfr()\n");
|
|||
|
|
|||
|
if (pSession == NULL)
|
|||
|
{
|
|||
|
RRCM_DBG_MSG ("RTCP : ERROR - Invalid RTP session", 0,
|
|||
|
__FILE__, __LINE__, DBG_ERROR);
|
|||
|
|
|||
|
IN_OUT_STR ("RTCP : Exit removeApplicationRtcpBfr()\n");
|
|||
|
|
|||
|
return NULL;
|
|||
|
}
|
|||
|
|
|||
|
pRTCPSess = (PRTCP_SESSION)pSession->pRTCPSession;
|
|||
|
if (pRTCPSess == NULL)
|
|||
|
{
|
|||
|
RRCM_DBG_MSG ("RTCP : ERROR - Invalid RTCP session", 0,
|
|||
|
__FILE__, __LINE__, DBG_ERROR);
|
|||
|
|
|||
|
IN_OUT_STR ("RTCP : Exit removeApplicationRtcpBfr()\n");
|
|||
|
|
|||
|
return NULL;
|
|||
|
}
|
|||
|
|
|||
|
pAppBfr = (PAPP_RTCP_BFR)removePcktFromHead (&(pRTCPSess->appRtcpBfrList),
|
|||
|
&pRTCPSess->critSect);
|
|||
|
|
|||
|
IN_OUT_STR ("RTCP : Exit removeApplicationRtcpBfr()\n");
|
|||
|
|
|||
|
return pAppBfr;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
// [EOF]
|
|||
|
|
|||
|
|
|||
|
|
|||
|
|
|||
|
|