annotate txd.c @ 5311:7742d5411c9d libavcodec

AC-3 decoder, soc revision 48, Aug 16 11:27:49 2006 UTC by cloud9 I realized that the bug was not in the imdct routine but in the get_transform_coeffs. Fixed it. Code now uses the ffmpeg's imdct routines. All the mplayer's ac3 samples are decoded successfully. Also improved downmixing. Now all the downmixing coeffcients for channels are normalized such that the sum of coefficients used to construct the output for single channel never exceeds 1.0.
author jbr
date Sat, 14 Jul 2007 15:58:42 +0000
parents 470601203f44
children fe2733d26e9b
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
rev   line source
4934
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
1 /*
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
2 * Renderware TeXture Dictionary (.txd) image decoder
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
3 * Copyright (c) 2007 Ivo van Poorten
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
4 *
5214
470601203f44 Group all copyright and author notices together.
diego
parents: 4934
diff changeset
5 * See also: http://wiki.multimedia.cx/index.php?title=TXD
470601203f44 Group all copyright and author notices together.
diego
parents: 4934
diff changeset
6 *
4934
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
7 * This file is part of FFmpeg.
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
8 *
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
9 * FFmpeg is free software; you can redistribute it and/or
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
10 * modify it under the terms of the GNU Lesser General Public
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
11 * License as published by the Free Software Foundation; either
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
12 * version 2.1 of the License, or (at your option) any later version.
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
13 *
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
14 * FFmpeg is distributed in the hope that it will be useful,
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
15 * but WITHOUT ANY WARRANTY; without even the implied warranty of
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
16 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
17 * Lesser General Public License for more details.
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
18 *
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
19 * You should have received a copy of the GNU Lesser General Public
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
20 * License along with FFmpeg; if not, write to the Free Software
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
21 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
22 */
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
23
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
24 #include "avcodec.h"
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
25 #include "s3tc.h"
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
26
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
27 typedef struct TXDContext {
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
28 AVFrame picture;
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
29 } TXDContext;
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
30
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
31 static int txd_init(AVCodecContext *avctx) {
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
32 TXDContext *s = avctx->priv_data;
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
33
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
34 avcodec_get_frame_defaults(&s->picture);
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
35 avctx->coded_frame = &s->picture;
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
36 s->picture.data[0] = NULL;
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
37
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
38 return 0;
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
39 }
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
40
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
41 static int txd_decode_frame(AVCodecContext *avctx, void *data, int *data_size,
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
42 uint8_t *buf, int buf_size) {
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
43 TXDContext * const s = avctx->priv_data;
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
44 AVFrame *picture = data;
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
45 AVFrame * const p = &s->picture;
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
46 unsigned int version, w, h, d3d_format, depth, stride, mipmap_count, flags;
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
47 unsigned int y, v;
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
48 uint8_t *ptr, *cur = buf;
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
49 uint32_t *palette = (uint32_t *)(cur + 88), *pal;
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
50
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
51 version = AV_RL32(cur);
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
52 d3d_format = AV_RL32(cur+76);
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
53 w = AV_RL16(cur+80);
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
54 h = AV_RL16(cur+82);
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
55 depth = AV_RL8 (cur+84);
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
56 mipmap_count = AV_RL8 (cur+85);
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
57 flags = AV_RL8 (cur+87);
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
58 cur += 92;
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
59
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
60 if (version < 8 || version > 9) {
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
61 av_log(avctx, AV_LOG_ERROR, "texture data version %i is unsupported\n",
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
62 version);
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
63 return -1;
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
64 }
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
65
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
66 if (depth == 8) {
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
67 avctx->pix_fmt = PIX_FMT_PAL8;
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
68 cur += 1024;
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
69 } else if (depth == 16 || depth == 32)
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
70 avctx->pix_fmt = PIX_FMT_RGB32;
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
71 else {
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
72 av_log(avctx, AV_LOG_ERROR, "depth of %i is unsupported\n", depth);
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
73 return -1;
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
74 }
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
75
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
76 if (p->data[0])
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
77 avctx->release_buffer(avctx, p);
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
78
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
79 if (avcodec_check_dimensions(avctx, w, h))
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
80 return -1;
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
81 if (w != avctx->width || h != avctx->height)
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
82 avcodec_set_dimensions(avctx, w, h);
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
83 if (avctx->get_buffer(avctx, p) < 0) {
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
84 av_log(avctx, AV_LOG_ERROR, "get_buffer() failed\n");
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
85 return -1;
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
86 }
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
87
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
88 p->pict_type = FF_I_TYPE;
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
89
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
90 ptr = p->data[0];
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
91 stride = p->linesize[0];
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
92
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
93 if (depth == 8) {
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
94 pal = (uint32_t *) p->data[1];
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
95 for (y=0; y<256; y++) {
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
96 v = AV_RB32(palette+y);
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
97 pal[y] = (v>>8) + (v<<24);
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
98 }
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
99 for (y=0; y<h; y++) {
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
100 memcpy(ptr, cur, w);
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
101 ptr += stride;
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
102 cur += w;
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
103 }
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
104 } else if (depth == 16) {
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
105 switch (d3d_format) {
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
106 case 0:
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
107 if (!flags&1) goto unsupported;
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
108 case FF_S3TC_DXT1:
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
109 ff_decode_dxt1(cur, ptr, w, h, stride);
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
110 break;
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
111 case FF_S3TC_DXT3:
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
112 ff_decode_dxt3(cur, ptr, w, h, stride);
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
113 break;
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
114 default:
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
115 goto unsupported;
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
116 }
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
117 } else if (depth == 32) {
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
118 switch (d3d_format) {
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
119 case 0x15:
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
120 case 0x16:
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
121 for (y=0; y<h; y++) {
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
122 memcpy(ptr, cur, w*4);
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
123 ptr += stride;
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
124 cur += w*4;
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
125 }
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
126 break;
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
127 default:
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
128 goto unsupported;
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
129 }
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
130 }
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
131
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
132 for (; mipmap_count > 1; mipmap_count--)
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
133 cur += AV_RL32(cur) + 4;
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
134
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
135 *picture = s->picture;
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
136 *data_size = sizeof(AVPicture);
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
137
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
138 return cur - buf;
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
139
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
140 unsupported:
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
141 av_log(avctx, AV_LOG_ERROR, "unsupported d3d format (%08x)\n", d3d_format);
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
142 return -1;
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
143 }
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
144
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
145 static int txd_end(AVCodecContext *avctx) {
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
146 TXDContext *s = avctx->priv_data;
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
147
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
148 if (s->picture.data[0])
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
149 avctx->release_buffer(avctx, &s->picture);
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
150
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
151 return 0;
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
152 }
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
153
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
154 AVCodec txd_decoder = {
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
155 "txd",
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
156 CODEC_TYPE_VIDEO,
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
157 CODEC_ID_TXD,
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
158 sizeof(TXDContext),
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
159 txd_init,
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
160 NULL,
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
161 txd_end,
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
162 txd_decode_frame,
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
163 0,
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
164 NULL
feba7e1adac5 Renderware TXD decoder
ivo
parents:
diff changeset
165 };