⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 header.c

📁 MPEG2的编码的加密和解码的C源程序
💻 C
📖 第 1 页 / 共 2 页
字号:
/* * 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 + -