]> git.sesse.net Git - ffmpeg/blob - libavcodec/wcmv.c
Merge commit '90adbf4abf336f8042aecdf1e18fdf76a96304b1'
[ffmpeg] / libavcodec / wcmv.c
1 /*
2  * WinCAM Motion Video decoder
3  *
4  * Copyright (c) 2018 Paul B Mahol
5  *
6  * This file is part of FFmpeg.
7  *
8  * FFmpeg is free software; you can redistribute it and/or
9  * modify it under the terms of the GNU Lesser General Public
10  * License as published by the Free Software Foundation; either
11  * version 2.1 of the License, or (at your option) any later version.
12  *
13  * FFmpeg is distributed in the hope that it will be useful,
14  * but WITHOUT ANY WARRANTY; without even the implied warranty of
15  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
16  * Lesser General Public License for more details.
17  *
18  * You should have received a copy of the GNU Lesser General Public
19  * License along with FFmpeg; if not, write to the Free Software
20  * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
21  */
22
23 #include <stdio.h>
24 #include <stdlib.h>
25 #include <string.h>
26
27 #include "libavutil/imgutils.h"
28
29 #include "avcodec.h"
30 #include "bytestream.h"
31 #include "internal.h"
32
33 #include <zlib.h>
34
35 typedef struct WCMVContext {
36     int         bpp;
37     z_stream    zstream;
38     AVFrame    *prev_frame;
39     uint8_t     block_data[65536*8];
40 } WCMVContext;
41
42 static int decode_frame(AVCodecContext *avctx,
43                         void *data, int *got_frame,
44                         AVPacket *avpkt)
45 {
46     WCMVContext *s = avctx->priv_data;
47     AVFrame *frame = data;
48     int skip, blocks, zret, ret, intra = 0, bpp = s->bpp;
49     GetByteContext gb;
50     uint8_t *dst;
51
52     ret = inflateReset(&s->zstream);
53     if (ret != Z_OK) {
54         av_log(avctx, AV_LOG_ERROR, "Inflate reset error: %d\n", ret);
55         return AVERROR_EXTERNAL;
56     }
57
58     bytestream2_init(&gb, avpkt->data, avpkt->size);
59
60     if ((ret = ff_get_buffer(avctx, frame, AV_GET_BUFFER_FLAG_REF)) < 0)
61         return ret;
62
63     if (s->prev_frame->data[0]) {
64         ret = av_frame_copy(frame, s->prev_frame);
65         if (ret < 0)
66             return ret;
67     } else {
68         ptrdiff_t linesize[4] = { frame->linesize[0], 0, 0, 0 };
69         av_image_fill_black(frame->data, linesize, avctx->pix_fmt, 0,
70                             avctx->width, avctx->height);
71     }
72
73     blocks = bytestream2_get_le16(&gb);
74     if (blocks > 5) {
75         GetByteContext bgb;
76         int x = 0, size;
77
78         if (blocks * 8 >= 0xFFFF)
79             size = bytestream2_get_le24(&gb);
80         else if (blocks * 8 >= 0xFF)
81             size = bytestream2_get_le16(&gb);
82         else
83             size = bytestream2_get_byte(&gb);
84
85         skip = bytestream2_tell(&gb);
86         if (size > avpkt->size - skip)
87             return AVERROR_INVALIDDATA;
88
89         s->zstream.next_in  = avpkt->data + skip;
90         s->zstream.avail_in = size;
91         s->zstream.next_out  = s->block_data;
92         s->zstream.avail_out = sizeof(s->block_data);
93
94         zret = inflate(&s->zstream, Z_FINISH);
95         if (zret != Z_STREAM_END) {
96             av_log(avctx, AV_LOG_ERROR,
97                    "Inflate failed with return code: %d.\n", zret);
98             return AVERROR_INVALIDDATA;
99         }
100
101         ret = inflateReset(&s->zstream);
102         if (ret != Z_OK) {
103             av_log(avctx, AV_LOG_ERROR, "Inflate reset error: %d\n", ret);
104             return AVERROR_EXTERNAL;
105         }
106
107         bytestream2_skip(&gb, size);
108         bytestream2_init(&bgb, s->block_data, blocks * 8);
109
110         for (int i = 0; i < blocks; i++) {
111             int w, h;
112
113             bytestream2_skip(&bgb, 4);
114             w = bytestream2_get_le16(&bgb);
115             h = bytestream2_get_le16(&bgb);
116             if (x + bpp * (int64_t)w * h > INT_MAX)
117                 return AVERROR_INVALIDDATA;
118             x += bpp * w * h;
119         }
120
121         if (x >= 0xFFFF)
122             bytestream2_skip(&gb, 3);
123         else if (x >= 0xFF)
124             bytestream2_skip(&gb, 2);
125         else
126             bytestream2_skip(&gb, 1);
127
128         skip = bytestream2_tell(&gb);
129
130         s->zstream.next_in  = avpkt->data + skip;
131         s->zstream.avail_in = avpkt->size - skip;
132
133         bytestream2_init(&gb, s->block_data, blocks * 8);
134     } else if (blocks) {
135         int x = 0;
136
137         bytestream2_seek(&gb, 2, SEEK_SET);
138
139         for (int i = 0; i < blocks; i++) {
140             int w, h;
141
142             bytestream2_skip(&gb, 4);
143             w = bytestream2_get_le16(&gb);
144             h = bytestream2_get_le16(&gb);
145             if (x + bpp * (int64_t)w * h > INT_MAX)
146                 return AVERROR_INVALIDDATA;
147             x += bpp * w * h;
148         }
149
150         if (x >= 0xFFFF)
151             bytestream2_skip(&gb, 3);
152         else if (x >= 0xFF)
153             bytestream2_skip(&gb, 2);
154         else
155             bytestream2_skip(&gb, 1);
156
157         skip = bytestream2_tell(&gb);
158
159         s->zstream.next_in  = avpkt->data + skip;
160         s->zstream.avail_in = avpkt->size - skip;
161
162         bytestream2_seek(&gb, 2, SEEK_SET);
163     }
164
165     for (int block = 0; block < blocks; block++) {
166         int x, y, w, h;
167
168         x = bytestream2_get_le16(&gb);
169         y = bytestream2_get_le16(&gb);
170         w = bytestream2_get_le16(&gb);
171         h = bytestream2_get_le16(&gb);
172
173         if (blocks == 1 && x == 0 && y == 0 && w == avctx->width && h == avctx->height)
174             intra = 1;
175
176         if (x + w > avctx->width || y + h > avctx->height)
177             return AVERROR_INVALIDDATA;
178
179         if (w > avctx->width || h > avctx->height)
180             return AVERROR_INVALIDDATA;
181
182         dst = frame->data[0] + (avctx->height - y - 1) * frame->linesize[0] + x * bpp;
183         for (int i = 0; i < h; i++) {
184             s->zstream.next_out  = dst;
185             s->zstream.avail_out = w * bpp;
186
187             zret = inflate(&s->zstream, Z_SYNC_FLUSH);
188             if (zret != Z_OK && zret != Z_STREAM_END) {
189                 av_log(avctx, AV_LOG_ERROR,
190                        "Inflate failed with return code: %d.\n", zret);
191                 return AVERROR_INVALIDDATA;
192             }
193
194             dst -= frame->linesize[0];
195         }
196     }
197
198     frame->key_frame = intra;
199     frame->pict_type = intra ? AV_PICTURE_TYPE_I : AV_PICTURE_TYPE_P;
200
201     av_frame_unref(s->prev_frame);
202     if ((ret = av_frame_ref(s->prev_frame, frame)) < 0)
203         return ret;
204
205     *got_frame = 1;
206
207     return avpkt->size;
208 }
209
210 static av_cold int decode_init(AVCodecContext *avctx)
211 {
212     WCMVContext *s = avctx->priv_data;
213     int zret;
214
215     switch (avctx->bits_per_coded_sample) {
216     case 16: avctx->pix_fmt = AV_PIX_FMT_RGB565LE; break;
217     case 24: avctx->pix_fmt = AV_PIX_FMT_BGR24;  break;
218     case 32: avctx->pix_fmt = AV_PIX_FMT_BGRA;   break;
219     default: av_log(avctx, AV_LOG_ERROR, "Unsupported bits_per_coded_sample: %d\n",
220                     avctx->bits_per_coded_sample);
221              return AVERROR_PATCHWELCOME;
222     }
223
224     s->bpp = avctx->bits_per_coded_sample >> 3;
225
226     s->zstream.zalloc = Z_NULL;
227     s->zstream.zfree = Z_NULL;
228     s->zstream.opaque = Z_NULL;
229     zret = inflateInit(&s->zstream);
230     if (zret != Z_OK) {
231         av_log(avctx, AV_LOG_ERROR, "Inflate init error: %d\n", zret);
232         return AVERROR_EXTERNAL;
233     }
234
235     s->prev_frame = av_frame_alloc();
236     if (!s->prev_frame)
237         return AVERROR(ENOMEM);
238
239     return 0;
240 }
241
242 static av_cold int decode_close(AVCodecContext *avctx)
243 {
244     WCMVContext *s = avctx->priv_data;
245
246     av_frame_free(&s->prev_frame);
247     inflateEnd(&s->zstream);
248
249     return 0;
250 }
251
252 AVCodec ff_wcmv_decoder = {
253     .name             = "wcmv",
254     .long_name        = NULL_IF_CONFIG_SMALL("WinCAM Motion Video"),
255     .type             = AVMEDIA_TYPE_VIDEO,
256     .id               = AV_CODEC_ID_WCMV,
257     .priv_data_size   = sizeof(WCMVContext),
258     .init             = decode_init,
259     .close            = decode_close,
260     .decode           = decode_frame,
261     .capabilities     = AV_CODEC_CAP_DR1,
262     .caps_internal    = FF_CODEC_CAP_INIT_THREADSAFE |
263                         FF_CODEC_CAP_INIT_CLEANUP,
264 };