mirror of
https://github.com/FFmpeg/FFmpeg.git
synced 2025-01-08 13:22:53 +02:00
a247ac640d
Given that the AVCodec.next pointer has now been removed, most of the AVCodecs are not modified at all any more and can therefore be made const (as this patch does); the only exceptions are the very few codecs for external libraries that have a init_static_data callback. Signed-off-by: Andreas Rheinhardt <andreas.rheinhardt@gmail.com> Signed-off-by: James Almer <jamrial@gmail.com>
683 lines
19 KiB
C
683 lines
19 KiB
C
/*
|
|
* ScreenPressor decoder
|
|
*
|
|
* Copyright (c) 2017 Paul B Mahol
|
|
*
|
|
* This file is part of FFmpeg.
|
|
*
|
|
* FFmpeg is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
* License as published by the Free Software Foundation; either
|
|
* version 2.1 of the License, or (at your option) any later version.
|
|
*
|
|
* FFmpeg is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
* Lesser General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
* License along with FFmpeg; if not, write to the Free Software
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
|
*/
|
|
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
|
|
#include "avcodec.h"
|
|
#include "bytestream.h"
|
|
#include "internal.h"
|
|
#include "scpr.h"
|
|
#include "scpr3.h"
|
|
|
|
#define TOP 0x01000000
|
|
#define BOT 0x010000
|
|
|
|
#include "scpr3.c"
|
|
|
|
static void init_rangecoder(RangeCoder *rc, GetByteContext *gb)
|
|
{
|
|
rc->code1 = 0;
|
|
rc->range = 0xFFFFFFFFU;
|
|
rc->code = bytestream2_get_be32(gb);
|
|
}
|
|
|
|
static void reinit_tables(SCPRContext *s)
|
|
{
|
|
int comp, i, j;
|
|
|
|
for (comp = 0; comp < 3; comp++) {
|
|
for (j = 0; j < 4096; j++) {
|
|
if (s->pixel_model[comp][j].total_freq != 256) {
|
|
for (i = 0; i < 256; i++)
|
|
s->pixel_model[comp][j].freq[i] = 1;
|
|
for (i = 0; i < 16; i++)
|
|
s->pixel_model[comp][j].lookup[i] = 16;
|
|
s->pixel_model[comp][j].total_freq = 256;
|
|
}
|
|
}
|
|
}
|
|
|
|
for (j = 0; j < 6; j++) {
|
|
uint32_t *p = s->run_model[j];
|
|
for (i = 0; i < 256; i++)
|
|
p[i] = 1;
|
|
p[256] = 256;
|
|
}
|
|
|
|
for (j = 0; j < 6; j++) {
|
|
uint32_t *op = s->op_model[j];
|
|
for (i = 0; i < 6; i++)
|
|
op[i] = 1;
|
|
op[6] = 6;
|
|
}
|
|
|
|
for (i = 0; i < 256; i++) {
|
|
s->range_model[i] = 1;
|
|
s->count_model[i] = 1;
|
|
}
|
|
s->range_model[256] = 256;
|
|
s->count_model[256] = 256;
|
|
|
|
for (i = 0; i < 5; i++) {
|
|
s->fill_model[i] = 1;
|
|
}
|
|
s->fill_model[5] = 5;
|
|
|
|
for (j = 0; j < 4; j++) {
|
|
for (i = 0; i < 16; i++) {
|
|
s->sxy_model[j][i] = 1;
|
|
}
|
|
s->sxy_model[j][16] = 16;
|
|
}
|
|
|
|
for (i = 0; i < 512; i++) {
|
|
s->mv_model[0][i] = 1;
|
|
s->mv_model[1][i] = 1;
|
|
}
|
|
s->mv_model[0][512] = 512;
|
|
s->mv_model[1][512] = 512;
|
|
}
|
|
|
|
static int decode(GetByteContext *gb, RangeCoder *rc, uint32_t cumFreq, uint32_t freq, uint32_t total_freq)
|
|
{
|
|
rc->code -= cumFreq * rc->range;
|
|
rc->range *= freq;
|
|
|
|
while (rc->range < TOP && bytestream2_get_bytes_left(gb) > 0) {
|
|
uint32_t byte = bytestream2_get_byteu(gb);
|
|
rc->code = (rc->code << 8) | byte;
|
|
rc->range <<= 8;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int get_freq(RangeCoder *rc, uint32_t total_freq, uint32_t *freq)
|
|
{
|
|
if (total_freq == 0)
|
|
return AVERROR_INVALIDDATA;
|
|
|
|
rc->range = rc->range / total_freq;
|
|
|
|
if (rc->range == 0)
|
|
return AVERROR_INVALIDDATA;
|
|
|
|
*freq = rc->code / rc->range;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int decode0(GetByteContext *gb, RangeCoder *rc, uint32_t cumFreq, uint32_t freq, uint32_t total_freq)
|
|
{
|
|
uint32_t t;
|
|
|
|
if (total_freq == 0)
|
|
return AVERROR_INVALIDDATA;
|
|
|
|
t = rc->range * (uint64_t)cumFreq / total_freq;
|
|
|
|
rc->code1 += t + 1;
|
|
rc->range = rc->range * (uint64_t)(freq + cumFreq) / total_freq - (t + 1);
|
|
|
|
while (rc->range < TOP && bytestream2_get_bytes_left(gb) > 0) {
|
|
uint32_t byte = bytestream2_get_byteu(gb);
|
|
rc->code = (rc->code << 8) | byte;
|
|
rc->code1 <<= 8;
|
|
rc->range <<= 8;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int get_freq0(RangeCoder *rc, uint32_t total_freq, uint32_t *freq)
|
|
{
|
|
if (rc->range == 0)
|
|
return AVERROR_INVALIDDATA;
|
|
|
|
*freq = total_freq * (uint64_t)(rc->code - rc->code1) / rc->range;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int decode_value(SCPRContext *s, uint32_t *cnt, uint32_t maxc, uint32_t step, uint32_t *rval)
|
|
{
|
|
GetByteContext *gb = &s->gb;
|
|
RangeCoder *rc = &s->rc;
|
|
uint32_t totfr = cnt[maxc];
|
|
uint32_t value;
|
|
uint32_t c = 0, cumfr = 0, cnt_c = 0;
|
|
int i, ret;
|
|
|
|
if ((ret = s->get_freq(rc, totfr, &value)) < 0)
|
|
return ret;
|
|
|
|
while (c < maxc) {
|
|
cnt_c = cnt[c];
|
|
if (value >= cumfr + cnt_c)
|
|
cumfr += cnt_c;
|
|
else
|
|
break;
|
|
c++;
|
|
}
|
|
|
|
if (c >= maxc)
|
|
return AVERROR_INVALIDDATA;
|
|
|
|
if ((ret = s->decode(gb, rc, cumfr, cnt_c, totfr)) < 0)
|
|
return ret;
|
|
|
|
cnt[c] = cnt_c + step;
|
|
totfr += step;
|
|
if (totfr > BOT) {
|
|
totfr = 0;
|
|
for (i = 0; i < maxc; i++) {
|
|
uint32_t nc = (cnt[i] >> 1) + 1;
|
|
cnt[i] = nc;
|
|
totfr += nc;
|
|
}
|
|
}
|
|
|
|
cnt[maxc] = totfr;
|
|
*rval = c;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int decode_unit(SCPRContext *s, PixelModel *pixel, uint32_t step, uint32_t *rval)
|
|
{
|
|
GetByteContext *gb = &s->gb;
|
|
RangeCoder *rc = &s->rc;
|
|
uint32_t totfr = pixel->total_freq;
|
|
uint32_t value, x = 0, cumfr = 0, cnt_x = 0;
|
|
int i, j, ret, c, cnt_c;
|
|
|
|
if ((ret = s->get_freq(rc, totfr, &value)) < 0)
|
|
return ret;
|
|
|
|
while (x < 16) {
|
|
cnt_x = pixel->lookup[x];
|
|
if (value >= cumfr + cnt_x)
|
|
cumfr += cnt_x;
|
|
else
|
|
break;
|
|
x++;
|
|
}
|
|
|
|
c = x * 16;
|
|
cnt_c = 0;
|
|
while (c < 256) {
|
|
cnt_c = pixel->freq[c];
|
|
if (value >= cumfr + cnt_c)
|
|
cumfr += cnt_c;
|
|
else
|
|
break;
|
|
c++;
|
|
}
|
|
if (x >= 16 || c >= 256) {
|
|
return AVERROR_INVALIDDATA;
|
|
}
|
|
|
|
if ((ret = s->decode(gb, rc, cumfr, cnt_c, totfr)) < 0)
|
|
return ret;
|
|
|
|
pixel->freq[c] = cnt_c + step;
|
|
pixel->lookup[x] = cnt_x + step;
|
|
totfr += step;
|
|
if (totfr > BOT) {
|
|
totfr = 0;
|
|
for (i = 0; i < 256; i++) {
|
|
uint32_t nc = (pixel->freq[i] >> 1) + 1;
|
|
pixel->freq[i] = nc;
|
|
totfr += nc;
|
|
}
|
|
for (i = 0; i < 16; i++) {
|
|
uint32_t sum = 0;
|
|
uint32_t i16_17 = i << 4;
|
|
for (j = 0; j < 16; j++)
|
|
sum += pixel->freq[i16_17 + j];
|
|
pixel->lookup[i] = sum;
|
|
}
|
|
}
|
|
pixel->total_freq = totfr;
|
|
|
|
*rval = c & s->cbits;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int decode_units(SCPRContext *s, uint32_t *r, uint32_t *g, uint32_t *b,
|
|
int *cx, int *cx1)
|
|
{
|
|
const int cxshift = s->cxshift;
|
|
int ret;
|
|
|
|
ret = decode_unit(s, &s->pixel_model[0][*cx + *cx1], 400, r);
|
|
if (ret < 0)
|
|
return ret;
|
|
|
|
*cx1 = (*cx << 6) & 0xFC0;
|
|
*cx = *r >> cxshift;
|
|
ret = decode_unit(s, &s->pixel_model[1][*cx + *cx1], 400, g);
|
|
if (ret < 0)
|
|
return ret;
|
|
|
|
*cx1 = (*cx << 6) & 0xFC0;
|
|
*cx = *g >> cxshift;
|
|
ret = decode_unit(s, &s->pixel_model[2][*cx + *cx1], 400, b);
|
|
if (ret < 0)
|
|
return ret;
|
|
|
|
*cx1 = (*cx << 6) & 0xFC0;
|
|
*cx = *b >> cxshift;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int decompress_i(AVCodecContext *avctx, uint32_t *dst, int linesize)
|
|
{
|
|
SCPRContext *s = avctx->priv_data;
|
|
GetByteContext *gb = &s->gb;
|
|
int cx = 0, cx1 = 0, k = 0;
|
|
int run, off, y = 0, x = 0, ret;
|
|
uint32_t clr = 0, r, g, b, backstep = linesize - avctx->width;
|
|
uint32_t lx, ly, ptype;
|
|
|
|
reinit_tables(s);
|
|
bytestream2_skip(gb, 2);
|
|
init_rangecoder(&s->rc, gb);
|
|
|
|
while (k < avctx->width + 1) {
|
|
ret = decode_units(s, &r, &g, &b, &cx, &cx1);
|
|
if (ret < 0)
|
|
return ret;
|
|
|
|
ret = decode_value(s, s->run_model[0], 256, 400, &run);
|
|
if (ret < 0)
|
|
return ret;
|
|
if (run <= 0)
|
|
return AVERROR_INVALIDDATA;
|
|
|
|
clr = (b << 16) + (g << 8) + r;
|
|
k += run;
|
|
while (run-- > 0) {
|
|
if (y >= avctx->height)
|
|
return AVERROR_INVALIDDATA;
|
|
|
|
dst[y * linesize + x] = clr;
|
|
lx = x;
|
|
ly = y;
|
|
x++;
|
|
if (x >= avctx->width) {
|
|
x = 0;
|
|
y++;
|
|
}
|
|
}
|
|
}
|
|
off = -linesize - 1;
|
|
ptype = 0;
|
|
|
|
while (x < avctx->width && y < avctx->height) {
|
|
ret = decode_value(s, s->op_model[ptype], 6, 1000, &ptype);
|
|
if (ret < 0)
|
|
return ret;
|
|
if (ptype == 0) {
|
|
ret = decode_units(s, &r, &g, &b, &cx, &cx1);
|
|
if (ret < 0)
|
|
return ret;
|
|
|
|
clr = (b << 16) + (g << 8) + r;
|
|
}
|
|
if (ptype > 5)
|
|
return AVERROR_INVALIDDATA;
|
|
ret = decode_value(s, s->run_model[ptype], 256, 400, &run);
|
|
if (ret < 0)
|
|
return ret;
|
|
if (run <= 0)
|
|
return AVERROR_INVALIDDATA;
|
|
|
|
ret = decode_run_i(avctx, ptype, run, &x, &y, clr,
|
|
dst, linesize, &lx, &ly,
|
|
backstep, off, &cx, &cx1);
|
|
if (ret < 0)
|
|
return ret;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int decompress_p(AVCodecContext *avctx,
|
|
uint32_t *dst, int linesize,
|
|
uint32_t *prev, int plinesize)
|
|
{
|
|
SCPRContext *s = avctx->priv_data;
|
|
GetByteContext *gb = &s->gb;
|
|
int ret, temp = 0, min, max, x, y, cx = 0, cx1 = 0;
|
|
int backstep = linesize - avctx->width;
|
|
|
|
if (bytestream2_get_byte(gb) == 0)
|
|
return 1;
|
|
bytestream2_skip(gb, 1);
|
|
init_rangecoder(&s->rc, gb);
|
|
|
|
ret = decode_value(s, s->range_model, 256, 1, &min);
|
|
ret |= decode_value(s, s->range_model, 256, 1, &temp);
|
|
if (ret < 0)
|
|
return ret;
|
|
|
|
min += temp << 8;
|
|
ret = decode_value(s, s->range_model, 256, 1, &max);
|
|
ret |= decode_value(s, s->range_model, 256, 1, &temp);
|
|
if (ret < 0)
|
|
return ret;
|
|
|
|
max += temp << 8;
|
|
if (min > max || min >= s->nbcount)
|
|
return AVERROR_INVALIDDATA;
|
|
|
|
memset(s->blocks, 0, sizeof(*s->blocks) * s->nbcount);
|
|
|
|
while (min <= max) {
|
|
int fill, count;
|
|
|
|
ret = decode_value(s, s->fill_model, 5, 10, &fill);
|
|
ret |= decode_value(s, s->count_model, 256, 20, &count);
|
|
if (ret < 0)
|
|
return ret;
|
|
if (count <= 0)
|
|
return AVERROR_INVALIDDATA;
|
|
|
|
while (min < s->nbcount && count-- > 0) {
|
|
s->blocks[min++] = fill;
|
|
}
|
|
}
|
|
|
|
ret = av_frame_copy(s->current_frame, s->last_frame);
|
|
if (ret < 0)
|
|
return ret;
|
|
|
|
for (y = 0; y < s->nby; y++) {
|
|
for (x = 0; x < s->nbx; x++) {
|
|
int sy1 = 0, sy2 = 16, sx1 = 0, sx2 = 16;
|
|
|
|
if (s->blocks[y * s->nbx + x] == 0)
|
|
continue;
|
|
|
|
if (((s->blocks[y * s->nbx + x] - 1) & 1) > 0) {
|
|
ret = decode_value(s, s->sxy_model[0], 16, 100, &sx1);
|
|
ret |= decode_value(s, s->sxy_model[1], 16, 100, &sy1);
|
|
ret |= decode_value(s, s->sxy_model[2], 16, 100, &sx2);
|
|
ret |= decode_value(s, s->sxy_model[3], 16, 100, &sy2);
|
|
if (ret < 0)
|
|
return ret;
|
|
|
|
sx2++;
|
|
sy2++;
|
|
}
|
|
if (((s->blocks[y * s->nbx + x] - 1) & 2) > 0) {
|
|
int i, j, by = y * 16, bx = x * 16;
|
|
int mvx, mvy;
|
|
|
|
ret = decode_value(s, s->mv_model[0], 512, 100, &mvx);
|
|
ret |= decode_value(s, s->mv_model[1], 512, 100, &mvy);
|
|
if (ret < 0)
|
|
return ret;
|
|
|
|
mvx -= 256;
|
|
mvy -= 256;
|
|
|
|
if (by + mvy + sy1 < 0 || bx + mvx + sx1 < 0 ||
|
|
by + mvy + sy1 >= avctx->height || bx + mvx + sx1 >= avctx->width)
|
|
return AVERROR_INVALIDDATA;
|
|
|
|
for (i = 0; i < sy2 - sy1 && (by + sy1 + i) < avctx->height && (by + mvy + sy1 + i) < avctx->height; i++) {
|
|
for (j = 0; j < sx2 - sx1 && (bx + sx1 + j) < avctx->width && (bx + mvx + sx1 + j) < avctx->width; j++) {
|
|
dst[(by + i + sy1) * linesize + bx + sx1 + j] = prev[(by + mvy + sy1 + i) * plinesize + bx + sx1 + mvx + j];
|
|
}
|
|
}
|
|
} else {
|
|
int run, bx = x * 16 + sx1, by = y * 16 + sy1;
|
|
uint32_t r, g, b, clr, ptype = 0;
|
|
|
|
for (; by < y * 16 + sy2 && by < avctx->height;) {
|
|
ret = decode_value(s, s->op_model[ptype], 6, 1000, &ptype);
|
|
if (ret < 0)
|
|
return ret;
|
|
if (ptype == 0) {
|
|
ret = decode_units(s, &r, &g, &b, &cx, &cx1);
|
|
if (ret < 0)
|
|
return ret;
|
|
|
|
clr = (b << 16) + (g << 8) + r;
|
|
}
|
|
if (ptype > 5)
|
|
return AVERROR_INVALIDDATA;
|
|
ret = decode_value(s, s->run_model[ptype], 256, 400, &run);
|
|
if (ret < 0)
|
|
return ret;
|
|
if (run <= 0)
|
|
return AVERROR_INVALIDDATA;
|
|
|
|
ret = decode_run_p(avctx, ptype, run, x, y, clr,
|
|
dst, prev, linesize, plinesize, &bx, &by,
|
|
backstep, sx1, sx2, &cx, &cx1);
|
|
if (ret < 0)
|
|
return ret;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int decode_frame(AVCodecContext *avctx, void *data, int *got_frame,
|
|
AVPacket *avpkt)
|
|
{
|
|
SCPRContext *s = avctx->priv_data;
|
|
GetByteContext *gb = &s->gb;
|
|
AVFrame *frame = data;
|
|
int ret, type;
|
|
|
|
if (avctx->bits_per_coded_sample == 16) {
|
|
if ((ret = ff_get_buffer(avctx, frame, 0)) < 0)
|
|
return ret;
|
|
}
|
|
|
|
if ((ret = ff_reget_buffer(avctx, s->current_frame, 0)) < 0)
|
|
return ret;
|
|
|
|
bytestream2_init(gb, avpkt->data, avpkt->size);
|
|
|
|
type = bytestream2_peek_byte(gb);
|
|
|
|
if (type == 2) {
|
|
s->version = 1;
|
|
s->get_freq = get_freq0;
|
|
s->decode = decode0;
|
|
frame->key_frame = 1;
|
|
ret = decompress_i(avctx, (uint32_t *)s->current_frame->data[0],
|
|
s->current_frame->linesize[0] / 4);
|
|
} else if (type == 18) {
|
|
s->version = 2;
|
|
s->get_freq = get_freq;
|
|
s->decode = decode;
|
|
frame->key_frame = 1;
|
|
ret = decompress_i(avctx, (uint32_t *)s->current_frame->data[0],
|
|
s->current_frame->linesize[0] / 4);
|
|
} else if (type == 34) {
|
|
frame->key_frame = 1;
|
|
s->version = 3;
|
|
ret = decompress_i3(avctx, (uint32_t *)s->current_frame->data[0],
|
|
s->current_frame->linesize[0] / 4);
|
|
} else if (type == 17 || type == 33) {
|
|
uint32_t clr, *dst = (uint32_t *)s->current_frame->data[0];
|
|
int y;
|
|
|
|
if (bytestream2_get_bytes_left(gb) < 3)
|
|
return AVERROR_INVALIDDATA;
|
|
|
|
frame->key_frame = 1;
|
|
bytestream2_skip(gb, 1);
|
|
if (avctx->bits_per_coded_sample == 16) {
|
|
uint16_t value = bytestream2_get_le16(gb);
|
|
int r, g, b;
|
|
|
|
r = (value ) & 31;
|
|
g = (value >> 5) & 31;
|
|
b = (value >> 10) & 31;
|
|
clr = (r << 16) + (g << 8) + b;
|
|
} else {
|
|
clr = bytestream2_get_le24(gb);
|
|
}
|
|
for (y = 0; y < avctx->height; y++) {
|
|
dst[0] = clr;
|
|
av_memcpy_backptr((uint8_t*)(dst+1), 4, 4*avctx->width - 4);
|
|
dst += s->current_frame->linesize[0] / 4;
|
|
}
|
|
} else if (type == 0 || type == 1) {
|
|
frame->key_frame = 0;
|
|
|
|
if (s->version == 1 || s->version == 2)
|
|
ret = decompress_p(avctx, (uint32_t *)s->current_frame->data[0],
|
|
s->current_frame->linesize[0] / 4,
|
|
(uint32_t *)s->last_frame->data[0],
|
|
s->last_frame->linesize[0] / 4);
|
|
else
|
|
ret = decompress_p3(avctx, (uint32_t *)s->current_frame->data[0],
|
|
s->current_frame->linesize[0] / 4,
|
|
(uint32_t *)s->last_frame->data[0],
|
|
s->last_frame->linesize[0] / 4);
|
|
if (ret == 1)
|
|
return avpkt->size;
|
|
} else {
|
|
return AVERROR_PATCHWELCOME;
|
|
}
|
|
|
|
if (ret < 0)
|
|
return ret;
|
|
|
|
if (bytestream2_get_bytes_left(gb) > 5)
|
|
return AVERROR_INVALIDDATA;
|
|
|
|
if (avctx->bits_per_coded_sample != 16) {
|
|
ret = av_frame_ref(data, s->current_frame);
|
|
if (ret < 0)
|
|
return ret;
|
|
} else {
|
|
uint8_t *dst = frame->data[0];
|
|
int x, y;
|
|
|
|
ret = av_frame_copy(frame, s->current_frame);
|
|
if (ret < 0)
|
|
return ret;
|
|
|
|
// scale up each sample by 8
|
|
for (y = 0; y < avctx->height; y++) {
|
|
// If the image is sufficiently aligned, compute 8 samples at once
|
|
if (!(((uintptr_t)dst) & 7)) {
|
|
uint64_t *dst64 = (uint64_t *)dst;
|
|
int w = avctx->width>>1;
|
|
for (x = 0; x < w; x++) {
|
|
dst64[x] = (dst64[x] << 3) & 0xFCFCFCFCFCFCFCFCULL;
|
|
}
|
|
x *= 8;
|
|
} else
|
|
x = 0;
|
|
for (; x < avctx->width * 4; x++) {
|
|
dst[x] = dst[x] << 3;
|
|
}
|
|
dst += frame->linesize[0];
|
|
}
|
|
}
|
|
|
|
frame->pict_type = frame->key_frame ? AV_PICTURE_TYPE_I : AV_PICTURE_TYPE_P;
|
|
|
|
FFSWAP(AVFrame *, s->current_frame, s->last_frame);
|
|
|
|
frame->data[0] += frame->linesize[0] * (avctx->height - 1);
|
|
frame->linesize[0] *= -1;
|
|
|
|
*got_frame = 1;
|
|
|
|
return avpkt->size;
|
|
}
|
|
|
|
static av_cold int decode_init(AVCodecContext *avctx)
|
|
{
|
|
SCPRContext *s = avctx->priv_data;
|
|
|
|
switch (avctx->bits_per_coded_sample) {
|
|
case 16: avctx->pix_fmt = AV_PIX_FMT_RGB0; break;
|
|
case 24:
|
|
case 32: avctx->pix_fmt = AV_PIX_FMT_BGR0; break;
|
|
default:
|
|
av_log(avctx, AV_LOG_ERROR, "Unsupported bitdepth %i\n", avctx->bits_per_coded_sample);
|
|
return AVERROR_INVALIDDATA;
|
|
}
|
|
|
|
s->get_freq = get_freq0;
|
|
s->decode = decode0;
|
|
|
|
s->cxshift = avctx->bits_per_coded_sample == 16 ? 0 : 2;
|
|
s->cbits = avctx->bits_per_coded_sample == 16 ? 0x1F : 0xFF;
|
|
s->nbx = (avctx->width + 15) / 16;
|
|
s->nby = (avctx->height + 15) / 16;
|
|
s->nbcount = s->nbx * s->nby;
|
|
s->blocks = av_malloc_array(s->nbcount, sizeof(*s->blocks));
|
|
if (!s->blocks)
|
|
return AVERROR(ENOMEM);
|
|
|
|
s->last_frame = av_frame_alloc();
|
|
s->current_frame = av_frame_alloc();
|
|
if (!s->last_frame || !s->current_frame)
|
|
return AVERROR(ENOMEM);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static av_cold int decode_close(AVCodecContext *avctx)
|
|
{
|
|
SCPRContext *s = avctx->priv_data;
|
|
|
|
av_freep(&s->blocks);
|
|
av_frame_free(&s->last_frame);
|
|
av_frame_free(&s->current_frame);
|
|
|
|
return 0;
|
|
}
|
|
|
|
const AVCodec ff_scpr_decoder = {
|
|
.name = "scpr",
|
|
.long_name = NULL_IF_CONFIG_SMALL("ScreenPressor"),
|
|
.type = AVMEDIA_TYPE_VIDEO,
|
|
.id = AV_CODEC_ID_SCPR,
|
|
.priv_data_size = sizeof(SCPRContext),
|
|
.init = decode_init,
|
|
.close = decode_close,
|
|
.decode = decode_frame,
|
|
.capabilities = AV_CODEC_CAP_DR1,
|
|
.caps_internal = FF_CODEC_CAP_INIT_THREADSAFE |
|
|
FF_CODEC_CAP_INIT_CLEANUP,
|
|
};
|