toxcore/toxav/rtp.c

529 lines
14 KiB
C
Raw Normal View History

/** rtp.c
2014-02-17 09:01:30 +08:00
*
* Copyright (C) 2013 Tox project All Rights Reserved.
*
* This file is part of Tox.
*
* Tox is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* Tox is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with Tox. If not, see <http://www.gnu.org/licenses/>.
*
*/
#ifdef HAVE_CONFIG_H
#include "config.h"
#endif /* HAVE_CONFIG_H */
2014-05-03 07:46:03 +08:00
#include "../toxcore/logger.h"
#include "../toxcore/util.h"
2014-05-03 07:46:03 +08:00
2014-02-10 06:06:44 +08:00
#include "rtp.h"
2014-02-01 19:52:48 +08:00
#include <stdlib.h>
2014-11-18 07:46:46 +08:00
void queue_message(RTPSession *_session, RTPMessage *_msg);
#define size_32 4
#define ADD_FLAG_VERSION(_h, _v) do { ( _h->flags ) &= 0x3F; ( _h->flags ) |= ( ( ( _v ) << 6 ) & 0xC0 ); } while(0)
#define ADD_FLAG_PADDING(_h, _v) do { if ( _v > 0 ) _v = 1; ( _h->flags ) &= 0xDF; ( _h->flags ) |= ( ( ( _v ) << 5 ) & 0x20 ); } while(0)
#define ADD_FLAG_EXTENSION(_h, _v) do { if ( _v > 0 ) _v = 1; ( _h->flags ) &= 0xEF;( _h->flags ) |= ( ( ( _v ) << 4 ) & 0x10 ); } while(0)
#define ADD_FLAG_CSRCC(_h, _v) do { ( _h->flags ) &= 0xF0; ( _h->flags ) |= ( ( _v ) & 0x0F ); } while(0)
#define ADD_SETTING_MARKER(_h, _v) do { if ( _v > 1 ) _v = 1; ( _h->marker_payloadt ) &= 0x7F; ( _h->marker_payloadt ) |= ( ( ( _v ) << 7 ) /*& 0x80 */ ); } while(0)
#define ADD_SETTING_PAYLOAD(_h, _v) do { if ( _v > 127 ) _v = 127; ( _h->marker_payloadt ) &= 0x80; ( _h->marker_payloadt ) |= ( ( _v ) /* & 0x7F */ ); } while(0)
#define GET_FLAG_VERSION(_h) (( _h->flags & 0xd0 ) >> 6)
#define GET_FLAG_PADDING(_h) (( _h->flags & 0x20 ) >> 5)
#define GET_FLAG_EXTENSION(_h) (( _h->flags & 0x10 ) >> 4)
#define GET_FLAG_CSRCC(_h) ( _h->flags & 0x0f )
#define GET_SETTING_MARKER(_h) (( _h->marker_payloadt ) >> 7)
#define GET_SETTING_PAYLOAD(_h) ((_h->marker_payloadt) & 0x7f)
/**
2014-11-18 07:46:46 +08:00
* Checks if message came in late.
*/
2014-11-18 07:46:46 +08:00
static int check_late_message (RTPSession *session, RTPMessage *msg)
{
/*
* Check Sequence number. If this new msg has lesser number then the session->rsequnum
* it shows that the message came in late. Also check timestamp to be 100% certain.
2014-02-17 09:01:30 +08:00
*
*/
return ( msg->header->sequnum < session->rsequnum && msg->header->timestamp < session->timestamp ) ? 0 : -1;
}
/**
2014-11-18 07:46:46 +08:00
* Extracts header from payload.
*/
2014-02-17 09:01:30 +08:00
RTPHeader *extract_header ( const uint8_t *payload, int length )
{
2014-02-01 19:52:48 +08:00
if ( !payload || !length ) {
2014-05-03 07:46:03 +08:00
LOGGER_WARNING("No payload to extract!");
return NULL;
}
2014-02-17 09:01:30 +08:00
2014-11-29 20:42:19 +08:00
RTPHeader *retu = calloc(1, sizeof (RTPHeader));
2014-06-29 10:29:39 +08:00
2014-11-29 20:42:19 +08:00
if ( !retu ) {
LOGGER_WARNING("Alloc failed! Program might misbehave!");
return NULL;
}
2014-06-29 10:29:39 +08:00
2014-11-29 20:42:19 +08:00
memcpy(&retu->sequnum, payload, sizeof(retu->sequnum));
retu->sequnum = ntohs(retu->sequnum);
2014-02-17 09:01:30 +08:00
2014-11-29 20:42:19 +08:00
const uint8_t *it = payload + 2;
2014-02-17 09:01:30 +08:00
2014-11-29 20:42:19 +08:00
retu->flags = *it;
++it;
2014-02-17 09:01:30 +08:00
2014-02-16 03:44:33 +08:00
/* This indicates if the first 2 bits are valid.
* Now it may happen that this is out of order but
* it cuts down chances of parsing some invalid value
*/
2014-02-17 09:01:30 +08:00
2014-11-29 20:42:19 +08:00
if ( GET_FLAG_VERSION(retu) != RTP_VERSION ) {
/* Deallocate */
2014-05-03 07:46:03 +08:00
LOGGER_WARNING("Invalid version!");
2014-11-29 20:42:19 +08:00
free(retu);
return NULL;
}
2014-02-17 09:01:30 +08:00
/*
* Added a check for the size of the header little sooner so
* I don't need to parse the other stuff if it's bad
*/
2014-11-29 20:42:19 +08:00
uint8_t cc = GET_FLAG_CSRCC ( retu );
int total = 12 /* Minimum header len */ + ( cc * 4 );
2014-02-17 09:01:30 +08:00
2014-11-29 20:42:19 +08:00
if ( length < total ) {
/* Deallocate */
2014-05-03 07:46:03 +08:00
LOGGER_WARNING("Length invalid!");
2014-11-29 20:42:19 +08:00
free(retu);
return NULL;
}
2014-02-17 09:01:30 +08:00
2014-11-29 20:42:19 +08:00
memset(retu->csrc, 0, 16 * sizeof (uint32_t));
2014-11-29 20:42:19 +08:00
retu->marker_payloadt = *it;
++it;
retu->length = total;
2014-02-17 09:01:30 +08:00
2014-11-29 20:42:19 +08:00
memcpy(&retu->timestamp, it, sizeof(retu->timestamp));
retu->timestamp = ntohl(retu->timestamp);
it += 4;
memcpy(&retu->ssrc, it, sizeof(retu->ssrc));
retu->ssrc = ntohl(retu->ssrc);
2014-02-17 09:01:30 +08:00
2014-11-29 20:42:19 +08:00
uint8_t x;
2014-02-17 09:01:30 +08:00
2014-11-29 20:42:19 +08:00
for ( x = 0; x < cc; x++ ) {
it += 4;
memcpy(&retu->csrc[x], it, sizeof(retu->csrc[x]));
retu->csrc[x] = ntohl(retu->csrc[x]);
}
2014-02-17 09:01:30 +08:00
2014-11-29 20:42:19 +08:00
return retu;
}
/**
2014-11-18 07:46:46 +08:00
* Extracts external header from payload. Must be called AFTER extract_header()!
*/
2014-02-17 09:01:30 +08:00
RTPExtHeader *extract_ext_header ( const uint8_t *payload, uint16_t length )
{
2014-11-29 20:42:19 +08:00
const uint8_t *it = payload;
2014-02-17 09:01:30 +08:00
2014-11-29 20:42:19 +08:00
RTPExtHeader *retu = calloc(1, sizeof (RTPExtHeader));
2014-06-29 10:29:39 +08:00
2014-11-29 20:42:19 +08:00
if ( !retu ) {
LOGGER_WARNING("Alloc failed! Program might misbehave!");
return NULL;
}
2014-02-17 09:01:30 +08:00
2014-11-29 20:42:19 +08:00
uint16_t ext_length;
memcpy(&ext_length, it, sizeof(ext_length));
ext_length = ntohs(ext_length);
it += 2;
2014-02-17 09:01:30 +08:00
2014-11-29 20:42:19 +08:00
if ( length < ( ext_length * sizeof(uint32_t) ) ) {
2014-05-03 07:46:03 +08:00
LOGGER_WARNING("Length invalid!");
2014-11-29 20:42:19 +08:00
free(retu);
return NULL;
}
2014-02-17 09:01:30 +08:00
2014-11-29 20:42:19 +08:00
retu->length = ext_length;
memcpy(&retu->type, it, sizeof(retu->type));
retu->type = ntohs(retu->type);
it += 2;
2014-02-17 09:01:30 +08:00
2014-11-29 20:42:19 +08:00
if ( !(retu->table = calloc(ext_length, sizeof (uint32_t))) ) {
LOGGER_WARNING("Alloc failed! Program might misbehave!");
2014-11-29 20:42:19 +08:00
free(retu);
return NULL;
}
2014-02-17 09:01:30 +08:00
2014-11-29 20:42:19 +08:00
uint16_t x;
2014-02-17 09:01:30 +08:00
2014-11-29 20:42:19 +08:00
for ( x = 0; x < ext_length; x++ ) {
it += 4;
memcpy(&(retu->table[x]), it, sizeof(retu->table[x]));
retu->table[x] = ntohl(retu->table[x]);
}
2014-02-17 09:01:30 +08:00
2014-11-29 20:42:19 +08:00
return retu;
}
/**
2014-11-18 07:46:46 +08:00
* Adds header to payload. Make sure _payload_ has enough space.
*/
2014-02-17 09:01:30 +08:00
uint8_t *add_header ( RTPHeader *header, uint8_t *payload )
{
2014-11-29 20:42:19 +08:00
uint8_t cc = GET_FLAG_CSRCC ( header );
uint8_t *it = payload;
uint16_t sequnum;
uint32_t timestamp;
uint32_t ssrc;
uint32_t csrc;
2014-02-17 09:01:30 +08:00
/* Add sequence number first */
sequnum = htons(header->sequnum);
2014-11-29 20:42:19 +08:00
memcpy(it, &sequnum, sizeof(sequnum));
it += 2;
2014-02-17 09:01:30 +08:00
2014-11-29 20:42:19 +08:00
*it = header->flags;
++it;
*it = header->marker_payloadt;
++it;
2014-02-17 09:01:30 +08:00
timestamp = htonl(header->timestamp);
2014-11-29 20:42:19 +08:00
memcpy(it, &timestamp, sizeof(timestamp));
it += 4;
ssrc = htonl(header->ssrc);
2014-11-29 20:42:19 +08:00
memcpy(it, &ssrc, sizeof(ssrc));
2014-02-17 09:01:30 +08:00
2014-11-29 20:42:19 +08:00
uint8_t x;
2014-02-17 09:01:30 +08:00
2014-11-29 20:42:19 +08:00
for ( x = 0; x < cc; x++ ) {
it += 4;
csrc = htonl(header->csrc[x]);
memcpy(it, &csrc, sizeof(csrc));
}
2014-02-17 09:01:30 +08:00
2014-11-29 20:42:19 +08:00
return it + 4;
}
/**
2014-11-18 07:46:46 +08:00
* Adds extension header to payload. Make sure _payload_ has enough space.
*/
2014-02-17 09:01:30 +08:00
uint8_t *add_ext_header ( RTPExtHeader *header, uint8_t *payload )
{
2014-11-29 20:42:19 +08:00
uint8_t *it = payload;
uint16_t length;
uint16_t type;
uint32_t entry;
2014-02-17 09:01:30 +08:00
length = htons(header->length);
2014-11-29 20:42:19 +08:00
memcpy(it, &length, sizeof(length));
it += 2;
type = htons(header->type);
2014-11-29 20:42:19 +08:00
memcpy(it, &type, sizeof(type));
it -= 2; /* Return to 0 position */
2014-02-17 09:01:30 +08:00
2014-02-07 07:10:55 +08:00
if ( header->table ) {
2014-02-17 09:01:30 +08:00
2014-11-29 20:42:19 +08:00
uint16_t x;
2014-11-29 20:42:19 +08:00
for ( x = 0; x < header->length; x++ ) {
it += 4;
entry = htonl(header->table[x]);
memcpy(it, &entry, sizeof(entry));
2014-02-07 07:10:55 +08:00
}
}
2014-02-17 09:01:30 +08:00
2014-11-29 20:42:19 +08:00
return it + 4;
}
/**
2014-11-18 07:46:46 +08:00
* Builds header from control session values.
*/
2014-02-17 09:01:30 +08:00
RTPHeader *build_header ( RTPSession *session )
{
2014-11-29 20:42:19 +08:00
RTPHeader *retu = calloc ( 1, sizeof (RTPHeader) );
2014-06-29 10:29:39 +08:00
2014-11-29 20:42:19 +08:00
if ( !retu ) {
LOGGER_WARNING("Alloc failed! Program might misbehave!");
return NULL;
}
2014-02-17 09:01:30 +08:00
2014-11-29 20:42:19 +08:00
ADD_FLAG_VERSION ( retu, session->version );
ADD_FLAG_PADDING ( retu, session->padding );
ADD_FLAG_EXTENSION ( retu, session->extension );
ADD_FLAG_CSRCC ( retu, session->cc );
ADD_SETTING_MARKER ( retu, session->marker );
ADD_SETTING_PAYLOAD ( retu, session->payload_type );
2014-02-17 09:01:30 +08:00
2014-11-29 20:42:19 +08:00
retu->sequnum = session->sequnum;
retu->timestamp = current_time_monotonic(); /* milliseconds */
retu->ssrc = session->ssrc;
2014-02-17 09:01:30 +08:00
int i;
2014-02-17 09:01:30 +08:00
for ( i = 0; i < session->cc; i++ )
2014-11-29 20:42:19 +08:00
retu->csrc[i] = session->csrc[i];
2014-11-29 20:42:19 +08:00
retu->length = 12 /* Minimum header len */ + ( session->cc * size_32 );
2014-02-17 09:01:30 +08:00
2014-11-29 20:42:19 +08:00
return retu;
}
/**
2014-11-18 07:46:46 +08:00
* Parses data into RTPMessage struct. Stores headers separately from the payload data
* and so the length variable is set accordingly.
*/
RTPMessage *msg_parse ( const uint8_t *data, int length )
2014-02-17 09:01:30 +08:00
{
2014-11-29 20:42:19 +08:00
RTPMessage *retu = calloc(1, sizeof (RTPMessage));
2014-02-17 09:01:30 +08:00
2014-11-29 20:42:19 +08:00
retu->header = extract_header ( data, length ); /* It allocates memory and all */
2014-02-17 09:01:30 +08:00
2014-11-29 20:42:19 +08:00
if ( !retu->header ) {
2014-05-03 07:46:03 +08:00
LOGGER_WARNING("Header failed to extract!");
2014-11-29 20:42:19 +08:00
free(retu);
return NULL;
}
2014-02-17 09:01:30 +08:00
2014-11-29 20:42:19 +08:00
uint16_t from_pos = retu->header->length;
retu->length = length - from_pos;
2014-02-17 09:01:30 +08:00
2014-11-29 20:42:19 +08:00
if ( GET_FLAG_EXTENSION ( retu->header ) ) {
retu->ext_header = extract_ext_header ( data + from_pos, length );
2014-02-17 09:01:30 +08:00
2014-11-29 20:42:19 +08:00
if ( retu->ext_header ) {
retu->length -= ( 4 /* Minimum ext header len */ + retu->ext_header->length * size_32 );
from_pos += ( 4 /* Minimum ext header len */ + retu->ext_header->length * size_32 );
2014-02-01 19:52:48 +08:00
} else { /* Error */
2014-05-03 07:46:03 +08:00
LOGGER_WARNING("Ext Header failed to extract!");
2014-11-29 20:42:19 +08:00
rtp_free_msg(NULL, retu);
return NULL;
}
} else {
2014-11-29 20:42:19 +08:00
retu->ext_header = NULL;
}
2014-02-07 07:10:55 +08:00
2014-11-29 20:42:19 +08:00
if ( length - from_pos <= MAX_RTP_SIZE )
memcpy ( retu->data, data + from_pos, length - from_pos );
2014-02-07 07:10:55 +08:00
else {
2014-05-03 07:46:03 +08:00
LOGGER_WARNING("Invalid length!");
2014-11-29 20:42:19 +08:00
rtp_free_msg(NULL, retu);
2014-02-07 07:10:55 +08:00
return NULL;
}
2014-02-17 09:01:30 +08:00
2014-11-29 20:42:19 +08:00
retu->next = NULL;
2014-02-17 09:01:30 +08:00
2014-11-29 20:42:19 +08:00
return retu;
}
/**
2014-11-18 07:46:46 +08:00
* Callback for networking core.
*/
int rtp_handle_packet ( Messenger *m, uint32_t friendnumber, const uint8_t *data, uint16_t length, void *object )
{
2014-11-29 20:42:19 +08:00
RTPSession *session = object;
RTPMessage *msg;
2014-02-17 09:01:30 +08:00
2014-11-29 20:42:19 +08:00
if ( !session || length < 13 ) { /* 12 is the minimum length for rtp + desc. byte */
2014-05-03 07:46:03 +08:00
LOGGER_WARNING("No session or invalid length of received buffer!");
return -1;
2014-05-03 07:46:03 +08:00
}
2014-11-29 20:42:19 +08:00
msg = msg_parse ( data + 1, length - 1 );
2014-02-17 09:01:30 +08:00
2014-11-29 20:42:19 +08:00
if ( !msg ) {
2014-05-03 07:46:03 +08:00
LOGGER_WARNING("Could not parse message!");
return -1;
}
2014-02-17 09:01:30 +08:00
/* Check if message came in late */
2014-11-29 20:42:19 +08:00
if ( check_late_message(session, msg) < 0 ) { /* Not late */
session->rsequnum = msg->header->sequnum;
session->timestamp = msg->header->timestamp;
}
2014-02-17 09:01:30 +08:00
2014-11-29 20:42:19 +08:00
queue_message(session, msg);
2014-02-17 09:01:30 +08:00
return 0;
}
/**
2014-11-18 07:46:46 +08:00
* Allocate message and store data there
*/
2014-02-17 09:01:30 +08:00
RTPMessage *rtp_new_message ( RTPSession *session, const uint8_t *data, uint32_t length )
{
2014-05-03 07:46:03 +08:00
if ( !session ) {
LOGGER_WARNING("No session!");
return NULL;
2014-05-03 07:46:03 +08:00
}
2014-02-17 09:01:30 +08:00
2014-11-29 20:42:19 +08:00
uint8_t *from_pos;
RTPMessage *retu = calloc(1, sizeof (RTPMessage));
2014-06-29 10:29:39 +08:00
2014-11-29 20:42:19 +08:00
if ( !retu ) {
LOGGER_WARNING("Alloc failed! Program might misbehave!");
return NULL;
}
2014-02-17 09:01:30 +08:00
2014-11-29 20:42:19 +08:00
/* Sets header values and copies the extension header in retu */
retu->header = build_header ( session ); /* It allocates memory and all */
retu->ext_header = session->ext_header;
2014-02-17 09:01:30 +08:00
2014-11-29 20:42:19 +08:00
uint32_t total_length = length + retu->header->length + 1;
2014-02-17 09:01:30 +08:00
2014-11-29 20:42:19 +08:00
retu->data[0] = session->prefix;
2014-02-17 09:01:30 +08:00
2014-11-29 20:42:19 +08:00
if ( retu->ext_header ) {
total_length += ( 4 /* Minimum ext header len */ + retu->ext_header->length * size_32 );
2014-02-17 09:01:30 +08:00
2014-11-29 20:42:19 +08:00
from_pos = add_header ( retu->header, retu->data + 1 );
from_pos = add_ext_header ( retu->ext_header, from_pos + 1 );
} else {
2014-11-29 20:42:19 +08:00
from_pos = add_header ( retu->header, retu->data + 1 );
}
2014-02-17 09:01:30 +08:00
/*
* Parses the extension header into the message
* Of course if any
*/
2014-02-17 09:01:30 +08:00
2014-11-29 20:42:19 +08:00
/* Appends data on to retu->data */
memcpy ( from_pos, data, length );
2014-02-17 09:01:30 +08:00
2014-11-29 20:42:19 +08:00
retu->length = total_length;
2014-02-17 09:01:30 +08:00
2014-11-29 20:42:19 +08:00
retu->next = NULL;
2014-02-17 09:01:30 +08:00
2014-11-29 20:42:19 +08:00
return retu;
}
2014-11-18 07:46:46 +08:00
2014-02-17 09:01:30 +08:00
int rtp_send_msg ( RTPSession *session, Messenger *messenger, const uint8_t *data, uint16_t length )
{
2014-02-17 09:01:30 +08:00
RTPMessage *msg = rtp_new_message (session, data, length);
2014-11-18 07:46:46 +08:00
if ( !msg ) return -1;
2014-02-17 09:01:30 +08:00
int ret = send_custom_lossy_packet(messenger, session->dest, msg->data, msg->length);
if ( 0 != ret) {
LOGGER_WARNING("Failed to send full packet (len: %d)! error: %i", length, ret);
2014-04-10 06:15:40 +08:00
rtp_free_msg ( session, msg );
2014-11-29 20:42:19 +08:00
return rtp_ErrorSending;
}
2014-02-17 09:01:30 +08:00
/* Set sequ number */
session->sequnum = session->sequnum >= MAX_SEQU_NUM ? 0 : session->sequnum + 1;
rtp_free_msg ( session, msg );
return 0;
}
2014-02-17 09:01:30 +08:00
void rtp_free_msg ( RTPSession *session, RTPMessage *msg )
{
2014-02-17 09:01:30 +08:00
if ( !session ) {
if ( msg->ext_header ) {
free ( msg->ext_header->table );
free ( msg->ext_header );
}
} else {
if ( msg->ext_header && session->ext_header != msg->ext_header ) {
free ( msg->ext_header->table );
free ( msg->ext_header );
}
}
2014-02-17 09:01:30 +08:00
free ( msg->header );
free ( msg );
}
2014-11-18 07:46:46 +08:00
RTPSession *rtp_new ( int payload_type, Messenger *messenger, int friend_num )
2014-02-17 09:01:30 +08:00
{
2014-11-29 20:42:19 +08:00
RTPSession *retu = calloc(1, sizeof(RTPSession));
2014-02-17 09:01:30 +08:00
2014-11-29 20:42:19 +08:00
if ( !retu ) {
LOGGER_WARNING("Alloc failed! Program might misbehave!");
return NULL;
}
2014-02-17 09:01:30 +08:00
if ( -1 == m_callback_rtp_packet(messenger, friend_num, payload_type, rtp_handle_packet, retu)) {
2014-05-03 07:46:03 +08:00
LOGGER_ERROR("Error setting custom register handler for rtp session");
2014-11-29 20:42:19 +08:00
free(retu);
2014-02-16 03:44:33 +08:00
return NULL;
}
2014-02-17 09:01:30 +08:00
2014-05-17 01:56:40 +08:00
LOGGER_DEBUG("Registered packet handler: pt: %d; fid: %d", payload_type, friend_num);
2014-02-17 09:01:30 +08:00
2014-11-29 20:42:19 +08:00
retu->version = RTP_VERSION; /* It's always 2 */
retu->padding = 0; /* If some additional data is needed about the packet */
retu->extension = 0; /* If extension to header is needed */
retu->cc = 1; /* Amount of contributors */
retu->csrc = NULL; /* Container */
retu->ssrc = random_int();
retu->marker = 0;
retu->payload_type = payload_type % 128;
2014-02-17 09:01:30 +08:00
2014-11-29 20:42:19 +08:00
retu->dest = friend_num;
2014-02-17 09:01:30 +08:00
2014-11-29 20:42:19 +08:00
retu->rsequnum = retu->sequnum = 0;
2014-02-17 09:01:30 +08:00
2014-11-29 20:42:19 +08:00
retu->ext_header = NULL; /* When needed allocate */
2014-02-17 09:01:30 +08:00
2014-11-29 20:42:19 +08:00
if ( !(retu->csrc = calloc(1, sizeof (uint32_t))) ) {
LOGGER_WARNING("Alloc failed! Program might misbehave!");
2014-11-29 20:42:19 +08:00
free(retu);
return NULL;
}
2014-02-17 09:01:30 +08:00
2014-11-29 20:42:19 +08:00
retu->csrc[0] = retu->ssrc; /* Set my ssrc to the list receive */
2014-02-17 09:01:30 +08:00
/* Also set payload type as prefix */
2014-11-29 20:42:19 +08:00
retu->prefix = payload_type;
2014-02-17 09:01:30 +08:00
/*
2014-02-17 09:01:30 +08:00
*
*/
2014-11-29 20:42:19 +08:00
return retu;
}
2014-11-18 07:46:46 +08:00
void rtp_kill ( RTPSession *session, Messenger *messenger )
{
if ( !session ) return;
m_callback_rtp_packet(messenger, session->dest, session->prefix, NULL, NULL);
free ( session->ext_header );
free ( session->csrc );
2014-02-17 09:01:30 +08:00
2014-07-05 21:11:25 +08:00
LOGGER_DEBUG("Terminated RTP session: %p", session);
/* And finally free session */
free ( session );
}