1 /*****************************************************************************
2 * asf.c: MMS access plug-in
3 *****************************************************************************
4 * Copyright (C) 2001-2004 the VideoLAN team
7 * Authors: Laurent Aimar <fenrir@via.ecp.fr>
9 * This program is free software; you can redistribute it and/or modify
10 * it under the terms of the GNU General Public License as published by
11 * the Free Software Foundation; either version 2 of the License, or
12 * (at your option) any later version.
14 * This program is distributed in the hope that it will be useful,
15 * but WITHOUT ANY WARRANTY; without even the implied warranty of
16 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
17 * GNU General Public License for more details.
19 * You should have received a copy of the GNU General Public License
20 * along with this program; if not, write to the Free Software
21 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA.
22 *****************************************************************************/
33 static int CmpGuid( const guid_t *p_guid1, const guid_t *p_guid2 )
35 return( ( p_guid1->v1 == p_guid2->v1 &&
36 p_guid1->v2 == p_guid2->v2 &&
37 p_guid1->v3 == p_guid2->v3 &&
38 p_guid1->v4[0] == p_guid2->v4[0] &&
39 p_guid1->v4[1] == p_guid2->v4[1] &&
40 p_guid1->v4[2] == p_guid2->v4[2] &&
41 p_guid1->v4[3] == p_guid2->v4[3] &&
42 p_guid1->v4[4] == p_guid2->v4[4] &&
43 p_guid1->v4[5] == p_guid2->v4[5] &&
44 p_guid1->v4[6] == p_guid2->v4[6] &&
45 p_guid1->v4[7] == p_guid2->v4[7] ) ? 1 : 0 );
48 void GenerateGuid ( guid_t *p_guid )
52 srand( mdate() & 0xffffffff );
54 /* FIXME should be generated using random data */
55 p_guid->v1 = 0xbabac001;
56 p_guid->v2 = ( (uint64_t)rand() << 16 ) / RAND_MAX;
57 p_guid->v3 = ( (uint64_t)rand() << 16 ) / RAND_MAX;
58 for( i = 0; i < 8; i++ )
60 p_guid->v4[i] = ( (uint64_t)rand() * 256 ) / RAND_MAX;
64 void asf_HeaderParse ( asf_header_t *hdr,
65 uint8_t *p_header, int i_header )
73 hdr->i_data_packets_count = 0;
74 hdr->i_min_data_packet_size = 0;
75 for( i = 0; i < 128; i++ )
77 hdr->stream[i].i_cat = ASF_STREAM_UNKNOWN;
78 hdr->stream[i].i_selected = 0;
79 hdr->stream[i].i_bitrate = -1;
82 var_buffer_initread( &buffer, p_header, i_header );
83 var_buffer_getguid( &buffer, &guid );
85 if( !CmpGuid( &guid, &asf_object_header_guid ) )
89 var_buffer_getmemory( &buffer, NULL, 30 - 16 );
93 var_buffer_getguid( &buffer, &guid );
94 i_size = var_buffer_get64( &buffer );
96 if( CmpGuid( &guid, &asf_object_file_properties_guid ) )
98 var_buffer_getmemory( &buffer, NULL, 16 );
99 hdr->i_file_size = var_buffer_get64( &buffer );
100 var_buffer_getmemory( &buffer, NULL, 8 );
101 hdr->i_data_packets_count = var_buffer_get64( &buffer );
102 var_buffer_getmemory( &buffer, NULL, 8+8+8+4);
103 hdr->i_min_data_packet_size = var_buffer_get32( &buffer );
105 var_buffer_getmemory( &buffer, NULL, i_size - 24 - 16 - 8 - 8 - 8 - 8-8-8-4 - 4);
107 else if( CmpGuid( &guid, &asf_object_header_extension_guid ) )
110 var_buffer_getmemory( &buffer, NULL, 46 - 24 );
112 else if( CmpGuid( &guid, &asf_object_extended_stream_properties_guid ) )
115 int16_t i_count1, i_count2;
119 var_buffer_getmemory( &buffer, NULL, 84 - 24 );
121 i_count1 = var_buffer_get16( &buffer );
122 i_count2 = var_buffer_get16( &buffer );
125 for( i = 0; i < i_count1; i++ )
129 var_buffer_get16( &buffer );
130 i_len = var_buffer_get16( &buffer );
131 var_buffer_getmemory( &buffer, NULL, i_len );
133 i_subsize = 4 + i_len;
136 for( i = 0; i < i_count2; i++ )
139 var_buffer_getmemory( &buffer, NULL, 16 + 2 );
140 i_len = var_buffer_get32( &buffer );
141 var_buffer_getmemory( &buffer, NULL, i_len );
143 i_subsize += 16 + 6 + i_len;
146 if( i_size - i_subsize <= 24 )
148 var_buffer_getmemory( &buffer, NULL, i_size - i_subsize );
150 /* It's a hack we just skip the first part of the object until
151 * the embed stream properties if any (ugly, but whose fault ?) */
153 else if( CmpGuid( &guid, &asf_object_stream_properties_guid ) )
158 var_buffer_getguid( &buffer, &stream_type );
159 var_buffer_getmemory( &buffer, NULL, 32 );
161 i_stream_id = var_buffer_get8( &buffer ) & 0x7f;
162 var_buffer_getmemory( &buffer, NULL, i_size - 24 - 32 - 16 - 1);
164 if( CmpGuid( &stream_type, &asf_object_stream_type_video ) )
166 hdr->stream[i_stream_id].i_cat = ASF_STREAM_VIDEO;
168 else if( CmpGuid( &stream_type, &asf_object_stream_type_audio ) )
170 hdr->stream[i_stream_id].i_cat = ASF_STREAM_AUDIO;
174 hdr->stream[i_stream_id].i_cat = ASF_STREAM_UNKNOWN;
177 else if ( CmpGuid( &guid, &asf_object_bitrate_properties_guid ) )
182 i_count = var_buffer_get16( &buffer );
186 i_stream_id = var_buffer_get16( &buffer )&0x7f;
187 hdr->stream[i_stream_id].i_bitrate = var_buffer_get32( &buffer );
191 var_buffer_getmemory( &buffer, NULL, i_size - 24 );
196 var_buffer_getmemory( &buffer, NULL, i_size - 24 );
199 if( var_buffer_readempty( &buffer ) )
204 void asf_StreamSelect ( asf_header_t *hdr,
206 bool b_all, bool b_audio, bool b_video )
208 /* XXX FIXME use mututal eclusion information */
210 int i_audio, i_video;
221 /* select all valid stream */
222 for( i = 1; i < 128; i++ )
224 if( hdr->stream[i].i_cat != ASF_STREAM_UNKNOWN )
226 hdr->stream[i].i_selected = 1;
233 for( i = 0; i < 128; i++ )
235 /* by default, not selected */
236 hdr->stream[i].i_selected = 0;
242 * - no audio nor video stream
244 * - if i_bitrate_max not set keep the highest bitrate
245 * - if i_bitrate_max is set, keep stream that make we used best
246 * quality regarding i_bitrate_max
249 * - it doesn't use mutual exclusion info..
250 * - when selecting a better stream we could select
251 * something that make i_bitrate_total> i_bitrate_max
253 for( i = 1; i < 128; i++ )
255 if( hdr->stream[i].i_cat == ASF_STREAM_UNKNOWN )
259 else if( hdr->stream[i].i_cat == ASF_STREAM_AUDIO && b_audio &&
261 ( ( ( hdr->stream[i].i_bitrate > hdr->stream[i_audio].i_bitrate &&
262 ( i_bitrate_total + hdr->stream[i].i_bitrate - hdr->stream[i_audio].i_bitrate
263 < i_bitrate_max || !i_bitrate_max) ) ||
264 ( hdr->stream[i].i_bitrate < hdr->stream[i_audio].i_bitrate &&
265 i_bitrate_max != 0 && i_bitrate_total > i_bitrate_max )
268 /* unselect old stream */
271 hdr->stream[i_audio].i_selected = 0;
272 if( hdr->stream[i_audio].i_bitrate> 0 )
274 i_bitrate_total -= hdr->stream[i_audio].i_bitrate;
278 hdr->stream[i].i_selected = 1;
279 if( hdr->stream[i].i_bitrate> 0 )
281 i_bitrate_total += hdr->stream[i].i_bitrate;
285 else if( hdr->stream[i].i_cat == ASF_STREAM_VIDEO && b_video &&
288 ( ( hdr->stream[i].i_bitrate > hdr->stream[i_video].i_bitrate &&
289 ( i_bitrate_total + hdr->stream[i].i_bitrate - hdr->stream[i_video].i_bitrate
290 < i_bitrate_max || !i_bitrate_max) ) ||
291 ( hdr->stream[i].i_bitrate < hdr->stream[i_video].i_bitrate &&
292 i_bitrate_max != 0 && i_bitrate_total > i_bitrate_max )
295 /* unselect old stream */
298 hdr->stream[i_video].i_selected = 0;
299 if( hdr->stream[i_video].i_bitrate> 0 )
301 i_bitrate_total -= hdr->stream[i_video].i_bitrate;
305 hdr->stream[i].i_selected = 1;
306 if( hdr->stream[i].i_bitrate> 0 )
308 i_bitrate_total += hdr->stream[i].i_bitrate;