Coverage Report

Created: 2026-04-01 07:42

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/ffmpeg/libavcodec/r210dec.c
Line
Count
Source
1
/*
2
 * R210 decoder
3
 *
4
 * Copyright (c) 2009 Reimar Doeffinger <Reimar.Doeffinger@gmx.de>
5
 *
6
 * This file is part of FFmpeg.
7
 *
8
 * FFmpeg is free software; you can redistribute it and/or
9
 * modify it under the terms of the GNU Lesser General Public
10
 * License as published by the Free Software Foundation; either
11
 * version 2.1 of the License, or (at your option) any later version.
12
 *
13
 * FFmpeg is distributed in the hope that it will be useful,
14
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
15
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
16
 * Lesser General Public License for more details.
17
 *
18
 * You should have received a copy of the GNU Lesser General Public
19
 * License along with FFmpeg; if not, write to the Free Software
20
 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
21
 */
22
23
#include "avcodec.h"
24
#include "codec_internal.h"
25
#include "config_components.h"
26
#include "decode.h"
27
#include "libavutil/bswap.h"
28
#include "libavutil/common.h"
29
30
static av_cold int decode_init(AVCodecContext *avctx)
31
2.42k
{
32
2.42k
    avctx->pix_fmt = AV_PIX_FMT_GBRP10;
33
2.42k
    avctx->bits_per_raw_sample = 10;
34
35
2.42k
    return 0;
36
2.42k
}
37
38
static int decode_frame(AVCodecContext *avctx, AVFrame *pic,
39
                        int *got_frame, AVPacket *avpkt)
40
1.93M
{
41
1.93M
    int h, w, ret;
42
1.93M
    const uint32_t *src = (const uint32_t *)avpkt->data;
43
1.93M
    int aligned_width = FFALIGN(avctx->width,
44
1.93M
                                avctx->codec_id == AV_CODEC_ID_R10K ? 1 : 64);
45
1.93M
    uint8_t *g_line, *b_line, *r_line;
46
1.93M
    int r10 = (avctx->codec_tag & 0xFFFFFF) == MKTAG('r', '1', '0', 0);
47
1.93M
    int le = avctx->codec_tag == MKTAG('R', '1', '0', 'k') &&
48
23.6k
             avctx->extradata_size >= 12 && !memcmp(&avctx->extradata[4], "DpxE", 4) &&
49
5.04k
             !avctx->extradata[11];
50
51
1.93M
    if (avpkt->size < 4 * aligned_width * avctx->height) {
52
1.68M
        av_log(avctx, AV_LOG_ERROR, "packet too small\n");
53
1.68M
        return AVERROR_INVALIDDATA;
54
1.68M
    }
55
56
248k
    if ((ret = ff_get_buffer(avctx, pic, 0)) < 0)
57
107k
        return ret;
58
59
140k
    g_line = pic->data[0];
60
140k
    b_line = pic->data[1];
61
140k
    r_line = pic->data[2];
62
63
421k
    for (h = 0; h < avctx->height; h++) {
64
280k
        uint16_t *dstg = (uint16_t *)g_line;
65
280k
        uint16_t *dstb = (uint16_t *)b_line;
66
280k
        uint16_t *dstr = (uint16_t *)r_line;
67
5.70M
        for (w = 0; w < avctx->width; w++) {
68
5.42M
            uint32_t pixel;
69
5.42M
            uint16_t r, g, b;
70
5.42M
            if (avctx->codec_id == AV_CODEC_ID_AVRP || r10 || le) {
71
2.51M
                pixel = av_le2ne32(*src++);
72
2.90M
            } else {
73
2.90M
                pixel = av_be2ne32(*src++);
74
2.90M
            }
75
5.42M
            if (avctx->codec_id == AV_CODEC_ID_R210) {
76
825k
                b =  pixel & 0x3ff;
77
825k
                g = (pixel >> 10) & 0x3ff;
78
825k
                r = (pixel >> 20) & 0x3ff;
79
4.59M
            } else if (r10) {
80
2.46k
                r =  pixel & 0x3ff;
81
2.46k
                g = (pixel >> 10) & 0x3ff;
82
2.46k
                b = (pixel >> 20) & 0x3ff;
83
4.59M
            } else {
84
4.59M
                b = (pixel >>  2) & 0x3ff;
85
4.59M
                g = (pixel >> 12) & 0x3ff;
86
4.59M
                r = (pixel >> 22) & 0x3ff;
87
4.59M
            }
88
5.42M
            *dstr++ = r;
89
5.42M
            *dstg++ = g;
90
5.42M
            *dstb++ = b;
91
5.42M
        }
92
280k
        src += aligned_width - avctx->width;
93
280k
        g_line += pic->linesize[0];
94
280k
        b_line += pic->linesize[1];
95
280k
        r_line += pic->linesize[2];
96
280k
    }
97
98
140k
    *got_frame      = 1;
99
100
140k
    return avpkt->size;
101
248k
}
102
103
#if CONFIG_R210_DECODER
104
const FFCodec ff_r210_decoder = {
105
    .p.name         = "r210",
106
    CODEC_LONG_NAME("Uncompressed RGB 10-bit"),
107
    .p.type         = AVMEDIA_TYPE_VIDEO,
108
    .p.id           = AV_CODEC_ID_R210,
109
    .init           = decode_init,
110
    FF_CODEC_DECODE_CB(decode_frame),
111
    .p.capabilities = AV_CODEC_CAP_DR1,
112
};
113
#endif
114
#if CONFIG_R10K_DECODER
115
const FFCodec ff_r10k_decoder = {
116
    .p.name         = "r10k",
117
    CODEC_LONG_NAME("AJA Kona 10-bit RGB Codec"),
118
    .p.type         = AVMEDIA_TYPE_VIDEO,
119
    .p.id           = AV_CODEC_ID_R10K,
120
    .init           = decode_init,
121
    FF_CODEC_DECODE_CB(decode_frame),
122
    .p.capabilities = AV_CODEC_CAP_DR1,
123
};
124
#endif
125
#if CONFIG_AVRP_DECODER
126
const FFCodec ff_avrp_decoder = {
127
    .p.name         = "avrp",
128
    CODEC_LONG_NAME("Avid 1:1 10-bit RGB Packer"),
129
    .p.type         = AVMEDIA_TYPE_VIDEO,
130
    .p.id           = AV_CODEC_ID_AVRP,
131
    .init           = decode_init,
132
    FF_CODEC_DECODE_CB(decode_frame),
133
    .p.capabilities = AV_CODEC_CAP_DR1,
134
};
135
#endif