/src/ffmpeg/libavcodec/eacmv.c
Line | Count | Source |
1 | | /* |
2 | | * Electronic Arts CMV Video Decoder |
3 | | * Copyright (c) 2007-2008 Peter Ross |
4 | | * |
5 | | * This file is part of FFmpeg. |
6 | | * |
7 | | * FFmpeg is free software; you can redistribute it and/or |
8 | | * modify it under the terms of the GNU Lesser General Public |
9 | | * License as published by the Free Software Foundation; either |
10 | | * version 2.1 of the License, or (at your option) any later version. |
11 | | * |
12 | | * FFmpeg is distributed in the hope that it will be useful, |
13 | | * but WITHOUT ANY WARRANTY; without even the implied warranty of |
14 | | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU |
15 | | * Lesser General Public License for more details. |
16 | | * |
17 | | * You should have received a copy of the GNU Lesser General Public |
18 | | * License along with FFmpeg; if not, write to the Free Software |
19 | | * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA |
20 | | */ |
21 | | |
22 | | /** |
23 | | * @file |
24 | | * Electronic Arts CMV Video Decoder |
25 | | * by Peter Ross (pross@xvid.org) |
26 | | * |
27 | | * Technical details here: |
28 | | * http://wiki.multimedia.cx/index.php?title=Electronic_Arts_CMV |
29 | | */ |
30 | | |
31 | | #include "libavutil/common.h" |
32 | | #include "libavutil/intreadwrite.h" |
33 | | #include "libavutil/imgutils.h" |
34 | | #include "avcodec.h" |
35 | | #include "codec_internal.h" |
36 | | #include "decode.h" |
37 | | |
38 | | typedef struct CmvContext { |
39 | | AVCodecContext *avctx; |
40 | | AVFrame *last_frame; ///< last |
41 | | AVFrame *last2_frame; ///< second-last |
42 | | int width, height; |
43 | | unsigned int palette[AVPALETTE_COUNT]; |
44 | | } CmvContext; |
45 | | |
46 | 1.45k | static av_cold int cmv_decode_init(AVCodecContext *avctx){ |
47 | 1.45k | CmvContext *s = avctx->priv_data; |
48 | | |
49 | 1.45k | s->avctx = avctx; |
50 | 1.45k | avctx->pix_fmt = AV_PIX_FMT_PAL8; |
51 | | |
52 | 1.45k | s->last_frame = av_frame_alloc(); |
53 | 1.45k | s->last2_frame = av_frame_alloc(); |
54 | 1.45k | if (!s->last_frame || !s->last2_frame) |
55 | 0 | return AVERROR(ENOMEM); |
56 | | |
57 | 1.45k | return 0; |
58 | 1.45k | } |
59 | | |
60 | | static void cmv_decode_intra(CmvContext * s, AVFrame *frame, |
61 | | const uint8_t *buf, const uint8_t *buf_end) |
62 | 122k | { |
63 | 122k | unsigned char *dst = frame->data[0]; |
64 | 122k | int i; |
65 | | |
66 | 978k | for (i=0; i < s->avctx->height && buf_end - buf >= s->avctx->width; i++) { |
67 | 856k | memcpy(dst, buf, s->avctx->width); |
68 | 856k | dst += frame->linesize[0]; |
69 | 856k | buf += s->avctx->width; |
70 | 856k | } |
71 | 122k | } |
72 | | |
73 | | static void cmv_motcomp(unsigned char *dst, ptrdiff_t dst_stride, |
74 | | const unsigned char *src, ptrdiff_t src_stride, |
75 | | int x, int y, |
76 | | int xoffset, int yoffset, |
77 | 262k | int width, int height){ |
78 | 262k | int i,j; |
79 | | |
80 | 1.31M | for(j=y;j<y+4;j++) |
81 | 5.25M | for(i=x;i<x+4;i++) |
82 | 4.20M | { |
83 | 4.20M | if (i+xoffset>=0 && i+xoffset<width && |
84 | 2.04M | j+yoffset>=0 && j+yoffset<height) { |
85 | 869k | dst[j*dst_stride + i] = src[(j+yoffset)*src_stride + i+xoffset]; |
86 | 3.33M | }else{ |
87 | 3.33M | dst[j*dst_stride + i] = 0; |
88 | 3.33M | } |
89 | 4.20M | } |
90 | 262k | } |
91 | | |
92 | | static void cmv_decode_inter(CmvContext *s, AVFrame *frame, const uint8_t *buf, |
93 | | const uint8_t *buf_end) |
94 | 1.75k | { |
95 | 1.75k | const uint8_t *raw = buf + (s->avctx->width*s->avctx->height/16); |
96 | 1.75k | int x,y,i; |
97 | | |
98 | 1.75k | i = 0; |
99 | 4.03M | for(y=0; y<s->avctx->height/4; y++) |
100 | 4.95M | for(x=0; x<s->avctx->width/4 && buf_end - buf > i; x++) { |
101 | 926k | if (buf[i]==0xFF) { |
102 | 30.1k | unsigned char *dst = frame->data[0] + (y*4)*frame->linesize[0] + x*4; |
103 | 30.1k | if (raw+16<buf_end && *raw==0xFF) { /* intra */ |
104 | 332 | raw++; |
105 | 332 | memcpy(dst, raw, 4); |
106 | 332 | memcpy(dst + frame->linesize[0], raw+4, 4); |
107 | 332 | memcpy(dst + 2 * frame->linesize[0], raw+8, 4); |
108 | 332 | memcpy(dst + 3 * frame->linesize[0], raw+12, 4); |
109 | 332 | raw+=16; |
110 | 29.8k | }else if(raw<buf_end) { /* inter using second-last frame as reference */ |
111 | 9.43k | int xoffset = (*raw & 0xF) - 7; |
112 | 9.43k | int yoffset = ((*raw >> 4)) - 7; |
113 | 9.43k | if (s->last2_frame->data[0]) |
114 | 7.56k | cmv_motcomp(frame->data[0], frame->linesize[0], |
115 | 7.56k | s->last2_frame->data[0], s->last2_frame->linesize[0], |
116 | 7.56k | x*4, y*4, xoffset, yoffset, s->avctx->width, s->avctx->height); |
117 | 9.43k | raw++; |
118 | 9.43k | } |
119 | 896k | }else{ /* inter using last frame as reference */ |
120 | 896k | int xoffset = (buf[i] & 0xF) - 7; |
121 | 896k | int yoffset = ((buf[i] >> 4)) - 7; |
122 | 896k | if (s->last_frame->data[0]) |
123 | 255k | cmv_motcomp(frame->data[0], frame->linesize[0], |
124 | 255k | s->last_frame->data[0], s->last_frame->linesize[0], |
125 | 255k | x*4, y*4, xoffset, yoffset, s->avctx->width, s->avctx->height); |
126 | 896k | } |
127 | 926k | i++; |
128 | 926k | } |
129 | 1.75k | } |
130 | | |
131 | | static int cmv_process_header(CmvContext *s, const uint8_t *buf, const uint8_t *buf_end) |
132 | 3.92k | { |
133 | 3.92k | int pal_start, pal_count, i, ret, fps; |
134 | | |
135 | 3.92k | if(buf_end - buf < 16) { |
136 | 485 | av_log(s->avctx, AV_LOG_WARNING, "truncated header\n"); |
137 | 485 | return AVERROR_INVALIDDATA; |
138 | 485 | } |
139 | | |
140 | 3.43k | s->width = AV_RL16(&buf[4]); |
141 | 3.43k | s->height = AV_RL16(&buf[6]); |
142 | | |
143 | 3.43k | if (s->width != s->avctx->width || |
144 | 2.88k | s->height != s->avctx->height) { |
145 | 2.88k | av_frame_unref(s->last_frame); |
146 | 2.88k | av_frame_unref(s->last2_frame); |
147 | 2.88k | } |
148 | | |
149 | 3.43k | ret = ff_set_dimensions(s->avctx, s->width, s->height); |
150 | 3.43k | if (ret < 0) |
151 | 768 | return ret; |
152 | | |
153 | 2.66k | fps = AV_RL16(&buf[10]); |
154 | 2.66k | if (fps > 0) |
155 | 1.63k | s->avctx->framerate = (AVRational){ fps, 1 }; |
156 | | |
157 | 2.66k | pal_start = AV_RL16(&buf[12]); |
158 | 2.66k | pal_count = AV_RL16(&buf[14]); |
159 | | |
160 | 2.66k | buf += 16; |
161 | 17.1k | for (i=pal_start; i<pal_start+pal_count && i<AVPALETTE_COUNT && buf_end - buf >= 3; i++) { |
162 | 14.4k | s->palette[i] = 0xFFU << 24 | AV_RB24(buf); |
163 | 14.4k | buf += 3; |
164 | 14.4k | } |
165 | | |
166 | 2.66k | return 0; |
167 | 3.43k | } |
168 | | |
169 | 279k | #define EA_PREAMBLE_SIZE 8 |
170 | 258k | #define MVIh_TAG MKTAG('M', 'V', 'I', 'h') |
171 | | |
172 | | static int cmv_decode_frame(AVCodecContext *avctx, AVFrame *frame, |
173 | | int *got_frame, AVPacket *avpkt) |
174 | 146k | { |
175 | 146k | const uint8_t *buf = avpkt->data; |
176 | 146k | int buf_size = avpkt->size; |
177 | 146k | CmvContext *s = avctx->priv_data; |
178 | 146k | const uint8_t *buf_end = buf + buf_size; |
179 | 146k | int ret; |
180 | | |
181 | 146k | if (buf_end - buf < EA_PREAMBLE_SIZE) |
182 | 15.0k | return AVERROR_INVALIDDATA; |
183 | | |
184 | 131k | if (AV_RL32(buf)==MVIh_TAG||AV_RB32(buf)==MVIh_TAG) { |
185 | 3.92k | unsigned size = AV_RL32(buf + 4); |
186 | 3.92k | ret = cmv_process_header(s, buf+EA_PREAMBLE_SIZE, buf_end); |
187 | 3.92k | if (ret < 0) |
188 | 1.25k | return ret; |
189 | 2.66k | if (size > buf_end - buf - EA_PREAMBLE_SIZE) |
190 | 749 | return AVERROR_INVALIDDATA; |
191 | 1.92k | buf += size; |
192 | 1.92k | } |
193 | | |
194 | 129k | if ((ret = av_image_check_size(s->width, s->height, 0, s->avctx)) < 0) |
195 | 2.69k | return ret; |
196 | | |
197 | 126k | buf += EA_PREAMBLE_SIZE; |
198 | 126k | if (!(buf[0]&1) && buf_end - buf < s->width * s->height * (int64_t)(100 - s->avctx->discard_damaged_percentage) / 100) |
199 | 1.86k | return AVERROR_INVALIDDATA; |
200 | | |
201 | 124k | if ((ret = ff_get_buffer(avctx, frame, AV_GET_BUFFER_FLAG_REF)) < 0) |
202 | 279 | return ret; |
203 | | |
204 | 124k | memcpy(frame->data[1], s->palette, AVPALETTE_SIZE); |
205 | | |
206 | 124k | if ((buf[0]&1)) { // subtype |
207 | 1.75k | cmv_decode_inter(s, frame, buf+2, buf_end); |
208 | 1.75k | frame->flags &= ~AV_FRAME_FLAG_KEY; |
209 | 1.75k | frame->pict_type = AV_PICTURE_TYPE_P; |
210 | 122k | }else{ |
211 | 122k | frame->flags |= AV_FRAME_FLAG_KEY; |
212 | 122k | frame->pict_type = AV_PICTURE_TYPE_I; |
213 | 122k | cmv_decode_intra(s, frame, buf+2, buf_end); |
214 | 122k | } |
215 | | |
216 | 124k | FFSWAP(AVFrame*, s->last2_frame, s->last_frame); |
217 | 124k | if ((ret = av_frame_replace(s->last_frame, frame)) < 0) |
218 | 0 | return ret; |
219 | | |
220 | 124k | *got_frame = 1; |
221 | | |
222 | 124k | return buf_size; |
223 | 124k | } |
224 | | |
225 | 1.45k | static av_cold int cmv_decode_end(AVCodecContext *avctx){ |
226 | 1.45k | CmvContext *s = avctx->priv_data; |
227 | | |
228 | 1.45k | av_frame_free(&s->last_frame); |
229 | 1.45k | av_frame_free(&s->last2_frame); |
230 | | |
231 | 1.45k | return 0; |
232 | 1.45k | } |
233 | | |
234 | | const FFCodec ff_eacmv_decoder = { |
235 | | .p.name = "eacmv", |
236 | | CODEC_LONG_NAME("Electronic Arts CMV video"), |
237 | | .p.type = AVMEDIA_TYPE_VIDEO, |
238 | | .p.id = AV_CODEC_ID_CMV, |
239 | | .priv_data_size = sizeof(CmvContext), |
240 | | .init = cmv_decode_init, |
241 | | .close = cmv_decode_end, |
242 | | FF_CODEC_DECODE_CB(cmv_decode_frame), |
243 | | .p.capabilities = AV_CODEC_CAP_DR1, |
244 | | .caps_internal = FF_CODEC_CAP_INIT_CLEANUP, |
245 | | }; |