1cabdff1aSopenharmony_ci/*
2cabdff1aSopenharmony_ci * TechSmith Screen Codec 2 (aka Dora) decoder
3cabdff1aSopenharmony_ci * Copyright (c) 2012 Konstantin Shishkov
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 Street, Fifth Floor, Boston, MA 02110-1301 USA
20cabdff1aSopenharmony_ci */
21cabdff1aSopenharmony_ci
22cabdff1aSopenharmony_ci/**
23cabdff1aSopenharmony_ci * @file
24cabdff1aSopenharmony_ci * TechSmith Screen Codec 2 decoder
25cabdff1aSopenharmony_ci */
26cabdff1aSopenharmony_ci
27cabdff1aSopenharmony_ci#include <inttypes.h>
28cabdff1aSopenharmony_ci
29cabdff1aSopenharmony_ci#include "libavutil/thread.h"
30cabdff1aSopenharmony_ci
31cabdff1aSopenharmony_ci#define BITSTREAM_READER_LE
32cabdff1aSopenharmony_ci#include "avcodec.h"
33cabdff1aSopenharmony_ci#include "bytestream.h"
34cabdff1aSopenharmony_ci#include "codec_internal.h"
35cabdff1aSopenharmony_ci#include "get_bits.h"
36cabdff1aSopenharmony_ci#include "internal.h"
37cabdff1aSopenharmony_ci#include "mathops.h"
38cabdff1aSopenharmony_ci#include "tscc2data.h"
39cabdff1aSopenharmony_ci
40cabdff1aSopenharmony_ci#define TSCC2_VLC_BITS 9
41cabdff1aSopenharmony_ci
42cabdff1aSopenharmony_citypedef struct TSCC2Context {
43cabdff1aSopenharmony_ci    AVCodecContext *avctx;
44cabdff1aSopenharmony_ci    AVFrame       *pic;
45cabdff1aSopenharmony_ci    int            mb_width, mb_height;
46cabdff1aSopenharmony_ci    uint8_t        *slice_quants;
47cabdff1aSopenharmony_ci    int            quant[2];
48cabdff1aSopenharmony_ci    int            q[2][3];
49cabdff1aSopenharmony_ci    GetBitContext  gb;
50cabdff1aSopenharmony_ci
51cabdff1aSopenharmony_ci    int            block[16];
52cabdff1aSopenharmony_ci} TSCC2Context;
53cabdff1aSopenharmony_ci
54cabdff1aSopenharmony_cistatic VLC dc_vlc, nc_vlc[NUM_VLC_SETS], ac_vlc[NUM_VLC_SETS];
55cabdff1aSopenharmony_ci
56cabdff1aSopenharmony_cistatic av_cold void tscc2_init_vlc(VLC *vlc, int *offset, int nb_codes,
57cabdff1aSopenharmony_ci                                   const uint8_t *lens, const void *syms,
58cabdff1aSopenharmony_ci                                   int sym_length)
59cabdff1aSopenharmony_ci{
60cabdff1aSopenharmony_ci    static VLCElem vlc_buf[15442];
61cabdff1aSopenharmony_ci
62cabdff1aSopenharmony_ci    vlc->table           = &vlc_buf[*offset];
63cabdff1aSopenharmony_ci    vlc->table_allocated = FF_ARRAY_ELEMS(vlc_buf) - *offset;
64cabdff1aSopenharmony_ci    ff_init_vlc_from_lengths(vlc, TSCC2_VLC_BITS, nb_codes,
65cabdff1aSopenharmony_ci                             lens, 1, syms, sym_length, sym_length, 0,
66cabdff1aSopenharmony_ci                             INIT_VLC_STATIC_OVERLONG | INIT_VLC_OUTPUT_LE, NULL);
67cabdff1aSopenharmony_ci    *offset += vlc->table_size;
68cabdff1aSopenharmony_ci}
69cabdff1aSopenharmony_ci
70cabdff1aSopenharmony_cistatic av_cold void tscc2_init_vlcs(void)
71cabdff1aSopenharmony_ci{
72cabdff1aSopenharmony_ci    const uint16_t *ac_vlc_syms = tscc2_ac_vlc_syms;
73cabdff1aSopenharmony_ci    const uint8_t  *ac_vlc_lens = tscc2_ac_vlc_lens;
74cabdff1aSopenharmony_ci    int i, offset = 0;
75cabdff1aSopenharmony_ci
76cabdff1aSopenharmony_ci    tscc2_init_vlc(&dc_vlc, &offset, DC_VLC_COUNT,
77cabdff1aSopenharmony_ci                   tscc2_dc_vlc_lens, tscc2_dc_vlc_syms, 2);
78cabdff1aSopenharmony_ci
79cabdff1aSopenharmony_ci    for (i = 0; i < NUM_VLC_SETS; i++) {
80cabdff1aSopenharmony_ci        tscc2_init_vlc(&nc_vlc[i], &offset, 16,
81cabdff1aSopenharmony_ci                       tscc2_nc_vlc_lens[i], tscc2_nc_vlc_syms[i], 1);
82cabdff1aSopenharmony_ci
83cabdff1aSopenharmony_ci        tscc2_init_vlc(&ac_vlc[i], &offset, tscc2_ac_vlc_sizes[i],
84cabdff1aSopenharmony_ci                       ac_vlc_lens, ac_vlc_syms, 2);
85cabdff1aSopenharmony_ci        ac_vlc_lens += tscc2_ac_vlc_sizes[i];
86cabdff1aSopenharmony_ci        ac_vlc_syms += tscc2_ac_vlc_sizes[i];
87cabdff1aSopenharmony_ci    }
88cabdff1aSopenharmony_ci}
89cabdff1aSopenharmony_ci
90cabdff1aSopenharmony_ci#define DEQUANT(val, q) (((q) * (val) + 0x80) >> 8)
91cabdff1aSopenharmony_ci#define DCT1D(d0, d1, d2, d3, s0, s1, s2, s3, OP) \
92cabdff1aSopenharmony_ci    OP(d0, 5 * ((s0) + (s1) + (s2)) + 2 * (s3));  \
93cabdff1aSopenharmony_ci    OP(d1, 5 * ((s0) - (s2) - (s3)) + 2 * (s1));  \
94cabdff1aSopenharmony_ci    OP(d2, 5 * ((s0) - (s2) + (s3)) - 2 * (s1));  \
95cabdff1aSopenharmony_ci    OP(d3, 5 * ((s0) - (s1) + (s2)) - 2 * (s3));  \
96cabdff1aSopenharmony_ci
97cabdff1aSopenharmony_ci#define COL_OP(a, b)  a = (b)
98cabdff1aSopenharmony_ci#define ROW_OP(a, b)  a = ((b) + 0x20) >> 6
99cabdff1aSopenharmony_ci
100cabdff1aSopenharmony_cistatic void tscc2_idct4_put(int *in, int q[3], uint8_t *dst, int stride)
101cabdff1aSopenharmony_ci{
102cabdff1aSopenharmony_ci    int i;
103cabdff1aSopenharmony_ci    int tblk[4 * 4];
104cabdff1aSopenharmony_ci    int t0, t1, t2, t3;
105cabdff1aSopenharmony_ci
106cabdff1aSopenharmony_ci    for (i = 0; i < 4; i++) {
107cabdff1aSopenharmony_ci        t0 = DEQUANT(q[0 + (i & 1)], in[0 * 4 + i]);
108cabdff1aSopenharmony_ci        t1 = DEQUANT(q[1 + (i & 1)], in[1 * 4 + i]);
109cabdff1aSopenharmony_ci        t2 = DEQUANT(q[0 + (i & 1)], in[2 * 4 + i]);
110cabdff1aSopenharmony_ci        t3 = DEQUANT(q[1 + (i & 1)], in[3 * 4 + i]);
111cabdff1aSopenharmony_ci        DCT1D(tblk[0 * 4 + i], tblk[1 * 4 + i],
112cabdff1aSopenharmony_ci              tblk[2 * 4 + i], tblk[3 * 4 + i],
113cabdff1aSopenharmony_ci              t0, t1, t2, t3, COL_OP);
114cabdff1aSopenharmony_ci    }
115cabdff1aSopenharmony_ci    for (i = 0; i < 4; i++) {
116cabdff1aSopenharmony_ci        DCT1D(dst[0], dst[1], dst[2], dst[3],
117cabdff1aSopenharmony_ci              tblk[i * 4 + 0], tblk[i * 4 + 1],
118cabdff1aSopenharmony_ci              tblk[i * 4 + 2], tblk[i * 4 + 3], ROW_OP);
119cabdff1aSopenharmony_ci        dst += stride;
120cabdff1aSopenharmony_ci    }
121cabdff1aSopenharmony_ci}
122cabdff1aSopenharmony_ci
123cabdff1aSopenharmony_cistatic int tscc2_decode_mb(TSCC2Context *c, int *q, int vlc_set,
124cabdff1aSopenharmony_ci                           uint8_t *dst, int stride, int plane)
125cabdff1aSopenharmony_ci{
126cabdff1aSopenharmony_ci    GetBitContext *gb = &c->gb;
127cabdff1aSopenharmony_ci    int prev_dc, dc, nc, ac, bpos, val;
128cabdff1aSopenharmony_ci    int i, j, k, l;
129cabdff1aSopenharmony_ci
130cabdff1aSopenharmony_ci    if (get_bits1(gb)) {
131cabdff1aSopenharmony_ci        if (get_bits1(gb)) {
132cabdff1aSopenharmony_ci            val = get_bits(gb, 8);
133cabdff1aSopenharmony_ci            for (i = 0; i < 8; i++, dst += stride)
134cabdff1aSopenharmony_ci                memset(dst, val, 16);
135cabdff1aSopenharmony_ci        } else {
136cabdff1aSopenharmony_ci            if (get_bits_left(gb) < 16 * 8 * 8)
137cabdff1aSopenharmony_ci                return AVERROR_INVALIDDATA;
138cabdff1aSopenharmony_ci            for (i = 0; i < 8; i++) {
139cabdff1aSopenharmony_ci                for (j = 0; j < 16; j++)
140cabdff1aSopenharmony_ci                    dst[j] = get_bits(gb, 8);
141cabdff1aSopenharmony_ci                dst += stride;
142cabdff1aSopenharmony_ci            }
143cabdff1aSopenharmony_ci        }
144cabdff1aSopenharmony_ci        return 0;
145cabdff1aSopenharmony_ci    }
146cabdff1aSopenharmony_ci
147cabdff1aSopenharmony_ci    prev_dc = 0;
148cabdff1aSopenharmony_ci    for (j = 0; j < 2; j++) {
149cabdff1aSopenharmony_ci        for (k = 0; k < 4; k++) {
150cabdff1aSopenharmony_ci            if (!(j | k)) {
151cabdff1aSopenharmony_ci                dc = get_bits(gb, 8);
152cabdff1aSopenharmony_ci            } else {
153cabdff1aSopenharmony_ci                dc = get_vlc2(gb, dc_vlc.table, TSCC2_VLC_BITS, 2);
154cabdff1aSopenharmony_ci                if (dc == 0x100)
155cabdff1aSopenharmony_ci                    dc = get_bits(gb, 8);
156cabdff1aSopenharmony_ci            }
157cabdff1aSopenharmony_ci            dc          = (dc + prev_dc) & 0xFF;
158cabdff1aSopenharmony_ci            prev_dc     = dc;
159cabdff1aSopenharmony_ci            c->block[0] = dc;
160cabdff1aSopenharmony_ci
161cabdff1aSopenharmony_ci            nc = get_vlc2(gb, nc_vlc[vlc_set].table, TSCC2_VLC_BITS, 1);
162cabdff1aSopenharmony_ci
163cabdff1aSopenharmony_ci            bpos = 1;
164cabdff1aSopenharmony_ci            memset(c->block + 1, 0, 15 * sizeof(*c->block));
165cabdff1aSopenharmony_ci            for (l = 0; l < nc; l++) {
166cabdff1aSopenharmony_ci                ac = get_vlc2(gb, ac_vlc[vlc_set].table, TSCC2_VLC_BITS, 2);
167cabdff1aSopenharmony_ci                if (ac == 0x1000)
168cabdff1aSopenharmony_ci                    ac = get_bits(gb, 12);
169cabdff1aSopenharmony_ci                bpos += ac & 0xF;
170cabdff1aSopenharmony_ci                if (bpos >= 16)
171cabdff1aSopenharmony_ci                    return AVERROR_INVALIDDATA;
172cabdff1aSopenharmony_ci                val = sign_extend(ac >> 4, 8);
173cabdff1aSopenharmony_ci                c->block[ff_zigzag_scan[bpos++]] = val;
174cabdff1aSopenharmony_ci            }
175cabdff1aSopenharmony_ci            tscc2_idct4_put(c->block, q, dst + k * 4, stride);
176cabdff1aSopenharmony_ci        }
177cabdff1aSopenharmony_ci        dst += 4 * stride;
178cabdff1aSopenharmony_ci    }
179cabdff1aSopenharmony_ci    return 0;
180cabdff1aSopenharmony_ci}
181cabdff1aSopenharmony_ci
182cabdff1aSopenharmony_cistatic int tscc2_decode_slice(TSCC2Context *c, int mb_y,
183cabdff1aSopenharmony_ci                              const uint8_t *buf, int buf_size)
184cabdff1aSopenharmony_ci{
185cabdff1aSopenharmony_ci    int i, mb_x, q, ret;
186cabdff1aSopenharmony_ci    int off;
187cabdff1aSopenharmony_ci
188cabdff1aSopenharmony_ci    if ((ret = init_get_bits8(&c->gb, buf, buf_size)) < 0)
189cabdff1aSopenharmony_ci        return ret;
190cabdff1aSopenharmony_ci
191cabdff1aSopenharmony_ci    for (mb_x = 0; mb_x < c->mb_width; mb_x++) {
192cabdff1aSopenharmony_ci        q = c->slice_quants[mb_x + c->mb_width * mb_y];
193cabdff1aSopenharmony_ci
194cabdff1aSopenharmony_ci        if (q == 0 || q == 3) // skip block
195cabdff1aSopenharmony_ci            continue;
196cabdff1aSopenharmony_ci        for (i = 0; i < 3; i++) {
197cabdff1aSopenharmony_ci            off = mb_x * 16 + mb_y * 8 * c->pic->linesize[i];
198cabdff1aSopenharmony_ci            ret = tscc2_decode_mb(c, c->q[q - 1], c->quant[q - 1] - 2,
199cabdff1aSopenharmony_ci                                  c->pic->data[i] + off, c->pic->linesize[i], i);
200cabdff1aSopenharmony_ci            if (ret)
201cabdff1aSopenharmony_ci                return ret;
202cabdff1aSopenharmony_ci        }
203cabdff1aSopenharmony_ci    }
204cabdff1aSopenharmony_ci
205cabdff1aSopenharmony_ci    return 0;
206cabdff1aSopenharmony_ci}
207cabdff1aSopenharmony_ci
208cabdff1aSopenharmony_cistatic int tscc2_decode_frame(AVCodecContext *avctx, AVFrame *rframe,
209cabdff1aSopenharmony_ci                              int *got_frame, AVPacket *avpkt)
210cabdff1aSopenharmony_ci{
211cabdff1aSopenharmony_ci    const uint8_t *buf = avpkt->data;
212cabdff1aSopenharmony_ci    int buf_size = avpkt->size;
213cabdff1aSopenharmony_ci    TSCC2Context *c = avctx->priv_data;
214cabdff1aSopenharmony_ci    GetByteContext gb;
215cabdff1aSopenharmony_ci    uint32_t frame_type, size;
216cabdff1aSopenharmony_ci    int i, val, len, pos = 0;
217cabdff1aSopenharmony_ci    int num_mb = c->mb_width * c->mb_height;
218cabdff1aSopenharmony_ci    int ret;
219cabdff1aSopenharmony_ci
220cabdff1aSopenharmony_ci    bytestream2_init(&gb, buf, buf_size);
221cabdff1aSopenharmony_ci    frame_type = bytestream2_get_byte(&gb);
222cabdff1aSopenharmony_ci    if (frame_type > 1) {
223cabdff1aSopenharmony_ci        av_log(avctx, AV_LOG_ERROR, "Incorrect frame type %"PRIu32"\n",
224cabdff1aSopenharmony_ci               frame_type);
225cabdff1aSopenharmony_ci        return AVERROR_INVALIDDATA;
226cabdff1aSopenharmony_ci    }
227cabdff1aSopenharmony_ci
228cabdff1aSopenharmony_ci    if (frame_type == 0) {
229cabdff1aSopenharmony_ci        // Skip duplicate frames
230cabdff1aSopenharmony_ci        return buf_size;
231cabdff1aSopenharmony_ci    }
232cabdff1aSopenharmony_ci
233cabdff1aSopenharmony_ci    if ((ret = ff_reget_buffer(avctx, c->pic, 0)) < 0) {
234cabdff1aSopenharmony_ci        return ret;
235cabdff1aSopenharmony_ci    }
236cabdff1aSopenharmony_ci
237cabdff1aSopenharmony_ci    if (bytestream2_get_bytes_left(&gb) < 4) {
238cabdff1aSopenharmony_ci        av_log(avctx, AV_LOG_ERROR, "Frame is too short\n");
239cabdff1aSopenharmony_ci        return AVERROR_INVALIDDATA;
240cabdff1aSopenharmony_ci    }
241cabdff1aSopenharmony_ci
242cabdff1aSopenharmony_ci    c->quant[0] = bytestream2_get_byte(&gb);
243cabdff1aSopenharmony_ci    c->quant[1] = bytestream2_get_byte(&gb);
244cabdff1aSopenharmony_ci    if (c->quant[0] < 2 || c->quant[0] > NUM_VLC_SETS + 1 ||
245cabdff1aSopenharmony_ci        c->quant[1] < 2 || c->quant[1] > NUM_VLC_SETS + 1) {
246cabdff1aSopenharmony_ci        av_log(avctx, AV_LOG_ERROR, "Invalid quantisers %d / %d\n",
247cabdff1aSopenharmony_ci               c->quant[0], c->quant[1]);
248cabdff1aSopenharmony_ci        return AVERROR_INVALIDDATA;
249cabdff1aSopenharmony_ci    }
250cabdff1aSopenharmony_ci
251cabdff1aSopenharmony_ci    for (i = 0; i < 3; i++) {
252cabdff1aSopenharmony_ci        c->q[0][i] = tscc2_quants[c->quant[0] - 2][i];
253cabdff1aSopenharmony_ci        c->q[1][i] = tscc2_quants[c->quant[1] - 2][i];
254cabdff1aSopenharmony_ci    }
255cabdff1aSopenharmony_ci
256cabdff1aSopenharmony_ci    bytestream2_skip(&gb, 1);
257cabdff1aSopenharmony_ci
258cabdff1aSopenharmony_ci    size = bytestream2_get_le32(&gb);
259cabdff1aSopenharmony_ci    if (size > bytestream2_get_bytes_left(&gb)) {
260cabdff1aSopenharmony_ci        av_log(avctx, AV_LOG_ERROR, "Slice properties chunk is too large\n");
261cabdff1aSopenharmony_ci        return AVERROR_INVALIDDATA;
262cabdff1aSopenharmony_ci    }
263cabdff1aSopenharmony_ci
264cabdff1aSopenharmony_ci    for (i = 0; i < size; i++) {
265cabdff1aSopenharmony_ci        val   = bytestream2_get_byte(&gb);
266cabdff1aSopenharmony_ci        len   = val & 0x3F;
267cabdff1aSopenharmony_ci        val >>= 6;
268cabdff1aSopenharmony_ci        if (pos + len > num_mb) {
269cabdff1aSopenharmony_ci            av_log(avctx, AV_LOG_ERROR, "Too many slice properties\n");
270cabdff1aSopenharmony_ci            return AVERROR_INVALIDDATA;
271cabdff1aSopenharmony_ci        }
272cabdff1aSopenharmony_ci        memset(c->slice_quants + pos, val, len);
273cabdff1aSopenharmony_ci        pos += len;
274cabdff1aSopenharmony_ci    }
275cabdff1aSopenharmony_ci    if (pos < num_mb) {
276cabdff1aSopenharmony_ci        av_log(avctx, AV_LOG_ERROR, "Too few slice properties (%d / %d)\n",
277cabdff1aSopenharmony_ci               pos, num_mb);
278cabdff1aSopenharmony_ci        return AVERROR_INVALIDDATA;
279cabdff1aSopenharmony_ci    }
280cabdff1aSopenharmony_ci
281cabdff1aSopenharmony_ci    for (i = 0; i < c->mb_height; i++) {
282cabdff1aSopenharmony_ci        size = bytestream2_peek_byte(&gb);
283cabdff1aSopenharmony_ci        if (size & 1) {
284cabdff1aSopenharmony_ci            size = bytestream2_get_byte(&gb) - 1;
285cabdff1aSopenharmony_ci        } else {
286cabdff1aSopenharmony_ci            size = bytestream2_get_le32(&gb) >> 1;
287cabdff1aSopenharmony_ci        }
288cabdff1aSopenharmony_ci        if (!size) {
289cabdff1aSopenharmony_ci            int skip_row = 1, j, off = i * c->mb_width;
290cabdff1aSopenharmony_ci            for (j = 0; j < c->mb_width; j++) {
291cabdff1aSopenharmony_ci                if (c->slice_quants[off + j] == 1 ||
292cabdff1aSopenharmony_ci                    c->slice_quants[off + j] == 2) {
293cabdff1aSopenharmony_ci                    skip_row = 0;
294cabdff1aSopenharmony_ci                    break;
295cabdff1aSopenharmony_ci                }
296cabdff1aSopenharmony_ci            }
297cabdff1aSopenharmony_ci            if (!skip_row) {
298cabdff1aSopenharmony_ci                av_log(avctx, AV_LOG_ERROR, "Non-skip row with zero size\n");
299cabdff1aSopenharmony_ci                return AVERROR_INVALIDDATA;
300cabdff1aSopenharmony_ci            }
301cabdff1aSopenharmony_ci        }
302cabdff1aSopenharmony_ci        if (bytestream2_get_bytes_left(&gb) < size) {
303cabdff1aSopenharmony_ci            av_log(avctx, AV_LOG_ERROR, "Invalid slice size (%"PRIu32"/%u)\n",
304cabdff1aSopenharmony_ci                   size, bytestream2_get_bytes_left(&gb));
305cabdff1aSopenharmony_ci            return AVERROR_INVALIDDATA;
306cabdff1aSopenharmony_ci        }
307cabdff1aSopenharmony_ci        ret = tscc2_decode_slice(c, i, buf + bytestream2_tell(&gb), size);
308cabdff1aSopenharmony_ci        if (ret) {
309cabdff1aSopenharmony_ci            av_log(avctx, AV_LOG_ERROR, "Error decoding slice %d\n", i);
310cabdff1aSopenharmony_ci            return ret;
311cabdff1aSopenharmony_ci        }
312cabdff1aSopenharmony_ci        bytestream2_skip(&gb, size);
313cabdff1aSopenharmony_ci    }
314cabdff1aSopenharmony_ci
315cabdff1aSopenharmony_ci    *got_frame      = 1;
316cabdff1aSopenharmony_ci    if ((ret = av_frame_ref(rframe, c->pic)) < 0)
317cabdff1aSopenharmony_ci        return ret;
318cabdff1aSopenharmony_ci
319cabdff1aSopenharmony_ci    /* always report that the buffer was completely consumed */
320cabdff1aSopenharmony_ci    return buf_size;
321cabdff1aSopenharmony_ci}
322cabdff1aSopenharmony_ci
323cabdff1aSopenharmony_cistatic av_cold int tscc2_decode_end(AVCodecContext *avctx)
324cabdff1aSopenharmony_ci{
325cabdff1aSopenharmony_ci    TSCC2Context * const c = avctx->priv_data;
326cabdff1aSopenharmony_ci
327cabdff1aSopenharmony_ci    av_frame_free(&c->pic);
328cabdff1aSopenharmony_ci    av_freep(&c->slice_quants);
329cabdff1aSopenharmony_ci
330cabdff1aSopenharmony_ci    return 0;
331cabdff1aSopenharmony_ci}
332cabdff1aSopenharmony_ci
333cabdff1aSopenharmony_cistatic av_cold int tscc2_decode_init(AVCodecContext *avctx)
334cabdff1aSopenharmony_ci{
335cabdff1aSopenharmony_ci    TSCC2Context * const c = avctx->priv_data;
336cabdff1aSopenharmony_ci    static AVOnce init_static_once = AV_ONCE_INIT;
337cabdff1aSopenharmony_ci
338cabdff1aSopenharmony_ci    c->avctx = avctx;
339cabdff1aSopenharmony_ci
340cabdff1aSopenharmony_ci    avctx->pix_fmt = AV_PIX_FMT_YUV444P;
341cabdff1aSopenharmony_ci
342cabdff1aSopenharmony_ci    c->mb_width     = FFALIGN(avctx->width,  16) >> 4;
343cabdff1aSopenharmony_ci    c->mb_height    = FFALIGN(avctx->height,  8) >> 3;
344cabdff1aSopenharmony_ci    c->slice_quants = av_malloc(c->mb_width * c->mb_height);
345cabdff1aSopenharmony_ci    if (!c->slice_quants) {
346cabdff1aSopenharmony_ci        av_log(avctx, AV_LOG_ERROR, "Cannot allocate slice information\n");
347cabdff1aSopenharmony_ci        return AVERROR(ENOMEM);
348cabdff1aSopenharmony_ci    }
349cabdff1aSopenharmony_ci
350cabdff1aSopenharmony_ci    c->pic = av_frame_alloc();
351cabdff1aSopenharmony_ci    if (!c->pic)
352cabdff1aSopenharmony_ci        return AVERROR(ENOMEM);
353cabdff1aSopenharmony_ci
354cabdff1aSopenharmony_ci    ff_thread_once(&init_static_once, tscc2_init_vlcs);
355cabdff1aSopenharmony_ci
356cabdff1aSopenharmony_ci    return 0;
357cabdff1aSopenharmony_ci}
358cabdff1aSopenharmony_ci
359cabdff1aSopenharmony_ciconst FFCodec ff_tscc2_decoder = {
360cabdff1aSopenharmony_ci    .p.name         = "tscc2",
361cabdff1aSopenharmony_ci    .p.long_name    = NULL_IF_CONFIG_SMALL("TechSmith Screen Codec 2"),
362cabdff1aSopenharmony_ci    .p.type         = AVMEDIA_TYPE_VIDEO,
363cabdff1aSopenharmony_ci    .p.id           = AV_CODEC_ID_TSCC2,
364cabdff1aSopenharmony_ci    .priv_data_size = sizeof(TSCC2Context),
365cabdff1aSopenharmony_ci    .init           = tscc2_decode_init,
366cabdff1aSopenharmony_ci    .close          = tscc2_decode_end,
367cabdff1aSopenharmony_ci    FF_CODEC_DECODE_CB(tscc2_decode_frame),
368cabdff1aSopenharmony_ci    .p.capabilities = AV_CODEC_CAP_DR1,
369cabdff1aSopenharmony_ci    .caps_internal  = FF_CODEC_CAP_INIT_CLEANUP | FF_CODEC_CAP_INIT_THREADSAFE,
370cabdff1aSopenharmony_ci};
371