/*****************************************************************************
* vpar_headers.c : headers parsing
- * (c)1999 VideoLAN
+ *****************************************************************************
+ * Copyright (C) 1999, 2000 VideoLAN
+ * $Id: vpar_headers.c,v 1.74 2001/01/18 05:13:23 sam Exp $
+ *
+ * Authors: Christophe Massiot <massiot@via.ecp.fr>
+ * Stéphane Borel <stef@via.ecp.fr>
+ *
+ * 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
+ * the Free Software Foundation; either version 2 of the License, or
+ * (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program; if not, write to the Free Software
+ * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111, USA.
*****************************************************************************/
-/* ?? passer en terminate/destroy avec les signaux supplémentaires */
-
/*****************************************************************************
* Preamble
*****************************************************************************/
-#include <errno.h>
-#include <stdlib.h>
-#include <stdio.h>
-#include <unistd.h>
-#include <string.h>
-#include <sys/uio.h>
-#include <X11/Xlib.h>
-#include <X11/extensions/XShm.h>
+#include "defs.h"
+
+#include <stdlib.h> /* free() */
#include "config.h"
#include "common.h"
+#include "threads.h"
#include "mtime.h"
-#include "vlc_thread.h"
+#include "plugins.h"
#include "intf_msg.h"
-#include "debug.h" /* ?? temporaire, requis par netlist.h */
-#include "input.h"
-#include "input_netlist.h"
-#include "decoder_fifo.h"
+#include "stream_control.h"
+#include "input_ext-dec.h"
+
#include "video.h"
#include "video_output.h"
-#include "video_parser.h"
-#include "undec_picture.h"
-#include "video_fifo.h"
#include "video_decoder.h"
+#include "vdec_motion.h"
+#include "../video_decoder/vdec_idct.h"
+
+#include "vpar_blocks.h"
+#include "../video_decoder/vpar_headers.h"
+#include "../video_decoder/vpar_synchro.h"
+#include "../video_decoder/video_parser.h"
+#include "../video_decoder/video_fifo.h"
/*
* Local prototypes
*/
+static __inline__ void NextStartCode( bit_stream_t * );
+static void SequenceHeader( vpar_thread_t * p_vpar );
+static void GroupHeader( vpar_thread_t * p_vpar );
+static void PictureHeader( vpar_thread_t * p_vpar );
+static void ExtensionAndUserData( vpar_thread_t * p_vpar );
+static void QuantMatrixExtension( vpar_thread_t * p_vpar );
+static void SequenceScalableExtension( vpar_thread_t * p_vpar );
+static void SequenceDisplayExtension( vpar_thread_t * p_vpar );
+static void PictureDisplayExtension( vpar_thread_t * p_vpar );
+static void PictureSpatialScalableExtension( vpar_thread_t * p_vpar );
+static void PictureTemporalScalableExtension( vpar_thread_t * p_vpar );
+static void CopyrightExtension( vpar_thread_t * p_vpar );
+
+/*
+ * Standard variables
+ */
/*****************************************************************************
- * vpar_NextSequenceHeader : Find the next sequence header
+ * pi_default_intra_quant : default quantization matrix
+ *****************************************************************************/
+u8 pi_default_intra_quant[] =
+{
+ 8, 16, 19, 22, 26, 27, 29, 34,
+ 16, 16, 22, 24, 27, 29, 34, 37,
+ 19, 22, 26, 27, 29, 34, 34, 38,
+ 22, 22, 26, 27, 29, 34, 37, 40,
+ 22, 26, 27, 29, 32, 35, 40, 48,
+ 26, 27, 29, 32, 35, 40, 48, 58,
+ 26, 27, 29, 34, 38, 46, 56, 69,
+ 27, 29, 35, 38, 46, 56, 69, 83
+};
+
+/*****************************************************************************
+ * pi_default_nonintra_quant : default quantization matrix
+ *****************************************************************************/
+u8 pi_default_nonintra_quant[] =
+{
+ 16, 16, 16, 16, 16, 16, 16, 16,
+ 16, 16, 16, 16, 16, 16, 16, 16,
+ 16, 16, 16, 16, 16, 16, 16, 16,
+ 16, 16, 16, 16, 16, 16, 16, 16,
+ 16, 16, 16, 16, 16, 16, 16, 16,
+ 16, 16, 16, 16, 16, 16, 16, 16,
+ 16, 16, 16, 16, 16, 16, 16, 16,
+ 16, 16, 16, 16, 16, 16, 16, 16
+};
+
+/*****************************************************************************
+ * pi_scan : zig-zag and alternate scan patterns
+ *****************************************************************************/
+u8 pi_scan[2][64] =
+{
+ { /* Zig-Zag pattern */
+ 0,1,8,16,9,2,3,10,17,24,32,25,18,11,4,5,
+ 12,19,26,33,40,48,41,34,27,20,13,6,7,14,21,28,
+ 35,42,49,56,57,50,43,36,29,22,15,23,30,37,44,51,
+ 58,59,52,45,38,31,39,46,53,60,61,54,47,55,62,63
+ },
+ { /* Alternate scan pattern */
+ 0,8,16,24,1,9,2,10,17,25,32,40,48,56,57,49,
+ 41,33,26,18,3,11,4,12,19,27,34,42,50,58,35,43,
+ 51,59,20,28,5,13,6,14,21,29,36,44,52,60,37,45,
+ 53,61,22,30,7,15,23,31,38,46,54,62,39,47,55,63
+ }
+};
+
+/*
+ * Local inline functions.
+ */
+
+/*****************************************************************************
+ * ReferenceUpdate : Update the reference pointers when we have a new picture
*****************************************************************************/
-void vpar_NextSequenceHeader( vpar_thread_t * p_vpar )
+static void __inline__ ReferenceUpdate( vpar_thread_t * p_vpar,
+ int i_coding_type,
+ picture_t * p_newref )
{
- while( !p_vpar->b_die )
+ if( i_coding_type != B_CODING_TYPE )
+ {
+ if( p_vpar->sequence.p_forward != NULL )
+ {
+ vout_UnlinkPicture( p_vpar->p_vout, p_vpar->sequence.p_forward );
+ }
+ if( p_vpar->sequence.p_backward != NULL )
+ {
+ vout_DatePicture( p_vpar->p_vout, p_vpar->sequence.p_backward,
+ vpar_SynchroDate( p_vpar ) );
+ }
+ p_vpar->sequence.p_forward = p_vpar->sequence.p_backward;
+ p_vpar->sequence.p_backward = p_newref;
+ if( p_newref != NULL )
+ {
+ vout_LinkPicture( p_vpar->p_vout, p_newref );
+ }
+ }
+ else if( p_newref != NULL )
{
- NextStartCode( p_vpar );
- if( ShowBits( &p_vpar->bit_stream, 32 ) == SEQUENCE_START_CODE )
+ /* Put date immediately. */
+ vout_DatePicture( p_vpar->p_vout, p_newref, vpar_SynchroDate(p_vpar) );
+ }
+}
+
+/*****************************************************************************
+ * ReferenceReplace : Replace the last reference pointer when we destroy
+ * a picture
+ *****************************************************************************/
+static void __inline__ ReferenceReplace( vpar_thread_t * p_vpar,
+ int i_coding_type,
+ picture_t * p_newref )
+{
+ if( i_coding_type != B_CODING_TYPE )
+ {
+ if( p_vpar->sequence.p_backward != NULL )
+ {
+ vout_UnlinkPicture( p_vpar->p_vout, p_vpar->sequence.p_backward );
+ }
+ p_vpar->sequence.p_backward = p_newref;
+ if( p_newref != NULL )
+ {
+ vout_LinkPicture( p_vpar->p_vout, p_newref );
+ }
+ }
+}
+
+/*****************************************************************************
+ * LoadMatrix : Load a quantization matrix
+ *****************************************************************************/
+static __inline__ void LoadMatrix( vpar_thread_t * p_vpar,
+ quant_matrix_t * p_matrix )
+{
+ int i_dummy;
+
+ if( !p_matrix->b_allocated )
+ {
+ /* Allocate a piece of memory to load the matrix. */
+ if( (p_matrix->pi_matrix = (u8 *)malloc( 64*sizeof(u8) )) == NULL )
+ {
+ intf_ErrMsg( "vpar error: allocation error in LoadMatrix()" );
+ p_vpar->p_fifo->b_error = 1;
return;
+ }
+ p_matrix->b_allocated = 1;
+ }
+
+ for( i_dummy = 0; i_dummy < 64; i_dummy++ )
+ {
+ p_matrix->pi_matrix[p_vpar->ppi_scan[SCAN_ZIGZAG][i_dummy]]
+ = GetBits( &p_vpar->bit_stream, 8 );
+ }
+
+#ifdef VDEC_DFT
+ /* Discrete Fourier Transform requires the quantization matrices to
+ * be normalized before using them. */
+ vdec_NormQuantMatrix( p_matrix->pi_matrix );
+#endif
+}
+
+/*****************************************************************************
+ * LinkMatrix : Link a quantization matrix to another
+ *****************************************************************************/
+static __inline__ void LinkMatrix( quant_matrix_t * p_matrix, u8 * pi_array )
+{
+ if( p_matrix->b_allocated )
+ {
+ /* Deallocate the piece of memory. */
+ free( p_matrix->pi_matrix );
+ p_matrix->b_allocated = 0;
+ }
+
+ p_matrix->pi_matrix = pi_array;
+}
+
+/*
+ * Exported functions.
+ */
+
+/*****************************************************************************
+ * vpar_NextSequenceHeader : Find the next sequence header
+ *****************************************************************************/
+int vpar_NextSequenceHeader( vpar_thread_t * p_vpar )
+{
+ while( !p_vpar->p_fifo->b_die )
+ {
+ NextStartCode( &p_vpar->bit_stream );
+ if( ShowBits( &p_vpar->bit_stream, 32 ) == SEQUENCE_HEADER_CODE )
+ {
+ return 0;
+ }
+ RemoveBits( &p_vpar->bit_stream, 8 );
}
+ return 1;
}
/*****************************************************************************
*****************************************************************************/
int vpar_ParseHeader( vpar_thread_t * p_vpar )
{
- while( !p_vpar->b_die )
+ while( !p_vpar->p_fifo->b_die )
{
- NextStartCode( p_vpar );
+ NextStartCode( &p_vpar->bit_stream );
switch( GetBits32( &p_vpar->bit_stream ) )
{
case SEQUENCE_HEADER_CODE:
+#ifdef STATS
+ p_vpar->c_sequences++;
+#endif
SequenceHeader( p_vpar );
return 0;
break;
break;
case SEQUENCE_END_CODE:
+ intf_DbgMsg("vpar debug: sequence end code received");
return 1;
break;
* Following functions are local
*/
-/*****************************************************************************
- * NextStartCode : Find the next start code
- *****************************************************************************/
-static __inline__ void NextStartCode( vpar_thread_t * p_vpar )
-{
- /* Re-align the buffer to an 8-bit boundary */
- DumpBits( &p_vpar->bit_stream, p_vpar->bit_stream.fifo.i_available & 7 );
-
- while( ShowBits( &p_vpar->bit_stream, 24 ) != 0x01L && !p_vpar->b_die )
- {
- DumpBits( &p_vpar->bit_stream, 8 );
- }
-}
-
/*****************************************************************************
* SequenceHeader : Parse the next sequence header
*****************************************************************************/
-#define RESERVED -1
-static double d_frame_rate_table[16] =
-{
- 0.0,
- ((23.0*1000.0)/1001.0),
- 24.0,
- 25.0,
- ((30.0*1000.0)/1001.0),
- 30.0,
- 50.0,
- ((60.0*1000.0)/1001.0),
- 60.0,
-
- RESERVED,
- RESERVED,
- RESERVED,
- RESERVED,
- RESERVED,
- RESERVED,
- RESERVED
-};
-
static void SequenceHeader( vpar_thread_t * p_vpar )
{
+#define RESERVED -1
+ static int i_frame_rate_table[16] =
+ {
+ 0,
+ 23 * 1000,
+ 24 * 1001,
+ 25 * 1001,
+ 30 * 1000,
+ 30 * 1001,
+ 50 * 1001,
+ 60 * 1000,
+ 60 * 1001,
+ RESERVED, RESERVED, RESERVED, RESERVED, RESERVED, RESERVED, RESERVED
+ };
+#undef RESERVED
+
int i_height_save, i_width_save;
-
+
i_height_save = p_vpar->sequence.i_height;
i_width_save = p_vpar->sequence.i_width;
- p_vpar->sequence.i_height = ntohl( GetBits( p_vpar->bit_stream, 12 ) );
- p_vpar->sequence.i_width = ntohl( GetBits( p_vpar->bit_stream, 12 ) );
- p_vpar->sequence.i_ratio = GetBits( p_vpar->bit_stream, 4 );
- p_vpar->sequence.d_frame_rate =
- d_frame_rate_table( GetBits( p_vpar->bit_stream, 4 ) );
+ p_vpar->sequence.i_width = GetBits( &p_vpar->bit_stream, 12 );
+ p_vpar->sequence.i_height = GetBits( &p_vpar->bit_stream, 12 );
+ p_vpar->sequence.i_aspect_ratio = GetBits( &p_vpar->bit_stream, 4 );
+ p_vpar->sequence.i_frame_rate =
+ i_frame_rate_table[ GetBits( &p_vpar->bit_stream, 4 ) ];
/* We don't need bit_rate_value, marker_bit, vbv_buffer_size,
* constrained_parameters_flag */
- DumpBits( p_vpar->bit_stream, 30 );
-
+ RemoveBits( &p_vpar->bit_stream, 30 );
+
/*
* Quantization matrices
*/
- if( GetBits( p_vpar->bit_stream, 1 ) ) /* load_intra_quantizer_matrix */
+ if( GetBits( &p_vpar->bit_stream, 1 ) ) /* load_intra_quantizer_matrix */
{
LoadMatrix( p_vpar, &p_vpar->sequence.intra_quant );
}
else
{
/* Use default matrix. */
- LinkMatrix( &p_vpar->sequence.intra_quant, pi_default_intra_quant );
+ LinkMatrix( &p_vpar->sequence.intra_quant,
+ p_vpar->pi_default_intra_quant );
}
-
- if( GetBits( p_vpar->bit_stream, 1 ) ) /* load_non_intra_quantizer_matrix */
+
+ if( GetBits( &p_vpar->bit_stream, 1 ) ) /* load_non_intra_quantizer_matrix */
{
LoadMatrix( p_vpar, &p_vpar->sequence.nonintra_quant );
}
else
{
/* Use default matrix. */
- LinkMatrix( &p_vpar->sequence.nonintra_quant, pi_default_nonintra_quant );
+ LinkMatrix( &p_vpar->sequence.nonintra_quant,
+ p_vpar->pi_default_nonintra_quant );
}
-
+
/* Unless later overwritten by a matrix extension, we have the same
* matrices for luminance and chrominance. */
LinkMatrix( &p_vpar->sequence.chroma_intra_quant,
/*
* Sequence Extension
*/
- if( ShowBits( p_vpar->bit_stream, 32 ) == EXTENSION_START_CODE )
+ NextStartCode( &p_vpar->bit_stream );
+ if( ShowBits( &p_vpar->bit_stream, 32 ) == EXTENSION_START_CODE )
{
- int i_dummy;
-
+ int i_dummy;
+
+ /* Turn the MPEG2 flag on */
+ p_vpar->sequence.b_mpeg2 = 1;
+
/* Parse sequence_extension */
- DumpBits32( &p_vpar->bit_stream );
+ RemoveBits32( &p_vpar->bit_stream );
/* extension_start_code_identifier, profile_and_level_indication */
- DumpBits( &p_vpar->bit_stream, 12 );
+ RemoveBits( &p_vpar->bit_stream, 12 );
p_vpar->sequence.b_progressive = GetBits( &p_vpar->bit_stream, 1 );
p_vpar->sequence.i_chroma_format = GetBits( &p_vpar->bit_stream, 2 );
p_vpar->sequence.i_width |= GetBits( &p_vpar->bit_stream, 2 ) << 12;
p_vpar->sequence.i_height |= GetBits( &p_vpar->bit_stream, 2 ) << 12;
/* bit_rate_extension, marker_bit, vbv_buffer_size_extension, low_delay */
- DumpBits( &p_vpar->bit_stream, 22 );
+ RemoveBits( &p_vpar->bit_stream, 22 );
/* frame_rate_extension_n */
i_dummy = GetBits( &p_vpar->bit_stream, 2 );
/* frame_rate_extension_d */
- p_vpar->sequence.d_frame_rate *= (i_dummy + 1)
+ p_vpar->sequence.i_frame_rate *= (i_dummy + 1)
/ (GetBits( &p_vpar->bit_stream, 5 ) + 1);
-
- /* Extension and User data */
- ExtensionAndUserData( p_vpar );
}
else
{
/* It's an MPEG-1 stream. Put adequate parameters. */
+
+ p_vpar->sequence.b_mpeg2 = 0;
p_vpar->sequence.b_progressive = 1;
- p_vpar->i_chroma_format = CHROMA_420;
+ p_vpar->sequence.i_chroma_format = CHROMA_420;
}
+ /* Update sizes */
p_vpar->sequence.i_mb_width = (p_vpar->sequence.i_width + 15) / 16;
p_vpar->sequence.i_mb_height = (p_vpar->sequence.b_progressive) ?
(p_vpar->sequence.i_height + 15) / 16 :
- 2 * (p_vpar->sequence.i_height + 31) / 32;
+ 2 * ((p_vpar->sequence.i_height + 31) / 32);
+ p_vpar->sequence.i_mb_size = p_vpar->sequence.i_mb_width
+ * p_vpar->sequence.i_mb_height;
p_vpar->sequence.i_width = (p_vpar->sequence.i_mb_width * 16);
p_vpar->sequence.i_height = (p_vpar->sequence.i_mb_height * 16);
+ p_vpar->sequence.i_size = p_vpar->sequence.i_width
+ * p_vpar->sequence.i_height;
+
+ /* Update chromatic information. */
+ switch( p_vpar->sequence.i_chroma_format )
+ {
+ case CHROMA_420:
+ p_vpar->sequence.i_chroma_nb_blocks = 2;
+ p_vpar->sequence.i_chroma_width = p_vpar->sequence.i_width >> 1;
+ p_vpar->sequence.i_chroma_mb_width = 8;
+ p_vpar->sequence.i_chroma_mb_height = 8;
+ break;
+ case CHROMA_422:
+ p_vpar->sequence.i_chroma_nb_blocks = 4;
+ p_vpar->sequence.i_chroma_width = p_vpar->sequence.i_width >> 1;
+ p_vpar->sequence.i_chroma_mb_width = 8;
+ p_vpar->sequence.i_chroma_mb_height = 16;
+ break;
+
+ case CHROMA_444:
+ p_vpar->sequence.i_chroma_nb_blocks = 8;
+ p_vpar->sequence.i_chroma_width = p_vpar->sequence.i_width;
+ p_vpar->sequence.i_chroma_mb_width = 16;
+ p_vpar->sequence.i_chroma_mb_height = 16;
+ }
+
+ /* Reset scalable_mode. */
+ p_vpar->sequence.i_scalable_mode = SC_NONE;
+
+#if 0
if( p_vpar->sequence.i_width != i_width_save
- || p_vpar->sequence.i_height != i_height_save
- || p_vpar->sequence.p_frame_lum_lookup == NULL )
+ || p_vpar->sequence.i_height != i_height_save )
{
- int i_x, i_y;
- pel_lookup_table * p_fr, p_fl;
- int i_fr, i_fl;
- static int pi_chroma_size[4] = {0, 2, 1, 0}
-#define Sequence p_vpar->sequence
+ /* FIXME: What do we do in case of a size change ?? */
+ }
+#endif
+
+ /* Extension and User data */
+ ExtensionAndUserData( p_vpar );
+}
+
+/*****************************************************************************
+ * GroupHeader : Parse the next group of pictures header
+ *****************************************************************************/
+static void GroupHeader( vpar_thread_t * p_vpar )
+{
+ /* Nothing to do, we don't care. */
+ RemoveBits( &p_vpar->bit_stream, 27 );
+ ExtensionAndUserData( p_vpar );
+}
- /* The size of the pictures has changed. Probably a new sequence.
- * We must recalculate the lookup matrices. */
+/*****************************************************************************
+ * PictureHeader : Parse the next picture header
+ *****************************************************************************/
+static void PictureHeader( vpar_thread_t * p_vpar )
+{
+ int i_structure;
+ int i_mb_base;
+ boolean_t b_parsable;
+#ifdef VDEC_SMP
+ int i_mb;
+#endif
- /* First unlink the previous lookup matrices so that they can
- * be freed in the future. */
- if( Sequence.p_frame_lum_lookup != NULL )
+ RemoveBits( &p_vpar->bit_stream, 10 ); /* temporal_reference */
+ p_vpar->picture.i_coding_type = GetBits( &p_vpar->bit_stream, 3 );
+ RemoveBits( &p_vpar->bit_stream, 16 ); /* vbv_delay */
+
+ if( p_vpar->picture.i_coding_type == P_CODING_TYPE
+ || p_vpar->picture.i_coding_type == B_CODING_TYPE )
+ {
+ p_vpar->picture.pb_full_pel_vector[0] = GetBits( &p_vpar->bit_stream, 1 );
+ p_vpar->picture.i_forward_f_code = GetBits( &p_vpar->bit_stream, 3 );
+ }
+ if( p_vpar->picture.i_coding_type == B_CODING_TYPE )
+ {
+ p_vpar->picture.pb_full_pel_vector[1] = GetBits( &p_vpar->bit_stream, 1 );
+ p_vpar->picture.i_backward_f_code = GetBits( &p_vpar->bit_stream, 3 );
+ }
+
+ /* extra_information_picture */
+ while( GetBits( &p_vpar->bit_stream, 1 ) )
+ {
+ RemoveBits( &p_vpar->bit_stream, 8 );
+ }
+
+ /*
+ * Picture Coding Extension
+ */
+ NextStartCode( &p_vpar->bit_stream );
+ if( ShowBits( &p_vpar->bit_stream, 32 ) == EXTENSION_START_CODE )
+ {
+ /* Parse picture_coding_extension */
+ RemoveBits32( &p_vpar->bit_stream );
+ /* extension_start_code_identifier */
+ RemoveBits( &p_vpar->bit_stream, 4 );
+
+ p_vpar->picture.ppi_f_code[0][0] = GetBits( &p_vpar->bit_stream, 4 );
+ p_vpar->picture.ppi_f_code[0][1] = GetBits( &p_vpar->bit_stream, 4 );
+ p_vpar->picture.ppi_f_code[1][0] = GetBits( &p_vpar->bit_stream, 4 );
+ p_vpar->picture.ppi_f_code[1][1] = GetBits( &p_vpar->bit_stream, 4 );
+ p_vpar->picture.i_intra_dc_precision = GetBits( &p_vpar->bit_stream, 2 );
+ i_structure = GetBits( &p_vpar->bit_stream, 2 );
+ p_vpar->picture.b_top_field_first = GetBits( &p_vpar->bit_stream, 1 );
+ p_vpar->picture.b_frame_pred_frame_dct
+ = GetBits( &p_vpar->bit_stream, 1 );
+ p_vpar->picture.b_concealment_mv = GetBits( &p_vpar->bit_stream, 1 );
+ p_vpar->picture.b_q_scale_type = GetBits( &p_vpar->bit_stream, 1 );
+ p_vpar->picture.b_intra_vlc_format = GetBits( &p_vpar->bit_stream, 1 );
+ p_vpar->picture.b_alternate_scan = GetBits( &p_vpar->bit_stream, 1 );
+ p_vpar->picture.b_repeat_first_field = GetBits( &p_vpar->bit_stream, 1 );
+ /* chroma_420_type (obsolete) */
+ RemoveBits( &p_vpar->bit_stream, 1 );
+ p_vpar->picture.b_progressive = GetBits( &p_vpar->bit_stream, 1 );
+
+ /* composite_display_flag */
+ if( GetBits( &p_vpar->bit_stream, 1 ) )
{
- UNLINK_LOOKUP( Sequence.p_frame_lum_lookup );
- UNLINK_LOOKUP( Sequence.p_field_lum_lookup );
- UNLINK_LOOKUP( Sequence.p_frame_chroma_lookup );
- UNLINK_LOOKUP( Sequence.p_field_chroma_lookup );
+ /* v_axis, field_sequence, sub_carrier, burst_amplitude,
+ * sub_carrier_phase */
+ RemoveBits( &p_vpar->bit_stream, 20 );
}
+ }
+ else
+ {
+ /* MPEG-1 compatibility flags */
+ p_vpar->picture.i_intra_dc_precision = 0; /* 8 bits */
+ i_structure = FRAME_STRUCTURE;
+ p_vpar->picture.b_top_field_first = 0;
+ p_vpar->picture.b_frame_pred_frame_dct = 1;
+ p_vpar->picture.b_concealment_mv = 0;
+ p_vpar->picture.b_q_scale_type = 0;
+ p_vpar->picture.b_intra_vlc_format = 0;
+ p_vpar->picture.b_alternate_scan = 0; /* zigzag */
+ p_vpar->picture.b_repeat_first_field = 1; /* FIXME! this contradicts
+ * ISO/IEC */
+ p_vpar->picture.b_progressive = 1;
+ }
- /* Allocate the new lookup tables. */
- Sequence.p_frame_lum_lookup
- = (pel_lookup_table_t *)malloc( sizeof( pel_lookup_table_t ) *
- Sequence.i_width * Sequence.i_height );
- Sequence.p_field_lum_lookup
- = (pel_lookup_table_t *)malloc( sizeof( pel_lookup_table_t ) *
- Sequence.i_width * Sequence.i_height );
- Sequence.p_frame_chroma_lookup
- = (pel_lookup_table_t *)malloc( sizeof( pel_lookup_table_t ) *
- Sequence.i_width * Sequence.i_height
- >> pi_chroma_size[Sequence.i_chroma_format] );
- Sequence.p_field_chroma_lookup
- = (pel_lookup_table_t *)malloc( sizeof( pel_lookup_table_t ) *
- Sequence.i_width * Sequence.i_height
- >> pi_chroma_size[Sequence.i_chroma_format] );
-
- if( !Sequence.p_frame_lum_lookup || !Sequence.p_field_lum_lookup
- || !Sequence.p_frame_chroma_lookup
- || !Sequence.p_field_chroma_lookup )
+#ifdef STATS
+ p_vpar->pc_pictures[p_vpar->picture.i_coding_type]++;
+#endif
+
+ if( p_vpar->picture.i_current_structure &&
+ (i_structure == FRAME_STRUCTURE ||
+ i_structure == p_vpar->picture.i_current_structure) )
+ {
+ /* We don't have the second field of the buffered frame. */
+ if( p_vpar->picture.p_picture != NULL )
{
- intf_DbgMsg("vpar error: not enough memory for lookup tables");
- p_vpar->b_error = 1;
- return;
+ ReferenceReplace( p_vpar,
+ p_vpar->picture.i_coding_type,
+ NULL );
+
+#ifdef VDEC_SMP
+ for( i_mb = 0; p_vpar->picture.pp_mb[i_mb] != NULL; i_mb++ )
+ {
+ vpar_DestroyMacroblock( &p_vpar->vfifo,
+ p_vpar->picture.pp_mb[i_mb] );
+ }
+#endif
+ vout_DestroyPicture( p_vpar->p_vout, p_vpar->picture.p_picture );
}
- /* Fill in the luminance lookup tables */
- p_fr = &Sequence.p_frame_lum_lookup->pi_pel;
- p_fl = &Sequence.p_field_lum_lookup->pi_pel;
- i_fr = i_fl = 0;
+ p_vpar->picture.i_current_structure = 0;
- for( i_y = 0; i_y < Sequence.i_height; i_y++ )
+ intf_DbgMsg("vpar debug: odd number of field picture.");
+ }
+
+ /* Do we have the reference pictures ? */
+ b_parsable = !(((p_vpar->picture.i_coding_type == P_CODING_TYPE) &&
+ (p_vpar->sequence.p_backward == NULL)) ||
+ /* p_backward will become p_forward later */
+ ((p_vpar->picture.i_coding_type == B_CODING_TYPE) &&
+ (p_vpar->sequence.p_forward == NULL ||
+ p_vpar->sequence.p_backward == NULL)));
+
+ if( p_vpar->picture.i_current_structure )
+ {
+ /* Second field of a frame. We will decode it if, and only if we
+ * have decoded the first field. */
+ if( b_parsable )
{
- int i_mb_y, i_b_y, i_pos_y;
- i_mb_y = i_y >> 4;
- i_b_y = (i_y & 15) >> 3;
- i_pos_y = (i_y & 7);
+ b_parsable = (p_vpar->picture.p_picture != NULL);
+ }
+ }
+ else
+ {
+ int i_repeat_field;
- for( i_x = 0; i_x < Sequence.i_width; i_x++ )
+ /* Compute the number of times the frame will be emitted by the
+ * decoder (number of periods). */
+ if( p_vpar->sequence.b_progressive )
+ {
+ i_repeat_field = 1 + p_vpar->picture.b_repeat_first_field
+ + p_vpar->picture.b_top_field_first;
+ }
+ else
+ {
+ if( p_vpar->picture.b_progressive )
{
- int i_mb_x, i_b_x, i_pos_x;
- i_mb_x = i_x >> 4;
- i_b_x = (i_x & 15) >> 3;
- i_pos_x = (i_x & 7);
-
- p_fl[i_fr + i_x] = p_fr[i_fr + i_x]
- = (i_mb_y*Sequence.i_mb_width + i_mb_y)*256
- + ((i_b_y << 1) + i_b_x)*64
- + i_pos_y*8 + i_pos_x;
+ i_repeat_field = 2 + p_vpar->picture.b_repeat_first_field;
}
- i_fr += Sequence.i_width;
- i_fl += Sequence.i_width << 1;
- if( i_fl == Sequence.i_width*Sequence.i_height )
+ else
{
- i_fl = Sequence.i_width;
+ i_repeat_field = 2;
}
}
-
- /* Fill in the chrominance lookup tables */
- p_fr = &Sequence.p_frame_chroma_lookup->pi_pel;
- p_fl = &Sequence.p_field_chroma_lookup->pi_pel;
- i_fr = i_fl = 0;
- switch( p_vpar->i_chroma_format )
+ /* Warn synchro we have a new picture (updates pictures index). */
+ vpar_SynchroNewPicture( p_vpar, p_vpar->picture.i_coding_type,
+ i_repeat_field );
+
+ if( b_parsable )
+ {
+ /* Does synchro say we have enough time to decode it ? */
+ b_parsable = vpar_SynchroChoose( p_vpar,
+ p_vpar->picture.i_coding_type, i_structure );
+ }
+ }
+
+ if( !b_parsable )
+ {
+ /* Update the reference pointers. */
+ ReferenceUpdate( p_vpar, p_vpar->picture.i_coding_type, NULL );
+
+ /* Update context. */
+ if( i_structure != FRAME_STRUCTURE )
{
- case CHROMA_444:
- /* That's the same as luminance */
- memcopy( &Sequence.p_frame_croma_lookup->pi_pel,
- &Sequence.p_frame_lum_lookup->pi_pel,
- sizeof(PEL_P)*Sequence.i_height*Sequence.i_width );
- memcopy( &Sequence.p_field_croma_lookup->pi_pel,
- &Sequence.p_field_lum_lookup->pi_pel,
- sizeof(PEL_P)*Sequence.i_height*Sequence.i_width );
-
- case CHROMA_422:
- for( i_y = 0; i_y < Sequence.i_height; i_y++ )
+ if( (p_vpar->picture.i_current_structure | i_structure)
+ == FRAME_STRUCTURE )
+ {
+ p_vpar->picture.i_current_structure = 0;
+ }
+ else
{
- int i_mb_y, i_b_y, i_pos_y;
- i_mb_y = i_y >> 4;
- i_b_y = (i_y & 15) >> 3;
- i_pos_y = (i_y & 7);
-
- for( i_x = 0; i_x < (Sequence.i_width >> 1); i_x++ )
- {
- int i_mb_x, i_pos_x;
- i_mb_x = i_x >> 3;
- i_pos_x = (i_x & 7);
-
- p_fl[i_fr + i_x] = p_fr[i_fr + i_x]
- = (i_mb_y*Sequence.i_mb_width + i_mb_y)*128
- + i_b_y*64
- + i_pos_y*8 + i_pos_x;
- }
- i_fr += Sequence.i_width >> 1;
- i_fl += Sequence.i_width;
- if( i_fl == (Sequence.i_width*Sequence.i_height >> 1) )
- {
- i_fl = (Sequence.i_width >> 1);
- }
+ /* The frame is complete. */
+ p_vpar->picture.i_current_structure = i_structure;
+
+ vpar_SynchroTrash( p_vpar, p_vpar->picture.i_coding_type, i_structure );
}
+ }
+ else
+ {
+ /* Warn Synchro we have trashed a picture. */
+ vpar_SynchroTrash( p_vpar, p_vpar->picture.i_coding_type, i_structure );
+ }
+ p_vpar->picture.p_picture = NULL;
+
+ return;
+ }
+
+ /* OK, now we are sure we will decode the picture. */
+#ifdef STATS
+ p_vpar->pc_decoded_pictures[p_vpar->picture.i_coding_type]++;
+#endif
+
+#define P_picture p_vpar->picture.p_picture
+ p_vpar->picture.b_error = 0;
+ p_vpar->picture.b_frame_structure = (i_structure == FRAME_STRUCTURE);
- case CHROMA_420:
- for( i_y = 0; i_y < (Sequence.i_height >> 1); i_y++ )
+ if( !p_vpar->picture.i_current_structure )
+ {
+ /* This is a new frame. Get a structure from the video_output. */
+ while( ( P_picture = vout_CreatePicture( p_vpar->p_vout,
+ 99+p_vpar->sequence.i_chroma_format, /*XXX??*/
+ p_vpar->sequence.i_width,
+ p_vpar->sequence.i_height ) )
+ == NULL )
+ {
+ intf_DbgMsg("vpar debug: allocation error in vout_CreatePicture, delaying");
+ if( p_vpar->p_fifo->b_die || p_vpar->p_fifo->b_error )
{
- int i_mb_y, i_pos_y;
- i_mb_y = i_y >> 3;
- i_pos_y = (i_y & 7);
-
- for( i_x = 0; i_x < (Sequence.i_width >> 1); i_x++ )
- {
- int i_mb_x, i_pos_x;
- i_mb_x = i_x >> 3;
- i_pos_x = (i_x & 7);
-
- p_fl[i_fr + i_x] = p_fr[i_fr + i_x]
- = (i_mb_y*Sequence.i_mb_width + i_mb_y)*64
- + i_pos_y*8 + i_pos_x;
- }
- i_fr += Sequence.i_width >> 1;
- i_fl += Sequence.i_width;
- if( i_fl == (Sequence.i_width*Sequence.i_height >> 2) )
- {
- i_fl = Sequence.i_width >> 1;
- }
+ return;
}
- } /* switch */
-
- /* Link the new lookup tables so that they don't get freed all
- * the time. */
- LINK_LOOKUP( Sequence.p_frame_lum_lookup );
- LINK_LOOKUP( Sequence.p_field_lum_lookup );
- LINK_LOOKUP( Sequence.p_frame_chroma_lookup );
- LINK_LOOKUP( Sequence.p_field_chroma_lookup );
-#undef Sequence
+ msleep( VPAR_OUTMEM_SLEEP );
+ }
+
+ /* Initialize values. */
+ vpar_SynchroDecode( p_vpar, p_vpar->picture.i_coding_type, i_structure );
+ P_picture->i_aspect_ratio = p_vpar->sequence.i_aspect_ratio;
+ P_picture->i_matrix_coefficients = p_vpar->sequence.i_matrix_coefficients;
+ p_vpar->picture.i_l_stride = ( p_vpar->sequence.i_width
+ << ( 1 - p_vpar->picture.b_frame_structure ) );
+ p_vpar->picture.i_c_stride = ( p_vpar->sequence.i_chroma_width
+ << ( 1 - p_vpar->picture.b_frame_structure ));
+
+ P_picture->i_deccount = p_vpar->sequence.i_mb_size;
+#ifdef VDEC_SMP
+ memset( p_vpar->picture.pp_mb, 0, MAX_MB*sizeof(macroblock_t *) );
+#endif
+/* FIXME ! remove asap ?? */
+//memset( P_picture->p_data, 0, (p_vpar->sequence.i_mb_size*384));
+
+ /* Update the reference pointers. */
+ ReferenceUpdate( p_vpar, p_vpar->picture.i_coding_type, P_picture );
+
+#ifdef VDEC_SMP
+ /* Link referenced pictures for the decoder
+ * They are unlinked in vpar_ReleaseMacroblock() & vpar_DestroyMacroblock() */
+ if( p_vpar->picture.i_coding_type == P_CODING_TYPE ||
+ p_vpar->picture.i_coding_type == B_CODING_TYPE )
+ {
+ vout_LinkPicture( p_vpar->p_vout, p_vpar->sequence.p_forward );
+ }
+ if( p_vpar->picture.i_coding_type == B_CODING_TYPE )
+ {
+ vout_LinkPicture( p_vpar->p_vout, p_vpar->sequence.p_backward );
+ }
+#endif
}
-}
+ p_vpar->picture.i_current_structure |= i_structure;
+ p_vpar->picture.i_structure = i_structure;
-/*****************************************************************************
- * GroupHeader : Parse the next group of pictures header
- *****************************************************************************/
-static void GroupHeader( vpar_thread_t * p_vpar )
-{
- /* Nothing to do, we don't care. */
- DumpBits( &p_vpar->bit_stream, 27 );
+ /* Initialize picture data for decoding. */
+ if( i_structure == BOTTOM_FIELD )
+ {
+ i_mb_base = p_vpar->sequence.i_mb_size >> 1;
+ p_vpar->mb.i_l_y = 1;
+ p_vpar->mb.i_c_y = 1;
+ }
+ else
+ {
+ i_mb_base = 0;
+ p_vpar->mb.i_l_y = p_vpar->mb.i_c_y = 0;
+ }
+ p_vpar->mb.i_l_x = p_vpar->mb.i_c_x = 0;
+
+ /* Extension and User data. */
ExtensionAndUserData( p_vpar );
-}
-/*****************************************************************************
- * PictureHeader : Parse the next picture header
- *****************************************************************************/
-static void PictureHeader( vpar_thread_t * p_vpar )
-{
- int i_coding_type;
- mtime_t i_pts;
- undec_picture_t * p_undec_p;
-
- DumpBits( &p_vpar->bit_stream, 10 ); /* temporal_reference */
- i_coding_type = GetBits( &p_vpar->bit_stream, 3 );
-
- if( ((i_coding_type == P_CODING_TYPE) &&
- (p_vpar->sequence.p_forward == NULL)) ||
- ((i_coding_type == B_CODING_TYPE) &&
- (p_vpar->sequence.p_forward == NULL ||
- p_vpar->sequence.p_backward == NULL)) )
- {
- /* The picture cannot be decoded because we lack one of the
- * reference frames */
-
- /* Update the reference pointers */
- ReferenceUpdate( p_vpar, i_coding_type, NULL );
-
- /* Warn Synchro we have trashed a picture */
- vpar_SynchroTrash( p_vpar, i_coding_type );
+ vpar_PictureData( p_vpar, i_mb_base );
+ if( p_vpar->p_fifo->b_die || p_vpar->p_fifo->b_error )
+ {
return;
}
- if( !(i_pts = vpar_SynchroChoose( p_vpar, i_coding_type )) )
+ if( p_vpar->picture.b_error )
{
- /* Synchro has decided not to decode the picture */
-
- /* Update the reference pointers */
- ReferenceUpdate( p_vpar, i_coding_type, NULL );
-
- return;
- }
+ /* Trash picture. */
+#ifdef VDEC_SMP
+ for( i_mb = 1; p_vpar->picture.pp_mb[i_mb] != NULL; i_mb++ )
+ {
+ vpar_DestroyMacroblock( &p_vpar->vfifo, p_vpar->picture.pp_mb[i_mb] );
+ }
+#endif
- /* OK, now we are sure we will decode the picture. Get a structure. */
- p_undec_p = vpar_NewPicture( &p_vpar->vfifo );
-
- /* Request a buffer from the video_output. */
- p_undec_p->p_picture = vout_CreatePicture( p_vpar->p_vout,
- SPLITTED_YUV_PICTURE,
- p_vpar->sequence.i_width,
- p_vpar->sequence.i_height,
- p_vpar->sequence.i_chroma_format );
-
- /* Initialize values */
- p_undec_p->i_coding_type = i_conding_type;
- p_undec_p->b_mpeg2 = p_vpar->sequence.b_mpeg2;
- p_undec_p->i_mb_height = p_vpar->sequence.i_mb_height;
- p_undec_p->i_mb_width = p_vpar->sequence.i_mb_width;
- p_undec_p->i_pts = i_pts;
-}
+#ifdef STATS
+ p_vpar->pc_malformed_pictures[p_vpar->picture.i_coding_type]++;
+#endif
-/*****************************************************************************
- * ReferenceUpdate : Update the reference pointers when we have a new picture
- *****************************************************************************/
-static void __inline__ ReferenceUpdate( vpar_thread_t * p_vpar,
- int i_coding_type,
- picture_t * p_newref )
-{
- if( i_coding_type != B_CODING_TYPE )
+ if( P_picture->i_deccount != 1 )
+ {
+ vpar_SynchroEnd( p_vpar, 1 );
+ vout_DestroyPicture( p_vpar->p_vout, P_picture );
+ }
+
+ ReferenceReplace( p_vpar, p_vpar->picture.i_coding_type, NULL );
+
+ /* Prepare context for the next picture. */
+ P_picture = NULL;
+ if( p_vpar->picture.i_current_structure == FRAME_STRUCTURE )
+ p_vpar->picture.i_current_structure = 0;
+ }
+ else if( p_vpar->picture.i_current_structure == FRAME_STRUCTURE )
{
- /* The P picture would have become the new p_backward reference. */
- vout_UnlinkPicture( p_vpar->p_vout, p_vpar->sequence.p_forward );
- p_vpar->sequence.p_forward = p_vpar->sequence.p_backward;
- p_vpar->sequence.p_backward = p_newref;
+ /* Frame completely parsed. */
+#ifdef VDEC_SMP
+ for( i_mb = 1; p_vpar->picture.pp_mb[i_mb] != NULL; i_mb++ )
+ {
+ vpar_DecodeMacroblock( &p_vpar->vfifo, p_vpar->picture.pp_mb[i_mb] );
+ }
+
+ /* Send signal to the video_decoder. */
+ vlc_mutex_lock( &p_vpar->vfifo.lock );
+ vlc_cond_signal( &p_vpar->vfifo.wait );
+ vlc_mutex_unlock( &p_vpar->vfifo.lock );
+#endif
+
+ /* Prepare context for the next picture. */
+ P_picture = NULL;
+ p_vpar->picture.i_current_structure = 0;
}
+#undef P_picture
}
/*****************************************************************************
*****************************************************************************/
static void ExtensionAndUserData( vpar_thread_t * p_vpar )
{
- while( !p_vpar->b_die )
+ while( !p_vpar->p_fifo->b_die )
{
- NextStartCode( p_vpar );
+ NextStartCode( &p_vpar->bit_stream );
switch( ShowBits( &p_vpar->bit_stream, 32 ) )
{
case EXTENSION_START_CODE:
- DumpBits32( &p_vpar->bit_stream );
+ RemoveBits32( &p_vpar->bit_stream );
switch( GetBits( &p_vpar->bit_stream, 4 ) )
{
case SEQUENCE_DISPLAY_EXTENSION_ID:
break;
case USER_DATA_START_CODE:
- DumpBits32( &p_vpar->bit_stream );
+ RemoveBits32( &p_vpar->bit_stream );
/* Wait for the next start code */
break;
}
}
+
/*****************************************************************************
- * SequenceDisplayExtension : Parse the sequence_display_extension structure
+ * SequenceDisplayExtension : Parse the sequence_display_extension structure *
*****************************************************************************/
+
static void SequenceDisplayExtension( vpar_thread_t * p_vpar )
{
-
+ /* We don't care sequence_display_extension. */
+ /* video_format */
+ RemoveBits( &p_vpar->bit_stream, 3 );
+ if( GetBits( &p_vpar->bit_stream, 1 ) )
+ {
+ /* Two bytes for color_desciption */
+ RemoveBits( &p_vpar->bit_stream, 16 );
+ p_vpar->sequence.i_matrix_coefficients = GetBits( &p_vpar->bit_stream, 8 );
+ }
+ /* display_horizontal and vertical_size and a marker_bit */
+ RemoveBits( &p_vpar->bit_stream, 29 );
}
+
/*****************************************************************************
- * LoadMatrix : Load a quantization matrix
+ * QuantMatrixExtension : Load quantization matrices for luminance *
+ * and chrominance *
*****************************************************************************/
-static void LoadMatrix( vpar_thread_t * p_vpar, quant_matrix_t * p_matrix )
+
+static void QuantMatrixExtension( vpar_thread_t * p_vpar )
{
- int i_dummy;
-
- if( !p_matrix->b_allocated )
+ if( GetBits( &p_vpar->bit_stream, 1 ) )
{
- /* Allocate a piece of memory to load the matrix. */
- p_matrix->pi_matrix = (int *)malloc( 64*sizeof(int) );
- p_matrix->b_allocated = TRUE;
+ /* Load intra_quantiser_matrix for luminance. */
+ LoadMatrix( p_vpar, &p_vpar->sequence.intra_quant );
}
-
- for( i_dummy = 0; i_dummy < 64; i_dummy++ )
+ else
{
- p_matrix->pi_matrix[pi_scan[SCAN_ZIGZAG][i_dummy]]
- = GetBits( p_vpar->bit_stream, 8 );
+ /* Use the default matrix. */
+ LinkMatrix( &p_vpar->sequence.intra_quant,
+ p_vpar->pi_default_intra_quant );
+ }
+ if( GetBits( &p_vpar->bit_stream, 1 ) )
+ {
+ /* Load non_intra_quantiser_matrix for luminance. */
+ LoadMatrix( p_vpar, &p_vpar->sequence.nonintra_quant );
+ }
+ else
+ {
+ /* Use the default matrix. */
+ LinkMatrix( &p_vpar->sequence.nonintra_quant,
+ p_vpar->pi_default_nonintra_quant );
+ }
+ if( GetBits( &p_vpar->bit_stream, 1 ) )
+ {
+ /* Load intra_quantiser_matrix for chrominance. */
+ LoadMatrix( p_vpar, &p_vpar->sequence.chroma_intra_quant );
+ }
+ else
+ {
+ /* Link the chrominance intra matrix to the luminance one. */
+ LinkMatrix( &p_vpar->sequence.chroma_intra_quant,
+ p_vpar->sequence.intra_quant.pi_matrix );
+ }
+ if( GetBits( &p_vpar->bit_stream, 1 ) )
+ {
+ /* Load non_intra_quantiser_matrix for chrominance. */
+ LoadMatrix( p_vpar, &p_vpar->sequence.chroma_nonintra_quant );
+ }
+ else
+ {
+ /* Link the chrominance intra matrix to the luminance one. */
+ LinkMatrix( &p_vpar->sequence.chroma_intra_quant,
+ p_vpar->sequence.intra_quant.pi_matrix );
+ }
+ if( GetBits( &p_vpar->bit_stream, 1 ) )
+ {
+ /* Load non_intra_quantiser_matrix for chrominance. */
+ LoadMatrix( p_vpar, &p_vpar->sequence.chroma_nonintra_quant );
+ }
+ else
+ {
+ /* Link the chrominance nonintra matrix to the luminance one. */
+ LinkMatrix( &p_vpar->sequence.chroma_nonintra_quant,
+ p_vpar->sequence.nonintra_quant.pi_matrix );
}
-
-#ifdef FOURIER_IDCT
- /* Discrete Fourier Transform requires the quantization matrices to
- * be normalized before using them. */
- vdec_NormQuantMatrix( p_matrix->pi_matrix );
-#endif
}
+
/*****************************************************************************
- * LinkMatrix : Link a quantization matrix to another
+ * SequenceScalableExtension : Parse the sequence_scalable_extension *
+ * structure to handle scalable coding *
*****************************************************************************/
-static void LinkMatrix( quant_matrix_t * p_matrix, int * pi_array )
+
+static void SequenceScalableExtension( vpar_thread_t * p_vpar )
{
- int i_dummy;
-
- if( p_matrix->b_allocated )
+ /* We don't care about anything scalable except the scalable mode. */
+ switch( p_vpar->sequence.i_scalable_mode = GetBits( &p_vpar->bit_stream, 2 ) )
+ /* The length of the structure depends on the value of the scalable_mode */
{
- /* Deallocate the piece of memory. */
- free( p_matrix->pi_matrix );
- p_matrix->b_allocated = FALSE;
+ case 1:
+ RemoveBits32( &p_vpar->bit_stream );
+ RemoveBits( &p_vpar->bit_stream, 21 );
+ break;
+ case 2:
+ RemoveBits( &p_vpar->bit_stream, 12 );
+ break;
+ default:
+ RemoveBits( &p_vpar->bit_stream, 4 );
}
-
- p_matrix->pi_matrix = pi_array;
+
}
+/*****************************************************************************
+ * PictureDisplayExtension : Parse the picture_display_extension structure *
+ *****************************************************************************/
+static void PictureDisplayExtension( vpar_thread_t * p_vpar )
+{
+ /* Number of frame center offset */
+ int i_nb, i_dummy;
+ /* I am not sure it works but it should
+ (fewer tests than shown in §6.3.12) */
+ i_nb = p_vpar->sequence.b_progressive ? p_vpar->sequence.b_progressive +
+ p_vpar->picture.b_repeat_first_field +
+ p_vpar->picture.b_top_field_first
+ : ( p_vpar->picture.b_frame_structure + 1 ) +
+ p_vpar->picture.b_repeat_first_field;
+ for( i_dummy = 0; i_dummy < i_nb; i_dummy++ )
+ {
+ RemoveBits( &p_vpar->bit_stream, 17 );
+ RemoveBits( &p_vpar->bit_stream, 17 );
+ }
+}
+
+
+/*****************************************************************************
+ * PictureSpatialScalableExtension *
+ *****************************************************************************/
+
+static void PictureSpatialScalableExtension( vpar_thread_t * p_vpar )
+{
+ /* That's scalable, so we trash it */
+ RemoveBits32( &p_vpar->bit_stream );
+ RemoveBits( &p_vpar->bit_stream, 16 );
+}
+
+
+/*****************************************************************************
+ * PictureTemporalScalableExtension *
+ *****************************************************************************/
+
+static void PictureTemporalScalableExtension( vpar_thread_t * p_vpar )
+{
+ /* Scalable again, trashed again */
+ RemoveBits( &p_vpar->bit_stream, 23 );
+}
+
+
+/*****************************************************************************
+ * CopyrightExtension : Keeps some legal informations *
+ *****************************************************************************/
+
+static void CopyrightExtension( vpar_thread_t * p_vpar )
+{
+ u32 i_copyright_nb_1, i_copyright_nb_2; /* local integers */
+ p_vpar->sequence.b_copyright_flag = GetBits( &p_vpar->bit_stream, 1 );
+ /* A flag that says whether the copyright information is significant */
+ p_vpar->sequence.i_copyright_id = GetBits( &p_vpar->bit_stream, 8 );
+ /* An identifier compliant with ISO/CEI JTC 1/SC 29 */
+ p_vpar->sequence.b_original = GetBits( &p_vpar->bit_stream, 1 );
+ /* Reserved bits */
+ RemoveBits( &p_vpar->bit_stream, 8 );
+ /* The copyright_number is split in three parts */
+ /* first part */
+ i_copyright_nb_1 = GetBits( &p_vpar->bit_stream, 20 );
+ RemoveBits( &p_vpar->bit_stream, 1 );
+ /* second part */
+ i_copyright_nb_2 = GetBits( &p_vpar->bit_stream, 22 );
+ RemoveBits( &p_vpar->bit_stream, 1 );
+ /* third part and sum */
+ p_vpar->sequence.i_copyright_nb = ( (u64)i_copyright_nb_1 << 44 ) |
+ ( (u64)i_copyright_nb_2 << 22 ) |
+ ( (u64)GetBits( &p_vpar->bit_stream, 22 ) );
+}