Coverage Report

Created: 2026-05-23 07:06

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/ffmpeg/libavformat/pva.c
Line
Count
Source
1
/*
2
 * TechnoTrend PVA (.pva) demuxer
3
 * Copyright (c) 2007, 2008 Ivo van Poorten
4
 *
5
 * This file is part of FFmpeg.
6
 *
7
 * FFmpeg is free software; you can redistribute it and/or
8
 * modify it under the terms of the GNU Lesser General Public
9
 * License as published by the Free Software Foundation; either
10
 * version 2.1 of the License, or (at your option) any later version.
11
 *
12
 * FFmpeg is distributed in the hope that it will be useful,
13
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
14
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
15
 * Lesser General Public License for more details.
16
 *
17
 * You should have received a copy of the GNU Lesser General Public
18
 * License along with FFmpeg; if not, write to the Free Software
19
 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
20
 */
21
22
#include "avformat.h"
23
#include "avio_internal.h"
24
#include "demux.h"
25
#include "internal.h"
26
#include "mpeg.h"
27
28
1.03M
#define PVA_MAX_PAYLOAD_LENGTH  0x17f8
29
4.11M
#define PVA_VIDEO_PAYLOAD       0x01
30
445k
#define PVA_AUDIO_PAYLOAD       0x02
31
2.96M
#define PVA_MAGIC               (('A' << 8) + 'V')
32
33
typedef struct PVAContext {
34
    int continue_pes;
35
} PVAContext;
36
37
959k
static int pva_check(const uint8_t *p) {
38
959k
    int length = AV_RB16(p + 6);
39
959k
    if (AV_RB16(p) != PVA_MAGIC || !p[2] || p[2] > 2 || p[4] != 0x55 ||
40
2.70k
        (p[5] & 0xe0) || length > PVA_MAX_PAYLOAD_LENGTH)
41
957k
        return -1;
42
1.87k
    return length + 8;
43
959k
}
44
45
958k
static int pva_probe(const AVProbeData * pd) {
46
958k
    const unsigned char *buf = pd->buf;
47
958k
    int len = pva_check(buf);
48
49
958k
    if (len < 0)
50
956k
        return 0;
51
52
1.71k
    if (pd->buf_size >= len + 8 &&
53
1.26k
        pva_check(buf + len) >= 0)
54
164
        return AVPROBE_SCORE_EXTENSION;
55
56
1.54k
    return AVPROBE_SCORE_MAX / 4;
57
1.71k
}
58
59
4.31k
static int pva_read_header(AVFormatContext *s) {
60
4.31k
    AVStream *st;
61
62
4.31k
    if (!(st = avformat_new_stream(s, NULL)))
63
0
        return AVERROR(ENOMEM);
64
4.31k
    st->codecpar->codec_type = AVMEDIA_TYPE_VIDEO;
65
4.31k
    st->codecpar->codec_id   = AV_CODEC_ID_MPEG2VIDEO;
66
4.31k
    ffstream(st)->need_parsing = AVSTREAM_PARSE_FULL;
67
4.31k
    avpriv_set_pts_info(st, 32, 1, 90000);
68
4.31k
    av_add_index_entry(st, 0, 0, 0, 0, AVINDEX_KEYFRAME);
69
70
4.31k
    if (!(st = avformat_new_stream(s, NULL)))
71
0
        return AVERROR(ENOMEM);
72
4.31k
    st->codecpar->codec_type = AVMEDIA_TYPE_AUDIO;
73
4.31k
    st->codecpar->codec_id   = AV_CODEC_ID_MP2;
74
4.31k
    ffstream(st)->need_parsing = AVSTREAM_PARSE_FULL;
75
4.31k
    avpriv_set_pts_info(st, 33, 1, 90000);
76
4.31k
    av_add_index_entry(st, 0, 0, 0, 0, AVINDEX_KEYFRAME);
77
78
    /* the parameters will be extracted from the compressed bitstream */
79
4.31k
    return 0;
80
4.31k
}
81
82
1.04M
#define pva_log if (read_packet) av_log
83
84
static int read_part_of_packet(AVFormatContext *s, int64_t *pts,
85
1.04M
                               int *len, int *strid, int read_packet) {
86
1.04M
    AVIOContext *pb = s->pb;
87
1.04M
    PVAContext *pvactx = s->priv_data;
88
1.04M
    int syncword, streamid, reserved, flags, length, pts_flag;
89
1.04M
    int64_t pva_pts = AV_NOPTS_VALUE, startpos;
90
1.04M
    int ret;
91
92
1.04M
recover:
93
1.04M
    startpos = avio_tell(pb);
94
95
1.04M
    syncword = avio_rb16(pb);
96
1.04M
    streamid = avio_r8(pb);
97
1.04M
    avio_r8(pb);               /* counter not used */
98
1.04M
    reserved = avio_r8(pb);
99
1.04M
    flags    = avio_r8(pb);
100
1.04M
    length   = avio_rb16(pb);
101
102
1.04M
    pts_flag = flags & 0x10;
103
104
1.04M
    if (syncword != PVA_MAGIC) {
105
17.2k
        pva_log(s, AV_LOG_ERROR, "invalid syncword\n");
106
17.2k
        return AVERROR_INVALIDDATA;
107
17.2k
    }
108
1.02M
    if (streamid != PVA_VIDEO_PAYLOAD && streamid != PVA_AUDIO_PAYLOAD) {
109
752
        pva_log(s, AV_LOG_ERROR, "invalid streamid\n");
110
752
        return AVERROR_INVALIDDATA;
111
752
    }
112
1.02M
    if (reserved != 0x55) {
113
1.02M
        pva_log(s, AV_LOG_WARNING, "expected reserved byte to be 0x55\n");
114
1.02M
    }
115
1.02M
    if (length > PVA_MAX_PAYLOAD_LENGTH) {
116
359
        pva_log(s, AV_LOG_ERROR, "invalid payload length %u\n", length);
117
359
        return AVERROR_INVALIDDATA;
118
359
    }
119
120
1.02M
    if (streamid == PVA_VIDEO_PAYLOAD && pts_flag) {
121
613k
        pva_pts = avio_rb32(pb);
122
613k
        length -= 4;
123
613k
    } else if (streamid == PVA_AUDIO_PAYLOAD) {
124
        /* PVA Audio Packets either start with a signaled PES packet or
125
         * are a continuation of the previous PES packet. New PES packets
126
         * always start at the beginning of a PVA Packet, never somewhere in
127
         * the middle. */
128
30.6k
        if (!pvactx->continue_pes) {
129
9.37k
            int pes_signal, pes_header_data_length, pes_packet_length,
130
9.37k
                pes_flags;
131
9.37k
            unsigned char pes_header_data[256];
132
133
9.37k
            pes_signal             = avio_rb24(pb);
134
9.37k
            avio_r8(pb);
135
9.37k
            pes_packet_length      = avio_rb16(pb);
136
9.37k
            pes_flags              = avio_rb16(pb);
137
9.37k
            pes_header_data_length = avio_r8(pb);
138
139
9.37k
            if (avio_feof(pb)) {
140
95
                return AVERROR_EOF;
141
95
            }
142
143
9.28k
            if (pes_signal != 1 || pes_header_data_length == 0) {
144
3.79k
                pva_log(s, AV_LOG_WARNING, "expected non empty signaled PES packet, "
145
3.79k
                                          "trying to recover\n");
146
3.79k
                avio_skip(pb, length - 9);
147
3.79k
                if (!read_packet)
148
0
                    return AVERROR_INVALIDDATA;
149
3.79k
                goto recover;
150
3.79k
            }
151
152
5.49k
            ret = ffio_read_size(pb, pes_header_data, pes_header_data_length);
153
5.49k
            if (ret < 0)
154
38
                return ret;
155
5.45k
            length -= 9 + pes_header_data_length;
156
157
5.45k
            pes_packet_length -= 3 + pes_header_data_length;
158
159
5.45k
            pvactx->continue_pes = pes_packet_length;
160
161
5.45k
            if (pes_flags & 0x80 && (pes_header_data[0] & 0xf0) == 0x20) {
162
2.62k
                if (pes_header_data_length < 5) {
163
393
                    pva_log(s, AV_LOG_ERROR, "header too short\n");
164
393
                    avio_skip(pb, length);
165
393
                    return AVERROR_INVALIDDATA;
166
393
                }
167
2.23k
                pva_pts = ff_parse_pes_pts(pes_header_data);
168
2.23k
            }
169
5.45k
        }
170
171
26.3k
        pvactx->continue_pes -= length;
172
173
26.3k
        if (pvactx->continue_pes < 0) {
174
3.90k
            pva_log(s, AV_LOG_WARNING, "audio data corruption\n");
175
3.90k
            pvactx->continue_pes = 0;
176
3.90k
        }
177
26.3k
    }
178
179
1.02M
    if (pva_pts != AV_NOPTS_VALUE)
180
615k
        av_add_index_entry(s->streams[streamid-1], startpos, pva_pts, 0, 0, AVINDEX_KEYFRAME);
181
182
1.02M
    *pts   = pva_pts;
183
1.02M
    *len   = length;
184
1.02M
    *strid = streamid;
185
1.02M
    return 0;
186
1.02M
}
187
188
1.04M
static int pva_read_packet(AVFormatContext *s, AVPacket *pkt) {
189
1.04M
    AVIOContext *pb = s->pb;
190
1.04M
    int64_t pva_pts;
191
1.04M
    int ret, length, streamid;
192
193
1.04M
    if (read_part_of_packet(s, &pva_pts, &length, &streamid, 1) < 0 ||
194
1.02M
       (ret = av_get_packet(pb, pkt, length)) <= 0)
195
24.8k
        return AVERROR_INVALIDDATA;
196
197
1.01M
    pkt->stream_index = streamid - 1;
198
1.01M
    pkt->pts = pva_pts;
199
200
1.01M
    return ret;
201
1.04M
}
202
203
static int64_t pva_read_timestamp(struct AVFormatContext *s, int stream_index,
204
0
                                          int64_t *pos, int64_t pos_limit) {
205
0
    AVIOContext *pb = s->pb;
206
0
    PVAContext *pvactx = s->priv_data;
207
0
    int length, streamid;
208
0
    int64_t res = AV_NOPTS_VALUE;
209
210
0
    pos_limit = FFMIN(*pos+PVA_MAX_PAYLOAD_LENGTH*8, (uint64_t)*pos+pos_limit);
211
212
0
    while (*pos < pos_limit) {
213
0
        res = AV_NOPTS_VALUE;
214
0
        avio_seek(pb, *pos, SEEK_SET);
215
216
0
        pvactx->continue_pes = 0;
217
0
        if (read_part_of_packet(s, &res, &length, &streamid, 0)) {
218
0
            (*pos)++;
219
0
            continue;
220
0
        }
221
0
        if (streamid - 1 != stream_index || res == AV_NOPTS_VALUE) {
222
0
            *pos = avio_tell(pb) + length;
223
0
            continue;
224
0
        }
225
0
        break;
226
0
    }
227
228
0
    pvactx->continue_pes = 0;
229
0
    return res;
230
0
}
231
232
const FFInputFormat ff_pva_demuxer = {
233
    .p.name         = "pva",
234
    .p.long_name    = NULL_IF_CONFIG_SMALL("TechnoTrend PVA"),
235
    .priv_data_size = sizeof(PVAContext),
236
    .read_probe     = pva_probe,
237
    .read_header    = pva_read_header,
238
    .read_packet    = pva_read_packet,
239
    .read_timestamp = pva_read_timestamp,
240
};