/*---------------------------------------------------------------------------- * File: RTPIO.C * Product: RTP/RTCP implementation * Description: Provides Session Creation/Deletion Functionality. * * * This listing is supplied under the terms * of a license agreement with Intel Corporation and * many not be copied nor disclosed except in accordance * with the terms of that agreement. * Copyright (c) 1995 Intel Corporation. *--------------------------------------------------------------------------*/ #include "rrcm.h" /*--------------------------------------------------------------------------- / Global Variables /--------------------------------------------------------------------------*/ /*--------------------------------------------------------------------------- / External Variables /--------------------------------------------------------------------------*/ extern PRTP_CONTEXT pRTPContext; #ifdef _DEBUG extern char debug_string[]; #endif /*---------------------------------------------------------------------------- * Function : CreateRTPSession * Description: Creates an RTP (and RTCP) session for a new stream. * * Input : RTPsocket : RTP socket descriptor * RTCPsd : RTCP socket descriptor * pRTCPTo : RTCP destination address * toRTCPLen : RTCP destination address length * pSdesInfo : -> to SDES information * dwStreamClock : Stream clocking frequency * ssrc : If set, user selected SSRC * pRRCMcallback : RRCM notification * dwCallbackInfo : User callback info * miscInfo : Miscelleanous information: * H.323Conf: 0x00000002 * Encrypt SR/RR: 0x00000004 * RTCPon: 0x00000008 * dwRtpSessionBw : RTP session bandwidth used for RTCP BW * *pStatus : -> to status information * * Return: handle to created session if successful * Otherwise(0) *pStatus = Initialization Error (see RRCM.H) ---------------------------------------------------------------------------*/ HANDLE WINAPI CreateRTPSession (SOCKET RTPsocket, SOCKET RTCPsocket, LPVOID pRTCPTo, DWORD toRTCPLen, PSDES_DATA pSdesInfo, DWORD dwStreamClock, PENCRYPT_INFO pEncryptInfo, DWORD ssrc, PRRCM_EVENT_CALLBACK pRRCMcallback, DWORD_PTR dwCallbackInfo, DWORD miscInfo, DWORD dwRtpSessionBw, DWORD *pStatus) { DWORD numCells = NUM_FREE_CONTEXT_CELLS; DWORD dwStatus; DWORD dwRTCPstatus; PRTP_SESSION pSession = NULL; PSSRC_ENTRY pSSRC; IN_OUT_STR ("RTP : Enter CreateRTPSession()\n"); // set status code *pStatus = dwStatus = RRCM_NoError; // If RTP context doesn't exist, report error and return. if (pRTPContext == NULL) { RRCM_DBG_MSG ("RTP : ERROR - No RTP Instance", 0, __FILE__, __LINE__, DBG_CRITICAL); IN_OUT_STR ("RTP : Exit CreateRTPSession()\n"); *pStatus = MAKE_RRCM_ERROR(RRCMError_RTPNoContext); return 0; } #if 0 // look for an existing session - Sender/Receiver for the same session // will be on two different graph under ActiveMovie if (pSession = findSessionID (RTPsocket)) { RRCM_DBG_MSG ("RTP : Session already created", 0, __FILE__, __LINE__, DBG_TRACE); IN_OUT_STR ("RTP : Exit CreateRTPSession()\n"); // return the unique RTP session ID return ((HANDLE)pSession); } #endif // Allocate a new session pointer. pSession = (PRTP_SESSION)GlobalAlloc (GMEM_FIXED | GMEM_ZEROINIT, sizeof(RTP_SESSION)); // Report error if could not allocate context if (pSession == NULL) { RRCM_DBG_MSG ("RTP : ERROR - Resource allocation failed", 0, __FILE__, __LINE__, DBG_CRITICAL); IN_OUT_STR ("RTP : Exit CreateRTPSession()\n"); *pStatus = MAKE_RRCM_ERROR(RRCMError_RTPSessResources); return 0; } // Initialize the RTP session's critical section InitializeCriticalSection(&pSession->critSect); // All seems OK, initialize RTCP for this session pSession->pRTCPSession = CreateRTCPSession(RTPsocket, RTCPsocket, pRTCPTo, toRTCPLen, pSdesInfo, dwStreamClock, pEncryptInfo, ssrc, pRRCMcallback, dwCallbackInfo, miscInfo, dwRtpSessionBw, &dwRTCPstatus); if (pSession->pRTCPSession == NULL) { dwStatus = dwRTCPstatus; // Can't proceed, return session pointer if (pSession) { GlobalFree (pSession); pSession = NULL; } } else { #if 0 // Associate the socket with the Session address dwStatus = createHashEntry (pSession, RTPsocket); #endif if (dwStatus == RRCM_NoError) { pSSRC = (PSSRC_ENTRY)pSession->pRTCPSession->XmtSSRCList.prev; if (pSSRC == NULL) { RRCM_DBG_MSG ("RTP : ERROR - No RTCP Xmt list", 0, __FILE__, __LINE__, DBG_CRITICAL); dwStatus = RRCMError_RTCPNoXmtList; } else { // Let's add this to our context addToHeadOfList(&(pRTPContext->pRTPSession), (PLINK_LIST)pSession, &pRTPContext->critSect); #ifdef _DEBUG wsprintf(debug_string, "RTP : Adding RTP Session. (Addr:0x%p)", pSession); RRCM_DBG_MSG (debug_string, 0, NULL, 0, DBG_TRACE); #endif } } } // set status code if (dwStatus != RRCM_NoError) *pStatus = MAKE_RRCM_ERROR(dwStatus); IN_OUT_STR ("RTP : Exit CreateRTPSession()\n"); // return the unique RTP session ID return ((HANDLE)pSession); } /*---------------------------------------------------------------------------- * Function : CloseRTPSession * Description: Terminates a local stream session. * * Input : RTPSession = RTP session ID * byeReason = -> to BYE reason * closeRTCPSocket = TRUE/FALSE. RTCP will close or not the socket * * Return: RRCM_NoError = OK. * Otherwise(!=0) = Error (see RRCM.H) ---------------------------------------------------------------------------*/ HRESULT WINAPI CloseRTPSession (HANDLE RTPSession, PCHAR byeReason, DWORD closeRTCPSocket) { PRTP_SESSION pSession; PSSRC_ENTRY pSSRCList; PSSRC_ENTRY pSSRC; DWORD dwStatus; IN_OUT_STR ("RTP : Enter CloseRTPSession()\n"); // If RTP context doesn't exist, report error and return. if (pRTPContext == NULL) { RRCM_DBG_MSG ("RTP : ERROR - No RTP Instance", 0, __FILE__, __LINE__, DBG_ERROR); IN_OUT_STR ("RTP : Exit CloseRTPSession()\n"); return (MAKE_RRCM_ERROR(RRCMError_RTPNoContext)); } // Cast Session ID to obtain the session pointer. pSession = (PRTP_SESSION)RTPSession; if (pSession == NULL) { RRCM_DBG_MSG ("RTP : ERROR - Invalid RTP session", 0, __FILE__, __LINE__, DBG_ERROR); IN_OUT_STR ("RTP : Exit CloseRTPSession()\n"); return (MAKE_RRCM_ERROR(RRCMError_RTPInvalidSession)); } // Remove the session from the linked list of sessions dwStatus = deleteRTPSession (pRTPContext, pSession); if (dwStatus != RRCM_NoError) { #ifdef _DEBUG wsprintf(debug_string, "RTP : ERROR - RTP session (Addr:0x%p) not found", RTPSession); RRCM_DBG_MSG (debug_string, 0, NULL, 0, DBG_ERROR); #endif return (MAKE_RRCM_ERROR(dwStatus)); } // lock out the session - it's on a free list now EnterCriticalSection (&pSession->critSect); #if 0 // clean up the Hash table for any stream still left in the Session for (pSSRCList = (PSSRC_ENTRY)pSession->pRTCPSession->XmtSSRCList.prev; pSSRCList != NULL; pSSRCList = (PSSRC_ENTRY)pSSRCList->SSRCList.next) { deleteHashEntry (pSSRCList->RTPsd); } #endif // All seems OK, close RTCP for each stream still open pSSRC = (PSSRC_ENTRY)pSession->pRTCPSession->XmtSSRCList.prev; if (pSSRC == NULL) { RRCM_DBG_MSG ("RTP : ERROR - No SSRC entry on the Xmt list", 0, __FILE__, __LINE__, DBG_ERROR); IN_OUT_STR ("RTP : Exit CloseRTPSession()\n"); return (MAKE_RRCM_ERROR(RRCMError_RTCPInvalidSSRCentry)); } // reset the close socket flag if (closeRTCPSocket) pSSRC->dwSSRCStatus |= CLOSE_RTCP_SOCKET; dwStatus = deleteRTCPSession (pSSRC->RTCPsd, byeReason); #ifdef _DEBUG if (dwStatus != RRCM_NoError) { wsprintf(debug_string, "RTP : ERROR - RTCP delete Session (Addr: x%p) error:%d", pSession, dwStatus); RRCM_DBG_MSG (debug_string, 0, NULL, 0, DBG_TRACE); } #endif #ifdef _DEBUG wsprintf(debug_string, "RTP : Deleting Session x%p", pSession); RRCM_DBG_MSG (debug_string, 0, NULL, 0, DBG_TRACE); #endif // lock out the session - it's on a free list now LeaveCriticalSection (&pSession->critSect); DeleteCriticalSection (&pSession->critSect); GlobalFree (pSession); pSession = NULL; IN_OUT_STR ("RTP : Exit CloseRTPSession()\n"); if (dwStatus != RRCM_NoError) dwStatus = MAKE_RRCM_ERROR(dwStatus); return (dwStatus); } /*-------------------------------------------------------------------------- ** Function : deleteRTPSession ** Description: Remove from RTP session queue and restore links for other ** sessions. ** ** Input : pRTPContext: -> to the RTP context ** pSession: -> to the RTP session ** ** Return: OK: RRCM_NoError ** !0: Error code (see RRCM.H) --------------------------------------------------------------------------*/ DWORD deleteRTPSession(PRTP_CONTEXT pRTPContext, PRTP_SESSION pSession) { PLINK_LIST pTmp; IN_OUT_STR ("RTP : Enter deleteRTPSession()\n"); // make sure the session exist pTmp = pRTPContext->pRTPSession.prev; while (pTmp) { if (pTmp == (PLINK_LIST)pSession) break; pTmp = pTmp->next; } if (pTmp == NULL) { RRCM_DBG_MSG ("RTP : ERROR - Invalid RTP session", 0, __FILE__, __LINE__, DBG_ERROR); IN_OUT_STR ("RTP : Exit deleteRTPSession()\n"); return (MAKE_RRCM_ERROR(RRCMError_RTPInvalidSession)); } // lock out queue access EnterCriticalSection (&pRTPContext->critSect); EnterCriticalSection (&pSession->critSect); if (pSession->RTPList.prev == NULL) // this was the first entry in the queue pRTPContext->pRTPSession.prev = pSession->RTPList.next; else (pSession->RTPList.prev)->next = pSession->RTPList.next; if (pSession->RTPList.next == NULL) // this was the last entry in the queue pRTPContext->pRTPSession.next = pSession->RTPList.prev; else (pSession->RTPList.next)->prev = pSession->RTPList.prev; // unlock out queue access LeaveCriticalSection (&pSession->critSect); LeaveCriticalSection (&pRTPContext->critSect); IN_OUT_STR ("RTP : Exit deleteRTPSession()\n"); return (RRCM_NoError); } // [EOF]