📄 header.c
字号:
/* * header.c * Copyright (C) 2000-2003 Michel Lespinasse <walken@zoy.org> * Copyright (C) 2003 Regis Duchesne <hpreg@zoy.org> * Copyright (C) 1999-2000 Aaron Holtzman <aholtzma@ess.engr.uvic.ca> * * This file is part of mpeg2dec, a free MPEG-2 video stream decoder. * See http://libmpeg2.sourceforge.net/ for updates. * * mpeg2dec 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. * * mpeg2dec 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-1307 USA */#include "config.h"#include <inttypes.h>#include <stdlib.h> /* defines NULL */#include <string.h> /* memcmp */#include "mpeg2.h"#include "attributes.h"#include "mpeg2_internal.h"#define SEQ_EXT 2#define SEQ_DISPLAY_EXT 4#define QUANT_MATRIX_EXT 8#define COPYRIGHT_EXT 0x10#define PIC_DISPLAY_EXT 0x80#define PIC_CODING_EXT 0x100/* default intra quant matrix, in zig-zag order */static const uint8_t default_intra_quantizer_matrix[64] ATTR_ALIGN(16) = { 8, 16, 16, 19, 16, 19, 22, 22, 22, 22, 22, 22, 26, 24, 26, 27, 27, 27, 26, 26, 26, 26, 27, 27, 27, 29, 29, 29, 34, 34, 34, 29, 29, 29, 27, 27, 29, 29, 32, 32, 34, 34, 37, 38, 37, 35, 35, 34, 35, 38, 38, 40, 40, 40, 48, 48, 46, 46, 56, 56, 58, 69, 69, 83};uint8_t mpeg2_scan_norm[64] ATTR_ALIGN(16) = { /* Zig-Zag scan 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};uint8_t mpeg2_scan_alt[64] ATTR_ALIGN(16) = { /* 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};void mpeg2_header_state_init (mpeg2dec_t * mpeg2dec){ if (mpeg2dec->sequence.width != (unsigned)-1) { int i; mpeg2dec->sequence.width = (unsigned)-1; if (!mpeg2dec->custom_fbuf) for (i = mpeg2dec->alloc_index_user; i < mpeg2dec->alloc_index; i++) { mpeg2_free (mpeg2dec->fbuf_alloc[i].fbuf.buf[0]); mpeg2_free (mpeg2dec->fbuf_alloc[i].fbuf.buf[1]); mpeg2_free (mpeg2dec->fbuf_alloc[i].fbuf.buf[2]); } if (mpeg2dec->convert_start) for (i = 0; i < 3; i++) { mpeg2_free (mpeg2dec->yuv_buf[i][0]); mpeg2_free (mpeg2dec->yuv_buf[i][1]); mpeg2_free (mpeg2dec->yuv_buf[i][2]); } if (mpeg2dec->decoder.convert_id) mpeg2_free (mpeg2dec->decoder.convert_id); } mpeg2dec->decoder.coding_type = I_TYPE; mpeg2dec->decoder.convert = NULL; mpeg2dec->decoder.convert_id = NULL; mpeg2dec->picture = mpeg2dec->pictures; mpeg2dec->fbuf[0] = &mpeg2dec->fbuf_alloc[0].fbuf; mpeg2dec->fbuf[1] = &mpeg2dec->fbuf_alloc[1].fbuf; mpeg2dec->fbuf[2] = &mpeg2dec->fbuf_alloc[2].fbuf; mpeg2dec->first = 1; mpeg2dec->alloc_index = 0; mpeg2dec->alloc_index_user = 0; mpeg2dec->first_decode_slice = 1; mpeg2dec->nb_decode_slices = 0xb0 - 1; mpeg2dec->convert = NULL; mpeg2dec->convert_start = NULL; mpeg2dec->custom_fbuf = 0; mpeg2dec->yuv_index = 0;}void mpeg2_reset_info (mpeg2_info_t * info){ info->current_picture = info->current_picture_2nd = NULL; info->display_picture = info->display_picture_2nd = NULL; info->current_fbuf = info->display_fbuf = info->discard_fbuf = NULL;}static void info_user_data (mpeg2dec_t * mpeg2dec){ if (mpeg2dec->user_data_len) { mpeg2dec->info.user_data = mpeg2dec->chunk_buffer; mpeg2dec->info.user_data_len = mpeg2dec->user_data_len - 3; }}int mpeg2_header_sequence (mpeg2dec_t * mpeg2dec){ uint8_t * buffer = mpeg2dec->chunk_start; mpeg2_sequence_t * sequence = &(mpeg2dec->new_sequence); static unsigned int frame_period[16] = { 0, 1126125, 1125000, 1080000, 900900, 900000, 540000, 450450, 450000, /* unofficial: xing 15 fps */ 1800000, /* unofficial: libmpeg3 "Unofficial economy rates" 5/10/12/15 fps */ 5400000, 2700000, 2250000, 1800000, 0, 0 }; int i; if ((buffer[6] & 0x20) != 0x20) /* missing marker_bit */ return 1; i = (buffer[0] << 16) | (buffer[1] << 8) | buffer[2]; if (! (sequence->display_width = sequence->picture_width = i >> 12)) return 1; if (! (sequence->display_height = sequence->picture_height = i & 0xfff)) return 1; sequence->width = (sequence->picture_width + 15) & ~15; sequence->height = (sequence->picture_height + 15) & ~15; sequence->chroma_width = sequence->width >> 1; sequence->chroma_height = sequence->height >> 1; sequence->flags = (SEQ_FLAG_PROGRESSIVE_SEQUENCE | SEQ_VIDEO_FORMAT_UNSPECIFIED); sequence->pixel_width = buffer[3] >> 4; /* aspect ratio */ sequence->frame_period = frame_period[buffer[3] & 15]; sequence->byte_rate = (buffer[4]<<10) | (buffer[5]<<2) | (buffer[6]>>6); sequence->vbv_buffer_size = ((buffer[6]<<16)|(buffer[7]<<8))&0x1ff800; if (buffer[7] & 4) sequence->flags |= SEQ_FLAG_CONSTRAINED_PARAMETERS; mpeg2dec->copy_matrix = 3; if (buffer[7] & 2) { for (i = 0; i < 64; i++) mpeg2dec->new_quantizer_matrix[0][mpeg2_scan_norm[i]] = (buffer[i+7] << 7) | (buffer[i+8] >> 1); buffer += 64; } else for (i = 0; i < 64; i++) mpeg2dec->new_quantizer_matrix[0][mpeg2_scan_norm[i]] = default_intra_quantizer_matrix[i]; if (buffer[7] & 1) for (i = 0; i < 64; i++) mpeg2dec->new_quantizer_matrix[1][mpeg2_scan_norm[i]] = buffer[i+8]; else memset (mpeg2dec->new_quantizer_matrix[1], 16, 64); sequence->profile_level_id = 0x80; sequence->colour_primaries = 0; sequence->transfer_characteristics = 0; sequence->matrix_coefficients = 0; mpeg2dec->ext_state = SEQ_EXT; mpeg2dec->state = STATE_SEQUENCE; mpeg2dec->display_offset_x = mpeg2dec->display_offset_y = 0; return 0;}static int sequence_ext (mpeg2dec_t * mpeg2dec){ uint8_t * buffer = mpeg2dec->chunk_start; mpeg2_sequence_t * sequence = &(mpeg2dec->new_sequence); uint32_t flags; if (!(buffer[3] & 1)) return 1; sequence->profile_level_id = (buffer[0] << 4) | (buffer[1] >> 4); sequence->display_width = sequence->picture_width += ((buffer[1] << 13) | (buffer[2] << 5)) & 0x3000; sequence->display_height = sequence->picture_height += (buffer[2] << 7) & 0x3000; sequence->width = (sequence->picture_width + 15) & ~15; sequence->height = (sequence->picture_height + 15) & ~15; flags = sequence->flags | SEQ_FLAG_MPEG2; if (!(buffer[1] & 8)) { flags &= ~SEQ_FLAG_PROGRESSIVE_SEQUENCE; sequence->height = (sequence->height + 31) & ~31; } if (buffer[5] & 0x80) flags |= SEQ_FLAG_LOW_DELAY; sequence->flags = flags; sequence->chroma_width = sequence->width; sequence->chroma_height = sequence->height; switch (buffer[1] & 6) { case 0: /* invalid */ return 1; case 2: /* 4:2:0 */ sequence->chroma_height >>= 1; case 4: /* 4:2:2 */ sequence->chroma_width >>= 1; } sequence->byte_rate += ((buffer[2]<<25) | (buffer[3]<<17)) & 0x3ffc0000; sequence->vbv_buffer_size |= buffer[4] << 21; sequence->frame_period = sequence->frame_period * ((buffer[5]&31)+1) / (((buffer[5]>>2)&3)+1); mpeg2dec->ext_state = SEQ_DISPLAY_EXT; return 0;}static int sequence_display_ext (mpeg2dec_t * mpeg2dec){ uint8_t * buffer = mpeg2dec->chunk_start; mpeg2_sequence_t * sequence = &(mpeg2dec->new_sequence); uint32_t flags; flags = ((sequence->flags & ~SEQ_MASK_VIDEO_FORMAT) | ((buffer[0]<<4) & SEQ_MASK_VIDEO_FORMAT)); if (buffer[0] & 1) { flags |= SEQ_FLAG_COLOUR_DESCRIPTION; sequence->colour_primaries = buffer[1]; sequence->transfer_characteristics = buffer[2]; sequence->matrix_coefficients = buffer[3]; buffer += 3; } if (!(buffer[2] & 2)) /* missing marker_bit */ return 1; if( (buffer[1] << 6) | (buffer[2] >> 2) ) sequence->display_width = (buffer[1] << 6) | (buffer[2] >> 2); if( ((buffer[2]& 1 ) << 13) | (buffer[3] << 5) | (buffer[4] >> 3) ) sequence->display_height = ((buffer[2]& 1 ) << 13) | (buffer[3] << 5) | (buffer[4] >> 3); return 0;}static inline void finalize_sequence (mpeg2_sequence_t * sequence){ int width; int height; sequence->byte_rate *= 50; if (sequence->flags & SEQ_FLAG_MPEG2) { switch (sequence->pixel_width) { case 1: /* square pixels */ sequence->pixel_width = sequence->pixel_height = 1; return; case 2: /* 4:3 aspect ratio */ width = 4; height = 3; break; case 3: /* 16:9 aspect ratio */ width = 16; height = 9; break; case 4: /* 2.21:1 aspect ratio */ width = 221; height = 100; break; default: /* illegal */ sequence->pixel_width = sequence->pixel_height = 0; return; } width *= sequence->display_height; height *= sequence->display_width; } else { if (sequence->byte_rate == 50 * 0x3ffff) sequence->byte_rate = 0; /* mpeg-1 VBR */ switch (sequence->pixel_width) { case 0: case 15: /* illegal */ sequence->pixel_width = sequence->pixel_height = 0; return; case 1: /* square pixels */ sequence->pixel_width = sequence->pixel_height = 1; return; case 3: /* 720x576 16:9 */ sequence->pixel_width = 64; sequence->pixel_height = 45; return; case 6: /* 720x480 16:9 */ sequence->pixel_width = 32; sequence->pixel_height = 27; return; case 12: /* 720*480 4:3 */ sequence->pixel_width = 8; sequence->pixel_height = 9; return; default: height = 88 * sequence->pixel_width + 1171; width = 2000; } } sequence->pixel_width = width; sequence->pixel_height = height; while (width) { /* find greatest common divisor */ int tmp = width; width = height % tmp; height = tmp; } sequence->pixel_width /= height; sequence->pixel_height /= height;}static void copy_matrix (mpeg2dec_t * mpeg2dec, int index){ if (memcmp (mpeg2dec->quantizer_matrix[index], mpeg2dec->new_quantizer_matrix[index], 64)) { memcpy (mpeg2dec->quantizer_matrix[index], mpeg2dec->new_quantizer_matrix[index], 64); mpeg2dec->scaled[index] = -1; }}static void finalize_matrix (mpeg2dec_t * mpeg2dec){ mpeg2_decoder_t * decoder = &(mpeg2dec->decoder); int i; for (i = 0; i < 2; i++) { if (mpeg2dec->copy_matrix & (1 << i)) copy_matrix (mpeg2dec, i); if ((mpeg2dec->copy_matrix & (4 << i)) && memcmp (mpeg2dec->quantizer_matrix[i], mpeg2dec->new_quantizer_matrix[i+2], 64)) { copy_matrix (mpeg2dec, i + 2); decoder->chroma_quantizer[i] = decoder->quantizer_prescale[i+2]; } else if (mpeg2dec->copy_matrix & (5 << i)) decoder->chroma_quantizer[i] = decoder->quantizer_prescale[i]; }}static mpeg2_state_t invalid_end_action (mpeg2dec_t * mpeg2dec){ mpeg2_reset_info (&(mpeg2dec->info)); mpeg2dec->info.gop = NULL; info_user_data (mpeg2dec); mpeg2_header_state_init (mpeg2dec); mpeg2dec->sequence = mpeg2dec->new_sequence; mpeg2dec->action = mpeg2_seek_header; mpeg2dec->state = STATE_SEQUENCE; return STATE_SEQUENCE;}void mpeg2_header_sequence_finalize (mpeg2dec_t * mpeg2dec){ mpeg2_sequence_t * sequence = &(mpeg2dec->new_sequence); mpeg2_decoder_t * decoder = &(mpeg2dec->decoder); finalize_sequence (sequence); finalize_matrix (mpeg2dec); decoder->mpeg1 = !(sequence->flags & SEQ_FLAG_MPEG2); decoder->width = sequence->width; decoder->height = sequence->height; decoder->vertical_position_extension = (sequence->picture_height > 2800); decoder->chroma_format = ((sequence->chroma_width == sequence->width) + (sequence->chroma_height == sequence->height)); if (mpeg2dec->sequence.width != (unsigned)-1) { unsigned int new_byte_rate; /* * According to 6.1.1.6, repeat sequence headers should be * identical to the original. However some DVDs dont respect * that and have different bitrates in the repeat sequence * headers. So we'll ignore that in the comparison and still * consider these as repeat sequence headers. * * However, be careful not to alter the current sequence when * returning STATE_INVALID_END. */ new_byte_rate = sequence->byte_rate; sequence->byte_rate = mpeg2dec->sequence.byte_rate; if (memcmp (&(mpeg2dec->sequence), sequence, sizeof (mpeg2_sequence_t))) { decoder->stride_frame = sequence->width; sequence->byte_rate = new_byte_rate; mpeg2_header_end (mpeg2dec); mpeg2dec->action = invalid_end_action; mpeg2dec->state = STATE_INVALID_END; return; } sequence->byte_rate = new_byte_rate; mpeg2dec->state = STATE_SEQUENCE_REPEATED; } else decoder->stride_frame = sequence->width; mpeg2dec->sequence = *sequence; mpeg2_reset_info (&(mpeg2dec->info)); mpeg2dec->info.sequence = &(mpeg2dec->sequence); mpeg2dec->info.gop = NULL; info_user_data (mpeg2dec);}int mpeg2_header_gop (mpeg2dec_t * mpeg2dec){ uint8_t * buffer = mpeg2dec->chunk_start; mpeg2_gop_t * gop = &(mpeg2dec->new_gop); if (! (buffer[1] & 8)) return 1; gop->hours = (buffer[0] >> 2) & 31; gop->minutes = ((buffer[0] << 4) | (buffer[1] >> 4)) & 63; gop->seconds = ((buffer[1] << 3) | (buffer[2] >> 5)) & 63; gop->pictures = ((buffer[2] << 1) | (buffer[3] >> 7)) & 63; gop->flags = (buffer[0] >> 7) | ((buffer[3] >> 4) & 6); mpeg2dec->state = STATE_GOP; return 0;}void mpeg2_header_gop_finalize (mpeg2dec_t * mpeg2dec){ mpeg2dec->gop = mpeg2dec->new_gop; mpeg2_reset_info (&(mpeg2dec->info)); mpeg2dec->info.gop = &(mpeg2dec->gop); info_user_data (mpeg2dec);}void mpeg2_set_fbuf (mpeg2dec_t * mpeg2dec, int b_type){
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -