]> git.sesse.net Git - ffmpeg/blob - libavcodec/wnv1.c
avcodec/wnv1: Apply offset during init, not later every time
[ffmpeg] / libavcodec / wnv1.c
1 /*
2  * Winnov WNV1 codec
3  * Copyright (c) 2005 Konstantin Shishkov
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 Street, Fifth Floor, Boston, MA 02110-1301 USA
20  */
21
22 /**
23  * @file
24  * Winnov WNV1 codec.
25  */
26
27 #define BITSTREAM_READER_LE
28 #include "avcodec.h"
29 #include "get_bits.h"
30 #include "internal.h"
31
32 static const uint8_t code_tab[16][2] = {
33     {  7, 1 }, {  8, 3 }, {  6, 3 }, { 9, 4 }, {  5, 4 }, { 10, 5 }, {  4, 5 },
34     { 11, 6 }, {  3, 6 }, { 12, 7 }, { 2, 7 }, { 13, 8 }, {  1, 8 }, { 14, 9 },
35     {  0, 9 }, { 15, 8 }
36 };
37
38 #define CODE_VLC_BITS 9
39 static VLC code_vlc;
40
41 /* returns modified base_value */
42 static inline int wnv1_get_code(GetBitContext *gb, int shift, int base_value)
43 {
44     int v = get_vlc2(gb, code_vlc.table, CODE_VLC_BITS, 1);
45
46     if (v == 8)
47         return get_bits(gb, 8 - shift) << shift;
48     else
49         return base_value + v * (1 << shift);
50 }
51
52 static int decode_frame(AVCodecContext *avctx,
53                         void *data, int *got_frame,
54                         AVPacket *avpkt)
55 {
56     const uint8_t *buf    = avpkt->data;
57     int buf_size          = avpkt->size;
58     AVFrame * const p     = data;
59     GetBitContext gb;
60     unsigned char *Y,*U,*V;
61     int i, j, ret, shift;
62     int prev_y = 0, prev_u = 0, prev_v = 0;
63
64     if (buf_size < 8 + avctx->height * (avctx->width/2)/8) {
65         av_log(avctx, AV_LOG_ERROR, "Packet size %d is too small\n", buf_size);
66         return AVERROR_INVALIDDATA;
67     }
68
69     if ((ret = ff_get_buffer(avctx, p, 0)) < 0)
70         return ret;
71     p->key_frame = 1;
72
73     if ((ret = init_get_bits8(&gb, buf + 8, buf_size - 8)) < 0)
74         return ret;
75
76     if (buf[2] >> 4 == 6)
77         shift = 2;
78     else {
79         shift = 8 - (buf[2] >> 4);
80         if (shift > 4) {
81             avpriv_request_sample(avctx,
82                                   "Unknown WNV1 frame header value %i",
83                                   buf[2] >> 4);
84             shift = 4;
85         }
86         if (shift < 1) {
87             avpriv_request_sample(avctx,
88                                   "Unknown WNV1 frame header value %i",
89                                   buf[2] >> 4);
90             shift = 1;
91         }
92     }
93
94     Y = p->data[0];
95     U = p->data[1];
96     V = p->data[2];
97     for (j = 0; j < avctx->height; j++) {
98         for (i = 0; i < avctx->width / 2; i++) {
99             Y[i * 2] = wnv1_get_code(&gb, shift, prev_y);
100             prev_u = U[i] = wnv1_get_code(&gb, shift, prev_u);
101             prev_y = Y[(i * 2) + 1] = wnv1_get_code(&gb, shift, Y[i * 2]);
102             prev_v = V[i] = wnv1_get_code(&gb, shift, prev_v);
103         }
104         Y += p->linesize[0];
105         U += p->linesize[1];
106         V += p->linesize[2];
107     }
108
109
110     *got_frame      = 1;
111
112     return buf_size;
113 }
114
115 static av_cold int decode_init(AVCodecContext *avctx)
116 {
117     avctx->pix_fmt = AV_PIX_FMT_YUV422P;
118
119     INIT_VLC_STATIC_FROM_LENGTHS(&code_vlc, CODE_VLC_BITS, 16,
120                                  &code_tab[0][1], 2,
121                                  &code_tab[0][0], 2, 1,
122                                  -7, INIT_VLC_OUTPUT_LE, 1 << CODE_VLC_BITS);
123     return 0;
124 }
125
126 AVCodec ff_wnv1_decoder = {
127     .name           = "wnv1",
128     .long_name      = NULL_IF_CONFIG_SMALL("Winnov WNV1"),
129     .type           = AVMEDIA_TYPE_VIDEO,
130     .id             = AV_CODEC_ID_WNV1,
131     .init           = decode_init,
132     .decode         = decode_frame,
133     .capabilities   = AV_CODEC_CAP_DR1,
134 };