X-Git-Url: https://git.sesse.net/?a=blobdiff_plain;f=src%2Finput%2Fmpeg_system.c;h=a630f8bb62e77946c9458020b45f79df74c9c258;hb=495436e7e14ef7a692124ce9ec8d1016a7dc18e1;hp=1ccfeec0dcd7783240fb0346327080755ade8d8f;hpb=4d26594b02a4b622db90d0367733afca3a1b6bab;p=vlc diff --git a/src/input/mpeg_system.c b/src/input/mpeg_system.c index 1ccfeec0dc..a630f8bb62 100644 --- a/src/input/mpeg_system.c +++ b/src/input/mpeg_system.c @@ -2,8 +2,13 @@ * mpeg_system.c: TS, PS and PES management ***************************************************************************** * Copyright (C) 1998, 1999, 2000 VideoLAN + * $Id: mpeg_system.c,v 1.54 2001/05/07 03:14:09 stef Exp $ * - * Authors: + * Authors: Christophe Massiot + * Michel Lespinasse + * BenoƮt Steiner + * Samuel Hocevar + * Henri Fallon * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by @@ -26,7 +31,8 @@ #include "defs.h" #include -#include +#include /* memcpy(), memset() */ +#include /* off_t */ #include "config.h" #include "common.h" @@ -40,76 +46,73 @@ #include "input_ext-dec.h" #include "input.h" - #include "mpeg_system.h" +#include "main.h" /* AC3/MPEG channel, SPU channel */ + /***************************************************************************** * Local prototypes *****************************************************************************/ +static void input_DecodePAT( input_thread_t *, es_descriptor_t *); +static void input_DecodePMT( input_thread_t *, es_descriptor_t *); /* * PES Packet management */ /***************************************************************************** - * input_DecodePES + * MoveChunk ***************************************************************************** - * Put a PES in the decoder's fifo. + * Small utility function used to parse discontinuous headers safely. Copies + * i_buf_len bytes of data to a buffer and returns the size copied. + * It also solves some alignment problems on non-IA-32, non-PPC processors. + * This is a variation on the theme of input_ext-dec.h:GetChunk(). *****************************************************************************/ -void input_DecodePES( input_thread_t * p_input, es_descriptor_t * p_es ) +static __inline__ size_t MoveChunk( byte_t * p_dest, + data_packet_t ** pp_data_src, + byte_t ** pp_src, + size_t i_buf_len ) { -#define p_pes (p_es->p_pes) - - /* FIXME: since we don't check the type of the stream anymore, we don't - * do the following : p_data->p_payload_start++; for DVD_SPU_ES, and - * DVD SPU support is BROKEN ! */ + ptrdiff_t i_available; - if( p_es->p_decoder_fifo != NULL ) + if( (i_available = (*pp_data_src)->p_payload_end - *pp_src) + >= i_buf_len ) { - vlc_mutex_lock( &p_es->p_decoder_fifo->data_lock ); + if( p_dest != NULL ) + memcpy( p_dest, *pp_src, i_buf_len ); + *pp_src += i_buf_len; + return( i_buf_len ); + } + else + { + size_t i_init_len = i_buf_len; -#if 0 - if( p_input->stream.b_pace_control ) + do { - /* FIXME : normally we shouldn't need this... */ - while( DECODER_FIFO_ISFULL( *p_es->p_decoder_fifo ) ) + if( p_dest != NULL ) + memcpy( p_dest, *pp_src, i_available ); + *pp_data_src = (*pp_data_src)->p_next; + i_buf_len -= i_available; + p_dest += i_available; + if( *pp_data_src == NULL ) { - vlc_mutex_unlock( &p_es->p_decoder_fifo->data_lock ); - msleep( 20000 ); - vlc_mutex_lock( &p_es->p_decoder_fifo->data_lock ); + *pp_src = NULL; + return( i_init_len - i_buf_len ); } + *pp_src = (*pp_data_src)->p_payload_start; } -#endif + while( (i_available = (*pp_data_src)->p_payload_end - *pp_src) + <= i_buf_len ); - if( !DECODER_FIFO_ISFULL( *p_es->p_decoder_fifo ) ) + if( i_buf_len ) { - //intf_DbgMsg("Putting %p into fifo %p/%d\n", - // p_pes, p_fifo, p_fifo->i_end); - p_es->p_decoder_fifo->buffer[p_es->p_decoder_fifo->i_end] = p_pes; - DECODER_FIFO_INCEND( *p_es->p_decoder_fifo ); - - /* Warn the decoder that it's got work to do. */ - vlc_cond_signal( &p_es->p_decoder_fifo->data_wait ); - } - else - { - /* The FIFO is full !!! This should not happen. */ - p_input->p_plugin->pf_delete_pes( p_input->p_method_data, p_pes ); - intf_ErrMsg( "PES trashed - fifo full ! (%d, %d)", - p_es->i_id, p_es->i_type); + if( p_dest != NULL ) + memcpy( p_dest, *pp_src, i_buf_len ); + *pp_src += i_buf_len; } - vlc_mutex_unlock( &p_es->p_decoder_fifo->data_lock ); - } - else - { - intf_ErrMsg("No fifo to receive PES %p (who wrote this damn code ?)", - p_pes); - p_input->p_plugin->pf_delete_pes( p_input->p_method_data, p_pes ); + return( i_init_len ); } - p_pes = NULL; - -#undef p_pes } /***************************************************************************** @@ -117,54 +120,38 @@ void input_DecodePES( input_thread_t * p_input, es_descriptor_t * p_es ) ***************************************************************************** * Parse a finished PES packet and analyze its header. *****************************************************************************/ -#define PES_HEADER_SIZE 14 +#define PES_HEADER_SIZE 7 void input_ParsePES( input_thread_t * p_input, es_descriptor_t * p_es ) { - data_packet_t * p_header_data; + data_packet_t * p_data; + byte_t * p_byte; byte_t p_header[PES_HEADER_SIZE]; - int i_done, i_todo; + int i_done; #define p_pes (p_es->p_pes) - //intf_DbgMsg("End of PES packet %p\n", p_pes); + //intf_DbgMsg("End of PES packet %p", p_pes); /* Parse the header. The header has a variable length, but in order * to improve the algorithm, we will read the 14 bytes we may be * interested in */ - p_header_data = p_pes->p_first; + p_data = p_pes->p_first; + p_byte = p_data->p_payload_start; i_done = 0; - for( ; ; ) - { - i_todo = p_header_data->p_payload_end - - p_header_data->p_payload_start; - if( i_todo > PES_HEADER_SIZE - i_done ) - i_todo = PES_HEADER_SIZE - i_done; - - memcpy( p_header + i_done, p_header_data->p_payload_start, - i_todo ); - i_done += i_todo; - - if( i_done < PES_HEADER_SIZE && p_header_data->p_next != NULL ) - { - p_header_data = p_header_data->p_next; - } - else - { - break; - } - } - - if( i_done != PES_HEADER_SIZE ) + if( MoveChunk( p_header, &p_data, &p_byte, PES_HEADER_SIZE ) + != PES_HEADER_SIZE ) { - intf_WarnMsg( 3, "PES packet too short to have a header" ); - p_input->p_plugin->pf_delete_pes( p_input->p_method_data, p_pes ); + intf_WarnMsg( 1, "PES packet too short to have a header" ); + p_input->pf_delete_pes( p_input->p_method_data, p_pes ); p_pes = NULL; return; } /* Get the PES size if defined */ - p_es->i_pes_real_size = U16_AT(p_header + 4) + 6; + p_es->i_pes_real_size = U16_AT(p_header + 4); + if( p_es->i_pes_real_size ) + p_es->i_pes_real_size += 6; /* First read the 6 header bytes common to all PES packets: * use them to test the PES validity */ @@ -172,7 +159,7 @@ void input_ParsePES( input_thread_t * p_input, es_descriptor_t * p_es ) { /* packet_start_code_prefix != 0x000001 */ intf_ErrMsg( "PES packet doesn't start with 0x000001 : data loss" ); - p_input->p_plugin->pf_delete_pes( p_input->p_method_data, p_pes ); + p_input->pf_delete_pes( p_input->p_method_data, p_pes ); p_pes = NULL; } else @@ -184,8 +171,7 @@ void input_ParsePES( input_thread_t * p_input, es_descriptor_t * p_es ) { /* PES_packet_length is set and != total received payload */ /* Warn the decoder that the data may be corrupt. */ - intf_WarnMsg( 3, "PES sizes do not match : packet corrupted" ); - p_pes->b_messed_up = 1; + intf_WarnMsg( 1, "PES sizes do not match : packet corrupted" ); } switch( p_es->i_stream_id ) @@ -207,51 +193,92 @@ void input_ParsePES( input_thread_t * p_input, es_descriptor_t * p_es ) if( (p_header[6] & 0xC0) == 0x80 ) { /* MPEG-2 : the PES header contains at least 3 more bytes. */ + size_t i_max_len; + boolean_t b_has_pts, b_has_dts; + byte_t p_full_header[12]; + p_pes->b_data_alignment = p_header[6] & 0x04; - p_pes->b_has_pts = p_header[7] & 0x80; - i_pes_header_size = p_header[8] + 9; - /* Now parse the optional header extensions (in the limit of - * the 14 bytes). */ - if( p_pes->b_has_pts ) + i_max_len = MoveChunk( p_full_header, &p_data, &p_byte, 12 ); + if( i_max_len < 2 ) { - p_pes->i_pts = - ( ((mtime_t)(p_header[9] & 0x0E) << 29) | - (((mtime_t)U16_AT(p_header + 10) << 14) - (1 << 14)) | - ((mtime_t)U16_AT(p_header + 12) >> 1) ) * 300; - p_pes->i_pts /= 27; + intf_WarnMsg( 1, + "PES packet too short to have a MPEG-2 header" ); + p_input->pf_delete_pes( p_input->p_method_data, + p_pes ); + p_pes = NULL; + return; + } + + b_has_pts = p_full_header[0] & 0x80; + b_has_dts = p_full_header[0] & 0x40; + i_pes_header_size = p_full_header[1] + 9; + + /* Now parse the optional header extensions */ + if( b_has_pts ) + { + if( i_max_len < 7 ) + { + intf_WarnMsg( 1, + "PES packet too short to have a MPEG-2 header" ); + p_input->pf_delete_pes( p_input->p_method_data, + p_pes ); + p_pes = NULL; + return; + } + p_pes->i_pts = input_ClockGetTS( p_input, p_es->p_pgrm, + ( ((mtime_t)(p_full_header[2] & 0x0E) << 29) | + ((mtime_t)(p_full_header[3]) << 22) | + ((mtime_t)(p_full_header[4] & 0xFE) << 14) | + ((mtime_t)p_full_header[5] << 7) | + ((mtime_t)p_full_header[6] >> 1) ) ); + + if( b_has_dts ) + { + if( i_max_len < 12 ) + { + intf_WarnMsg( 1, + "PES packet too short to have a MPEG-2 header" ); + p_input->pf_delete_pes( p_input->p_method_data, + p_pes ); + p_pes = NULL; + return; + } + p_pes->i_dts = input_ClockGetTS( p_input, p_es->p_pgrm, + ( ((mtime_t)(p_full_header[7] & 0x0E) << 29) | + (((mtime_t)U16_AT(p_full_header + 8) << 14) + - (1 << 14)) | + ((mtime_t)U16_AT(p_full_header + 10) >> 1) ) ); + } } } else { /* Probably MPEG-1 */ - byte_t * p_byte; - data_packet_t * p_data; + boolean_t b_has_pts, b_has_dts; i_pes_header_size = 6; p_data = p_pes->p_first; - p_byte = p_data->p_buffer + 6; - while( *p_byte == 0xFF && i_pes_header_size < 22 ) + p_byte = p_data->p_payload_start; + /* Cannot fail because the previous one succeeded. */ + MoveChunk( NULL, &p_data, &p_byte, 6 ); + + while( *p_byte == 0xFF && i_pes_header_size < 23 ) { i_pes_header_size++; - p_byte++; - if( p_byte >= p_data->p_payload_end ) + if( MoveChunk( NULL, &p_data, &p_byte, 1 ) != 1 ) { - p_data = p_data->p_next; - if( p_data == NULL ) - { - intf_ErrMsg( "MPEG-1 packet too short for header" ); - p_input->p_plugin->pf_delete_pes( p_input->p_method_data, p_pes ); - p_pes = NULL; - return; - } - p_byte = p_data->p_payload_start; + intf_WarnMsg( 1, + "PES packet too short to have a MPEG-1 header" ); + p_input->pf_delete_pes( p_input->p_method_data, p_pes ); + p_pes = NULL; + return; } } - if( i_pes_header_size == 22 ) + if( i_pes_header_size == 23 ) { intf_ErrMsg( "Too much MPEG-1 stuffing" ); - p_input->p_plugin->pf_delete_pes( p_input->p_method_data, p_pes ); + p_input->pf_delete_pes( p_input->p_method_data, p_pes ); p_pes = NULL; return; } @@ -259,130 +286,127 @@ void input_ParsePES( input_thread_t * p_input, es_descriptor_t * p_es ) if( (*p_byte & 0xC0) == 0x40 ) { /* Don't ask why... --Meuuh */ - p_byte += 2; + /* Erm... why ? --Sam */ + /* Well... According to the recommendation, it is for + * STD_buffer_scale and STD_buffer_size. --Meuuh */ i_pes_header_size += 2; - if( p_byte >= p_data->p_payload_end ) + if( MoveChunk( NULL, &p_data, &p_byte, 2 ) != 2 ) { - int i_plus = p_byte - p_data->p_payload_end; - p_data = p_data->p_next; - if( p_data == NULL ) - { - intf_ErrMsg( "MPEG-1 packet too short for header" ); - p_input->p_plugin->pf_delete_pes( p_input->p_method_data, p_pes ); - p_pes = NULL; - return; - } - p_byte = p_data->p_payload_start + i_plus; + intf_WarnMsg( 1, + "PES packet too short to have a MPEG-1 header" ); + p_input->pf_delete_pes( p_input->p_method_data, p_pes ); + p_pes = NULL; + return; } } i_pes_header_size++; - p_pes->b_has_pts = *p_byte & 0x20; - if( *p_byte & 0x10 ) - { - /* DTS */ - i_pes_header_size += 5; - } - if( *p_byte & 0x20 ) + b_has_pts = *p_byte & 0x20; + b_has_dts = *p_byte & 0x10; + + if( b_has_pts ) { - /* PTS */ - byte_t p_pts[5]; - int i; + byte_t p_ts[5]; i_pes_header_size += 4; - p_pts[0] = *p_byte; - for( i = 1; i < 5; i++ ) + if( MoveChunk( p_ts, &p_data, &p_byte, 5 ) != 5 ) + { + intf_WarnMsg( 1, + "PES packet too short to have a MPEG-1 header" ); + p_input->pf_delete_pes( p_input->p_method_data, p_pes ); + p_pes = NULL; + return; + } + + p_pes->i_pts = input_ClockGetTS( p_input, p_es->p_pgrm, + ( ((mtime_t)(p_ts[0] & 0x0E) << 29) | + (((mtime_t)U32_AT(p_ts) & 0xFFFE00) << 6) | + ((mtime_t)p_ts[3] << 7) | + ((mtime_t)p_ts[4] >> 1) ) ); + + if( b_has_dts ) { - p_byte++; - if( p_byte >= p_data->p_payload_end ) + i_pes_header_size += 5; + if( MoveChunk( p_ts, &p_data, &p_byte, 5 ) != 5 ) { - p_data = p_data->p_next; - if( p_data == NULL ) - { - intf_ErrMsg( "MPEG-1 packet too short for header" ); - p_input->p_plugin->pf_delete_pes( p_input->p_method_data, p_pes ); - p_pes = NULL; - return; - } - p_byte = p_data->p_payload_start; + intf_WarnMsg( 1, + "PES packet too short to have a MPEG-1 header" ); + p_input->pf_delete_pes( p_input->p_method_data, + p_pes ); + p_pes = NULL; + return; } - p_pts[i] = *p_byte; + p_pes->i_dts = input_ClockGetTS( p_input, + p_es->p_pgrm, + ( ((mtime_t)(p_ts[0] & 0x0E) << 29) | + (((mtime_t)U32_AT(p_ts) & 0xFFFE00) << 6) | + ((mtime_t)p_ts[3] << 7) | + ((mtime_t)p_ts[4] >> 1) ) ); } - p_pes->i_pts = - ( ((mtime_t)(p_pts[0] & 0x0E) << 29) | - (((mtime_t)U16_AT(p_pts + 1) << 14) - (1 << 14)) | - ((mtime_t)U16_AT(p_pts + 3) >> 1) ) * 300; - p_pes->i_pts /= 27; } } - /* PTS management */ - if( p_pes->b_has_pts ) - { - //intf_Msg("%lld\n", p_pes->i_pts); - switch( p_es->p_pgrm->i_synchro_state ) - { - case SYNCHRO_NOT_STARTED: - case SYNCHRO_START: - p_pes->b_has_pts = 0; - break; - - case SYNCHRO_REINIT: /* We skip a PES | Why ?? --Meuuh */ - p_pes->b_has_pts = 0; - p_es->p_pgrm->i_synchro_state = SYNCHRO_START; - break; - - case SYNCHRO_OK: - p_pes->i_pts += p_es->p_pgrm->delta_cr - + p_es->p_pgrm->delta_absolute - + DEFAULT_PTS_DELAY; - break; - } - } break; } + if( p_es->i_stream_id == 0xbd ) + { + /* With private stream 1, the first byte of the payload + * is a stream_private_id, so skip it. */ + i_pes_header_size++; + } + /* Now we've parsed the header, we just have to indicate in some * specific data packets where the PES payload begins (renumber * p_payload_start), so that the decoders can find the beginning * of their data right out of the box. */ - p_header_data = p_pes->p_first; - i_payload_size = p_header_data->p_payload_end - - p_header_data->p_payload_start; + p_data = p_pes->p_first; + i_payload_size = p_data->p_payload_end + - p_data->p_payload_start; while( i_pes_header_size > i_payload_size ) { /* These packets are entirely filled by the PES header. */ i_pes_header_size -= i_payload_size; - p_header_data->p_payload_start = p_header_data->p_payload_end; + p_data->p_payload_start = p_data->p_payload_end; /* Go to the next data packet. */ - if( (p_header_data = p_header_data->p_next) == NULL ) + if( (p_data = p_data->p_next) == NULL ) { intf_ErrMsg( "PES header bigger than payload" ); - p_input->p_plugin->pf_delete_pes( p_input->p_method_data, - p_pes ); + p_input->pf_delete_pes( p_input->p_method_data, p_pes ); p_pes = NULL; return; } - i_payload_size = p_header_data->p_payload_end - - p_header_data->p_payload_start; + i_payload_size = p_data->p_payload_end + - p_data->p_payload_start; } /* This last packet is partly header, partly payload. */ if( i_payload_size < i_pes_header_size ) { intf_ErrMsg( "PES header bigger than payload" ); - p_input->p_plugin->pf_delete_pes( p_input->p_method_data, p_pes ); + p_input->pf_delete_pes( p_input->p_method_data, p_pes ); p_pes = NULL; return; } - p_header_data->p_payload_start += i_pes_header_size; + p_data->p_payload_start += i_pes_header_size; /* Now we can eventually put the PES packet in the decoder's * PES fifo */ - input_DecodePES( p_input, p_es ); + if( p_es->p_decoder_fifo != NULL ) + { + input_DecodePES( p_es->p_decoder_fifo, p_pes ); + } + else + { + intf_ErrMsg("No fifo to receive PES %p (who wrote this damn code ?)", + p_pes); + p_input->pf_delete_pes( p_input->p_method_data, p_pes ); + } + p_pes = NULL; } #undef p_pes + } /***************************************************************************** @@ -396,52 +420,19 @@ void input_GatherPES( input_thread_t * p_input, data_packet_t * p_data, { #define p_pes (p_es->p_pes) - //intf_DbgMsg("PES-demultiplexing %p (%p)\n", p_ts_packet, p_pes); + //intf_DbgMsg("PES-demultiplexing %p (%p)", p_ts_packet, p_pes); - /* If we lost data, insert an NULL data packet (philosophy : 0 is quite + /* If we lost data, insert a NULL data packet (philosophy : 0 is quite * often an escape sequence in decoders, so that should make them wait * for the next start code). */ - if( b_packet_lost || p_es->b_discontinuity ) + if( b_packet_lost ) { - data_packet_t * p_pad_data; - - if( (p_pad_data = p_input->p_plugin->pf_new_packet( - p_input->p_method_data, - PADDING_PACKET_SIZE )) == NULL ) - { - intf_ErrMsg("Out of memory\n"); - p_input->b_error = 1; - return; - } - memset( p_data->p_buffer, 0, PADDING_PACKET_SIZE ); - p_pad_data->b_discard_payload = 1; - - if( p_pes != NULL ) - { - p_pes->b_messed_up = p_pes->b_discontinuity = 1; - input_GatherPES( p_input, p_pad_data, p_es, 0, 0 ); - } - else - { - if( (p_pes = p_input->p_plugin->pf_new_pes( - p_input->p_method_data )) == NULL ) - { - intf_ErrMsg("Out of memory\n"); - p_input->b_error = 1; - return; - } - - p_pes->p_first = p_pad_data; - p_pes->b_messed_up = p_pes->b_discontinuity = 1; - input_DecodePES( p_input, p_es ); - } - - p_es->b_discontinuity = 0; + input_NullPacket( p_input, p_es ); } if( b_unit_start && p_pes != NULL ) { - /* If the TS packet contains the begining of a new PES packet, and + /* If the data packet contains the begining of a new PES packet, and * if we were reassembling a PES packet, then the PES should be * complete now, so parse its header and give it to the decoders. */ input_ParsePES( p_input, p_es ); @@ -450,7 +441,7 @@ void input_GatherPES( input_thread_t * p_input, data_packet_t * p_data, if( !b_unit_start && p_pes == NULL ) { /* Random access... */ - p_input->p_plugin->pf_delete_packet( p_input->p_method_data, p_data ); + p_input->pf_delete_packet( p_input->p_method_data, p_data ); } else { @@ -461,15 +452,15 @@ void input_GatherPES( input_thread_t * p_input, data_packet_t * p_data, * packet. This is also here that we can synchronize with the * stream if we lost packets or if the decoder has just * started. */ - if( (p_pes = p_input->p_plugin->pf_new_pes( p_input->p_method_data ) ) == NULL ) + if( (p_pes = p_input->pf_new_pes( p_input->p_method_data ) ) == NULL ) { intf_ErrMsg("Out of memory"); p_input->b_error = 1; return; } - //intf_DbgMsg("New PES packet %p (first data: %p)\n", p_pes, p_data); + p_pes->i_rate = p_input->stream.control.i_rate; p_pes->p_first = p_data; - + /* If the PES header fits in the first data packet, we can * already set p_gather->i_pes_real_size. */ if( p_data->p_payload_end - p_data->p_payload_start @@ -477,11 +468,12 @@ void input_GatherPES( input_thread_t * p_input, data_packet_t * p_data, { p_es->i_pes_real_size = U16_AT(p_data->p_payload_start + 4) + 6; + } else - { + { p_es->i_pes_real_size = 0; - } + } } else { @@ -489,7 +481,6 @@ void input_GatherPES( input_thread_t * p_input, data_packet_t * p_data, p_es->p_last->p_next = p_data; } - p_data->p_next = NULL; p_es->p_last = p_data; /* Size of the payload carried in the data packet */ @@ -508,224 +499,292 @@ void input_GatherPES( input_thread_t * p_input, data_packet_t * p_data, /* - * Pace control - */ - -/* - * DISCUSSION : SYNCHRONIZATION METHOD - * - * In some cases we can impose the pace of reading (when reading from a - * file or a pipe), and for the synchronization we simply sleep() until - * it is time to deliver the packet to the decoders. When reading from - * the network, we must be read at the same pace as the server writes, - * otherwise the kernel's buffer will trash packets. The risk is now to - * overflow the input buffers in case the server goes too fast, that is - * why we do these calculations : - * - * We compute an average for the pcr because we want to eliminate the - * network jitter and keep the low frequency variations. The average is - * in fact a low pass filter and the jitter is a high frequency signal - * that is why it is eliminated by the filter/average. - * - * The low frequency variations enable us to synchronize the client clock - * with the server clock because they represent the time variation between - * the 2 clocks. Those variations (ie the filtered pcr) are used to compute - * the presentation dates for the audio and video frames. With those dates - * we can decode (or trash) the MPEG2 stream at "exactly" the same rate - * as it is sent by the server and so we keep the synchronization between - * the server and the client. - * - * It is a very important matter if you want to avoid underflow or overflow - * in all the FIFOs, but it may be not enough. + * PS Demultiplexing */ /***************************************************************************** - * Constants - *****************************************************************************/ - -/* Maximum number of samples used to compute the dynamic average value, - * it is also the maximum of c_average_count in pgrm_ts_data_t. - * We use the following formula : - * new_average = (old_average * c_average + new_sample_value) / (c_average +1) */ -#define CR_MAX_AVERAGE_COUNTER 40 - -/* Maximum gap allowed between two CRs. */ -#define CR_MAX_GAP 1000000 - -/***************************************************************************** - * CRReInit : Reinitialize the clock reference + * GetID: Get the ID of a stream *****************************************************************************/ -static void CRReInit( pgrm_descriptor_t * p_pgrm ) +static u16 GetID( data_packet_t * p_data ) { - p_pgrm->delta_cr = 0; - p_pgrm->last_cr = 0; - p_pgrm->c_average_count = 0; + u16 i_id; + + i_id = p_data->p_payload_start[3]; /* stream_id */ + if( i_id == 0xBD ) + { + /* FIXME : this is not valid if the header is split in multiple + * packets */ + /* stream_private_id */ + i_id |= p_data->p_payload_start[ 9 + p_data->p_payload_start[8] ] << 8; + } + return( i_id ); } -/* FIXME: find a better name */ /***************************************************************************** - * CRDecode : Decode a clock reference + * DecodePSM: Decode the Program Stream Map information + ***************************************************************************** + * FIXME : loads are not aligned in this function *****************************************************************************/ -static void CRDecode( input_thread_t * p_input, pgrm_descriptor_t * p_pgrm, - mtime_t cr_time ) +static void DecodePSM( input_thread_t * p_input, data_packet_t * p_data ) { - if( p_pgrm->i_synchro_state != SYNCHRO_OK ) + stream_ps_data_t * p_demux = + (stream_ps_data_t *)p_input->stream.p_demux_data; + byte_t * p_byte; + byte_t * p_end; + int i; + int i_new_es_number = 0; + + if( p_data->p_payload_start + 10 > p_data->p_payload_end ) { - switch( p_pgrm->i_synchro_state ) - { - case SYNCHRO_START: - p_pgrm->delta_absolute = mdate() - cr_time; - p_pgrm->i_synchro_state = SYNCHRO_OK; - break; + intf_ErrMsg( "PSM too short : packet corrupt" ); + return; + } - case SYNCHRO_NOT_STARTED: - p_pgrm->i_synchro_state = SYNCHRO_START; - break; + if( p_demux->b_has_PSM + && p_demux->i_PSM_version == (p_data->p_payload_start[6] & 0x1F) ) + { + /* Already got that one. */ + return; + } - default: - break; - } + intf_DbgMsg( "Building PSM" ); + p_demux->b_has_PSM = 1; + p_demux->i_PSM_version = p_data->p_payload_start[6] & 0x1F; + + /* Go to elementary_stream_map_length, jumping over + * program_stream_info. */ + p_byte = p_data->p_payload_start + 10 + + U16_AT(&p_data->p_payload_start[8]); + if( p_byte > p_data->p_payload_end ) + { + intf_ErrMsg( "PSM too short : packet corrupt" ); + return; } - else + /* This is the full size of the elementary_stream_map. + * 2 == elementary_stream_map_length + * Please note that CRC_32 is not included in the length. */ + p_end = p_byte + 2 + U16_AT(p_byte); + p_byte += 2; + if( p_end > p_data->p_payload_end ) { - if( p_pgrm->b_discontinuity || - ( p_pgrm->last_cr != 0 && - ( (p_pgrm->last_cr - cr_time) > CR_MAX_GAP - || (p_pgrm->last_cr - cr_time) < - CR_MAX_GAP ) ) ) - { - int i_es; + intf_ErrMsg( "PSM too short : packet corrupt" ); + return; + } - /* Stream discontinuity. */ - intf_WarnMsg( 3, "CR re-initialiazed" ); - CRReInit( p_pgrm ); - p_pgrm->i_synchro_state = SYNCHRO_REINIT; - p_pgrm->b_discontinuity = 0; + vlc_mutex_lock( &p_input->stream.stream_lock ); - /* Warn all the elementary streams */ - for( i_es = 0; i_es < p_pgrm->i_es_number; i_es++ ) + /* 4 == minimum useful size of a section */ + while( p_byte + 4 <= p_end ) + { + es_descriptor_t * p_es = NULL; + u8 i_stream_id = p_byte[1]; + /* FIXME: there will be a problem with private streams... (same + * stream_id) */ + + /* Look for the ES in the ES table */ + for( i = i_new_es_number; + i < p_input->stream.pp_programs[0]->i_es_number; + i++ ) + { + if( p_input->stream.pp_programs[0]->pp_es[i]->i_stream_id + == i_stream_id ) { - p_pgrm->pp_es[i_es]->b_discontinuity = 1; + p_es = p_input->stream.pp_programs[0]->pp_es[i]; + if( p_es->i_type != p_byte[0] ) + { + input_DelES( p_input, p_es ); + p_es = NULL; + } + else + { + /* Move the ES to the beginning. */ + p_input->stream.pp_programs[0]->pp_es[i] + = p_input->stream.pp_programs[0]->pp_es[ i_new_es_number ]; + p_input->stream.pp_programs[0]->pp_es[ i_new_es_number ] + = p_es; + i_new_es_number++; + } + break; } } - p_pgrm->last_cr = cr_time; - if( p_input->stream.b_pace_control ) + /* The goal is to have all the ES we have just read in the + * beginning of the pp_es table, and all the others at the end, + * so that we can close them more easily at the end. */ + if( p_es == NULL ) { - /* Wait a while before delivering the packets to the decoder. */ - mwait( cr_time + p_pgrm->delta_absolute ); + p_es = input_AddES( p_input, p_input->stream.pp_programs[0], + i_stream_id, 0 ); + p_es->i_type = p_byte[0]; + p_es->b_audio = ( p_es->i_type == MPEG1_AUDIO_ES + || p_es->i_type == MPEG2_AUDIO_ES + || p_es->i_type == AC3_AUDIO_ES + || p_es->i_type == LPCM_AUDIO_ES + ); + + /* input_AddES has inserted the new element at the end. */ + p_input->stream.pp_programs[0]->pp_es[ + p_input->stream.pp_programs[0]->i_es_number ] + = p_input->stream.pp_programs[0]->pp_es[ i_new_es_number ]; + p_input->stream.pp_programs[0]->pp_es[ i_new_es_number ] = p_es; + i_new_es_number++; } - else - { - mtime_t sys_time, delta_cr; - - sys_time = mdate(); - delta_cr = sys_time - cr_time; + p_byte += 4 + U16_AT(&p_byte[2]); + } - if( p_pgrm->c_average_count == CR_MAX_AVERAGE_COUNTER ) - { - p_pgrm->delta_cr = ( delta_cr + (p_pgrm->delta_cr - * (CR_MAX_AVERAGE_COUNTER - 1)) ) - / CR_MAX_AVERAGE_COUNTER; - } - else - { - p_pgrm->delta_cr = ( delta_cr + (p_pgrm->delta_cr - * p_pgrm->c_average_count) ) - / ( p_pgrm->c_average_count + 1 ); - p_pgrm->c_average_count++; - } - } + /* Un-select the streams that are no longer parts of the program. */ + for( i = i_new_es_number; + i < p_input->stream.pp_programs[0]->i_es_number; + i++ ) + { + /* We remove pp_es[i_new_es_member] and not pp_es[i] because the + * list will be emptied starting from the end */ + input_DelES( p_input, + p_input->stream.pp_programs[0]->pp_es[i_new_es_number] ); } -} +#ifdef STATS + intf_Msg( "input info: The stream map after the PSM is now :" ); + input_DumpStream( p_input ); +#endif -/* - * PS Demultiplexing - */ + vlc_mutex_unlock( &p_input->stream.stream_lock ); +} /***************************************************************************** - * DecodePSM: Decode the Program Stream Map information + * input_ParsePS: read the PS header *****************************************************************************/ -static void DecodePSM( input_thread_t * p_input, data_packet_t * p_data ) +es_descriptor_t * input_ParsePS( input_thread_t * p_input, + data_packet_t * p_data ) { - stream_ps_data_t * p_demux = - (stream_ps_data_t *)p_input->stream.p_demux_data; + u32 i_code; + es_descriptor_t * p_es = NULL; - if( !p_demux->b_is_PSM_complete ) + i_code = p_data->p_payload_start[3]; + + if( i_code > 0xBC ) /* ES start code */ { - byte_t * p_byte; - byte_t * p_end; - int i_es = 0; + u16 i_id; + int i_dummy; - intf_DbgMsg( "Building PSM" ); - if( p_data->p_payload_start + 10 > p_data->p_payload_end ) - { - intf_ErrMsg( "PSM too short : packet corrupt" ); - return; - } - /* Go to elementary_stream_map_length, jumping over - * program_stream_info. */ - p_byte = p_data->p_payload_start + 10 - + U16_AT(&p_data->p_payload_start[8]); - if( p_byte > p_data->p_payload_end ) - { - intf_ErrMsg( "PSM too short : packet corrupt" ); - return; - } - /* This is the full size of the elementary_stream_map. - * 2 == elementary_stream_map_length - * 4 == CRC_32 */ - p_end = p_byte + 2 + U16_AT(p_byte) - 4; - p_byte += 2; - if( p_end > p_data->p_payload_end ) - { - intf_ErrMsg( "PSM too short : packet corrupt" ); - return; - } + /* This is a PES packet. Find out if we want it or not. */ + i_id = GetID( p_data ); vlc_mutex_lock( &p_input->stream.stream_lock ); - - /* 4 == minimum useful size of a section */ - while( p_byte + 4 <= p_end ) + if( p_input->stream.pp_programs[0]->b_is_ok ) { - p_input->p_es[i_es].i_id - = p_input->p_es[i_es].i_stream_id - = p_byte[1]; - p_input->p_es[i_es].i_type = p_byte[0]; - p_input->p_es[i_es].p_pgrm = p_input->stream.pp_programs[0]; - p_input->p_es[i_es].p_pes = NULL; - p_byte += 4 + U16_AT(&p_byte[2]); - -#ifdef AUTO_SPAWN - switch( p_input->p_es[i_es].i_type ) + /* Look only at the selected ES. */ + for( i_dummy = 0; i_dummy < p_input->stream.i_selected_es_number; + i_dummy++ ) { - case MPEG1_AUDIO_ES: - case MPEG2_AUDIO_ES: - /* Spawn audio thread. */ - intf_DbgMsg( "Starting an MPEG-audio decoder" ); - break; - - case MPEG1_VIDEO_ES: - case MPEG2_VIDEO_ES: - /* Spawn video thread. */ - intf_DbgMsg( "Starting an MPEG-video decoder" ); + if( p_input->stream.pp_selected_es[i_dummy] != NULL + && p_input->stream.pp_selected_es[i_dummy]->i_id == i_id ) + { + p_es = p_input->stream.pp_selected_es[i_dummy]; break; + } } -#endif - - i_es++; } + else + { + stream_ps_data_t * p_demux = + (stream_ps_data_t *)p_input->stream.pp_programs[0]->p_demux_data; + /* Search all ES ; if not found -> AddES */ + p_es = input_FindES( p_input, i_id ); + + if( p_es == NULL && !p_demux->b_has_PSM ) + { + p_es = input_AddES( p_input, p_input->stream.pp_programs[0], + i_id, 0 ); + if( p_es != NULL ) + { + p_es->i_stream_id = p_data->p_payload_start[3]; + + /* Set stream type and auto-spawn. */ + if( (i_id & 0xF0) == 0xE0 ) + { + /* MPEG video */ + p_es->i_type = MPEG2_VIDEO_ES; + p_es->i_cat = VIDEO_ES; +#ifdef AUTO_SPAWN + if( !p_input->stream.b_seekable ) + input_SelectES( p_input, p_es ); +#endif + } + else if( (i_id & 0xE0) == 0xC0 ) + { + /* MPEG audio */ + p_es->i_type = MPEG2_AUDIO_ES; + p_es->b_audio = 1; + p_es->i_cat = AUDIO_ES; +#ifdef AUTO_SPAWN + if( !p_input->stream.b_seekable ) + if( main_GetIntVariable( INPUT_CHANNEL_VAR, 0 ) + == (p_es->i_id & 0x1F) ) + switch( main_GetIntVariable( INPUT_AUDIO_VAR, 0 ) ) + { + case 0: + main_PutIntVariable( INPUT_CHANNEL_VAR, + REQUESTED_MPEG ); + case REQUESTED_MPEG: + input_SelectES( p_input, p_es ); + } +#endif + } + else if( (i_id & 0xF0FF) == 0x80BD ) + { + /* AC3 audio (0x80->0x8F) */ + p_es->i_type = AC3_AUDIO_ES; + p_es->b_audio = 1; + p_es->i_cat = AUDIO_ES; +#ifdef AUTO_SPAWN + if( !p_input->stream.b_seekable ) + if( main_GetIntVariable( INPUT_CHANNEL_VAR, 0 ) + == ((p_es->i_id & 0xF00) >> 8) ) + switch( main_GetIntVariable( INPUT_AUDIO_VAR, 0 ) ) + { + case 0: + main_PutIntVariable( INPUT_CHANNEL_VAR, + REQUESTED_AC3 ); + case REQUESTED_AC3: + input_SelectES( p_input, p_es ); + } +#endif + } + else if( (i_id & 0xE0FF) == 0x20BD ) + { + /* Subtitles video (0x20->0x3F) */ + p_es->i_type = DVD_SPU_ES; + p_es->i_cat = SPU_ES; +#ifdef AUTO_SPAWN + if( main_GetIntVariable( INPUT_SUBTITLE_VAR, -1 ) + == ((p_es->i_id & 0x1F00) >> 8) ) + { + if( !p_input->stream.b_seekable ) + input_SelectES( p_input, p_es ); + } +#endif + } + else if( (i_id & 0xF0FF) == 0xA0BD ) + { + /* LPCM audio (0xA0->0xAF) */ + p_es->i_type = LPCM_AUDIO_ES; + p_es->b_audio = 1; + p_es->i_cat = AUDIO_ES; + /* FIXME : write the decoder */ + } + else + { + p_es->i_type = UNKNOWN_ES; + } + } + } + } /* stream.b_is_ok */ vlc_mutex_unlock( &p_input->stream.stream_lock ); - p_demux->i_PSM_version = p_data->p_buffer[6] & 0x1F; - p_demux->b_is_PSM_complete = 1; - } - else if( p_demux->i_PSM_version != (p_data->p_buffer[6] & 0x1F) ) - { - /* FIXME */ - intf_ErrMsg( "PSM changed, this is not supported yet !" ); - p_demux->i_PSM_version = p_data->p_buffer[6] & 0x1F; - } + } /* i_code > 0xBC */ + + return( p_es ); } /***************************************************************************** @@ -737,41 +796,76 @@ void input_DemuxPS( input_thread_t * p_input, data_packet_t * p_data ) boolean_t b_trash = 0; es_descriptor_t * p_es = NULL; - i_code = U32_AT( p_data->p_buffer ); - if( i_code >= 0x1B9 && i_code <= 0x1BC ) + i_code = U32_AT( p_data->p_payload_start ); + if( i_code <= 0x1BC ) { switch( i_code ) { case 0x1BA: /* PACK_START_CODE */ { - /* Convert the SCR in microseconds. */ + /* Read the SCR. */ mtime_t scr_time; + u32 i_mux_rate; - if( (p_data->p_buffer[4] & 0xC0) == 0x40 ) + if( (p_data->p_payload_start[4] & 0xC0) == 0x40 ) { /* MPEG-2 */ + byte_t p_header[14]; + byte_t * p_byte; + p_byte = p_data->p_payload_start; + + if( MoveChunk( p_header, &p_data, &p_byte, 14 ) != 14 ) + { + intf_WarnMsg( 1, "Packet too short to have a header" ); + b_trash = 1; + break; + } scr_time = - (( ((mtime_t)(p_data->p_buffer[4] & 0x38) << 27) | - ((mtime_t)(U32_AT(p_data->p_buffer + 4) & 0x03FFF800) + ((mtime_t)(p_header[4] & 0x38) << 27) | + ((mtime_t)(U32_AT(p_header + 4) & 0x03FFF800) << 4) | - ((mtime_t)(U32_AT(p_data->p_buffer + 6) & 0x03FFF800) - >> 11) - ) * 300) / 27; + ((( ((mtime_t)U16_AT(p_header + 6) << 16) + | (mtime_t)U16_AT(p_header + 8) ) & 0x03FFF800) + >> 11); + + /* mux_rate */ + i_mux_rate = ((u32)U16_AT(p_header + 10) << 6) + | (p_header[12] >> 2); } else { - /* MPEG-1 SCR is like PTS */ + /* MPEG-1 SCR is like PTS. */ + byte_t p_header[12]; + byte_t * p_byte; + p_byte = p_data->p_payload_start; + + if( MoveChunk( p_header, &p_data, &p_byte, 12 ) != 12 ) + { + intf_WarnMsg( 1, "Packet too short to have a header" ); + b_trash = 1; + break; + } scr_time = - (( ((mtime_t)(p_data->p_buffer[4] & 0x0E) << 29) | - (((mtime_t)U16_AT(p_data->p_buffer + 5) << 14) - - (1 << 14)) | - ((mtime_t)U16_AT(p_data->p_buffer + 7) >> 1) - ) * 300) / 27; + ((mtime_t)(p_header[4] & 0x0E) << 29) | + (((mtime_t)U32_AT(p_header + 4) & 0xFFFE00) << 6) | + ((mtime_t)p_header[7] << 7) | + ((mtime_t)p_header[8] >> 1); + + /* mux_rate */ + i_mux_rate = (U32_AT(p_header + 8) & 0x7FFFFE) >> 1; } /* Call the pace control. */ - //intf_Msg("+%lld\n", scr_time); - CRDecode( p_input, p_input->stream.pp_programs[0], - scr_time ); + input_ClockManageRef( p_input, p_input->stream.pp_programs[0], + scr_time ); + + if( i_mux_rate != p_input->stream.i_mux_rate + && p_input->stream.i_mux_rate ) + { + intf_WarnMsg(2, + "Mux_rate changed - expect cosmetic errors"); + } + p_input->stream.i_mux_rate = i_mux_rate; + b_trash = 1; } break; @@ -781,7 +875,6 @@ void input_DemuxPS( input_thread_t * p_input, data_packet_t * p_data ) break; case 0x1BC: /* PROGRAM_STREAM_MAP_CODE */ - intf_ErrMsg("meuuuuh\n"); DecodePSM( p_input, p_data ); b_trash = 1; break; @@ -793,163 +886,42 @@ void input_DemuxPS( input_thread_t * p_input, data_packet_t * p_data ) default: /* This should not happen */ b_trash = 1; - intf_WarnMsg( 1, "Unwanted packet received with start code %x", + intf_WarnMsg( 3, "Unwanted packet received with start code 0x%.8x", i_code ); } } else { - u16 i_id; - int i_dummy; - - /* This is a PES packet. Find out if we want it or not. */ - i_id = p_data->p_buffer[3]; /* ID of the stream. */ - - vlc_mutex_lock( &p_input->stream.stream_lock ); - for( i_dummy = 0; i_dummy < INPUT_MAX_ES; i_dummy++ ) - { - if( p_input->p_es[i_dummy].i_id == i_id ) - { - p_es = &p_input->p_es[i_dummy]; - break; - } - } - vlc_mutex_unlock( &p_input->stream.stream_lock ); + p_es = input_ParsePS( p_input, p_data ); - if( p_es == NULL ) - { -#if 1 - /* FIXME ! */ - if( (i_id & 0xC0L) == 0xC0L ) - { - /* MPEG video and audio */ - for( i_dummy = 0; i_dummy < INPUT_MAX_ES; i_dummy++ ) - { - if( p_input->p_es[i_dummy].i_id == EMPTY_ID ) - { - p_es = &p_input->p_es[i_dummy]; - break; - } - } - - if( p_es != NULL && (i_id & 0xF0L) == 0xE0L ) - { - /* MPEG video */ - vdec_config_t * p_config; - p_es->i_id = p_es->i_stream_id = i_id; - p_es->i_type = MPEG2_VIDEO_ES; - p_es->p_pgrm = p_input->stream.pp_programs[0]; - p_es->p_pes = NULL; - -#ifdef AUTO_SPAWN - p_config = (vdec_config_t *)malloc( sizeof(vdec_config_t) ); - p_config->p_vout = p_input->p_default_vout; - /* FIXME ! */ - p_config->decoder_config.i_stream_id = i_id; - p_config->decoder_config.i_type = MPEG2_VIDEO_ES; - p_config->decoder_config.p_stream_ctrl = - &p_input->stream.control; - p_config->decoder_config.p_decoder_fifo = - (decoder_fifo_t *)malloc( sizeof(decoder_fifo_t) ); - vlc_mutex_init(&p_config->decoder_config.p_decoder_fifo->data_lock); - vlc_cond_init(&p_config->decoder_config.p_decoder_fifo->data_wait); - p_config->decoder_config.p_decoder_fifo->i_start = - p_config->decoder_config.p_decoder_fifo->i_end = 0; - p_config->decoder_config.p_decoder_fifo->b_die = 0; - p_config->decoder_config.p_decoder_fifo->p_packets_mgt = - p_input->p_method_data; - p_config->decoder_config.p_decoder_fifo->pf_delete_pes = - p_input->p_plugin->pf_delete_pes; - p_es->p_decoder_fifo = p_config->decoder_config.p_decoder_fifo; - p_config->decoder_config.pf_init_bit_stream = - InitBitstream; - for( i_dummy = 0; i_dummy < INPUT_MAX_SELECTED_ES; i_dummy++ ) - { - if( p_input->pp_selected_es[i_dummy] == NULL ) - { - p_input->pp_selected_es[i_dummy] = p_es; - break; - } - } - - p_es->thread_id = vpar_CreateThread( p_config ); -#endif - } - else if( p_es != NULL && (i_id & 0xE0) == 0xC0 ) - { - /* MPEG audio */ - adec_config_t * p_config; - p_es->i_id = p_es->i_stream_id = i_id; - p_es->i_type = MPEG2_AUDIO_ES; - p_es->p_pgrm = p_input->stream.pp_programs[0]; - p_es->p_pes = NULL; - -#ifdef AUTO_SPAWN - p_config = (adec_config_t *)malloc( sizeof(adec_config_t) ); - p_config->p_aout = p_input->p_default_aout; - /* FIXME ! */ - p_config->decoder_config.i_stream_id = i_id; - p_config->decoder_config.i_type = MPEG2_AUDIO_ES; - p_config->decoder_config.p_stream_ctrl = - &p_input->stream.control; - p_config->decoder_config.p_decoder_fifo = - (decoder_fifo_t *)malloc( sizeof(decoder_fifo_t) ); - vlc_mutex_init(&p_config->decoder_config.p_decoder_fifo->data_lock); - vlc_cond_init(&p_config->decoder_config.p_decoder_fifo->data_wait); - p_config->decoder_config.p_decoder_fifo->i_start = - p_config->decoder_config.p_decoder_fifo->i_end = 0; - p_config->decoder_config.p_decoder_fifo->b_die = 0; - p_config->decoder_config.p_decoder_fifo->p_packets_mgt = - p_input->p_method_data; - p_config->decoder_config.p_decoder_fifo->pf_delete_pes = - p_input->p_plugin->pf_delete_pes; - p_es->p_decoder_fifo = p_config->decoder_config.p_decoder_fifo; - p_config->decoder_config.pf_init_bit_stream = - InitBitstream; - for( i_dummy = 0; i_dummy < INPUT_MAX_SELECTED_ES; i_dummy++ ) - { - if( p_input->pp_selected_es[i_dummy] == NULL ) - { - p_input->pp_selected_es[i_dummy] = p_es; - break; - } - } - - p_es->thread_id = adec_CreateThread( p_config ); -#endif - } - else - { - b_trash = 1; - } - } - else - b_trash = 1; -#else - b_trash = 1; -#endif - } - - if( p_es != NULL ) + vlc_mutex_lock( &p_input->stream.control.control_lock ); + if( p_es != NULL && p_es->p_decoder_fifo != NULL + && (!p_es->b_audio || !p_input->stream.control.b_mute) ) { + vlc_mutex_unlock( &p_input->stream.control.control_lock ); #ifdef STATS p_es->c_packets++; #endif input_GatherPES( p_input, p_data, p_es, 1, 0 ); } + else + { + vlc_mutex_unlock( &p_input->stream.control.control_lock ); + b_trash = 1; + } } /* Trash the packet if it has no payload or if it isn't selected */ if( b_trash ) { - p_input->p_plugin->pf_delete_packet( p_input, p_data ); + p_input->pf_delete_packet( p_input->p_method_data, p_data ); #ifdef STATS p_input->c_packets_trashed++; #endif } } - + /* * TS Demultiplexing */ @@ -959,21 +931,19 @@ void input_DemuxPS( input_thread_t * p_input, data_packet_t * p_data ) *****************************************************************************/ void input_DemuxTS( input_thread_t * p_input, data_packet_t * p_data ) { - int i_pid, i_dummy; + u16 i_pid; + int i_dummy; boolean_t b_adaptation; /* Adaptation field is present */ boolean_t b_payload; /* Packet carries payload */ boolean_t b_unit_start; /* A PSI or a PES start in the packet */ boolean_t b_trash = 0; /* Is the packet unuseful ? */ boolean_t b_lost = 0; /* Was there a packet loss ? */ + boolean_t b_psi = 0; /* Is this a PSI ? */ es_descriptor_t * p_es = NULL; es_ts_data_t * p_es_demux = NULL; pgrm_ts_data_t * p_pgrm_demux = NULL; -#define p (p_data->p_buffer) - - //intf_DbgMsg("input debug: TS-demultiplexing packet %p, pid %d\n", - // p_ts_packet, U16_AT(&p[1]) & 0x1fff); - + #define p (p_data->p_buffer) /* Extract flags values from TS common header. */ i_pid = U16_AT(&p[1]) & 0x1fff; b_unit_start = (p[1] & 0x40); @@ -982,34 +952,44 @@ void input_DemuxTS( input_thread_t * p_input, data_packet_t * p_data ) /* Find out the elementary stream. */ vlc_mutex_lock( &p_input->stream.stream_lock ); - for( i_dummy = 0; i_dummy < INPUT_MAX_ES; i_dummy++ ) + + p_es= input_FindES( p_input, i_pid ); + + if( (p_es != NULL) && (p_es->p_demux_data != NULL) ) { - if( p_input->p_es[i_dummy].i_id != EMPTY_ID ) - { - if( p_input->p_es[i_dummy].i_id == i_pid ) - { - p_es = &p_input->p_es[i_dummy]; - p_es_demux = (es_ts_data_t *)p_es->p_demux_data; - p_pgrm_demux = (pgrm_ts_data_t *)p_es->p_pgrm->p_demux_data; - break; - } - } + p_es_demux = (es_ts_data_t *)p_es->p_demux_data; + + if( p_es_demux->b_psi ) + b_psi = 1; + else + p_pgrm_demux = (pgrm_ts_data_t *)p_es->p_pgrm->p_demux_data; } - vlc_mutex_unlock( &p_input->stream.stream_lock ); - -#ifdef STATS - p_es->c_packets++; -#endif - if( p_es->p_decoder_fifo == NULL ) + vlc_mutex_lock( &p_input->stream.control.control_lock ); + if( ( p_es == NULL ) || (p_es->b_audio && p_input->stream.control.b_mute) ) { /* Not selected. Just read the adaptation field for a PCR. */ b_trash = 1; } + else if( p_es->p_decoder_fifo == NULL && !b_psi ) + b_trash =1; - if( (p_es->p_decoder_fifo != NULL) || (p_pgrm_demux->i_pcr_pid == i_pid) ) + vlc_mutex_unlock( &p_input->stream.control.control_lock ); + vlc_mutex_unlock( &p_input->stream.stream_lock ); + + + /* Don't change the order of the tests : if b_psi then p_pgrm_demux + * may still be null. Who said it was ugly ? */ + if( ( p_es != NULL ) && + ((p_es->p_decoder_fifo != NULL) || b_psi + || (p_pgrm_demux->i_pcr_pid == i_pid) ) ) { +#ifdef STATS + p_es->c_packets++; +#endif + /* Extract adaptation field information if any */ + if( !b_adaptation ) { /* We don't have any adaptation_field, so payload starts @@ -1049,15 +1029,15 @@ void input_DemuxTS( input_thread_t * p_input, data_packet_t * p_data ) if( p[5] & 0x80 ) { intf_WarnMsg( 2, - "discontinuity_indicator" \ - " encountered by TS demux (position read: %d," \ + "discontinuity_indicator" + " encountered by TS demux (position read: %d," " saved: %d)", p[5] & 0x80, p_es_demux->i_continuity_counter ); /* If the PID carries the PCR, there will be a system * time-based discontinuity. We let the PCR decoder * handle that. */ - p_es->p_pgrm->b_discontinuity = 1; + p_es->p_pgrm->i_synchro_state = SYNCHRO_REINIT; /* There also may be a continuity_counter * discontinuity: resynchronise our counter with @@ -1067,43 +1047,43 @@ void input_DemuxTS( input_thread_t * p_input, data_packet_t * p_data ) /* If this is a PCR_PID, and this TS packet contains a * PCR, we pass it along to the PCR decoder. */ - if( (p_pgrm_demux->i_pcr_pid == i_pid) && (p[5] & 0x10) ) + + if( !b_psi && (p_pgrm_demux->i_pcr_pid == i_pid) && (p[5] & 0x10) ) { /* There should be a PCR field in the packet, check * if the adaptation field is long enough to carry * it. */ if( p[4] >= 7 ) { - /* Convert the PCR in microseconds. - * WARNING: do not remove the casts in the - * following calculation ! */ + /* Read the PCR. */ mtime_t pcr_time; pcr_time = - ( (( (mtime_t)U32_AT((u32*)&p[6]) << 1 ) - | ( p[10] >> 7 )) * 300 ) / 27; + ( (mtime_t)U32_AT((u32*)&p[6]) << 1 ) + | ( p[10] >> 7 ); /* Call the pace control. */ - CRDecode( p_input, p_es->p_pgrm, pcr_time ); + input_ClockManageRef( p_input, p_es->p_pgrm, + pcr_time ); } } /* PCR ? */ } /* valid TS adaptation field ? */ } /* length > 0 */ } /* has adaptation field */ - /* Check the continuity of the stream. */ i_dummy = ((p[3] & 0x0f) - p_es_demux->i_continuity_counter) & 0x0f; if( i_dummy == 1 ) { /* Everything is ok, just increase our counter */ - p_es_demux->i_continuity_counter++; + (p_es_demux->i_continuity_counter)++; } else { if( !b_payload && i_dummy == 0 ) { - /* This is a packet without payload, this is allowed by the draft. - * As there is nothing interesting in this packet (except PCR that - * have already been handled), we can trash the packet. */ - intf_WarnMsg( 1, + /* This is a packet without payload, this is allowed by the + * draft. As there is nothing interesting in this packet + * (except PCR that have already been handled), we can trash + * the packet. */ + intf_WarnMsg( 3, "Packet without payload received by TS demux" ); b_trash = 1; } @@ -1111,15 +1091,15 @@ void input_DemuxTS( input_thread_t * p_input, data_packet_t * p_data ) { /* FIXME: this can never happen, can it ? --Meuuh */ /* Duplicate packet: mark it as being to be trashed. */ - intf_WarnMsg( 1, "Duplicate packet received by TS demux" ); + intf_WarnMsg( 3, "Duplicate packet received by TS demux" ); b_trash = 1; } else if( p_es_demux->i_continuity_counter == 0xFF ) { - /* This means that the packet is the first one we receive for this - * ES since the continuity counter ranges between 0 and 0x0F - * excepts when it has been initialized by the input: Init the - * counter to the correct value. */ + /* This means that the packet is the first one we receive for + * this ES since the continuity counter ranges between 0 and + * 0x0F excepts when it has been initialized by the input: + * init the counter to the correct value. */ intf_DbgMsg( "First packet for PID %d received by TS demux", p_es->i_id ); p_es_demux->i_continuity_counter = (p[3] & 0x0f); @@ -1127,11 +1107,11 @@ void input_DemuxTS( input_thread_t * p_input, data_packet_t * p_data ) else { /* This can indicate that we missed a packet or that the - * continuity_counter wrapped and we received a dup packet: as we - * don't know, do as if we missed a packet to be sure to recover - * from this situation */ + * continuity_counter wrapped and we received a dup packet: + * as we don't know, do as if we missed a packet to be sure + * to recover from this situation */ intf_WarnMsg( 2, - "Packet lost by TS demux: current %d, packet %d\n", + "Packet lost by TS demux: current %d, packet %d", p_es_demux->i_continuity_counter & 0x0f, p[3] & 0x0f ); b_lost = 1; @@ -1139,32 +1119,420 @@ void input_DemuxTS( input_thread_t * p_input, data_packet_t * p_data ) } /* not continuous */ } /* continuity */ } /* if selected or PCR */ - + /* Trash the packet if it has no payload or if it isn't selected */ if( b_trash ) { - p_input->p_plugin->pf_delete_packet( p_input, p_data ); + p_input->pf_delete_packet( p_input->p_method_data, p_data ); #ifdef STATS p_input->c_packets_trashed++; #endif } else { - if( p_es_demux->b_psi ) + if( b_psi ) { /* The payload contains PSI tables */ -#if 0 input_DemuxPSI( p_input, p_data, p_es, b_unit_start, b_lost ); -#endif + } else { /* The payload carries a PES stream */ - if( b_unit_start ) - input_GatherPES( p_input, p_data, p_es, b_unit_start, b_lost ); + input_GatherPES( p_input, p_data, p_es, b_unit_start, b_lost ); + } + + } + +#undef p + +} + +/* + * PSI demultiplexing and decoding + */ + +/***************************************************************************** + * DemuxPSI : makes up complete PSI data + *****************************************************************************/ +void input_DemuxPSI( input_thread_t * p_input, data_packet_t * p_data, + es_descriptor_t * p_es, boolean_t b_unit_start, boolean_t b_lost ) +{ + es_ts_data_t * p_demux_data; + + p_demux_data = (es_ts_data_t *)p_es->p_demux_data; + +#define p_psi (p_demux_data->p_psi_section) +#define p (p_data->p_payload_start) + + if( b_unit_start ) + { + /* unit_start set to 1 -> presence of a pointer field + * (see ISO/IEC 13818 (2.4.4.2) which should be set to 0x00 */ + if( (u8)p[0] != 0x00 ) + { + intf_WarnMsg( 2, + "Non zero pointer field found. Trying to continue" ); + p+=(u8)p[0]; + } + else + p++; + + /* This is the begining of a new section */ + + if( ((u8)(p[1]) & 0xc0) != 0x80 ) + { + intf_WarnMsg( 2, "Invalid PSI packet" ); + p_psi->b_trash = 1; + } + else + { + p_psi->i_section_length = U16_AT(p+1) & 0x0fff; + p_psi->b_section_complete = 0; + p_psi->i_read_in_section = 0; + p_psi->i_section_number = (u8)p[6]; + + if( p_psi->b_is_complete || p_psi->i_section_number == 0 ) + { + /* This is a new PSI packet */ + p_psi->b_is_complete = 0; + p_psi->b_trash = 0; + p_psi->i_version_number = ( p[5] >> 1 ) & 0x1f; + p_psi->i_last_section_number = (u8)p[7]; + + /* We'll write at the begining of the buffer */ + p_psi->p_current = p_psi->buffer; + } + else + { + if( p_psi->b_section_complete ) + { + /* New Section of an already started PSI */ + p_psi->b_section_complete = 0; + + if( p_psi->i_version_number != (( p[5] >> 1 ) & 0x1f) ) + { + intf_WarnMsg( 2,"PSI version differs inside same PAT" ); + p_psi->b_trash = 1; + } + if( p_psi->i_section_number + 1 != (u8)p[6] ) + { + intf_WarnMsg( 2, + "PSI Section discontinuity. Packet lost ?"); + p_psi->b_trash = 1; + } + else + p_psi->i_section_number++; + } + else + { + intf_WarnMsg( 2, "Received unexpected new PSI section" ); + p_psi->b_trash = 1; + } + } + } + } /* b_unit_start */ + + if( !p_psi->b_trash ) + { + /* read */ + if( (p_data->p_payload_end - p) >= + ( p_psi->i_section_length - p_psi->i_read_in_section ) ) + { + /* The end of the section is in this TS packet */ + memcpy( p_psi->p_current, p, + (p_psi->i_section_length - p_psi->i_read_in_section) ); + + p_psi->b_section_complete = 1; + p_psi->p_current += + (p_psi->i_section_length - p_psi->i_read_in_section); + + if( p_psi->i_section_number == p_psi->i_last_section_number ) + { + /* This was the last section of PSI */ + p_psi->b_is_complete = 1; + } + } + else + { + memcpy( p_psi->buffer, p, p_data->p_payload_end - p ); + p_psi->i_read_in_section+= p_data->p_payload_end - p; + + p_psi->p_current += p_data->p_payload_end - p; } } + if ( p_psi->b_is_complete ) + { + switch( p_demux_data->i_psi_type) + { + case PSI_IS_PAT: + input_DecodePAT( p_input, p_es ); + break; + case PSI_IS_PMT: + input_DecodePMT( p_input, p_es ); + break; + default: + intf_WarnMsg(2, "Received unknown PSI in demuxPSI"); + } + } +#undef p_psi #undef p + + return ; +} + +/***************************************************************************** + * DecodePAT : Decodes Programm association table and deal with it + *****************************************************************************/ +static void input_DecodePAT( input_thread_t * p_input, es_descriptor_t * p_es ) +{ + + stream_ts_data_t * p_stream_data; + es_ts_data_t * p_demux_data; + + p_demux_data = (es_ts_data_t *)p_es->p_demux_data; + p_stream_data = (stream_ts_data_t *)p_input->stream.p_demux_data; + +#define p_psi (p_demux_data->p_psi_section) + + if( p_stream_data->i_pat_version != p_psi->i_version_number ) + { + /* PAT has changed. We are going to delete all programms and + * create new ones. We chose not to only change what was needed + * as a PAT change may mean the stream is radically changing and + * this is a secure method to avoid krashes */ + pgrm_descriptor_t * p_pgrm; + es_descriptor_t * p_current_es; + es_ts_data_t * p_es_demux; + pgrm_ts_data_t * p_pgrm_demux; + byte_t * p_current_data; + + int i_section_length,i_program_id,i_pmt_pid; + int i_loop, i_current_section; + + p_current_data = p_psi->buffer; + + + for( i_loop = 0; i_loop < p_input->stream.i_pgrm_number; i_loop++ ) + { + input_DelProgram( p_input, p_input->stream.pp_programs[i_loop] ); + } + + do + { + i_section_length = U16_AT(p_current_data+1) & 0x0fff; + i_current_section = (u8)p_current_data[6]; + + for( i_loop = 0; i_loop < (i_section_length-9)/4 ; i_loop++ ) + { + i_program_id = U16_AT(p_current_data + i_loop*4 + 8); + i_pmt_pid = U16_AT( p_current_data + i_loop*4 + 10) & 0x1fff; + + /* If program = 0, we're having info about NIT not PMT */ + if( i_program_id ) + { + /* Add this program */ + p_pgrm = input_AddProgram( p_input, i_program_id, + sizeof( pgrm_ts_data_t ) ); + + /* whatis the PID of the PMT of this program */ + p_pgrm_demux = (pgrm_ts_data_t *)p_pgrm->p_demux_data; + p_pgrm_demux->i_pmt_version = PMT_UNINITIALIZED; + + /* Add the PMT ES to this program */ + p_current_es = input_AddES( p_input, p_pgrm,(u16)i_pmt_pid, + sizeof( es_ts_data_t) ); + p_es_demux = (es_ts_data_t *)p_current_es->p_demux_data; + p_es_demux->b_psi = 1; + p_es_demux->i_psi_type = PSI_IS_PMT; + + p_es_demux->p_psi_section = + malloc( sizeof( psi_section_t ) ); + p_es_demux->p_psi_section->b_is_complete = 0; + } + } + + p_current_data+=3+i_section_length; + + } while( i_current_section < p_psi->i_last_section_number ); + + /* Go to the beginning of the next section*/ + p_stream_data->i_pat_version = p_psi->i_version_number; + + } +#undef p_psi + +} + +/***************************************************************************** + * DecodePMT : decode a given Program Stream Map + * *************************************************************************** + * When the PMT changes, it may mean a deep change in the stream, and it is + * careful to delete the ES and add them again. If the PMT doesn't change, + * there no need to do anything. + *****************************************************************************/ +static void input_DecodePMT( input_thread_t * p_input, es_descriptor_t * p_es ) +{ + + pgrm_ts_data_t * p_pgrm_data; + es_ts_data_t * p_demux_data; + + p_demux_data = (es_ts_data_t *)p_es->p_demux_data; + p_pgrm_data = (pgrm_ts_data_t *)p_es->p_pgrm->p_demux_data; + +#define p_psi (p_demux_data->p_psi_section) + + if( p_psi->i_version_number != p_pgrm_data->i_pmt_version ) + { + es_descriptor_t * p_new_es; + es_ts_data_t * p_es_demux; + byte_t * p_current_data, * p_current_section; + int i_section_length,i_current_section; + int i_prog_info_length, i_loop; + int i_es_info_length, i_pid, i_stream_type; + int i_audio_es, i_spu_es; + int i_required_audio_es, i_required_spu_es; + + p_current_section = p_psi->buffer; + p_current_data = p_psi->buffer; + + p_pgrm_data->i_pcr_pid = U16_AT(p_current_section + 8) & 0x1fff; + + i_audio_es = 0; + i_spu_es = 0; + + /* Lock stream information */ + vlc_mutex_lock( &p_input->stream.stream_lock ); + + /* Get the number of the required audio stream */ + if( p_main->b_audio ) + { + /* Default is the first one */ + i_required_audio_es = main_GetIntVariable( INPUT_CHANNEL_VAR, 1 ); + if( i_required_audio_es < 0 ) + { + main_PutIntVariable( INPUT_CHANNEL_VAR, 1 ); + i_required_audio_es = 1; + } + } + else + { + i_required_audio_es = 0; + } + + /* Same thing for subtitles */ + if( p_main->b_video ) + { + /* for spu, default is none */ + i_required_spu_es = main_GetIntVariable( INPUT_SUBTITLE_VAR, 0 ); + if( i_required_spu_es < 0 ) + { + main_PutIntVariable( INPUT_SUBTITLE_VAR, 0 ); + i_required_spu_es = 0; + } + } + else + { + i_required_spu_es = 0; + } + + /* Delete all ES in this program except the PSI */ + for( i_loop=0; i_loop < p_es->p_pgrm->i_es_number; i_loop++ ) + { + p_es_demux = (es_ts_data_t *) + p_es->p_pgrm->pp_es[i_loop]->p_demux_data; + if ( ! p_es_demux->b_psi ) + input_DelES( p_input, p_es->p_pgrm->pp_es[i_loop] ); + } + + /* Then add what we received in this PMT */ + do + { + + i_section_length = U16_AT(p_current_data+1) & 0x0fff; + i_current_section = (u8)p_current_data[6]; + i_prog_info_length = U16_AT(p_current_data+10) & 0x0fff; + + /* For the moment we ignore program descriptors */ + p_current_data += 12+i_prog_info_length; + + /* The end of the section, before the CRC is at + * p_current_section + i_section_length -1 */ + while( p_current_data < p_current_section + i_section_length -1 ) + { + i_stream_type = (int)p_current_data[0]; + i_pid = U16_AT( p_current_data + 1 ) & 0x1fff; + i_es_info_length = U16_AT( p_current_data + 3 ) & 0x0fff; + + /* Add this ES to the program */ + p_new_es = input_AddES( p_input, p_es->p_pgrm, + (u16)i_pid, sizeof( es_ts_data_t ) ); + + /* Tell the decoders what kind of stream it is */ + p_new_es->i_type = i_stream_type; + + /* Tell the interface what kind of stream it is and select + * the required ones */ + switch( i_stream_type ) + { + case MPEG1_VIDEO_ES: + case MPEG2_VIDEO_ES: + p_new_es->i_cat = VIDEO_ES; + input_SelectES( p_input, p_new_es ); + break; + case MPEG1_AUDIO_ES: + case MPEG2_AUDIO_ES: + case LPCM_AUDIO_ES : + p_new_es->i_cat = AUDIO_ES; + i_audio_es += 1; + if( i_audio_es == i_required_audio_es ) + input_SelectES( p_input, p_new_es ); + break; + case AC3_AUDIO_ES : + p_new_es->i_stream_id = 0xBD; + p_new_es->i_cat = AUDIO_ES; + i_audio_es += 1; + if( i_audio_es == i_required_audio_es ) + input_SelectES( p_input, p_new_es ); + break; + /* Not sure this one is fully norm-compliant */ + case DVD_SPU_ES : + p_new_es->i_cat = SPU_ES; + i_spu_es += 1; + if( i_spu_es == i_required_spu_es ) + input_SelectES( p_input, p_new_es ); + break; + default : + p_new_es->i_cat = UNKNOWN_ES; + break; + } + + p_current_data += 5 + i_es_info_length; + } + + /* Go to the beginning of the next section*/ + p_current_data += 3+i_section_length; + + p_current_section+=1; + + } while( i_current_section < p_psi->i_last_section_number ); + + if( i_required_audio_es > i_audio_es ) + intf_WarnMsg( 2, "TS input: Non-existing audio es required." ); + + if( i_required_spu_es > i_spu_es ) + intf_WarnMsg( 2, "TS input: Non-existing subtitles es required." ); + + p_pgrm_data->i_pmt_version = p_psi->i_version_number; + + /* inform interface that stream has changed */ + p_input->stream.b_changed = 1; + } + +#undef p_psi + + /* Remove lock */ + vlc_mutex_unlock( &p_input->stream.stream_lock ); }