annotate eamad.c @ 12530:63edd10ad4bc libavcodec tip

Try to fix crashes introduced by r25218 r25218 made assumptions about the existence of past reference frames that weren't necessarily true.
author darkshikari
date Tue, 28 Sep 2010 09:06:22 +0000
parents ffb3668ff7af
children
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
rev   line source
9848
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
1 /*
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
2 * Electronic Arts Madcow Video Decoder
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
3 * Copyright (c) 2007-2009 Peter Ross
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
4 *
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
5 * This file is part of FFmpeg.
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
6 *
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
7 * FFmpeg is free software; you can redistribute it and/or
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
8 * modify it under the terms of the GNU Lesser General Public
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
9 * License as published by the Free Software Foundation; either
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
10 * version 2.1 of the License, or (at your option) any later version.
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
11 *
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
12 * FFmpeg is distributed in the hope that it will be useful,
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
13 * but WITHOUT ANY WARRANTY; without even the implied warranty of
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
15 * Lesser General Public License for more details.
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
16 *
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
17 * You should have received a copy of the GNU Lesser General Public
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
18 * License along with FFmpeg; if not, write to the Free Software
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
19 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
20 */
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
21
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
22 /**
11644
7dd2a45249a9 Remove explicit filename from Doxygen @file commands.
diego
parents: 11560
diff changeset
23 * @file
9848
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
24 * Electronic Arts Madcow Video Decoder
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
25 * by Peter Ross <pross@xvid.org>
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
26 *
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
27 * Technical details here:
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
28 * http://wiki.multimedia.cx/index.php?title=Electronic_Arts_MAD
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
29 */
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
30
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
31 #include "avcodec.h"
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
32 #include "get_bits.h"
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
33 #include "dsputil.h"
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
34 #include "aandcttab.h"
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
35 #include "mpeg12.h"
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
36 #include "mpeg12data.h"
12372
914f484bb476 Remove use of the deprecated function avcodec_check_dimensions(), use
stefano
parents: 12129
diff changeset
37 #include "libavcore/imgutils.h"
9848
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
38
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
39 #define EA_PREAMBLE_SIZE 8
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
40 #define MADk_TAG MKTAG('M', 'A', 'D', 'k') /* MAD i-frame */
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
41 #define MADm_TAG MKTAG('M', 'A', 'D', 'm') /* MAD p-frame */
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
42 #define MADe_TAG MKTAG('M', 'A', 'D', 'e') /* MAD lqp-frame */
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
43
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
44 typedef struct MadContext {
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
45 MpegEncContext s;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
46 AVFrame frame;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
47 AVFrame last_frame;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
48 void *bitstream_buf;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
49 unsigned int bitstream_buf_size;
11369
98970e51365a Remove DECLARE_ALIGNED_{8,16} macros
mru
parents: 10961
diff changeset
50 DECLARE_ALIGNED(16, DCTELEM, block)[64];
9848
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
51 } MadContext;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
52
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
53 static void bswap16_buf(uint16_t *dst, const uint16_t *src, int count)
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
54 {
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
55 int i;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
56 for (i=0; i<count; i++)
12129
8b28e74de2c0 Add av_ prefix to bswap macros
mru
parents: 11644
diff changeset
57 dst[i] = av_bswap16(src[i]);
9848
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
58 }
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
59
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
60 static av_cold int decode_init(AVCodecContext *avctx)
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
61 {
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
62 MadContext *t = avctx->priv_data;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
63 MpegEncContext *s = &t->s;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
64 s->avctx = avctx;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
65 avctx->pix_fmt = PIX_FMT_YUV420P;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
66 if (avctx->idct_algo == FF_IDCT_AUTO)
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
67 avctx->idct_algo = FF_IDCT_EA;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
68 dsputil_init(&s->dsp, avctx);
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
69 ff_init_scantable(s->dsp.idct_permutation, &s->intra_scantable, ff_zigzag_direct);
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
70 ff_mpeg12_init_vlcs();
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
71 return 0;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
72 }
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
73
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
74 static inline void comp(unsigned char *dst, int dst_stride,
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
75 unsigned char *src, int src_stride, int add)
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
76 {
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
77 int j, i;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
78 for (j=0; j<8; j++)
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
79 for (i=0; i<8; i++)
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
80 dst[j*dst_stride + i] = av_clip_uint8(src[j*src_stride + i] + add);
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
81 }
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
82
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
83 static inline void comp_block(MadContext *t, int mb_x, int mb_y,
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
84 int j, int mv_x, int mv_y, int add)
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
85 {
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
86 MpegEncContext *s = &t->s;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
87 if (j < 4) {
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
88 comp(t->frame.data[0] + (mb_y*16 + ((j&2)<<2))*t->frame.linesize[0] + mb_x*16 + ((j&1)<<3),
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
89 t->frame.linesize[0],
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
90 t->last_frame.data[0] + (mb_y*16 + ((j&2)<<2) + mv_y)*t->last_frame.linesize[0] + mb_x*16 + ((j&1)<<3) + mv_x,
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
91 t->last_frame.linesize[0], add);
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
92 } else if (!(s->avctx->flags & CODEC_FLAG_GRAY)) {
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
93 int index = j - 3;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
94 comp(t->frame.data[index] + (mb_y*8)*t->frame.linesize[index] + mb_x * 8,
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
95 t->frame.linesize[index],
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
96 t->last_frame.data[index] + (mb_y * 8 + (mv_y/2))*t->last_frame.linesize[index] + mb_x * 8 + (mv_x/2),
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
97 t->last_frame.linesize[index], add);
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
98 }
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
99 }
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
100
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
101 static inline void idct_put(MadContext *t, DCTELEM *block, int mb_x, int mb_y, int j)
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
102 {
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
103 MpegEncContext *s = &t->s;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
104 if (j < 4) {
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
105 s->dsp.idct_put(
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
106 t->frame.data[0] + (mb_y*16 + ((j&2)<<2))*t->frame.linesize[0] + mb_x*16 + ((j&1)<<3),
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
107 t->frame.linesize[0], block);
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
108 } else if (!(s->avctx->flags & CODEC_FLAG_GRAY)) {
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
109 int index = j - 3;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
110 s->dsp.idct_put(
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
111 t->frame.data[index] + (mb_y*8)*t->frame.linesize[index] + mb_x*8,
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
112 t->frame.linesize[index], block);
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
113 }
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
114 }
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
115
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
116 static inline void decode_block_intra(MadContext * t, DCTELEM * block)
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
117 {
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
118 MpegEncContext *s = &t->s;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
119 int level, i, j, run;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
120 RLTable *rl = &ff_rl_mpeg1;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
121 const uint8_t *scantable = s->intra_scantable.permutated;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
122 int16_t *quant_matrix = s->intra_matrix;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
123
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
124 block[0] = (128 + get_sbits(&s->gb, 8)) * quant_matrix[0];
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
125
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
126 /* The RL decoder is derived from mpeg1_decode_block_intra;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
127 Escaped level and run values a decoded differently */
9854
2428d32533f6 Initialise MAD decode_block_intra() loop iterator to zero.
pross
parents: 9848
diff changeset
128 i = 0;
9848
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
129 {
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
130 OPEN_READER(re, &s->gb);
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
131 /* now quantify & encode AC coefficients */
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
132 for (;;) {
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
133 UPDATE_CACHE(re, &s->gb);
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
134 GET_RL_VLC(level, run, re, &s->gb, rl->rl_vlc[0], TEX_VLC_BITS, 2, 0);
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
135
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
136 if (level == 127) {
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
137 break;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
138 } else if (level != 0) {
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
139 i += run;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
140 j = scantable[i];
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
141 level = (level*quant_matrix[j]) >> 4;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
142 level = (level-1)|1;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
143 level = (level ^ SHOW_SBITS(re, &s->gb, 1)) - SHOW_SBITS(re, &s->gb, 1);
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
144 LAST_SKIP_BITS(re, &s->gb, 1);
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
145 } else {
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
146 /* escape */
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
147 UPDATE_CACHE(re, &s->gb);
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
148 level = SHOW_SBITS(re, &s->gb, 10); SKIP_BITS(re, &s->gb, 10);
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
149
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
150 UPDATE_CACHE(re, &s->gb);
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
151 run = SHOW_UBITS(re, &s->gb, 6)+1; LAST_SKIP_BITS(re, &s->gb, 6);
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
152
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
153 i += run;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
154 j = scantable[i];
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
155 if (level < 0) {
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
156 level = -level;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
157 level = (level*quant_matrix[j]) >> 4;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
158 level = (level-1)|1;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
159 level = -level;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
160 } else {
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
161 level = (level*quant_matrix[j]) >> 4;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
162 level = (level-1)|1;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
163 }
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
164 }
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
165 if (i > 63) {
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
166 av_log(s->avctx, AV_LOG_ERROR, "ac-tex damaged at %d %d\n", s->mb_x, s->mb_y);
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
167 return;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
168 }
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
169
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
170 block[j] = level;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
171 }
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
172 CLOSE_READER(re, &s->gb);
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
173 }
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
174 }
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
175
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
176 static int decode_motion(GetBitContext *gb)
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
177 {
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
178 int value = 0;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
179 if (get_bits1(gb)) {
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
180 if (get_bits1(gb))
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
181 value = -17;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
182 value += get_bits(gb, 4) + 1;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
183 }
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
184 return value;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
185 }
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
186
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
187 static void decode_mb(MadContext *t, int inter)
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
188 {
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
189 MpegEncContext *s = &t->s;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
190 int mv_map = 0;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
191 int mv_x, mv_y;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
192 int j;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
193
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
194 if (inter) {
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
195 int v = decode210(&s->gb);
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
196 if (v < 2) {
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
197 mv_map = v ? get_bits(&s->gb, 6) : 63;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
198 mv_x = decode_motion(&s->gb);
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
199 mv_y = decode_motion(&s->gb);
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
200 } else {
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
201 mv_map = 0;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
202 }
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
203 }
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
204
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
205 for (j=0; j<6; j++) {
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
206 if (mv_map & (1<<j)) { // mv_x and mv_y are guarded by mv_map
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
207 int add = 2*decode_motion(&s->gb);
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
208 comp_block(t, s->mb_x, s->mb_y, j, mv_x, mv_y, add);
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
209 } else {
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
210 s->dsp.clear_block(t->block);
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
211 decode_block_intra(t, t->block);
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
212 idct_put(t, t->block, s->mb_x, s->mb_y, j);
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
213 }
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
214 }
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
215 }
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
216
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
217 static void calc_intra_matrix(MadContext *t, int qscale)
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
218 {
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
219 MpegEncContext *s = &t->s;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
220 int i;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
221
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
222 if (s->avctx->idct_algo == FF_IDCT_EA) {
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
223 s->intra_matrix[0] = (ff_inv_aanscales[0]*ff_mpeg1_default_intra_matrix[0]) >> 11;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
224 for (i=1; i<64; i++)
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
225 s->intra_matrix[i] = (ff_inv_aanscales[i]*ff_mpeg1_default_intra_matrix[i]*qscale + 32) >> 10;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
226 } else {
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
227 s->intra_matrix[0] = ff_mpeg1_default_intra_matrix[0];
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
228 for (i=1; i<64; i++)
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
229 s->intra_matrix[i] = (ff_mpeg1_default_intra_matrix[i]*qscale) << 1;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
230 }
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
231 }
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
232
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
233 static int decode_frame(AVCodecContext *avctx,
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
234 void *data, int *data_size,
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
235 AVPacket *avpkt)
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
236 {
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
237 const uint8_t *buf = avpkt->data;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
238 int buf_size = avpkt->size;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
239 const uint8_t *buf_end = buf+buf_size;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
240 MadContext *t = avctx->priv_data;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
241 MpegEncContext *s = &t->s;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
242 int chunk_type;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
243 int inter;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
244
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
245 if (buf_size < 17) {
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
246 av_log(avctx, AV_LOG_ERROR, "Input buffer too small\n");
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
247 *data_size = 0;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
248 return -1;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
249 }
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
250
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
251 chunk_type = AV_RL32(&buf[0]);
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
252 inter = (chunk_type == MADm_TAG || chunk_type == MADe_TAG);
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
253 buf += 8;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
254
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
255 av_reduce(&avctx->time_base.num, &avctx->time_base.den,
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
256 AV_RL16(&buf[6]), 1000, 1<<30);
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
257
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
258 s->width = AV_RL16(&buf[8]);
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
259 s->height = AV_RL16(&buf[10]);
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
260 calc_intra_matrix(t, buf[13]);
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
261 buf += 16;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
262
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
263 if (avctx->width != s->width || avctx->height != s->height) {
12462
ffb3668ff7af Use new imgutils.h API names, fix deprecation warnings.
stefano
parents: 12372
diff changeset
264 if (av_image_check_size(s->width, s->height, 0, avctx) < 0)
9848
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
265 return -1;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
266 avcodec_set_dimensions(avctx, s->width, s->height);
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
267 if (t->frame.data[0])
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
268 avctx->release_buffer(avctx, &t->frame);
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
269 }
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
270
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
271 t->frame.reference = 1;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
272 if (!t->frame.data[0]) {
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
273 if (avctx->get_buffer(avctx, &t->frame) < 0) {
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
274 av_log(avctx, AV_LOG_ERROR, "get_buffer() failed\n");
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
275 return -1;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
276 }
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
277 }
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
278
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
279 av_fast_malloc(&t->bitstream_buf, &t->bitstream_buf_size, (buf_end-buf) + FF_INPUT_BUFFER_PADDING_SIZE);
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
280 if (!t->bitstream_buf)
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
281 return AVERROR(ENOMEM);
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
282 bswap16_buf(t->bitstream_buf, (const uint16_t*)buf, (buf_end-buf)/2);
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
283 init_get_bits(&s->gb, t->bitstream_buf, 8*(buf_end-buf));
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
284
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
285 for (s->mb_y=0; s->mb_y < (avctx->height+15)/16; s->mb_y++)
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
286 for (s->mb_x=0; s->mb_x < (avctx->width +15)/16; s->mb_x++)
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
287 decode_mb(t, inter);
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
288
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
289 *data_size = sizeof(AVFrame);
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
290 *(AVFrame*)data = t->frame;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
291
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
292 if (chunk_type != MADe_TAG)
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
293 FFSWAP(AVFrame, t->frame, t->last_frame);
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
294
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
295 return buf_size;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
296 }
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
297
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
298 static av_cold int decode_end(AVCodecContext *avctx)
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
299 {
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
300 MadContext *t = avctx->priv_data;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
301 if (t->frame.data[0])
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
302 avctx->release_buffer(avctx, &t->frame);
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
303 if (t->last_frame.data[0])
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
304 avctx->release_buffer(avctx, &t->last_frame);
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
305 av_free(t->bitstream_buf);
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
306 return 0;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
307 }
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
308
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
309 AVCodec eamad_decoder = {
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
310 "eamad",
11560
8a4984c5cacc Define AVMediaType enum, and use it instead of enum CodecType, which
stefano
parents: 11369
diff changeset
311 AVMEDIA_TYPE_VIDEO,
9848
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
312 CODEC_ID_MAD,
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
313 sizeof(MadContext),
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
314 decode_init,
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
315 NULL,
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
316 decode_end,
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
317 decode_frame,
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
318 CODEC_CAP_DR1,
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
319 .long_name = NULL_IF_CONFIG_SMALL("Electronic Arts Madcow Video")
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
320 };