modules/misc/sap.c:

 * more verbose header parsing code
 * support for compressed announces if zlib is available (using some code
   from Mosu from the matroska project)
 * if the sdp packet contains more than one m= line, pass it to the
   livedotcom module
configure.ac:
 * link sap with zlib if available
modules/demux/livedotcom.c:
 * accept sdp from the sap module
parent b38ec882
dnl Autoconf settings for vlc dnl Autoconf settings for vlc
dnl $Id: configure.ac,v 1.113 2003/11/15 01:21:48 massiot Exp $ dnl $Id: configure.ac,v 1.114 2003/11/20 23:13:28 sigmunau Exp $
AC_INIT(vlc,0.6.3-cvs) AC_INIT(vlc,0.6.3-cvs)
...@@ -1905,7 +1905,7 @@ dnl ...@@ -1905,7 +1905,7 @@ dnl
dnl MP4 module dnl MP4 module
dnl dnl
AC_CHECK_HEADERS(zlib.h, [ AC_CHECK_HEADERS(zlib.h, [
AX_ADD_LDFLAGS([mp4 skins],[-lz]) AX_ADD_LDFLAGS([mp4 skins sap],[-lz])
] ) ] )
......
...@@ -2,7 +2,7 @@ ...@@ -2,7 +2,7 @@
* live.cpp : live.com support. * live.cpp : live.com support.
***************************************************************************** *****************************************************************************
* Copyright (C) 2003 VideoLAN * Copyright (C) 2003 VideoLAN
* $Id: livedotcom.cpp,v 1.8 2003/11/20 22:10:55 fenrir Exp $ * $Id: livedotcom.cpp,v 1.9 2003/11/20 23:13:28 sigmunau Exp $
* *
* Authors: Laurent Aimar <fenrir@via.ecp.fr> * Authors: Laurent Aimar <fenrir@via.ecp.fr>
* *
...@@ -65,6 +65,7 @@ vlc_module_begin(); ...@@ -65,6 +65,7 @@ vlc_module_begin();
add_submodule(); add_submodule();
set_description( _("RTSP/RTP describe") ); set_description( _("RTSP/RTP describe") );
add_shortcut( "rtsp" ); add_shortcut( "rtsp" );
add_shortcut( "sdp" );
set_capability( "access", 0 ); set_capability( "access", 0 );
set_callbacks( AccessOpen, AccessClose ); set_callbacks( AccessOpen, AccessClose );
add_category_hint( N_("RTSP"), NULL, VLC_TRUE ); add_category_hint( N_("RTSP"), NULL, VLC_TRUE );
...@@ -134,6 +135,7 @@ struct demux_sys_t ...@@ -134,6 +135,7 @@ struct demux_sys_t
}; };
static ssize_t Read ( input_thread_t *, byte_t *, size_t ); static ssize_t Read ( input_thread_t *, byte_t *, size_t );
static ssize_t MRLRead( input_thread_t *, byte_t *, size_t );
static int Demux ( input_thread_t * ); static int Demux ( input_thread_t * );
static int Control( input_thread_t *, int, va_list ); static int Control( input_thread_t *, int, va_list );
...@@ -155,86 +157,108 @@ static int AccessOpen( vlc_object_t *p_this ) ...@@ -155,86 +157,108 @@ static int AccessOpen( vlc_object_t *p_this )
vlc_value_t val; vlc_value_t val;
char *psz_url; char *psz_url;
if( p_input->psz_access == NULL || strcasecmp( p_input->psz_access, "rtsp" ) ) if( p_input->psz_access == NULL || ( strcasecmp( p_input->psz_access, "rtsp" ) && strcasecmp( p_input->psz_access, "sdp" ) ) )
{ {
msg_Warn( p_input, "RTSP access discarded" ); msg_Warn( p_input, "RTSP access discarded" );
return VLC_EGENERIC; return VLC_EGENERIC;
} }
if( ( scheduler = BasicTaskScheduler::createNew() ) == NULL ) if( !strcasecmp( p_input->psz_access, "rtsp" ) )
{ {
msg_Err( p_input, "BasicTaskScheduler::createNew failed" ); if( ( scheduler = BasicTaskScheduler::createNew() ) == NULL )
return VLC_EGENERIC; {
} msg_Err( p_input, "BasicTaskScheduler::createNew failed" );
if( ( env = BasicUsageEnvironment::createNew(*scheduler) ) == NULL ) return VLC_EGENERIC;
{ }
delete scheduler; if( ( env = BasicUsageEnvironment::createNew(*scheduler) ) == NULL )
msg_Err( p_input, "BasicUsageEnvironment::createNew failed" ); {
return VLC_EGENERIC; delete scheduler;
} msg_Err( p_input, "BasicUsageEnvironment::createNew failed" );
if( ( rtsp = RTSPClient::createNew(*env, 1/*verbose*/, "VLC Media Player" ) ) == NULL ) return VLC_EGENERIC;
{ }
delete env; if( ( rtsp = RTSPClient::createNew(*env, 1/*verbose*/, "VLC Media Player" ) ) == NULL )
delete scheduler; {
msg_Err( p_input, "RTSPClient::createNew failed" ); delete env;
return VLC_EGENERIC; delete scheduler;
} msg_Err( p_input, "RTSPClient::createNew failed" );
return VLC_EGENERIC;
}
psz_url = (char*)malloc( strlen( p_input->psz_name ) + 8 ); psz_url = (char*)malloc( strlen( p_input->psz_name ) + 8 );
sprintf( psz_url, "rtsp://%s", p_input->psz_name ); sprintf( psz_url, "rtsp://%s", p_input->psz_name );
p_sys = (access_sys_t*)malloc( sizeof( access_sys_t ) ); p_sys = (access_sys_t*)malloc( sizeof( access_sys_t ) );
p_sys->p_sdp = rtsp->describeURL( psz_url ); p_sys->p_sdp = rtsp->describeURL( psz_url );
if( p_sys->p_sdp == NULL ) if( p_sys->p_sdp == NULL )
{ {
msg_Err( p_input, "describeURL failed (%s)", env->getResultMsg() ); msg_Err( p_input, "describeURL failed (%s)", env->getResultMsg() );
free( psz_url );
delete env;
delete scheduler;
free( p_sys );
return VLC_EGENERIC;
}
free( psz_url ); free( psz_url );
p_sys->i_sdp = strlen( p_sys->p_sdp );
p_sys->i_pos = 0;
//fprintf( stderr, "sdp=%s\n", p_sys->p_sdp );
delete env; delete env;
delete scheduler; delete scheduler;
free( p_sys );
return VLC_EGENERIC;
}
free( psz_url );
p_sys->i_sdp = strlen( p_sys->p_sdp );
p_sys->i_pos = 0;
//fprintf( stderr, "sdp=%s\n", p_sys->p_sdp ); var_Create( p_input, "rtsp-tcp", VLC_VAR_BOOL|VLC_VAR_DOINHERIT );
var_Get( p_input, "rtsp-tcp", &val );
delete env; p_input->p_access_data = p_sys;
delete scheduler; p_input->i_mtu = 0;
var_Create( p_input, "rtsp-tcp", VLC_VAR_BOOL|VLC_VAR_DOINHERIT ); /* Set exported functions */
var_Get( p_input, "rtsp-tcp", &val ); p_input->pf_read = Read;
p_input->pf_seek = NULL;
p_input->pf_set_program = input_SetProgram;
p_input->pf_set_area = NULL;
p_input->p_private = NULL;
p_input->p_access_data = p_sys; p_input->psz_demux = "live";
p_input->i_mtu = 0;
/* Set exported functions */
p_input->pf_read = Read;
p_input->pf_seek = NULL;
p_input->pf_set_program = input_SetProgram;
p_input->pf_set_area = NULL;
p_input->p_private = NULL;
p_input->psz_demux = "live";
/* Finished to set some variable */
vlc_mutex_lock( &p_input->stream.stream_lock );
/* FIXME that's not true but eg over tcp, server send data too fast */
p_input->stream.b_pace_control = val.b_bool;
p_input->stream.p_selected_area->i_tell = 0;
p_input->stream.b_seekable = 1; /* Hack to display time */
p_input->stream.p_selected_area->i_size = 0;
p_input->stream.i_method = INPUT_METHOD_NETWORK;
vlc_mutex_unlock( &p_input->stream.stream_lock );
/* Update default_pts to a suitable value for RTSP access */
var_Create( p_input, "rtsp-caching", VLC_VAR_INTEGER|VLC_VAR_DOINHERIT );
var_Get( p_input, "rtsp-caching", &val );
p_input->i_pts_delay = val.i_int * 1000;
return VLC_SUCCESS; /* Finished to set some variable */
vlc_mutex_lock( &p_input->stream.stream_lock );
/* FIXME that's not true but eg over tcp, server send data too fast */
p_input->stream.b_pace_control = val.b_bool;
p_input->stream.p_selected_area->i_tell = 0;
p_input->stream.b_seekable = 1; /* Hack to display time */
p_input->stream.p_selected_area->i_size = 0;
p_input->stream.i_method = INPUT_METHOD_NETWORK;
vlc_mutex_unlock( &p_input->stream.stream_lock );
/* Update default_pts to a suitable value for RTSP access */
var_Create( p_input, "rtsp-caching", VLC_VAR_INTEGER|VLC_VAR_DOINHERIT );
var_Get( p_input, "rtsp-caching", &val );
p_input->i_pts_delay = val.i_int * 1000;
return VLC_SUCCESS;
}
else
{
p_input->p_access_data = (access_sys_t*)0;
p_input->i_mtu = 0;
p_input->pf_read = MRLRead;
p_input->pf_seek = NULL;
p_input->pf_set_program = input_SetProgram;
p_input->pf_set_area = NULL;
p_input->p_private = NULL;
p_input->psz_demux = "live";
/* Finished to set some variable */
vlc_mutex_lock( &p_input->stream.stream_lock );
p_input->stream.b_pace_control = VLC_TRUE;
p_input->stream.p_selected_area->i_tell = 0;
p_input->stream.b_seekable = VLC_FALSE;
p_input->stream.p_selected_area->i_size = 0;
p_input->stream.i_method = INPUT_METHOD_NETWORK;
vlc_mutex_unlock( &p_input->stream.stream_lock );
}
} }
/***************************************************************************** /*****************************************************************************
...@@ -244,9 +268,11 @@ static void AccessClose( vlc_object_t *p_this ) ...@@ -244,9 +268,11 @@ static void AccessClose( vlc_object_t *p_this )
{ {
input_thread_t *p_input = (input_thread_t *)p_this; input_thread_t *p_input = (input_thread_t *)p_this;
access_sys_t *p_sys = p_input->p_access_data; access_sys_t *p_sys = p_input->p_access_data;
if( !strcasecmp( p_input->psz_access, "rtsp" ) )
delete[] p_sys->p_sdp; {
free( p_sys ); delete[] p_sys->p_sdp;
free( p_sys );
}
} }
/***************************************************************************** /*****************************************************************************
...@@ -264,6 +290,22 @@ static ssize_t Read ( input_thread_t *p_input, byte_t *p_buffer, size_t i_len ) ...@@ -264,6 +290,22 @@ static ssize_t Read ( input_thread_t *p_input, byte_t *p_buffer, size_t i_len )
} }
return i_copy; return i_copy;
} }
/*****************************************************************************
* MRLRead: read data from the mrl
*****************************************************************************/
static ssize_t MRLRead ( input_thread_t *p_input, byte_t *p_buffer, size_t i_len )
{
int i_done = (int)p_input->p_access_data;
int i_copy = __MIN( (int)i_len, strlen(p_input->psz_name) - i_done );
if( i_copy > 0 )
{
memcpy( p_buffer, &p_input->psz_name[i_done], i_copy );
i_done += i_copy;
p_input->p_access_data = (access_sys_t*)i_done;
}
return i_copy;
}
/***************************************************************************** /*****************************************************************************
......
...@@ -2,7 +2,7 @@ ...@@ -2,7 +2,7 @@
* sap.c : SAP interface module * sap.c : SAP interface module
***************************************************************************** *****************************************************************************
* Copyright (C) 2001 VideoLAN * Copyright (C) 2001 VideoLAN
* $Id: sap.c,v 1.35 2003/11/12 20:01:01 gbazin Exp $ * $Id: sap.c,v 1.36 2003/11/20 23:13:28 sigmunau Exp $
* *
* Authors: Arnaud Schauly <gitan@via.ecp.fr> * Authors: Arnaud Schauly <gitan@via.ecp.fr>
* Clment Stenac <zorglub@via.ecp.fr> * Clment Stenac <zorglub@via.ecp.fr>
...@@ -66,6 +66,10 @@ ...@@ -66,6 +66,10 @@
#include "network.h" #include "network.h"
#ifdef HAVE_ZLIB_H
# include <zlib.h>
#endif
#define MAX_LINE_LENGTH 256 #define MAX_LINE_LENGTH 256
/* SAP is always on that port */ /* SAP is always on that port */
...@@ -136,6 +140,7 @@ struct sess_descr_t ...@@ -136,6 +140,7 @@ struct sess_descr_t
int i_version; int i_version;
char *psz_sessionname; char *psz_sessionname;
char *psz_connection; char *psz_connection;
char *psz_sdp;
int i_media; int i_media;
media_descr_t **pp_media; media_descr_t **pp_media;
...@@ -165,6 +170,47 @@ struct intf_sys_t ...@@ -165,6 +170,47 @@ struct intf_sys_t
int i_group; int i_group;
}; };
#ifdef HAVE_ZLIB_H
int do_decompress(unsigned char *src, unsigned char **_dst, int slen) {
int result, dstsize, n;
unsigned char *dst;
z_stream d_stream;
d_stream.zalloc = (alloc_func)0;
d_stream.zfree = (free_func)0;
d_stream.opaque = (voidpf)0;
result = inflateInit(&d_stream);
if (result != Z_OK) {
printf("inflateInit() failed. Result: %d\n", result);
return(-1);
}
d_stream.next_in = (Bytef *)src;
d_stream.avail_in = slen;
n = 0;
dst = NULL;
do {
n++;
dst = (unsigned char *)realloc(dst, n * 1000);
d_stream.next_out = (Bytef *)&dst[(n - 1) * 1000];
d_stream.avail_out = 1000;
result = inflate(&d_stream, Z_NO_FLUSH);
if ((result != Z_OK) && (result != Z_STREAM_END)) {
printf("Zlib decompression failed. Result: %d\n", result);
return(-1);
}
} while ((d_stream.avail_out == 0) && (d_stream.avail_in != 0) &&
(result != Z_STREAM_END));
dstsize = d_stream.total_out;
inflateEnd(&d_stream);
*_dst = (unsigned char *)realloc(dst, dstsize);
return dstsize;
}
#endif
/***************************************************************************** /*****************************************************************************
* Open: initialize and create stuff * Open: initialize and create stuff
*****************************************************************************/ *****************************************************************************/
...@@ -290,19 +336,29 @@ static void Close( vlc_object_t *p_this ) ...@@ -290,19 +336,29 @@ static void Close( vlc_object_t *p_this )
***************************************************************************** *****************************************************************************
* Listens to SAP packets, and sends them to packet_handle * Listens to SAP packets, and sends them to packet_handle
*****************************************************************************/ *****************************************************************************/
#define MAX_SAP_BUFFER 2000 #define MAX_SAP_BUFFER 5000
static void Run( intf_thread_t *p_intf ) static void Run( intf_thread_t *p_intf )
{ {
intf_sys_t *p_sys = p_intf->p_sys; intf_sys_t *p_sys = p_intf->p_sys;
uint8_t buffer[MAX_SAP_BUFFER + 1]; uint8_t buffer[MAX_SAP_BUFFER + 1];
uint8_t *p_end;
/* read SAP packets */ /* read SAP packets */
while( !p_intf->b_die ) while( !p_intf->b_die )
{ {
int i_read = NetRead( p_intf, p_sys->fd, buffer, MAX_SAP_BUFFER ); int i_read = NetRead( p_intf, p_sys->fd, buffer, MAX_SAP_BUFFER );
uint8_t *p_sdp; uint8_t *p_sdp;
int i_version;
int i_address_type;
int b_reserved;
int b_message_type;
int b_encrypted;
int b_compressed;
int i_mesg_id_hash;
unsigned char *p_decompressed_buffer;
int i_decompressed_size;
/* Minimum length is > 6 */ /* Minimum length is > 6 */
if( i_read <= 6 ) if( i_read <= 6 )
{ {
...@@ -314,13 +370,57 @@ static void Run( intf_thread_t *p_intf ) ...@@ -314,13 +370,57 @@ static void Run( intf_thread_t *p_intf )
} }
buffer[i_read] = '\0'; buffer[i_read] = '\0';
p_end = &buffer[i_read];
/* Parse the SAP header */ /* Parse the SAP header */
i_version = buffer[0] >> 5;
if( i_version != 1 )
{
msg_Warn( p_intf, "strange sap version %d found", i_version );
}
i_address_type = buffer[0] & 0x10;
b_reserved = buffer[0] & 0x08;
if( b_reserved != 0 )
{
msg_Warn( p_intf, "reserved bit incorrectly set" );
}
b_message_type = buffer[0] & 0x04;
if( b_message_type != 0 )
{
msg_Warn( p_intf, "got session deletion packet" );
}
b_encrypted = buffer[0] & 0x02;
if( b_encrypted )
{
msg_Warn( p_intf, "encrypted packet" );
}
b_compressed = buffer[0] & 0x01;
p_sdp = &buffer[4]; p_sdp = &buffer[4];
p_sdp += (buffer[0]&0x10) ? 16 : 4; if( i_address_type == 0 ) /* ipv4 source address */
p_sdp += buffer[1]; {
p_sdp += 4;
while( p_sdp < &buffer[i_read-1] && *p_sdp != '\0' && p_sdp[0] != 'v' && p_sdp[1] != '=' ) }
else /* ipv6 source address */
{
p_sdp += 16;
}
if( b_compressed )
{
#ifdef HAVE_ZLIB_H
i_decompressed_size = do_decompress( p_sdp, &p_decompressed_buffer, i_read - ( p_sdp - buffer ) );
if( i_decompressed_size > 0 )
{
memcpy( p_sdp, p_decompressed_buffer, i_decompressed_size );
p_sdp[i_decompressed_size] = '\0';
p_end = &p_sdp[i_decompressed_size];
free( p_decompressed_buffer );
}
#else
msg_Warn( p_intf, "Ignoring compressed sap packet" );
#endif
}
p_sdp += buffer[1]; /* size of signature */
while( p_sdp < p_end - 1 && *p_sdp != '\0' && p_sdp[0] != 'v' && p_sdp[1] != '=' )
{ {
p_sdp++; p_sdp++;
} }
...@@ -328,7 +428,8 @@ static void Run( intf_thread_t *p_intf ) ...@@ -328,7 +428,8 @@ static void Run( intf_thread_t *p_intf )
{ {
p_sdp++; p_sdp++;
} }
if( p_sdp < &buffer[i_read] )
if( p_sdp < p_end )
{ {
sess_descr_t *p_sd = parse_sdp( p_intf, p_sdp ); sess_descr_t *p_sd = parse_sdp( p_intf, p_sdp );
if( p_sd ) if( p_sd )
...@@ -337,6 +438,10 @@ static void Run( intf_thread_t *p_intf ) ...@@ -337,6 +438,10 @@ static void Run( intf_thread_t *p_intf )
free_sd ( p_sd ); free_sd ( p_sd );
} }
} }
else
{
msg_Warn( p_intf, "ditching sap packet" );
}
} }
} }
...@@ -467,6 +572,46 @@ static void sess_toitem( intf_thread_t * p_intf, sess_descr_t * p_sd ) ...@@ -467,6 +572,46 @@ static void sess_toitem( intf_thread_t * p_intf, sess_descr_t * p_sd )
playlist_t *p_playlist = NULL; playlist_t *p_playlist = NULL;
psz_uri_default = NULL; psz_uri_default = NULL;
if( p_sd->i_media > 1 )
{
p_item = malloc( sizeof( playlist_item_t ) );
if( p_item == NULL )
{
msg_Warn( p_intf, "out of memory" );
return;
}
p_item->psz_name = strdup( p_sd->psz_sessionname );
p_item->psz_uri = NULL;
p_item->i_duration = -1;
p_item->ppsz_options= NULL;
p_item->i_options = 0;
p_item->i_type = 0;
p_item->i_status = 0;
p_item->b_autodeletion = VLC_FALSE;
p_item->b_enabled = VLC_TRUE;
p_item->i_group = p_intf->p_sys->i_group;
p_item->psz_author = strdup( "" );
psz_uri = malloc( strlen( p_sd->psz_sdp ) + 7 );
if( psz_uri == NULL )
{
msg_Warn( p_intf, "out of memory" );
free( p_item->psz_name );
free( p_item->psz_author );
free( p_item );
return;
}
p_item->psz_uri = psz_uri;
memcpy( psz_uri, "sdp://", 6 );
psz_uri += 6;
memcpy( psz_uri, p_sd->psz_sdp, strlen( p_sd->psz_sdp ) + 1 );
/* Enqueueing p_item in the playlist */
p_playlist = vlc_object_find( p_intf, VLC_OBJECT_PLAYLIST, FIND_ANYWHERE );
playlist_AddItem ( p_playlist, p_item, PLAYLIST_CHECK_INSERT, PLAYLIST_END );
vlc_object_release( p_playlist );
return;
}
cfield_parse( p_sd->psz_connection, &psz_uri_default ); cfield_parse( p_sd->psz_connection, &psz_uri_default );
for( i_count = 0 ; i_count < p_sd->i_media ; i_count++ ) for( i_count = 0 ; i_count < p_sd->i_media ; i_count++ )
...@@ -618,6 +763,8 @@ static sess_descr_t * parse_sdp( intf_thread_t * p_intf, char *p_packet ) ...@@ -618,6 +763,8 @@ static sess_descr_t * parse_sdp( intf_thread_t * p_intf, char *p_packet )
sd = malloc( sizeof( sess_descr_t ) ); sd = malloc( sizeof( sess_descr_t ) );
sd->psz_sessionname = NULL; sd->psz_sessionname = NULL;
sd->psz_connection = NULL; sd->psz_connection = NULL;
sd->psz_sdp = strdup( p_packet );
sd->i_media = 0; sd->i_media = 0;
sd->pp_media = NULL; sd->pp_media = NULL;
sd->i_attributes = 0; sd->i_attributes = 0;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment