xref: /third_party/ffmpeg/libavcodec/pgxdec.c (revision cabdff1a)
1cabdff1aSopenharmony_ci/*
2cabdff1aSopenharmony_ci * PGX image format
3cabdff1aSopenharmony_ci * Copyright (c) 2020 Gautam Ramakrishnan
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#include "avcodec.h"
23cabdff1aSopenharmony_ci#include "internal.h"
24cabdff1aSopenharmony_ci#include "bytestream.h"
25cabdff1aSopenharmony_ci#include "codec_internal.h"
26cabdff1aSopenharmony_ci#include "libavutil/imgutils.h"
27cabdff1aSopenharmony_ci
28cabdff1aSopenharmony_cistatic int pgx_get_number(AVCodecContext *avctx, GetByteContext *g, int *number) {
29cabdff1aSopenharmony_ci    int ret = AVERROR_INVALIDDATA;
30cabdff1aSopenharmony_ci    char digit;
31cabdff1aSopenharmony_ci
32cabdff1aSopenharmony_ci    *number = 0;
33cabdff1aSopenharmony_ci    while (1) {
34cabdff1aSopenharmony_ci        uint64_t temp;
35cabdff1aSopenharmony_ci        if (bytestream2_get_bytes_left(g) <= 0)
36cabdff1aSopenharmony_ci            return AVERROR_INVALIDDATA;
37cabdff1aSopenharmony_ci        digit = bytestream2_get_byteu(g);
38cabdff1aSopenharmony_ci        if (digit == ' ' || digit == 0xA || digit == 0xD)
39cabdff1aSopenharmony_ci            break;
40cabdff1aSopenharmony_ci        else if (digit < '0' || digit > '9')
41cabdff1aSopenharmony_ci            return AVERROR_INVALIDDATA;
42cabdff1aSopenharmony_ci
43cabdff1aSopenharmony_ci        temp = (uint64_t)10 * (*number) + (digit - '0');
44cabdff1aSopenharmony_ci        if (temp > INT_MAX)
45cabdff1aSopenharmony_ci            return AVERROR_INVALIDDATA;
46cabdff1aSopenharmony_ci        *number = temp;
47cabdff1aSopenharmony_ci        ret = 0;
48cabdff1aSopenharmony_ci    }
49cabdff1aSopenharmony_ci
50cabdff1aSopenharmony_ci    return ret;
51cabdff1aSopenharmony_ci}
52cabdff1aSopenharmony_ci
53cabdff1aSopenharmony_cistatic int pgx_decode_header(AVCodecContext *avctx, GetByteContext *g,
54cabdff1aSopenharmony_ci                             int *depth, int *width, int *height,
55cabdff1aSopenharmony_ci                             int *sign)
56cabdff1aSopenharmony_ci{
57cabdff1aSopenharmony_ci    int byte;
58cabdff1aSopenharmony_ci
59cabdff1aSopenharmony_ci    if (bytestream2_get_bytes_left(g) < 12)
60cabdff1aSopenharmony_ci        return AVERROR_INVALIDDATA;
61cabdff1aSopenharmony_ci
62cabdff1aSopenharmony_ci    bytestream2_skipu(g, 6);
63cabdff1aSopenharmony_ci
64cabdff1aSopenharmony_ci    // Is the component signed?
65cabdff1aSopenharmony_ci    byte = bytestream2_peek_byteu(g);
66cabdff1aSopenharmony_ci    if (byte == '+') {
67cabdff1aSopenharmony_ci        *sign = 0;
68cabdff1aSopenharmony_ci        bytestream2_skipu(g, 1);
69cabdff1aSopenharmony_ci    } else if (byte == '-') {
70cabdff1aSopenharmony_ci        *sign = 1;
71cabdff1aSopenharmony_ci        bytestream2_skipu(g, 1);
72cabdff1aSopenharmony_ci    }
73cabdff1aSopenharmony_ci
74cabdff1aSopenharmony_ci    byte = bytestream2_peek_byteu(g);
75cabdff1aSopenharmony_ci    if (byte == ' ')
76cabdff1aSopenharmony_ci        bytestream2_skipu(g, 1);
77cabdff1aSopenharmony_ci
78cabdff1aSopenharmony_ci    if (pgx_get_number(avctx, g, depth))
79cabdff1aSopenharmony_ci        goto error;
80cabdff1aSopenharmony_ci    if (pgx_get_number(avctx, g, width))
81cabdff1aSopenharmony_ci        goto error;
82cabdff1aSopenharmony_ci    if (pgx_get_number(avctx, g, height))
83cabdff1aSopenharmony_ci        goto error;
84cabdff1aSopenharmony_ci
85cabdff1aSopenharmony_ci    if (bytestream2_peek_byte(g) == 0xA)
86cabdff1aSopenharmony_ci        bytestream2_skip(g, 1);
87cabdff1aSopenharmony_ci    return 0;
88cabdff1aSopenharmony_ci
89cabdff1aSopenharmony_cierror:
90cabdff1aSopenharmony_ci    av_log(avctx, AV_LOG_ERROR, "Error in decoding header.\n");
91cabdff1aSopenharmony_ci    return AVERROR_INVALIDDATA;
92cabdff1aSopenharmony_ci}
93cabdff1aSopenharmony_ci
94cabdff1aSopenharmony_ci#define WRITE_FRAME(D, PIXEL, suffix)                                                       \
95cabdff1aSopenharmony_ci    static inline void write_frame_ ##D(AVFrame *frame, GetByteContext *g, \
96cabdff1aSopenharmony_ci                                        int width, int height, int sign, int depth)         \
97cabdff1aSopenharmony_ci    {                                                                                       \
98cabdff1aSopenharmony_ci        const unsigned offset = sign ? (1 << (D - 1)) : 0;                                  \
99cabdff1aSopenharmony_ci        int i, j;                                                                           \
100cabdff1aSopenharmony_ci        for (i = 0; i < height; i++) {                                                      \
101cabdff1aSopenharmony_ci            PIXEL *line = (PIXEL*)(frame->data[0] + i * frame->linesize[0]);                \
102cabdff1aSopenharmony_ci            for (j = 0; j < width; j++) {                                                   \
103cabdff1aSopenharmony_ci                unsigned val = bytestream2_get_ ##suffix##u(g) << (D - depth);              \
104cabdff1aSopenharmony_ci                val ^= offset;                                                              \
105cabdff1aSopenharmony_ci                *(line + j) = val;                                                          \
106cabdff1aSopenharmony_ci            }                                                                               \
107cabdff1aSopenharmony_ci        }                                                                                   \
108cabdff1aSopenharmony_ci    }                                                                                       \
109cabdff1aSopenharmony_ci
110cabdff1aSopenharmony_ciWRITE_FRAME(8, uint8_t, byte)
111cabdff1aSopenharmony_ciWRITE_FRAME(16, uint16_t, be16)
112cabdff1aSopenharmony_ci
113cabdff1aSopenharmony_cistatic int pgx_decode_frame(AVCodecContext *avctx, AVFrame *p,
114cabdff1aSopenharmony_ci                            int *got_frame, AVPacket *avpkt)
115cabdff1aSopenharmony_ci{
116cabdff1aSopenharmony_ci    int ret;
117cabdff1aSopenharmony_ci    int bpp;
118cabdff1aSopenharmony_ci    int width, height, depth;
119cabdff1aSopenharmony_ci    int sign = 0;
120cabdff1aSopenharmony_ci    GetByteContext g;
121cabdff1aSopenharmony_ci    bytestream2_init(&g, avpkt->data, avpkt->size);
122cabdff1aSopenharmony_ci
123cabdff1aSopenharmony_ci    if ((ret = pgx_decode_header(avctx, &g, &depth, &width, &height, &sign)) < 0)
124cabdff1aSopenharmony_ci        return ret;
125cabdff1aSopenharmony_ci
126cabdff1aSopenharmony_ci    if ((ret = ff_set_dimensions(avctx, width, height)) < 0)
127cabdff1aSopenharmony_ci        return ret;
128cabdff1aSopenharmony_ci
129cabdff1aSopenharmony_ci    if (depth > 0 && depth <= 8) {
130cabdff1aSopenharmony_ci        avctx->pix_fmt = AV_PIX_FMT_GRAY8;
131cabdff1aSopenharmony_ci        bpp = 8;
132cabdff1aSopenharmony_ci    } else if (depth > 0 && depth <= 16) {
133cabdff1aSopenharmony_ci        avctx->pix_fmt = AV_PIX_FMT_GRAY16;
134cabdff1aSopenharmony_ci        bpp = 16;
135cabdff1aSopenharmony_ci    } else {
136cabdff1aSopenharmony_ci        av_log(avctx, AV_LOG_ERROR, "depth %d is invalid or unsupported.\n", depth);
137cabdff1aSopenharmony_ci        return AVERROR_PATCHWELCOME;
138cabdff1aSopenharmony_ci    }
139cabdff1aSopenharmony_ci    if (bytestream2_get_bytes_left(&g) < width * height * (bpp >> 3))
140cabdff1aSopenharmony_ci        return AVERROR_INVALIDDATA;
141cabdff1aSopenharmony_ci    if ((ret = ff_get_buffer(avctx, p, 0)) < 0)
142cabdff1aSopenharmony_ci        return ret;
143cabdff1aSopenharmony_ci    p->pict_type = AV_PICTURE_TYPE_I;
144cabdff1aSopenharmony_ci    p->key_frame = 1;
145cabdff1aSopenharmony_ci    avctx->bits_per_raw_sample = depth;
146cabdff1aSopenharmony_ci    if (bpp == 8)
147cabdff1aSopenharmony_ci        write_frame_8(p, &g, width, height, sign, depth);
148cabdff1aSopenharmony_ci    else if (bpp == 16)
149cabdff1aSopenharmony_ci        write_frame_16(p, &g, width, height, sign, depth);
150cabdff1aSopenharmony_ci    *got_frame = 1;
151cabdff1aSopenharmony_ci    return 0;
152cabdff1aSopenharmony_ci}
153cabdff1aSopenharmony_ci
154cabdff1aSopenharmony_ciconst FFCodec ff_pgx_decoder = {
155cabdff1aSopenharmony_ci    .p.name         = "pgx",
156cabdff1aSopenharmony_ci    .p.long_name    = NULL_IF_CONFIG_SMALL("PGX (JPEG2000 Test Format)"),
157cabdff1aSopenharmony_ci    .p.type         = AVMEDIA_TYPE_VIDEO,
158cabdff1aSopenharmony_ci    .p.id           = AV_CODEC_ID_PGX,
159cabdff1aSopenharmony_ci    .p.capabilities = AV_CODEC_CAP_DR1,
160cabdff1aSopenharmony_ci    FF_CODEC_DECODE_CB(pgx_decode_frame),
161cabdff1aSopenharmony_ci};
162