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