annotate eamad.c @ 11762:91b9bd17e79c libavcodec

aacenc: Trellis over scalefactors using an estimated codebook rather than every codebook. The minimal codebook to encode the band without clipping is used (as is done in the TLS).
author alexc
date Tue, 25 May 2010 18:31:55 +0000
parents 7dd2a45249a9
children 8b28e74de2c0
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"
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
37
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
38 #define EA_PREAMBLE_SIZE 8
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
39 #define MADk_TAG MKTAG('M', 'A', 'D', 'k') /* MAD i-frame */
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
40 #define MADm_TAG MKTAG('M', 'A', 'D', 'm') /* MAD p-frame */
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
41 #define MADe_TAG MKTAG('M', 'A', 'D', 'e') /* MAD lqp-frame */
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
42
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
43 typedef struct MadContext {
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
44 MpegEncContext s;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
45 AVFrame frame;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
46 AVFrame last_frame;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
47 void *bitstream_buf;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
48 unsigned int bitstream_buf_size;
11369
98970e51365a Remove DECLARE_ALIGNED_{8,16} macros
mru
parents: 10961
diff changeset
49 DECLARE_ALIGNED(16, DCTELEM, block)[64];
9848
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
50 } MadContext;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
51
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
52 static void bswap16_buf(uint16_t *dst, const uint16_t *src, int count)
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
53 {
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
54 int i;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
55 for (i=0; i<count; i++)
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
56 dst[i] = bswap_16(src[i]);
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
57 }
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
58
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
59 static av_cold int decode_init(AVCodecContext *avctx)
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
60 {
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
61 MadContext *t = avctx->priv_data;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
62 MpegEncContext *s = &t->s;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
63 s->avctx = avctx;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
64 avctx->pix_fmt = PIX_FMT_YUV420P;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
65 if (avctx->idct_algo == FF_IDCT_AUTO)
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
66 avctx->idct_algo = FF_IDCT_EA;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
67 dsputil_init(&s->dsp, avctx);
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
68 ff_init_scantable(s->dsp.idct_permutation, &s->intra_scantable, ff_zigzag_direct);
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
69 ff_mpeg12_init_vlcs();
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
70 return 0;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
71 }
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
72
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
73 static inline void comp(unsigned char *dst, int dst_stride,
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
74 unsigned char *src, int src_stride, int add)
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
75 {
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
76 int j, i;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
77 for (j=0; j<8; j++)
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
78 for (i=0; i<8; i++)
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
79 dst[j*dst_stride + i] = av_clip_uint8(src[j*src_stride + i] + add);
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
80 }
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
81
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
82 static inline void comp_block(MadContext *t, int mb_x, int mb_y,
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
83 int j, int mv_x, int mv_y, int add)
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
84 {
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
85 MpegEncContext *s = &t->s;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
86 if (j < 4) {
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
87 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
88 t->frame.linesize[0],
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
89 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
90 t->last_frame.linesize[0], add);
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
91 } else if (!(s->avctx->flags & CODEC_FLAG_GRAY)) {
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
92 int index = j - 3;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
93 comp(t->frame.data[index] + (mb_y*8)*t->frame.linesize[index] + mb_x * 8,
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
94 t->frame.linesize[index],
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
95 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
96 t->last_frame.linesize[index], add);
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
97 }
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 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
101 {
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
102 MpegEncContext *s = &t->s;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
103 if (j < 4) {
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
104 s->dsp.idct_put(
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
105 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
106 t->frame.linesize[0], block);
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
107 } else if (!(s->avctx->flags & CODEC_FLAG_GRAY)) {
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
108 int index = j - 3;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
109 s->dsp.idct_put(
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
110 t->frame.data[index] + (mb_y*8)*t->frame.linesize[index] + mb_x*8,
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
111 t->frame.linesize[index], block);
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
112 }
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 static inline void decode_block_intra(MadContext * t, DCTELEM * block)
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
116 {
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
117 MpegEncContext *s = &t->s;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
118 int level, i, j, run;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
119 RLTable *rl = &ff_rl_mpeg1;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
120 const uint8_t *scantable = s->intra_scantable.permutated;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
121 int16_t *quant_matrix = s->intra_matrix;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
122
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
123 block[0] = (128 + get_sbits(&s->gb, 8)) * quant_matrix[0];
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
124
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
125 /* The RL decoder is derived from mpeg1_decode_block_intra;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
126 Escaped level and run values a decoded differently */
9854
2428d32533f6 Initialise MAD decode_block_intra() loop iterator to zero.
pross
parents: 9848
diff changeset
127 i = 0;
9848
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
128 {
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
129 OPEN_READER(re, &s->gb);
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
130 /* now quantify & encode AC coefficients */
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
131 for (;;) {
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
132 UPDATE_CACHE(re, &s->gb);
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
133 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
134
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
135 if (level == 127) {
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
136 break;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
137 } else if (level != 0) {
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
138 i += run;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
139 j = scantable[i];
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
140 level = (level*quant_matrix[j]) >> 4;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
141 level = (level-1)|1;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
142 level = (level ^ SHOW_SBITS(re, &s->gb, 1)) - SHOW_SBITS(re, &s->gb, 1);
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
143 LAST_SKIP_BITS(re, &s->gb, 1);
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
144 } else {
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
145 /* escape */
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
146 UPDATE_CACHE(re, &s->gb);
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
147 level = SHOW_SBITS(re, &s->gb, 10); SKIP_BITS(re, &s->gb, 10);
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
148
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
149 UPDATE_CACHE(re, &s->gb);
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
150 run = SHOW_UBITS(re, &s->gb, 6)+1; LAST_SKIP_BITS(re, &s->gb, 6);
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
151
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
152 i += run;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
153 j = scantable[i];
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
154 if (level < 0) {
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
155 level = -level;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
156 level = (level*quant_matrix[j]) >> 4;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
157 level = (level-1)|1;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
158 level = -level;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
159 } else {
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
160 level = (level*quant_matrix[j]) >> 4;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
161 level = (level-1)|1;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
162 }
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
163 }
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
164 if (i > 63) {
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
165 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
166 return;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
167 }
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
168
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
169 block[j] = level;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
170 }
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
171 CLOSE_READER(re, &s->gb);
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
172 }
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 static int decode_motion(GetBitContext *gb)
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
176 {
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
177 int value = 0;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
178 if (get_bits1(gb)) {
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
179 if (get_bits1(gb))
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
180 value = -17;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
181 value += get_bits(gb, 4) + 1;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
182 }
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
183 return value;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
184 }
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
185
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
186 static void decode_mb(MadContext *t, int inter)
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
187 {
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
188 MpegEncContext *s = &t->s;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
189 int mv_map = 0;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
190 int mv_x, mv_y;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
191 int j;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
192
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
193 if (inter) {
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
194 int v = decode210(&s->gb);
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
195 if (v < 2) {
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
196 mv_map = v ? get_bits(&s->gb, 6) : 63;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
197 mv_x = decode_motion(&s->gb);
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
198 mv_y = decode_motion(&s->gb);
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
199 } else {
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
200 mv_map = 0;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
201 }
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 for (j=0; j<6; j++) {
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
205 if (mv_map & (1<<j)) { // mv_x and mv_y are guarded by mv_map
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
206 int add = 2*decode_motion(&s->gb);
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
207 comp_block(t, s->mb_x, s->mb_y, j, mv_x, mv_y, add);
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
208 } else {
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
209 s->dsp.clear_block(t->block);
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
210 decode_block_intra(t, t->block);
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
211 idct_put(t, t->block, s->mb_x, s->mb_y, j);
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
212 }
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 static void calc_intra_matrix(MadContext *t, int qscale)
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
217 {
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
218 MpegEncContext *s = &t->s;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
219 int i;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
220
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
221 if (s->avctx->idct_algo == FF_IDCT_EA) {
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
222 s->intra_matrix[0] = (ff_inv_aanscales[0]*ff_mpeg1_default_intra_matrix[0]) >> 11;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
223 for (i=1; i<64; i++)
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
224 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
225 } else {
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
226 s->intra_matrix[0] = ff_mpeg1_default_intra_matrix[0];
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
227 for (i=1; i<64; i++)
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
228 s->intra_matrix[i] = (ff_mpeg1_default_intra_matrix[i]*qscale) << 1;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
229 }
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 static int decode_frame(AVCodecContext *avctx,
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
233 void *data, int *data_size,
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
234 AVPacket *avpkt)
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
235 {
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
236 const uint8_t *buf = avpkt->data;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
237 int buf_size = avpkt->size;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
238 const uint8_t *buf_end = buf+buf_size;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
239 MadContext *t = avctx->priv_data;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
240 MpegEncContext *s = &t->s;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
241 int chunk_type;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
242 int inter;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
243
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
244 if (buf_size < 17) {
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
245 av_log(avctx, AV_LOG_ERROR, "Input buffer too small\n");
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
246 *data_size = 0;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
247 return -1;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
248 }
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
249
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
250 chunk_type = AV_RL32(&buf[0]);
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
251 inter = (chunk_type == MADm_TAG || chunk_type == MADe_TAG);
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
252 buf += 8;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
253
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
254 av_reduce(&avctx->time_base.num, &avctx->time_base.den,
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
255 AV_RL16(&buf[6]), 1000, 1<<30);
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
256
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
257 s->width = AV_RL16(&buf[8]);
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
258 s->height = AV_RL16(&buf[10]);
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
259 calc_intra_matrix(t, buf[13]);
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
260 buf += 16;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
261
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
262 if (avctx->width != s->width || avctx->height != s->height) {
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
263 if (avcodec_check_dimensions(avctx, s->width, s->height) < 0)
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
264 return -1;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
265 avcodec_set_dimensions(avctx, s->width, s->height);
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
266 if (t->frame.data[0])
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
267 avctx->release_buffer(avctx, &t->frame);
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
268 }
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
269
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
270 t->frame.reference = 1;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
271 if (!t->frame.data[0]) {
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
272 if (avctx->get_buffer(avctx, &t->frame) < 0) {
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
273 av_log(avctx, AV_LOG_ERROR, "get_buffer() failed\n");
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
274 return -1;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
275 }
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 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
279 if (!t->bitstream_buf)
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
280 return AVERROR(ENOMEM);
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
281 bswap16_buf(t->bitstream_buf, (const uint16_t*)buf, (buf_end-buf)/2);
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
282 init_get_bits(&s->gb, t->bitstream_buf, 8*(buf_end-buf));
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
283
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
284 for (s->mb_y=0; s->mb_y < (avctx->height+15)/16; s->mb_y++)
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
285 for (s->mb_x=0; s->mb_x < (avctx->width +15)/16; s->mb_x++)
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
286 decode_mb(t, inter);
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
287
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
288 *data_size = sizeof(AVFrame);
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
289 *(AVFrame*)data = t->frame;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
290
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
291 if (chunk_type != MADe_TAG)
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
292 FFSWAP(AVFrame, t->frame, t->last_frame);
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
293
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
294 return buf_size;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
295 }
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
296
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
297 static av_cold int decode_end(AVCodecContext *avctx)
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
298 {
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
299 MadContext *t = avctx->priv_data;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
300 if (t->frame.data[0])
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
301 avctx->release_buffer(avctx, &t->frame);
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
302 if (t->last_frame.data[0])
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
303 avctx->release_buffer(avctx, &t->last_frame);
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
304 av_free(t->bitstream_buf);
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
305 return 0;
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
306 }
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
307
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
308 AVCodec eamad_decoder = {
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
309 "eamad",
11560
8a4984c5cacc Define AVMediaType enum, and use it instead of enum CodecType, which
stefano
parents: 11369
diff changeset
310 AVMEDIA_TYPE_VIDEO,
9848
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
311 CODEC_ID_MAD,
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
312 sizeof(MadContext),
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
313 decode_init,
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
314 NULL,
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
315 decode_end,
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
316 decode_frame,
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
317 CODEC_CAP_DR1,
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
318 .long_name = NULL_IF_CONFIG_SMALL("Electronic Arts Madcow Video")
173ec947655f Electronic Arts Madcow decoder
pross
parents:
diff changeset
319 };