Coverage Report

Created: 2026-02-14 06:59

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.56k
{
32
2.56k
    avctx->pix_fmt = AV_PIX_FMT_GBRP10;
33
2.56k
    avctx->bits_per_raw_sample = 10;
34
35
2.56k
    return 0;
36
2.56k
}
37
38
static int decode_frame(AVCodecContext *avctx, AVFrame *pic,
39
                        int *got_frame, AVPacket *avpkt)
40
1.90M
{
41
1.90M
    int h, w, ret;
42
1.90M
    const uint32_t *src = (const uint32_t *)avpkt->data;
43
1.90M
    int aligned_width = FFALIGN(avctx->width,
44
1.90M
                                avctx->codec_id == AV_CODEC_ID_R10K ? 1 : 64);
45
1.90M
    uint8_t *g_line, *b_line, *r_line;
46
1.90M
    int r10 = (avctx->codec_tag & 0xFFFFFF) == MKTAG('r', '1', '0', 0);
47
1.90M
    int le = avctx->codec_tag == MKTAG('R', '1', '0', 'k') &&
48
28.4k
             avctx->extradata_size >= 12 && !memcmp(&avctx->extradata[4], "DpxE", 4) &&
49
3.88k
             !avctx->extradata[11];
50
51
1.90M
    if (avpkt->size < 4 * aligned_width * avctx->height) {
52
1.71M
        av_log(avctx, AV_LOG_ERROR, "packet too small\n");
53
1.71M
        return AVERROR_INVALIDDATA;
54
1.71M
    }
55
56
186k
    if ((ret = ff_get_buffer(avctx, pic, 0)) < 0)
57
56.7k
        return ret;
58
59
129k
    g_line = pic->data[0];
60
129k
    b_line = pic->data[1];
61
129k
    r_line = pic->data[2];
62
63
447k
    for (h = 0; h < avctx->height; h++) {
64
317k
        uint16_t *dstg = (uint16_t *)g_line;
65
317k
        uint16_t *dstb = (uint16_t *)b_line;
66
317k
        uint16_t *dstr = (uint16_t *)r_line;
67
6.13M
        for (w = 0; w < avctx->width; w++) {
68
5.81M
            uint32_t pixel;
69
5.81M
            uint16_t r, g, b;
70
5.81M
            if (avctx->codec_id == AV_CODEC_ID_AVRP || r10 || le) {
71
2.61M
                pixel = av_le2ne32(*src++);
72
3.20M
            } else {
73
3.20M
                pixel = av_be2ne32(*src++);
74
3.20M
            }
75
5.81M
            if (avctx->codec_id == AV_CODEC_ID_R210) {
76
849k
                b =  pixel & 0x3ff;
77
849k
                g = (pixel >> 10) & 0x3ff;
78
849k
                r = (pixel >> 20) & 0x3ff;
79
4.96M
            } else if (r10) {
80
1.86k
                r =  pixel & 0x3ff;
81
1.86k
                g = (pixel >> 10) & 0x3ff;
82
1.86k
                b = (pixel >> 20) & 0x3ff;
83
4.96M
            } else {
84
4.96M
                b = (pixel >>  2) & 0x3ff;
85
4.96M
                g = (pixel >> 12) & 0x3ff;
86
4.96M
                r = (pixel >> 22) & 0x3ff;
87
4.96M
            }
88
5.81M
            *dstr++ = r;
89
5.81M
            *dstg++ = g;
90
5.81M
            *dstb++ = b;
91
5.81M
        }
92
317k
        src += aligned_width - avctx->width;
93
317k
        g_line += pic->linesize[0];
94
317k
        b_line += pic->linesize[1];
95
317k
        r_line += pic->linesize[2];
96
317k
    }
97
98
129k
    *got_frame      = 1;
99
100
129k
    return avpkt->size;
101
186k
}
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