Commit 881768ad authored by Laurent Aimar's avatar Laurent Aimar

* all: begin to add mms over udp support. It begin to work (at least

for me) but it still needs some work.
parent 1158db5f
...@@ -2,7 +2,7 @@ ...@@ -2,7 +2,7 @@
* buffer.c: MMS access plug-in * buffer.c: MMS access plug-in
***************************************************************************** *****************************************************************************
* Copyright (C) 2001, 2002 VideoLAN * Copyright (C) 2001, 2002 VideoLAN
* $Id: buffer.c,v 1.1 2002/11/22 18:35:57 sam Exp $ * $Id: buffer.c,v 1.2 2002/11/25 00:22:04 fenrir Exp $
* *
* Authors: Laurent Aimar <fenrir@via.ecp.fr> * Authors: Laurent Aimar <fenrir@via.ecp.fr>
* *
...@@ -193,7 +193,7 @@ uint64_t var_buffer_get64( var_buffer_t *p_buf ) ...@@ -193,7 +193,7 @@ uint64_t var_buffer_get64( var_buffer_t *p_buf )
return( i_dw1 + ( i_dw2 << 32 ) ); return( i_dw1 + ( i_dw2 << 32 ) );
} }
int var_buffer_getmemory ( var_buffer_t *p_buf, void *p_mem, int i_mem ) int var_buffer_getmemory ( var_buffer_t *p_buf, void *p_mem, int64_t i_mem )
{ {
int i_copy; int i_copy;
...@@ -202,6 +202,11 @@ int var_buffer_getmemory ( var_buffer_t *p_buf, void *p_mem, int i_mem ) ...@@ -202,6 +202,11 @@ int var_buffer_getmemory ( var_buffer_t *p_buf, void *p_mem, int i_mem )
{ {
memcpy( p_mem, p_buf + p_buf->i_data, i_copy ); memcpy( p_mem, p_buf + p_buf->i_data, i_copy );
} }
if( i_copy < 0 )
{
// fprintf( stderr, "\n**************arrrrrrggggg\n" );
i_copy = 0;
}
p_buf->i_data += i_copy; p_buf->i_data += i_copy;
return( i_copy ); return( i_copy );
} }
......
...@@ -2,7 +2,7 @@ ...@@ -2,7 +2,7 @@
* buffer.h: MMS access plug-in * buffer.h: MMS access plug-in
***************************************************************************** *****************************************************************************
* Copyright (C) 2001, 2002 VideoLAN * Copyright (C) 2001, 2002 VideoLAN
* $Id: buffer.h,v 1.1 2002/11/22 18:35:57 sam Exp $ * $Id: buffer.h,v 1.2 2002/11/25 00:22:04 fenrir Exp $
* *
* Authors: Laurent Aimar <fenrir@via.ecp.fr> * Authors: Laurent Aimar <fenrir@via.ecp.fr>
* *
...@@ -50,7 +50,7 @@ uint8_t var_buffer_get8 ( var_buffer_t *p_buf ); ...@@ -50,7 +50,7 @@ uint8_t var_buffer_get8 ( var_buffer_t *p_buf );
uint16_t var_buffer_get16( var_buffer_t *p_buf ); uint16_t var_buffer_get16( var_buffer_t *p_buf );
uint32_t var_buffer_get32( var_buffer_t *p_buf ); uint32_t var_buffer_get32( var_buffer_t *p_buf );
uint64_t var_buffer_get64( var_buffer_t *p_buf ); uint64_t var_buffer_get64( var_buffer_t *p_buf );
int var_buffer_getmemory ( var_buffer_t *p_buf, void *p_mem, int i_mem ); int var_buffer_getmemory ( var_buffer_t *p_buf, void *p_mem, int64_t i_mem );
int var_buffer_readempty( var_buffer_t *p_buf ); int var_buffer_readempty( var_buffer_t *p_buf );
void var_buffer_getguid( var_buffer_t *p_buf, guid_t *p_guid ); void var_buffer_getguid( var_buffer_t *p_buf, guid_t *p_guid );
...@@ -2,7 +2,7 @@ ...@@ -2,7 +2,7 @@
* mms.c: MMS access plug-in * mms.c: MMS access plug-in
***************************************************************************** *****************************************************************************
* Copyright (C) 2001, 2002 VideoLAN * Copyright (C) 2001, 2002 VideoLAN
* $Id: mms.c,v 1.6 2002/11/22 18:35:57 sam Exp $ * $Id: mms.c,v 1.7 2002/11/25 00:22:04 fenrir Exp $
* *
* Authors: Laurent Aimar <fenrir@via.ecp.fr> * Authors: Laurent Aimar <fenrir@via.ecp.fr>
* *
...@@ -76,8 +76,6 @@ static void Close ( vlc_object_t * ); ...@@ -76,8 +76,6 @@ static void Close ( vlc_object_t * );
static int Read ( input_thread_t * p_input, byte_t * p_buffer, static int Read ( input_thread_t * p_input, byte_t * p_buffer,
size_t i_len ); size_t i_len );
static ssize_t NetRead ( input_thread_t * p_input, input_socket_t * p_socket,
byte_t * p_buffer, size_t i_len );
static void Seek ( input_thread_t *, off_t ); static void Seek ( input_thread_t *, off_t );
static int SetProgram ( input_thread_t *, pgrm_descriptor_t * ); static int SetProgram ( input_thread_t *, pgrm_descriptor_t * );
...@@ -85,7 +83,7 @@ static int SetProgram ( input_thread_t *, pgrm_descriptor_t * ); ...@@ -85,7 +83,7 @@ static int SetProgram ( input_thread_t *, pgrm_descriptor_t * );
static int MMSOpen( input_thread_t *, url_t *, int, char * ); static int MMSOpen( input_thread_t *, url_t *, int, char * );
static int MMSStart ( input_thread_t *, uint32_t ); static int MMSStart ( input_thread_t *, uint32_t );
static int MMSStop ( input_thread_t *p_input ); // Not used static int MMSStop ( input_thread_t *p_input );
static int MMSClose ( input_thread_t * ); static int MMSClose ( input_thread_t * );
...@@ -109,7 +107,9 @@ static void mms_ParseURL( url_t *p_url, char *psz_url ); ...@@ -109,7 +107,9 @@ static void mms_ParseURL( url_t *p_url, char *psz_url );
/* /*
* Ok, ok, j'le ferai plus... * Ok, ok, j'le ferai plus...
*/ */
/*
* Merci :))
*/
/***************************************************************************** /*****************************************************************************
* Module descriptor * Module descriptor
...@@ -122,8 +122,6 @@ vlc_module_begin(); ...@@ -122,8 +122,6 @@ vlc_module_begin();
"force selection of all streams", "force selection of all streams",
"force selection of all streams" ); "force selection of all streams" );
// add_bool( "mms-modem", 0, NULL, "Set maxbitrate to 56Kb/s","" );
add_string( "mms-stream", NULL, NULL, add_string( "mms-stream", NULL, NULL,
"streams selection", "streams selection",
"force this stream selection" ); "force this stream selection" );
...@@ -152,8 +150,6 @@ static int Open( vlc_object_t *p_this ) ...@@ -152,8 +150,6 @@ static int Open( vlc_object_t *p_this )
(void*)p_access = malloc( sizeof( access_t ) ); (void*)p_access = malloc( sizeof( access_t ) );
memset( p_access, 0, sizeof( access_t ) ); memset( p_access, 0, sizeof( access_t ) );
p_access->p_cmd = malloc( BUF_SIZE );
/* *** Parse URL and get server addr/port and path *** */ /* *** Parse URL and get server addr/port and path *** */
mms_ParseURL( &p_access->url, p_input->psz_name ); mms_ParseURL( &p_access->url, p_input->psz_name );
...@@ -161,13 +157,16 @@ static int Open( vlc_object_t *p_this ) ...@@ -161,13 +157,16 @@ static int Open( vlc_object_t *p_this )
!( *p_access->url.psz_server_addr ) ) !( *p_access->url.psz_server_addr ) )
{ {
FREE( p_access->url.psz_private ); FREE( p_access->url.psz_private );
FREE( p_access->p_cmd );
msg_Err( p_input, "invalid server name" ); msg_Err( p_input, "invalid server name" );
return( -1 ); return( -1 );
} }
if( p_access->url.i_server_port == 0 ) if( p_access->url.i_server_port == 0 )
{ {
p_access->url.i_server_port = 1755; // default port p_access->url.i_server_port = 1755; /* default port */
}
if( p_access->url.i_bind_port == 0 )
{
p_access->url.i_bind_port = 7000; /* default port */
} }
...@@ -197,11 +196,11 @@ static int Open( vlc_object_t *p_this ) ...@@ -197,11 +196,11 @@ static int Open( vlc_object_t *p_this )
} }
/* 3: connect */ /* 3: connect */
if( i_proto == MMS_PROTO_AUTO ) if( i_proto == MMS_PROTO_AUTO )
{ // first try with TCP { /* first try with TCP */
i_status = i_status =
MMSOpen( p_input, &p_access->url, MMS_PROTO_TCP, psz_network ); MMSOpen( p_input, &p_access->url, MMS_PROTO_TCP, psz_network );
if( i_status < 0 ) if( i_status < 0 )
{ // then with UDP { /* then with UDP */
i_status = i_status =
MMSOpen( p_input, &p_access->url, MMS_PROTO_UDP, psz_network ); MMSOpen( p_input, &p_access->url, MMS_PROTO_UDP, psz_network );
} }
...@@ -215,16 +214,12 @@ static int Open( vlc_object_t *p_this ) ...@@ -215,16 +214,12 @@ static int Open( vlc_object_t *p_this )
if( i_status < 0 ) if( i_status < 0 )
{ {
// all sockets are closed
msg_Err( p_input, "cannot connect to server" ); msg_Err( p_input, "cannot connect to server" );
FREE( p_access->url.psz_private ); FREE( p_access->url.psz_private );
FREE( p_access->p_cmd );
return( -1 ); return( -1 );
} }
msg_Dbg( p_input, "connected to %s", p_access->url.psz_server_addr ); msg_Dbg( p_input, "connected to %s", p_access->url.psz_server_addr );
// all sockets are open
/* *** set exported functions *** */ /* *** set exported functions *** */
p_input->pf_read = Read; p_input->pf_read = Read;
...@@ -232,7 +227,7 @@ static int Open( vlc_object_t *p_this ) ...@@ -232,7 +227,7 @@ static int Open( vlc_object_t *p_this )
p_input->pf_set_program = SetProgram; p_input->pf_set_program = SetProgram;
p_input->pf_set_area = NULL; p_input->pf_set_area = NULL;
p_input->p_private = NULL; // XXX ?? p_input->p_private = NULL;
/* *** finished to set some variable *** */ /* *** finished to set some variable *** */
vlc_mutex_lock( &p_input->stream.stream_lock ); vlc_mutex_lock( &p_input->stream.stream_lock );
...@@ -261,7 +256,6 @@ static int Open( vlc_object_t *p_this ) ...@@ -261,7 +256,6 @@ static int Open( vlc_object_t *p_this )
msg_Err( p_input, "cannot start stream" ); msg_Err( p_input, "cannot start stream" );
MMSClose( p_input ); MMSClose( p_input );
FREE( p_access->url.psz_private ); FREE( p_access->url.psz_private );
FREE( p_access->p_cmd );
return( -1 ); return( -1 );
} }
...@@ -281,7 +275,6 @@ static void Close( vlc_object_t *p_this ) ...@@ -281,7 +275,6 @@ static void Close( vlc_object_t *p_this )
/* free memory */ /* free memory */
FREE( p_access->url.psz_private ); FREE( p_access->url.psz_private );
FREE( p_access->p_cmd );
} }
/***************************************************************************** /*****************************************************************************
...@@ -404,60 +397,6 @@ static int Read ( input_thread_t * p_input, byte_t * p_buffer, ...@@ -404,60 +397,6 @@ static int Read ( input_thread_t * p_input, byte_t * p_buffer,
return( i_data ); return( i_data );
} }
/*****************************************************************************
* NetRead: read on a file descriptor, checking b_die periodically
*****************************************************************************/
static ssize_t NetRead( input_thread_t * p_input, input_socket_t * p_socket,
byte_t * p_buffer, size_t i_len )
{
#ifdef UNDER_CE
return -1;
#else
struct timeval timeout;
fd_set fds;
int i_ret;
/* Initialize file descriptor set */
FD_ZERO( &fds );
FD_SET( p_socket->i_handle, &fds );
/* We'll wait 0.5 second if nothing happens */
timeout.tv_sec = 0;
timeout.tv_usec = 500000;
/* Find if some data is available */
i_ret = select( p_socket->i_handle + 1, &fds,
NULL, NULL, &timeout );
if( i_ret == -1 && errno != EINTR )
{
msg_Err( p_input, "network select error (%s)", strerror(errno) );
}
else if( i_ret > 0 )
{
ssize_t i_recv = recv( p_socket->i_handle, p_buffer, i_len, 0 );
if( i_recv > 0 )
{
vlc_mutex_lock( &p_input->stream.stream_lock );
p_input->stream.p_selected_area->i_tell += i_recv;
vlc_mutex_unlock( &p_input->stream.stream_lock );
}
if( i_recv < 0 )
{
msg_Err( p_input, "recv failed (%s)", strerror(errno) );
}
return i_recv;
}
return 0;
#endif
}
static void asf_HeaderParse( mms_stream_t stream[128], static void asf_HeaderParse( mms_stream_t stream[128],
uint8_t *p_header, int i_header ) uint8_t *p_header, int i_header )
{ {
...@@ -471,17 +410,21 @@ static void asf_HeaderParse( mms_stream_t stream[128], ...@@ -471,17 +410,21 @@ static void asf_HeaderParse( mms_stream_t stream[128],
stream[i].i_cat = MMS_STREAM_UNKNOWN; stream[i].i_cat = MMS_STREAM_UNKNOWN;
} }
// fprintf( stderr, " ---------------------header:%d\n", i_header );
var_buffer_initread( &buffer, p_header, i_header ); var_buffer_initread( &buffer, p_header, i_header );
var_buffer_getguid( &buffer, &guid ); var_buffer_getguid( &buffer, &guid );
if( !CmpGuid( &guid, &asf_object_header_guid ) ) if( !CmpGuid( &guid, &asf_object_header_guid ) )
{ {
// XXX Error // XXX Error
// fprintf( stderr, " ---------------------ERROR------\n" );
} }
var_buffer_getmemory( &buffer, NULL, 30 - 16 ); var_buffer_getmemory( &buffer, NULL, 30 - 16 );
for( ;; ) for( ;; )
{ {
// fprintf( stderr, " ---------------------data:%d\n", buffer.i_data );
if( var_buffer_readempty( &buffer ) ) if( var_buffer_readempty( &buffer ) )
{ {
return; return;
...@@ -493,13 +436,14 @@ static void asf_HeaderParse( mms_stream_t stream[128], ...@@ -493,13 +436,14 @@ static void asf_HeaderParse( mms_stream_t stream[128],
{ {
int i_stream_id; int i_stream_id;
guid_t stream_type; guid_t stream_type;
// msg_Dbg( p_input, "found stream_properties" ); // msg_Dbg( p_input, "found stream_properties" );
var_buffer_getguid( &buffer, &stream_type ); var_buffer_getguid( &buffer, &stream_type );
var_buffer_getmemory( &buffer, NULL, 32 ); var_buffer_getmemory( &buffer, NULL, 32 );
i_stream_id = var_buffer_get8( &buffer ) & 0x7f; i_stream_id = var_buffer_get8( &buffer ) & 0x7f;
var_buffer_getmemory( &buffer, NULL, i_size - 24 - 32 - 16 - 1 );
// fprintf( stderr, " 1---------------------skip:%lld\n", i_size - 24 - 32 - 16 - 1 );
var_buffer_getmemory( &buffer, NULL, i_size - 24 - 32 - 16 - 1);
if( CmpGuid( &stream_type, &asf_object_stream_type_video ) ) if( CmpGuid( &stream_type, &asf_object_stream_type_video ) )
{ {
...@@ -531,12 +475,14 @@ static void asf_HeaderParse( mms_stream_t stream[128], ...@@ -531,12 +475,14 @@ static void asf_HeaderParse( mms_stream_t stream[128],
i_count--; i_count--;
i_size -= 6; i_size -= 6;
} }
// fprintf( stderr, " 2---------------------skip:%lld\n", i_size - 24);
var_buffer_getmemory( &buffer, NULL, i_size - 24 ); var_buffer_getmemory( &buffer, NULL, i_size - 24 );
} }
else else
{ {
// skip unknown guid // skip unknown guid
var_buffer_getmemory( &buffer, NULL, i_size - 24 ); var_buffer_getmemory( &buffer, NULL, i_size - 24 );
// fprintf( stderr, " 3---------------------skip:%lld\n", i_size - 24);
} }
} }
} }
...@@ -731,40 +677,64 @@ static int MMSOpen( input_thread_t *p_input, ...@@ -731,40 +677,64 @@ static int MMSOpen( input_thread_t *p_input,
p_input->p_private = (void*)&socket_desc; p_input->p_private = (void*)&socket_desc;
if( !( p_network = module_Need( p_input, "network", psz_network ) ) ) if( !( p_network = module_Need( p_input, "network", psz_network ) ) )
{ {
msg_Err( p_input, "failed to open a connection" ); msg_Err( p_input, "failed to open a connection (tcp)" );
return( -1 ); return( -1 );
} }
module_Unneed( p_input, p_network ); module_Unneed( p_input, p_network );
p_access->socket_server.i_handle = socket_desc.i_handle; p_access->socket_tcp.i_handle = socket_desc.i_handle;
p_input->i_mtu = socket_desc.i_mtu; // FIXME p_input->i_mtu = 0; /*socket_desc.i_mtu;*/
msg_Dbg( p_input, msg_Dbg( p_input,
"connection with \"%s:%d\" successful", "connection(tcp) with \"%s:%d\" successful",
p_url->psz_server_addr, p_url->psz_server_addr,
p_url->i_server_port ); p_url->i_server_port );
/* *** Bind port if UDP protocol is selected *** */ /* *** Bind port if UDP protocol is selected *** */
// TODO
if( b_udp ) if( b_udp )
{ {
msg_Err( p_input, msg_Err( p_input,
"MMS/UDP not yet implemented" ); "MMS/UDP not yet functionnal, anyway trying..." );
// close socket if( !p_url->psz_bind_addr || !*p_url->psz_bind_addr )
{
msg_Err( p_input, "for udp you have to provide bind address (mms://<server_addr>@<bind_addr/<path> (FIXME)" );
#if defined( UNDER_CE )
CloseHandle( (HANDLE)p_access->socket_tcp.i_handle );
#elif defined( WIN32 )
closesocket( p_access->socket_tcp.i_handle );
#else
close( p_access->socket_tcp.i_handle );
#endif
return( -1 );
}
socket_desc.i_type = NETWORK_UDP;
socket_desc.psz_server_addr = "";
socket_desc.i_server_port = 0;
socket_desc.psz_bind_addr = p_url->psz_bind_addr;
socket_desc.i_bind_port = p_url->i_bind_port;
p_input->p_private = (void*)&socket_desc;
if( !( p_network = module_Need( p_input, "network", psz_network ) ) )
{
msg_Err( p_input, "failed to open a connection (udp)" );
#if defined( UNDER_CE ) #if defined( UNDER_CE )
CloseHandle( (HANDLE)p_access->socket_server.i_handle ); CloseHandle( (HANDLE)p_access->socket_tcp.i_handle );
#elif defined( WIN32 ) #elif defined( WIN32 )
closesocket( p_access->socket_server.i_handle ); closesocket( p_access->socket_tcp.i_handle );
#else #else
close( p_access->socket_server.i_handle ); close( p_access->socket_tcp.i_handle );
#endif #endif
return( -1 ); return( -1 );
} }
module_Unneed( p_input, p_network );
p_access->socket_udp.i_handle = socket_desc.i_handle;
p_input->i_mtu = 0;/*socket_desc.i_mtu; FIXME */
}
/* *** Init context for mms prototcol *** */ /* *** Init context for mms prototcol *** */
GenerateGuid( &p_access->guid ); // used to identify client by server GenerateGuid( &p_access->guid ); /* used to identify client by server */
msg_Dbg( p_input, msg_Dbg( p_input,
"generated guid: "GUID_FMT, "generated guid: "GUID_FMT,
GUID_PRINT( p_access->guid ) ); GUID_PRINT( p_access->guid ) );
p_access->i_command_level = 1; // updated after 0x1A command p_access->i_command_level = 1; /* updated after 0x1A command */
p_access->i_seq_num = 0; p_access->i_seq_num = 0;
p_access->i_media_packet_id_type = 0x04; p_access->i_media_packet_id_type = 0x04;
p_access->i_header_packet_id_type = 0x02; p_access->i_header_packet_id_type = 0x02;
...@@ -777,6 +747,10 @@ static int MMSOpen( input_thread_t *p_input, ...@@ -777,6 +747,10 @@ static int MMSOpen( input_thread_t *p_input,
p_access->i_media_used = 0; p_access->i_media_used = 0;
p_access->i_pos = 0; p_access->i_pos = 0;
p_access->i_buffer_tcp = 0;
p_access->i_buffer_udp = 0;
p_access->p_cmd = NULL;
p_access->i_cmd = 0;
/* *** send command 1 : connection request *** */ /* *** send command 1 : connection request *** */
var_buffer_initwrite( &buffer, 0 ); var_buffer_initwrite( &buffer, 0 );
...@@ -831,8 +805,17 @@ static int MMSOpen( input_thread_t *p_input, ...@@ -831,8 +805,17 @@ static int MMSOpen( input_thread_t *p_input,
var_buffer_add32( &buffer, 0x00000000 ); var_buffer_add32( &buffer, 0x00000000 );
var_buffer_add32( &buffer, 0x000a0000 ); var_buffer_add32( &buffer, 0x000a0000 );
var_buffer_add32( &buffer, 0x00000002 ); var_buffer_add32( &buffer, 0x00000002 );
// FIXME wrong for UDP FIXME if( b_udp )
sprintf( tmp, "\\\\127.0.0.1\\%s\\1242", b_udp ? "UDP" : "TCP" ); {
sprintf( tmp,
"\\\\%s\\UDP\\%d",
p_url->psz_bind_addr,
p_url->i_bind_port );
}
else
{
sprintf( tmp, "\\\\127.0.0.1\\TCP\\1242" );
}
var_buffer_addUTF16( &buffer, tmp ); var_buffer_addUTF16( &buffer, tmp );
var_buffer_add16( &buffer, '0' ); var_buffer_add16( &buffer, '0' );
...@@ -844,7 +827,7 @@ static int MMSOpen( input_thread_t *p_input, ...@@ -844,7 +827,7 @@ static int MMSOpen( input_thread_t *p_input,
buffer.i_data ); buffer.i_data );
/* *** response from server, should be 0x02 or 0x03 *** */ /* *** response from server, should be 0x02 or 0x03 *** */
mms_CommandRead( p_input, 0x02, 0 ); mms_CommandRead( p_input, 0x02, 0x03 );
if( p_access->i_command == 0x03 ) if( p_access->i_command == 0x03 )
{ {
msg_Err( p_input, msg_Err( p_input,
...@@ -861,7 +844,6 @@ static int MMSOpen( input_thread_t *p_input, ...@@ -861,7 +844,6 @@ static int MMSOpen( input_thread_t *p_input,
/* *** send command 5 : media file name/path requested *** */ /* *** send command 5 : media file name/path requested *** */
var_buffer_reinitwrite( &buffer, 0 ); var_buffer_reinitwrite( &buffer, 0 );
var_buffer_add64( &buffer, 0 ); var_buffer_add64( &buffer, 0 );
// var_buffer_addUTF16( &buffer, "/" );
var_buffer_addUTF16( &buffer, p_url->psz_path ); var_buffer_addUTF16( &buffer, p_url->psz_path );
mms_CommandSend( p_input, mms_CommandSend( p_input,
...@@ -878,7 +860,7 @@ static int MMSOpen( input_thread_t *p_input, ...@@ -878,7 +860,7 @@ static int MMSOpen( input_thread_t *p_input,
if( p_access->i_command == 0x1A ) if( p_access->i_command == 0x1A )
{ {
msg_Err( p_input, "id/password requested (not yet supported)" ); msg_Err( p_input, "id/password requested (not yet supported)" );
// FIXME /* FIXME */
var_buffer_free( &buffer ); var_buffer_free( &buffer );
MMSClose( p_input ); MMSClose( p_input );
return( -1 ); return( -1 );
...@@ -893,7 +875,7 @@ static int MMSOpen( input_thread_t *p_input, ...@@ -893,7 +875,7 @@ static int MMSOpen( input_thread_t *p_input,
return( -1 ); return( -1 );
} }
// 1 for file ok, 2 for authen ok /* 1 for file ok, 2 for authen ok */
switch( GetDWLE( p_access->p_cmd + MMS_CMD_HEADERSIZE ) ) switch( GetDWLE( p_access->p_cmd + MMS_CMD_HEADERSIZE ) )
{ {
case 0x0001: case 0x0001:
...@@ -933,6 +915,9 @@ static int MMSOpen( input_thread_t *p_input, ...@@ -933,6 +915,9 @@ static int MMSOpen( input_thread_t *p_input,
p_access->i_max_bit_rate, p_access->i_max_bit_rate,
p_access->i_header_size ); p_access->i_header_size );
/* XXX XXX dirty hack XXX XXX */
p_input->i_mtu = 3 * p_access->i_packet_length;
/* *** send command 15 *** */ /* *** send command 15 *** */
var_buffer_reinitwrite( &buffer, 0 ); var_buffer_reinitwrite( &buffer, 0 );
...@@ -968,15 +953,15 @@ static int MMSOpen( input_thread_t *p_input, ...@@ -968,15 +953,15 @@ static int MMSOpen( input_thread_t *p_input,
return( -1 ); return( -1 );
} }
/* *** parse header and get stream and their id *** */ /* *** parse header and get stream and their id *** */
// get all streams properties, /* get all streams properties,
// *
// TODO : stream bitrates properties(optional) * TODO : stream bitrates properties(optional)
// and bitrate mutual exclusion(optional) * and bitrate mutual exclusion(optional) */
asf_HeaderParse( p_access->stream, asf_HeaderParse( p_access->stream,
p_access->p_header, p_access->i_header ); p_access->p_header, p_access->i_header );
mms_StreamSelect( p_input, p_access->stream ); mms_StreamSelect( p_input, p_access->stream );
/* *** now select stream we want to receive *** */ /* *** now select stream we want to receive *** */
// TODO take care of stream bitrate TODO /* TODO take care of stream bitrate TODO */
i_streams = 0; i_streams = 0;
i_first = -1; i_first = -1;
var_buffer_reinitwrite( &buffer, 0 ); var_buffer_reinitwrite( &buffer, 0 );
...@@ -1061,9 +1046,8 @@ static int MMSStart ( input_thread_t *p_input, uint32_t i_packet ) ...@@ -1061,9 +1046,8 @@ static int MMSStart ( input_thread_t *p_input, uint32_t i_packet )
/* *** start stream from packet 0 *** */ /* *** start stream from packet 0 *** */
var_buffer_initwrite( &buffer, 0 ); var_buffer_initwrite( &buffer, 0 );
var_buffer_add64( &buffer, 0 ); // seek point in second var_buffer_add64( &buffer, 0 ); /* seek point in second */
var_buffer_add32( &buffer, 0xffffffff ); var_buffer_add32( &buffer, 0xffffffff );
// var_buffer_add32( &buffer, 0xffffffff ); // begin from start
var_buffer_add32( &buffer, i_packet ); // begin from start var_buffer_add32( &buffer, i_packet ); // begin from start
var_buffer_add8( &buffer, 0xff ); // stream time limit var_buffer_add8( &buffer, 0xff ); // stream time limit
var_buffer_add8( &buffer, 0xff ); // on 3bytes ... var_buffer_add8( &buffer, 0xff ); // on 3bytes ...
...@@ -1127,24 +1111,25 @@ static int MMSClose ( input_thread_t *p_input ) ...@@ -1127,24 +1111,25 @@ static int MMSClose ( input_thread_t *p_input )
NULL, 0 ); NULL, 0 );
/* *** close sockets *** */ /* *** close sockets *** */
#if defined( UNDER_CE ) #if defined( UNDER_CE )
CloseHandle( (HANDLE)p_access->socket_server.i_handle ); CloseHandle( (HANDLE)p_access->socket_tcp.i_handle );
#elif defined( WIN32 ) #elif defined( WIN32 )
closesocket( p_access->socket_server.i_handle ); closesocket( p_access->socket_tcp.i_handle );
#else #else
close( p_access->socket_server.i_handle ); close( p_access->socket_tcp.i_handle );
#endif #endif
if( p_access->i_proto == MMS_PROTO_UDP ) if( p_access->i_proto == MMS_PROTO_UDP )
{ {
#if defined( UNDER_CE ) #if defined( UNDER_CE )
CloseHandle( (HANDLE)p_access->socket_data.i_handle ); CloseHandle( (HANDLE)p_access->socket_udp.i_handle );
#elif defined( WIN32 ) #elif defined( WIN32 )
closesocket( p_access->socket_data.i_handle ); closesocket( p_access->socket_udp.i_handle );
#else #else
close( p_access->socket_data.i_handle ); close( p_access->socket_udp.i_handle );
#endif #endif
} }
FREE( p_access->p_cmd );
FREE( p_access->p_media ); FREE( p_access->p_media );
FREE( p_access->p_header ); FREE( p_access->p_header );
...@@ -1174,7 +1159,8 @@ static void mms_ParseURL( url_t *p_url, char *psz_url ) ...@@ -1174,7 +1159,8 @@ static void mms_ParseURL( url_t *p_url, char *psz_url )
} }
p_url->psz_server_addr = psz_parser; p_url->psz_server_addr = psz_parser;
while( *psz_parser && *psz_parser != ':' && *psz_parser != '/' ) while( *psz_parser &&
*psz_parser != ':' && *psz_parser != '/' && *psz_parser != '@' )
{ {
psz_parser++; psz_parser++;
} }
...@@ -1194,46 +1180,68 @@ static void mms_ParseURL( url_t *p_url, char *psz_url ) ...@@ -1194,46 +1180,68 @@ static void mms_ParseURL( url_t *p_url, char *psz_url )
{ {
psz_server_port = ""; psz_server_port = "";
} }
if( *psz_parser == '/' )
if( *psz_parser == '@' )
{ {
char *psz_bind_port;
*psz_parser = '\0'; *psz_parser = '\0';
psz_parser++; psz_parser++;
p_url->psz_path = psz_parser;
p_url->psz_bind_addr = psz_parser;
while( *psz_parser && *psz_parser != ':' && *psz_parser != '/' )
{
psz_parser++;
} }
if( *psz_server_port ) if( *psz_parser == ':' )
{ {
p_url->i_server_port = strtol( psz_server_port, &psz_parser, 10 ); *psz_parser = '\0';
psz_parser++;
psz_bind_port = psz_parser;
while( *psz_parser && *psz_parser != '/' )
{
psz_parser++;
}
} }
else else
{ {
p_url->i_server_port = 0; psz_bind_port = "";
}
if( *psz_bind_port )
{
p_url->i_bind_port = strtol( psz_bind_port, &psz_parser, 10 );
}
else
{
p_url->i_bind_port = 0;
}
}
else
{
p_url->psz_bind_addr = "";
p_url->i_bind_port = 0;
} }
}
static int mms_ReadData( input_thread_t *p_input,
uint8_t *p_data,
int i_data )
{
access_t *p_access = (access_t*)p_input->p_access_data;
int i_read;
while( i_data > 0 ) if( *psz_parser == '/' )
{ {
i_read = NetRead( p_input, &p_access->socket_server, p_data, i_data ); *psz_parser = '\0';
if( i_read < 0 ) psz_parser++;
p_url->psz_path = psz_parser;
}
if( *psz_server_port )
{ {
msg_Err( p_input, "failed to read data" ); p_url->i_server_port = strtol( psz_server_port, &psz_parser, 10 );
return( -1 );
} }
i_data -= i_read; else
p_data += i_read; {
p_url->i_server_port = 0;
} }
return( 0 );
} }
/**************************************************************************** /****************************************************************************
* *
* MMS specific functions * MMS specific functions
...@@ -1255,11 +1263,11 @@ static int mms_CommandSend( input_thread_t *p_input, ...@@ -1255,11 +1263,11 @@ static int mms_CommandSend( input_thread_t *p_input,
/* first init uffer */ /* first init uffer */
var_buffer_initwrite( &buffer, 0 ); var_buffer_initwrite( &buffer, 0 );
var_buffer_add32( &buffer, 0x00000001 ); // start sequence var_buffer_add32( &buffer, 0x00000001 ); /* start sequence */
var_buffer_add32( &buffer, 0xB00BFACE ); // ... var_buffer_add32( &buffer, 0xB00BFACE );
// size after protocol type /* size after protocol type */
var_buffer_add32( &buffer, i_data + MMS_CMD_HEADERSIZE - 16 ); var_buffer_add32( &buffer, i_data + MMS_CMD_HEADERSIZE - 16 );
var_buffer_add32( &buffer, 0x20534d4d ); // protocol "MMS " var_buffer_add32( &buffer, 0x20534d4d ); /* protocol "MMS " */
var_buffer_add32( &buffer, i_data_by8 + 4 ); var_buffer_add32( &buffer, i_data_by8 + 4 );
var_buffer_add32( &buffer, p_access->i_seq_num ); p_access->i_seq_num++; var_buffer_add32( &buffer, p_access->i_seq_num ); p_access->i_seq_num++;
var_buffer_add64( &buffer, 0 ); var_buffer_add64( &buffer, 0 );
...@@ -1275,7 +1283,7 @@ static int mms_CommandSend( input_thread_t *p_input, ...@@ -1275,7 +1283,7 @@ static int mms_CommandSend( input_thread_t *p_input,
} }
/* send it */ /* send it */
if( send( p_access->socket_server.i_handle, if( send( p_access->socket_tcp.i_handle,
buffer.p_data, buffer.p_data,
buffer.i_data, buffer.i_data,
0 ) == -1 ) 0 ) == -1 )
...@@ -1288,183 +1296,243 @@ static int mms_CommandSend( input_thread_t *p_input, ...@@ -1288,183 +1296,243 @@ static int mms_CommandSend( input_thread_t *p_input,
return( 0 ); return( 0 );
} }
static int mms_ReceiveCommand( input_thread_t *p_input ) static int NetFillBuffer( input_thread_t *p_input )
{ {
#define GET32( i_pos ) \ #ifdef UNDER_CE
( p_access->p_cmd[i_pos] + ( p_access->p_cmd[i_pos +1] << 8 ) + \ return -1;
( p_access->p_cmd[i_pos + 2] << 16 ) + \ #else
( p_access->p_cmd[i_pos + 3] << 24 ) )
access_t *p_access = (access_t*)p_input->p_access_data; access_t *p_access = (access_t*)p_input->p_access_data;
struct timeval timeout;
fd_set fds;
int i_ret;
do /* FIXME when using udp */
{ ssize_t i_tcp, i_udp;
int i_length; ssize_t i_tcp_read, i_udp_read;
// see for UDP mode int i_handle_max;
/* Initialize file descriptor set */
FD_ZERO( &fds );
i_tcp = MMS_BUFFER_SIZE/2 - p_access->i_buffer_tcp;
/* *** Read complete command *** */ if( p_access->i_proto == MMS_PROTO_UDP )
p_access->i_cmd = NetRead( p_input, &p_access->socket_server,
p_access->p_cmd, BUF_SIZE );
if( p_access->i_cmd < 12 )
{ {
msg_Warn( p_input, "failed to receive command" ); i_udp = MMS_BUFFER_SIZE/2 - p_access->i_buffer_udp;
p_access->i_command = 0;
return( -1 );
} }
i_length = GetDWLE( p_access->p_cmd + 8 ) + 16; else
if( i_length > p_access->i_cmd )
{
if( mms_ReadData( p_input,
p_access->p_cmd + p_access->i_cmd,
i_length - p_access->i_cmd ) < 0 )
{ {
msg_Warn( p_input, "failed to receive command" ); i_udp = 0; /* there isn't udp socket */
p_access->i_command = 0;
return( -1 );
}
} }
msg_Dbg( p_input, "received %d bytes", p_access->i_cmd ); i_handle_max = 0;
if( i_tcp > 0 )
p_access->i_command = GET32( 36 ) & 0xffff;
msg_Dbg( p_input,
"recv command start_sequence:0x%8.8x command_id:0x%8.8x length:%d len8:%d sequence 0x%8.8x len8_II:%d dir_comm:0x%8.8x",
GET32( 0 ),
GET32( 4 ),
GET32( 8 ),
GET32( 16 ),
GET32( 20 ),
GET32( 32 ),
GET32( 36 ),
GET32( 40 ) );
if( p_access->i_command == 0x1b )
{ {
mms_CommandSend( p_input, 0x1b, 0, 0, NULL, 0 ); FD_SET( p_access->socket_tcp.i_handle, &fds );
i_handle_max = __MAX( i_handle_max, p_access->socket_tcp.i_handle );
}
if( i_udp > 0 )
{
FD_SET( p_access->socket_udp.i_handle, &fds );
i_handle_max = __MAX( i_handle_max, p_access->socket_udp.i_handle );
} }
} while( p_access->i_command == 0x1b ); if( i_handle_max == 0 )
{
msg_Warn( p_input, "nothing to read %d:%d", i_tcp, i_udp );
return( 0 ); return( 0 );
} }
else
{
msg_Warn( p_input, "ask for tcp:%d udp:%d", i_tcp, i_udp );
}
#define MMS_RETRY_MAX 10 /* We'll wait 0.5 second if nothing happens */
#define MMS_RETRY_SLEEP 50000 timeout.tv_sec = 0;
timeout.tv_usec = 500000;
static int mms_CommandRead( input_thread_t *p_input, int i_command1, int i_command2 ) /* Find if some data is available */
{ i_ret = select( i_handle_max + 1,
access_t *p_access = (access_t*)p_input->p_access_data; &fds,
int i_count; NULL, NULL, &timeout );
int i_status;
for( i_count = 0; i_count < MMS_RETRY_MAX; ) if( i_ret == -1 && errno != EINTR )
{ {
msg_Err( p_input, "network select error (%s)", strerror(errno) );
return -1;
}
i_status = mms_ReceiveCommand( p_input ); if( i_tcp > 0 && FD_ISSET( p_access->socket_tcp.i_handle, &fds ) )
if( i_status < 0 || p_access->i_command == 0 )
{ {
i_count++; i_tcp_read =
msleep( MMS_RETRY_SLEEP ); recv( p_access->socket_tcp.i_handle,
p_access->buffer_tcp + p_access->i_buffer_tcp,
i_tcp + MMS_BUFFER_SIZE/2, 0 );
} }
else if( i_command1 == 0 && i_command2 == 0) else
{ {
return( 0 ); i_tcp_read = 0;
} }
else if( p_access->i_command == i_command1 || p_access->i_command == i_command2 )
if( i_udp > 0 && FD_ISSET( p_access->socket_udp.i_handle, &fds ) )
{ {
return( 0 ); i_udp_read = recv( p_access->socket_udp.i_handle,
p_access->buffer_udp + p_access->i_buffer_udp,
i_udp + MMS_BUFFER_SIZE/2, 0 );
} }
else else
{ {
switch( p_access->i_command ) i_udp_read = 0;
{
case 0x03:
msg_Warn( p_input, "socket closed by server" );
return( -1 );
case 0x1e:
msg_Warn( p_input, "end of media stream" );
return( -1 );
default:
break;
}
} }
}
msg_Warn( p_input, "failed to receive command (abording)" );
return( -1 );
}
p_access->i_buffer_tcp += i_tcp_read;
p_access->i_buffer_udp += i_udp_read;
msg_Dbg( p_input,
"filling TCP buffer with %d bytes (buffer:%d)",
i_tcp_read,
p_access->i_buffer_tcp );
if( p_access->i_proto == MMS_PROTO_UDP )
{
msg_Dbg( p_input,
"filling UDP buffer with %d bytes (buffer:%d)",
i_udp_read,
p_access->i_buffer_udp );
}
static int mms_ReceivePacket( input_thread_t *p_input ) return( i_tcp_read + i_udp_read);
#endif
}
static int mms_ParseCommand( input_thread_t *p_input,
uint8_t *p_data,
int i_data,
int *pi_used )
{ {
#define GET32( i_pos ) \
( p_access->p_cmd[i_pos] + ( p_access->p_cmd[i_pos +1] << 8 ) + \
( p_access->p_cmd[i_pos + 2] << 16 ) + \
( p_access->p_cmd[i_pos + 3] << 24 ) )
access_t *p_access = (access_t*)p_input->p_access_data; access_t *p_access = (access_t*)p_input->p_access_data;
uint8_t preheader[8]; int i_length;
int i_read; uint32_t i_id;
if( p_access->i_proto == MMS_PROTO_UDP ) if( p_access->p_cmd )
{ {
return( -1 ); free( p_access->p_cmd );
} }
else p_access->i_cmd = i_data;
p_access->p_cmd = malloc( i_data );
memcpy( p_access->p_cmd, p_data, i_data );
*pi_used = i_data; /* by default */
if( i_data < MMS_CMD_HEADERSIZE )
{ {
for( ;; ) msg_Warn( p_input, "truncated command (header incomplete)" );
p_access->i_command = 0;
return( -1 );
}
i_id = GetDWLE( p_data + 4 );
i_length = GetDWLE( p_data + 8 ) + 16;
if( i_id != 0xb00bface )
{ {
i_read = NetRead( p_input, &p_access->socket_server, preheader, 8 ); msg_Err( p_input,
if( i_read < 8 ) "incorrect command header (0x%x)", i_id );
p_access->i_command = 0;
return( -1 );
}
if( i_length > p_access->i_cmd )
{ {
msg_Warn( p_input, "cannot read preheader" ); msg_Warn( p_input,
"truncated command (missing %d bytes)",
i_length - i_data );
p_access->i_command = 0;
return( -1 ); return( -1 );
} }
/* preheader format : else if( i_length < p_access->i_cmd )
* u32 i_sequence_number
* u8 i_packet_id
* u8 i_udp_sequence/i_tcp_flags
* u16 i_length
*/
if( preheader[4] == p_access->i_header_packet_id_type ||
preheader[4] == p_access->i_media_packet_id_type ||
preheader[4] == 0xff )// udp timing pair
{ {
p_access->i_cmd = i_length;
*pi_used = i_length;
}
msg_Dbg( p_input,
"recv command start_sequence:0x%8.8x command_id:0x%8.8x length:%d len8:%d sequence 0x%8.8x len8_II:%d dir_comm:0x%8.8x",
GET32( 0 ),
GET32( 4 ),
GET32( 8 ),
GET32( 16 ),
GET32( 20 ),
GET32( 32 ),
GET32( 36 ),
GET32( 40 ) );
p_access->i_command = GET32( 36 ) & 0xffff;
return( MMS_PACKET_CMD );
}
static int mms_ParsePacket( input_thread_t *p_input,
uint8_t *p_data, int i_data,
int *pi_used )
{
access_t *p_access = (access_t*)p_input->p_access_data;
int i_packet_seq_num; int i_packet_seq_num;
int i_packet_length; int i_packet_length;
int i_packet_id; int i_packet_id;
uint8_t *p_packet; uint8_t *p_packet;
i_packet_seq_num = GetDWLE( preheader );
i_packet_length = GetWLE( preheader + 6 );
i_packet_id = preheader[4];
/* *** read complete packet *** */ *pi_used = i_data; /* default */
if( i_packet_length <= 8 ) if( i_data <= 8 )
{ {
msg_Err( p_input, msg_Warn( p_input, "truncated packet (header incomplete)" );
"empty or broken packet" );
return( -1 ); return( -1 );
} }
p_packet = malloc( i_packet_length - 8 );
if( mms_ReadData( p_input, i_packet_id = p_data[4];
p_packet, i_packet_seq_num = GetDWLE( p_data );
i_packet_length - 8 ) < 0 ) i_packet_length = GetWLE( p_data + 6 );
if( i_packet_length > i_data || i_packet_length <= 8)
{ {
msg_Err( p_input, msg_Warn( p_input,
"cannot read data" ); "truncated packet (missing %d bytes)",
i_packet_length - i_data );
return( -1 );
}
else if( i_packet_length < i_data )
{
*pi_used = i_packet_length;
} }
if( i_packet_id == 0xff ) if( i_packet_id == 0xff )
{ {
msg_Warn( p_input, msg_Warn( p_input,
"receive MMS UDP pair timing" ); "receive MMS UDP pair timing" );
free( p_packet );
return( MMS_PACKET_UDP_TIMING ); return( MMS_PACKET_UDP_TIMING );
} }
else
if( i_packet_id != p_access->i_header_packet_id_type &&
i_packet_id != p_access->i_media_packet_id_type )
{ {
msg_Warn( p_input, "incorrect Packet Id Type (0x%x)", i_packet_id );
return( -1 );
}
/* we now have a media or a header packet */
p_packet = malloc( i_packet_length - 8 ); // don't bother with preheader
memcpy( p_packet, p_data + 8, i_packet_length - 8 );
if( i_packet_seq_num != p_access->i_packet_seq_num ) if( i_packet_seq_num != p_access->i_packet_seq_num )
{ {
// FIXME for udp could be just wrong order ? /* FIXME for udp could be just wrong order ? */
msg_Warn( p_input, msg_Warn( p_input,
"detected packet lost (%d != %d)", "detected packet lost (%d != %d)",
i_packet_seq_num, i_packet_seq_num,
...@@ -1478,6 +1546,10 @@ static int mms_ReceivePacket( input_thread_t *p_input ) ...@@ -1478,6 +1546,10 @@ static int mms_ReceivePacket( input_thread_t *p_input )
FREE( p_access->p_header ); FREE( p_access->p_header );
p_access->p_header = p_packet; p_access->p_header = p_packet;
p_access->i_header = i_packet_length - 8; p_access->i_header = i_packet_length - 8;
msg_Dbg( p_input,
"receive header packet (%d bytes)",
i_packet_length - 8 );
return( MMS_PACKET_HEADER ); return( MMS_PACKET_HEADER );
} }
else else
...@@ -1486,51 +1558,199 @@ static int mms_ReceivePacket( input_thread_t *p_input ) ...@@ -1486,51 +1558,199 @@ static int mms_ReceivePacket( input_thread_t *p_input )
p_access->p_media = p_packet; p_access->p_media = p_packet;
p_access->i_media = i_packet_length - 8; p_access->i_media = i_packet_length - 8;
p_access->i_media_used = 0; p_access->i_media_used = 0;
msg_Dbg( p_input,
"receive media packet (%d bytes)",
i_packet_length - 8 );
return( MMS_PACKET_MEDIA ); return( MMS_PACKET_MEDIA );
} }
}
static int mms_ReceivePacket( input_thread_t *p_input )
{
access_t *p_access = (access_t*)p_input->p_access_data;
int i_packet_tcp_type;
int i_packet_udp_type;
for( ;; )
{
if( NetFillBuffer( p_input ) < 0 )
{
msg_Warn( p_input, "cannot fill buffer" );
continue;
} }
/* TODO udp */
i_packet_tcp_type = -1;
i_packet_udp_type = -1;
if( p_access->i_buffer_tcp > 0 )
{
int i_used;
if( GetDWLE( p_access->buffer_tcp + 4 ) == 0xb00bface )
{
i_packet_tcp_type =
mms_ParseCommand( p_input,
p_access->buffer_tcp,
p_access->i_buffer_tcp,
&i_used );
} }
else else
{ {
int i_packet_length; i_packet_tcp_type =
// command ? mms_ParsePacket( p_input,
if( GetDWLE( preheader + 4 ) != 0xb00bface ) p_access->buffer_tcp,
p_access->i_buffer_tcp,
&i_used );
}
if( i_used < MMS_BUFFER_SIZE )
{ {
msg_Err( p_input, memmove( p_access->buffer_tcp,
"incorrect command header (0x%x)", p_access->buffer_tcp + i_used,
GetDWLE( preheader + 4 ) ); MMS_BUFFER_SIZE - i_used );
}
p_access->i_buffer_tcp -= i_used;
} }
memcpy( p_access->p_cmd, preheader, 8 ); else if( p_access->i_buffer_udp > 0 )
if( mms_ReadData( p_input,
p_access->p_cmd + 8,
8 ) < 0 )
{ {
msg_Err( p_input, int i_used;
"cannot read data" ); #if 0
if( GetDWLE( p_access->buffer_tcp + 4 ) == 0xb00bface )
{
i_packet_tcp_type =
mms_ParseCommand( p_input,
p_access->buffer_tcp,
p_access->i_buffer_tcp,
&i_used );
} }
p_access->i_cmd = 16; else
i_packet_length = GetDWLE( p_access->p_cmd + 8 ); #endif
if( mms_ReadData( p_input,
p_access->p_cmd + 16,
i_packet_length ) < 0 )
{ {
msg_Err( p_input, i_packet_tcp_type =
"cannot read data" ); mms_ParsePacket( p_input,
p_access->buffer_udp,
p_access->i_buffer_udp,
&i_used );
}
if( i_used < MMS_BUFFER_SIZE )
{
memmove( p_access->buffer_udp,
p_access->buffer_udp + i_used,
MMS_BUFFER_SIZE - i_used );
}
p_access->i_buffer_udp -= i_used;
}
else
{
i_packet_udp_type = -1;
}
if( i_packet_tcp_type == MMS_PACKET_CMD &&
p_access->i_command == 0x1b )
{
mms_CommandSend( p_input, 0x1b, 0, 0, NULL, 0 );
i_packet_tcp_type = -1;
} }
p_access->i_cmd += i_packet_length;
p_access->i_command = GetDWLE( p_access->p_cmd + 36 ) & 0xffff; if( i_packet_tcp_type != -1 )
{
return( i_packet_tcp_type );
}
else if( i_packet_udp_type != -1 )
{
return( i_packet_udp_type );
}
}
}
static int mms_ReceiveCommand( input_thread_t *p_input )
{
access_t *p_access = (access_t*)p_input->p_access_data;
for( ;; )
{
int i_used;
int i_status;
NetFillBuffer( p_input );
i_status = mms_ParseCommand( p_input,
p_access->buffer_tcp,
p_access->i_buffer_tcp,
&i_used );
if( i_used < MMS_BUFFER_SIZE )
{
memmove( p_access->buffer_tcp,
p_access->buffer_tcp + i_used,
MMS_BUFFER_SIZE - i_used );
}
p_access->i_buffer_tcp -= i_used;
if( i_status < 0 )
{
return( -1 );
}
if( p_access->i_command == 0x1b ) if( p_access->i_command == 0x1b )
{ {
mms_CommandSend( p_input, 0x1b, 0, 0, NULL, 0 ); mms_CommandSend( p_input, 0x1b, 0, 0, NULL, 0 );
} }
else else
{ {
return( MMS_PACKET_CMD ); break;
}
} }
return( 0 );
}
#define MMS_RETRY_MAX 10
#define MMS_RETRY_SLEEP 50000
static int mms_CommandRead( input_thread_t *p_input, int i_command1, int i_command2 )
{
access_t *p_access = (access_t*)p_input->p_access_data;
int i_count;
int i_status;
for( i_count = 0; i_count < MMS_RETRY_MAX; )
{
i_status = mms_ReceiveCommand( p_input );
if( i_status < 0 || p_access->i_command == 0 )
{
i_count++;
msleep( MMS_RETRY_SLEEP );
}
else if( i_command1 == 0 && i_command2 == 0)
{
return( 0 );
}
else if( p_access->i_command == i_command1 || p_access->i_command == i_command2 )
{
return( 0 );
}
else
{
switch( p_access->i_command )
{
case 0x03:
msg_Warn( p_input, "socket closed by server" );
return( -1 );
case 0x1e:
msg_Warn( p_input, "end of media stream" );
return( -1 );
default:
break;
} }
} }
} }
msg_Warn( p_input, "failed to receive command (abording)" );
return( -1 );
} }
...@@ -1551,7 +1771,7 @@ static int mms_HeaderMediaRead( input_thread_t *p_input, int i_type ) ...@@ -1551,7 +1771,7 @@ static int mms_HeaderMediaRead( input_thread_t *p_input, int i_type )
"cannot receive header (%d/%d)", i_count, MMS_RETRY_MAX ); "cannot receive header (%d/%d)", i_count, MMS_RETRY_MAX );
msleep( MMS_RETRY_SLEEP ); msleep( MMS_RETRY_SLEEP );
} }
else if( i_status == i_type ) else if( i_status == i_type || i_type == MMS_PACKET_ANY )
{ {
return( 0 ); return( 0 );
} }
...@@ -1583,6 +1803,3 @@ static int mms_HeaderMediaRead( input_thread_t *p_input, int i_type ) ...@@ -1583,6 +1803,3 @@ static int mms_HeaderMediaRead( input_thread_t *p_input, int i_type )
return( -1 ); return( -1 );
} }
...@@ -2,7 +2,7 @@ ...@@ -2,7 +2,7 @@
* mms.h: MMS access plug-in * mms.h: MMS access plug-in
***************************************************************************** *****************************************************************************
* Copyright (C) 2001, 2002 VideoLAN * Copyright (C) 2001, 2002 VideoLAN
* $Id: mms.h,v 1.3 2002/11/13 20:28:13 fenrir Exp $ * $Id: mms.h,v 1.4 2002/11/25 00:22:04 fenrir Exp $
* *
* Authors: Laurent Aimar <fenrir@via.ecp.fr> * Authors: Laurent Aimar <fenrir@via.ecp.fr>
* *
...@@ -26,9 +26,13 @@ typedef struct url_s ...@@ -26,9 +26,13 @@ typedef struct url_s
{ {
char *psz_server_addr; char *psz_server_addr;
int i_server_port; int i_server_port;
char *psz_bind_addr;
int i_bind_port;
char *psz_path; char *psz_path;
// private /* private */
char *psz_private; char *psz_private;
} url_t; } url_t;
...@@ -39,10 +43,11 @@ typedef struct url_s ...@@ -39,10 +43,11 @@ typedef struct url_s
#define MMS_PROTO_TCP 1 #define MMS_PROTO_TCP 1
#define MMS_PROTO_UDP 2 #define MMS_PROTO_UDP 2
#define MMS_PACKET_CMD 0 #define MMS_PACKET_ANY 0
#define MMS_PACKET_HEADER 1 #define MMS_PACKET_CMD 1
#define MMS_PACKET_MEDIA 2 #define MMS_PACKET_HEADER 2
#define MMS_PACKET_UDP_TIMING 3 #define MMS_PACKET_MEDIA 3
#define MMS_PACKET_UDP_TIMING 4
#define MMS_STREAM_VIDEO 0x0001 #define MMS_STREAM_VIDEO 0x0001
...@@ -54,26 +59,33 @@ typedef struct url_s ...@@ -54,26 +59,33 @@ typedef struct url_s
typedef struct mms_stream_s typedef struct mms_stream_s
{ {
int i_id; // 1 -> 127 int i_id; /* 1 -> 127 */
int i_cat; // MMS_STREAM_VIDEO, MMS_STREAM_AUDIO int i_cat; /* MMS_STREAM_VIDEO, MMS_STREAM_AUDIO */
int i_bitrate; // -1 if unknown int i_bitrate; /* -1 if unknown */
int i_selected; int i_selected;
} mms_stream_t; } mms_stream_t;
#define MMS_BUFFER_SIZE 100000
typedef struct access_s typedef struct access_s
{ {
int i_proto; // MMS_PROTO_TCP, MMS_PROTO_UDP int i_proto; /* MMS_PROTO_TCP, MMS_PROTO_UDP */
input_socket_t socket_server; // TCP socket for communication with server input_socket_t socket_tcp; /* TCP socket for communication with server */
input_socket_t socket_data; // Optional UDP socket for data(media/header packet) input_socket_t socket_udp; /* Optional UDP socket for data(media/header packet) */
// send by server /* send by server */
url_t url; /* connect to this server */
mms_stream_t stream[128]; /* in asf never more than 1->127 streams */
url_t url; // connect to this server off_t i_pos; /* position of next byte to be read */
mms_stream_t stream[128]; //in asf never more than 1->127 streams /* */
uint8_t buffer_tcp[MMS_BUFFER_SIZE];
int i_buffer_tcp;
off_t i_pos; // position of next byte to be read uint8_t buffer_udp[MMS_BUFFER_SIZE];
int i_buffer_udp;
/* data necessary to send data to server */ /* data necessary to send data to server */
guid_t guid; guid_t guid;
...@@ -84,26 +96,26 @@ typedef struct access_s ...@@ -84,26 +96,26 @@ typedef struct access_s
int i_packet_seq_num; int i_packet_seq_num;
uint8_t *p_cmd; // latest command read uint8_t *p_cmd; /* latest command read */
int i_cmd; // allocated at the begining int i_cmd; /* allocated at the begining */
uint8_t *p_header; // allocated by mms_ReadPacket uint8_t *p_header; /* allocated by mms_ReadPacket */
int i_header; int i_header;
uint8_t *p_media; // allocated by mms_ReadPacket uint8_t *p_media; /* allocated by mms_ReadPacket */
int i_media; int i_media;
int i_media_used; int i_media_used;
// extracted informations /* extracted informations */
int i_command; int i_command;
// from 0x01 answer (not yet set) /* from 0x01 answer (not yet set) */
char *psz_server_version; char *psz_server_version;
char *psz_tool_version; char *psz_tool_version;
char *psz_update_player_url; char *psz_update_player_url;
char *psz_encryption_type; char *psz_encryption_type;
// from 0x06 answer /* from 0x06 answer */
uint32_t i_flags_broadcast; uint32_t i_flags_broadcast;
uint32_t i_media_length; uint32_t i_media_length;
int i_packet_length; int i_packet_length;
......
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