annotate mpegvideo_parser.c @ 5062:2dd00b1cc94b libavcodec

Remove mdct.o and fft.o from fft-test prerequisites list. Both objects were added to the link command, resulting in multiple definitions of symbols. Now linking works in the general case when mdct.o and fft.o are compiled into libavcodec.a.
author diego
date Tue, 22 May 2007 07:08:38 +0000
parents 0d1cc37d9430
children 9f4f88218b78
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 /* the packet only represents half a frame
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
107 XXX,FIXME maybe find a different solution */
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
108 if(picture_structure != 3)
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
109 s->repeat_pict = -1;
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
110 }
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
111 break;
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
112 }
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
113 }
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
114 break;
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
115 case -1:
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
116 goto the_end;
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
117 default:
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
118 /* we stop parsing when we encounter a slice. It ensures
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
119 that this function takes a negligible amount of time */
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
120 if (start_code >= SLICE_MIN_START_CODE &&
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
121 start_code <= SLICE_MAX_START_CODE)
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
122 goto the_end;
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
123 break;
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
124 }
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
125 }
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
126 the_end: ;
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
127 }
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 static int mpegvideo_parse(AVCodecParserContext *s,
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
130 AVCodecContext *avctx,
4931
0d1cc37d9430 make some parser parameters const to avoid casting const to non-const
aurel
parents: 4916
diff changeset
131 const uint8_t **poutbuf, int *poutbuf_size,
4915
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
132 const uint8_t *buf, int buf_size)
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
133 {
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
134 ParseContext1 *pc1 = s->priv_data;
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
135 ParseContext *pc= &pc1->pc;
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
136 int next;
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
137
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
138 if(s->flags & PARSER_FLAG_COMPLETE_FRAMES){
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
139 next= buf_size;
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
140 }else{
4916
13ef168891b0 add a ff_ prefix to the now exported mpeg1_find_frame_end() function
aurel
parents: 4915
diff changeset
141 next= ff_mpeg1_find_frame_end(pc, buf, buf_size);
4915
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
142
4931
0d1cc37d9430 make some parser parameters const to avoid casting const to non-const
aurel
parents: 4916
diff changeset
143 if (ff_combine_frame(pc, next, &buf, &buf_size) < 0) {
4915
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
144 *poutbuf = NULL;
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
145 *poutbuf_size = 0;
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
146 return buf_size;
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
147 }
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
148
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
149 }
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
150 /* 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
151 to have the full timing information. The time take by this
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
152 function should be negligible for uncorrupted streams */
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
153 mpegvideo_extract_headers(s, avctx, buf, buf_size);
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
154 #if 0
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
155 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
156 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
157 #endif
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
158
4931
0d1cc37d9430 make some parser parameters const to avoid casting const to non-const
aurel
parents: 4916
diff changeset
159 *poutbuf = buf;
4915
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
160 *poutbuf_size = buf_size;
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
161 return next;
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
162 }
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
163
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
164 static int mpegvideo_split(AVCodecContext *avctx,
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
165 const uint8_t *buf, int buf_size)
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
166 {
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
167 int i;
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
168 uint32_t state= -1;
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 for(i=0; i<buf_size; i++){
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
171 state= (state<<8) | buf[i];
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
172 if(state != 0x1B3 && state != 0x1B5 && state < 0x200 && state >= 0x100)
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
173 return i-3;
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
174 }
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
175 return 0;
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
176 }
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
177
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
178 AVCodecParser mpegvideo_parser = {
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
179 { CODEC_ID_MPEG1VIDEO, CODEC_ID_MPEG2VIDEO },
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
180 sizeof(ParseContext1),
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
181 NULL,
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
182 mpegvideo_parse,
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
183 ff_parse1_close,
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
184 mpegvideo_split,
c22e10113015 move mpegvideo_parser in it's own file
aurel
parents:
diff changeset
185 };