annotate mpegvideo_parser.c @ 6582:7d40da88f0c7 libavcodec

Crop parameters are unsigned, having them negative could be bad and lead to crashes or maybe exploits (not checked at all if this is possible).
author michael
date Fri, 11 Apr 2008 12:57:49 +0000
parents cb31fe7bd914
children 5a5ded44808a
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
rev   line source
4915
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
1 /*
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
2 * MPEG1 / MPEG2 video parser
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
3 * Copyright (c) 2000,2001 Fabrice Bellard.
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
4 * Copyright (c) 2002-2004 Michael Niedermayer <michaelni@gmx.at>
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
5 *
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
6 * This file is part of FFmpeg.
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
7 *
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
8 * FFmpeg is free software; you can redistribute it and/or
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
9 * modify it under the terms of the GNU Lesser General Public
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
10 * License as published by the Free Software Foundation; either
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
11 * version 2.1 of the License, or (at your option) any later version.
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
12 *
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
13 * FFmpeg is distributed in the hope that it will be useful,
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
14 * but WITHOUT ANY WARRANTY; without even the implied warranty of
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
15 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
16 * Lesser General Public License for more details.
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
17 *
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
18 * You should have received a copy of the GNU Lesser General Public
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
19 * License along with FFmpeg; if not, write to the Free Software
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
20 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
21 */
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
22
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
23 #include "parser.h"
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
24 #include "mpegvideo.h"
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
25
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
26 static void mpegvideo_extract_headers(AVCodecParserContext *s,
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
27 AVCodecContext *avctx,
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
28 const uint8_t *buf, int buf_size)
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
29 {
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
30 ParseContext1 *pc = s->priv_data;
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
31 const uint8_t *buf_end;
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
32 uint32_t start_code;
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
33 int frame_rate_index, ext_type, bytes_left;
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
34 int frame_rate_ext_n, frame_rate_ext_d;
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
35 int picture_structure, top_field_first, repeat_first_field, progressive_frame;
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
36 int horiz_size_ext, vert_size_ext, bit_rate_ext;
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
37 //FIXME replace the crap with get_bits()
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
38 s->repeat_pict = 0;
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
39 buf_end = buf + buf_size;
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
40 while (buf < buf_end) {
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
41 start_code= -1;
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
42 buf= ff_find_start_code(buf, buf_end, &start_code);
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
43 bytes_left = buf_end - buf;
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
44 switch(start_code) {
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
45 case PICTURE_START_CODE:
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
46 if (bytes_left >= 2) {
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
47 s->pict_type = (buf[1] >> 3) & 7;
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
48 }
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
49 break;
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
50 case SEQ_START_CODE:
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
51 if (bytes_left >= 7) {
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
52 pc->width = (buf[0] << 4) | (buf[1] >> 4);
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
53 pc->height = ((buf[1] & 0x0f) << 8) | buf[2];
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
54 avcodec_set_dimensions(avctx, pc->width, pc->height);
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
55 frame_rate_index = buf[3] & 0xf;
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
56 pc->frame_rate.den = avctx->time_base.den = ff_frame_rate_tab[frame_rate_index].num;
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
57 pc->frame_rate.num = avctx->time_base.num = ff_frame_rate_tab[frame_rate_index].den;
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
58 avctx->bit_rate = ((buf[4]<<10) | (buf[5]<<2) | (buf[6]>>6))*400;
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
59 avctx->codec_id = CODEC_ID_MPEG1VIDEO;
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
60 avctx->sub_id = 1;
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
61 }
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
62 break;
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
63 case EXT_START_CODE:
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
64 if (bytes_left >= 1) {
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
65 ext_type = (buf[0] >> 4);
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
66 switch(ext_type) {
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
67 case 0x1: /* sequence extension */
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
68 if (bytes_left >= 6) {
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
69 horiz_size_ext = ((buf[1] & 1) << 1) | (buf[2] >> 7);
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
70 vert_size_ext = (buf[2] >> 5) & 3;
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
71 bit_rate_ext = ((buf[2] & 0x1F)<<7) | (buf[3]>>1);
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
72 frame_rate_ext_n = (buf[5] >> 5) & 3;
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
73 frame_rate_ext_d = (buf[5] & 0x1f);
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
74 pc->progressive_sequence = buf[1] & (1 << 3);
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
75 avctx->has_b_frames= !(buf[5] >> 7);
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
76
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
77 pc->width |=(horiz_size_ext << 12);
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
78 pc->height |=( vert_size_ext << 12);
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
79 avctx->bit_rate += (bit_rate_ext << 18) * 400;
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
80 avcodec_set_dimensions(avctx, pc->width, pc->height);
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
81 avctx->time_base.den = pc->frame_rate.den * (frame_rate_ext_n + 1);
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
82 avctx->time_base.num = pc->frame_rate.num * (frame_rate_ext_d + 1);
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
83 avctx->codec_id = CODEC_ID_MPEG2VIDEO;
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
84 avctx->sub_id = 2; /* forces MPEG2 */
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
85 }
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
86 break;
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
87 case 0x8: /* picture coding extension */
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
88 if (bytes_left >= 5) {
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
89 picture_structure = buf[2]&3;
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
90 top_field_first = buf[3] & (1 << 7);
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
91 repeat_first_field = buf[3] & (1 << 1);
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
92 progressive_frame = buf[4] & (1 << 7);
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
93
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
94 /* check if we must repeat the frame */
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
95 if (repeat_first_field) {
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
96 if (pc->progressive_sequence) {
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
97 if (top_field_first)
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
98 s->repeat_pict = 4;
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
99 else
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
100 s->repeat_pict = 2;
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
101 } else if (progressive_frame) {
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
102 s->repeat_pict = 1;
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
103 }
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
104 }
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
105 }
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
106 break;
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
107 }
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
108 }
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
109 break;
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
110 case -1:
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
111 goto the_end;
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
112 default:
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
113 /* we stop parsing when we encounter a slice. It ensures
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
114 that this function takes a negligible amount of time */
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
115 if (start_code >= SLICE_MIN_START_CODE &&
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
116 start_code <= SLICE_MAX_START_CODE)
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
117 goto the_end;
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
118 break;
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
119 }
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
120 }
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
121 the_end: ;
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
122 }
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
123
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
124 static int mpegvideo_parse(AVCodecParserContext *s,
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
125 AVCodecContext *avctx,
4931
0d1cc37d9430 make some parser parameters const to avoid casting const to non-const
aurel
parents: 4916
diff changeset
126 const uint8_t **poutbuf, int *poutbuf_size,
4915
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
127 const uint8_t *buf, int buf_size)
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
128 {
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
129 ParseContext1 *pc1 = s->priv_data;
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
130 ParseContext *pc= &pc1->pc;
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
131 int next;
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
132
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
133 if(s->flags & PARSER_FLAG_COMPLETE_FRAMES){
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
134 next= buf_size;
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
135 }else{
4916
13ef168891b0 add a ff_ prefix to the now exported mpeg1_find_frame_end() function
aurel
parents: 4915
diff changeset
136 next= ff_mpeg1_find_frame_end(pc, buf, buf_size);
4915
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
137
4931
0d1cc37d9430 make some parser parameters const to avoid casting const to non-const
aurel
parents: 4916
diff changeset
138 if (ff_combine_frame(pc, next, &buf, &buf_size) < 0) {
4915
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
139 *poutbuf = NULL;
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
140 *poutbuf_size = 0;
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
141 return buf_size;
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
142 }
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
143
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
144 }
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
145 /* we have a full frame : we just parse the first few MPEG headers
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
146 to have the full timing information. The time take by this
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
147 function should be negligible for uncorrupted streams */
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
148 mpegvideo_extract_headers(s, avctx, buf, buf_size);
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
149 #if 0
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
150 printf("pict_type=%d frame_rate=%0.3f repeat_pict=%d\n",
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
151 s->pict_type, (double)avctx->time_base.den / avctx->time_base.num, s->repeat_pict);
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
152 #endif
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
153
4931
0d1cc37d9430 make some parser parameters const to avoid casting const to non-const
aurel
parents: 4916
diff changeset
154 *poutbuf = buf;
4915
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
155 *poutbuf_size = buf_size;
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
156 return next;
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
157 }
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
158
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
159 static int mpegvideo_split(AVCodecContext *avctx,
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
160 const uint8_t *buf, int buf_size)
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
161 {
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
162 int i;
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
163 uint32_t state= -1;
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
164
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
165 for(i=0; i<buf_size; i++){
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
166 state= (state<<8) | buf[i];
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
167 if(state != 0x1B3 && state != 0x1B5 && state < 0x200 && state >= 0x100)
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
168 return i-3;
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
169 }
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
170 return 0;
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
171 }
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
172
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
173 AVCodecParser mpegvideo_parser = {
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
174 { CODEC_ID_MPEG1VIDEO, CODEC_ID_MPEG2VIDEO },
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
175 sizeof(ParseContext1),
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
176 NULL,
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
177 mpegvideo_parse,
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
178 ff_parse1_close,
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
179 };