Skip to content

Commit

Permalink
avcodec: add Simbiosis IMX video decoder
Browse files Browse the repository at this point in the history
  • Loading branch information
richardpl committed Feb 20, 2021
1 parent 5a4f0d9 commit 8651bf8
Show file tree
Hide file tree
Showing 7 changed files with 165 additions and 1 deletion.
1 change: 1 addition & 0 deletions Changelog
Original file line number Diff line number Diff line change
Expand Up @@ -75,6 +75,7 @@ version <next>:
- setts bitstream filter
- vif video filter
- OpenEXR image encoder
- Simbiosis IMX decoder


version 4.3:
Expand Down
1 change: 1 addition & 0 deletions libavcodec/Makefile
Original file line number Diff line number Diff line change
Expand Up @@ -620,6 +620,7 @@ OBJS-$(CONFIG_SIPR_DECODER) += sipr.o acelp_pitch_delay.o \
acelp_filters.o celp_filters.o \
sipr16k.o
OBJS-$(CONFIG_SIREN_DECODER) += siren.o
OBJS-$(CONFIG_SIMBIOSIS_IMX_DECODER) += imx.o
OBJS-$(CONFIG_SMACKAUD_DECODER) += smacker.o
OBJS-$(CONFIG_SMACKER_DECODER) += smacker.o
OBJS-$(CONFIG_SMC_DECODER) += smc.o
Expand Down
1 change: 1 addition & 0 deletions libavcodec/allcodecs.c
Original file line number Diff line number Diff line change
Expand Up @@ -293,6 +293,7 @@ extern AVCodec ff_sgi_encoder;
extern AVCodec ff_sgi_decoder;
extern AVCodec ff_sgirle_decoder;
extern AVCodec ff_sheervideo_decoder;
extern AVCodec ff_simbiosis_imx_decoder;
extern AVCodec ff_smacker_decoder;
extern AVCodec ff_smc_decoder;
extern AVCodec ff_smvjpeg_decoder;
Expand Down
7 changes: 7 additions & 0 deletions libavcodec/codec_desc.c
Original file line number Diff line number Diff line change
Expand Up @@ -1842,6 +1842,13 @@ static const AVCodecDescriptor codec_descriptors[] = {
.long_name = NULL_IF_CONFIG_SMALL("Cintel RAW"),
.props = AV_CODEC_PROP_INTRA_ONLY | AV_CODEC_PROP_LOSSY | AV_CODEC_PROP_LOSSLESS,
},
{
.id = AV_CODEC_ID_SIMBIOSIS_IMX,
.type = AVMEDIA_TYPE_VIDEO,
.name = "simbiosis_imx",
.long_name = NULL_IF_CONFIG_SMALL("Simbiosis Interactive IMX Video"),
.props = AV_CODEC_PROP_LOSSY,
},

/* various PCM "codecs" */
{
Expand Down
1 change: 1 addition & 0 deletions libavcodec/codec_id.h
Original file line number Diff line number Diff line change
Expand Up @@ -305,6 +305,7 @@ enum AVCodecID {
AV_CODEC_ID_IPU,
AV_CODEC_ID_ARGO,
AV_CODEC_ID_CRI,
AV_CODEC_ID_SIMBIOSIS_IMX,

/* various PCM "codecs" */
AV_CODEC_ID_FIRST_AUDIO = 0x10000, ///< A dummy id pointing at the start of audio codecs
Expand Down
153 changes: 153 additions & 0 deletions libavcodec/imx.c
Original file line number Diff line number Diff line change
@@ -0,0 +1,153 @@
/*
* Copyright (c) 2021 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 "libavutil/common.h"
#include "avcodec.h"
#include "bytestream.h"
#include "internal.h"

typedef struct SimbiosisIMXContext {
uint32_t pal[256];
uint8_t history[32768];
int pos;
} SimbiosisIMXContext;

static av_cold int imx_decode_init(AVCodecContext *avctx)
{
avctx->pix_fmt = AV_PIX_FMT_PAL8;
avctx->width = 320;
avctx->height = 160;
return 0;
}

static int imx_decode_frame(AVCodecContext *avctx, void *data,
int *got_frame, AVPacket *avpkt)
{
SimbiosisIMXContext *imx = avctx->priv_data;
int ret, x, y, pal_size;
const uint8_t *pal = av_packet_get_side_data(avpkt, AV_PKT_DATA_PALETTE, &pal_size);
AVFrame *frame = data;
GetByteContext gb;

if ((ret = ff_get_buffer(avctx, frame, 0)) < 0)
return ret;

if (pal && pal_size == AVPALETTE_SIZE) {
memcpy(imx->pal, pal, pal_size);
frame->palette_has_changed = 1;
}

bytestream2_init(&gb, avpkt->data, avpkt->size);

memcpy(frame->data[1], imx->pal, AVPALETTE_SIZE);

x = 0, y = 0;
while (bytestream2_get_bytes_left(&gb) > 0 &&
x < 320 && y < 160) {
int b = bytestream2_get_byte(&gb);
int len = b & 0x3f;
int op = b >> 6;
int fill;

switch (op) {
case 3:
len = len * 64 + bytestream2_get_byte(&gb);
case 0:
while (len > 0) {
x++;
len--;
if (x >= 320) {
x = 0;
y++;
}
if (y >= 160)
break;
}
break;
case 1:
if (len == 0) {
int offset = bytestream2_get_le16(&gb);

if (offset < 0 || offset >= 32768)
return AVERROR_INVALIDDATA;

len = bytestream2_get_byte(&gb);
while (len > 0 && offset < 32768) {
frame->data[0][x + y * frame->linesize[0]] = imx->history[offset++];
x++;
len--;
if (x >= 320) {
x = 0;
y++;
}
if (y >= 160)
break;
}
} else {
while (len > 0) {
fill = bytestream2_get_byte(&gb);
frame->data[0][x + y * frame->linesize[0]] = fill;
if (imx->pos < 32768)
imx->history[imx->pos++] = fill;
x++;
len--;
if (x >= 320) {
x = 0;
y++;
}
if (y >= 160)
break;
}
}
break;
case 2:
fill = bytestream2_get_byte(&gb);

while (len > 0) {
frame->data[0][x + y * frame->linesize[0]] = fill;
x++;
len--;
if (x >= 320) {
x = 0;
y++;
}
if (y >= 160)
break;
}
break;
}
}

*got_frame = 1;

return avpkt->size;
}

AVCodec ff_simbiosis_imx_decoder = {
.name = "simbiosis_imx",
.long_name = NULL_IF_CONFIG_SMALL("Simbiosis Interactive IMX Video"),
.type = AVMEDIA_TYPE_VIDEO,
.id = AV_CODEC_ID_SIMBIOSIS_IMX,
.priv_data_size = sizeof(SimbiosisIMXContext),
.init = imx_decode_init,
.decode = imx_decode_frame,
.capabilities = AV_CODEC_CAP_DR1,
.caps_internal = FF_CODEC_CAP_INIT_THREADSAFE,
};
2 changes: 1 addition & 1 deletion libavcodec/version.h
Original file line number Diff line number Diff line change
Expand Up @@ -28,7 +28,7 @@
#include "libavutil/version.h"

#define LIBAVCODEC_VERSION_MAJOR 58
#define LIBAVCODEC_VERSION_MINOR 124
#define LIBAVCODEC_VERSION_MINOR 125
#define LIBAVCODEC_VERSION_MICRO 100

#define LIBAVCODEC_VERSION_INT AV_VERSION_INT(LIBAVCODEC_VERSION_MAJOR, \
Expand Down

0 comments on commit 8651bf8

Please sign in to comment.