Coverage Report

Created: 2026-01-16 07:48

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/ffmpeg/libavformat/ilbc.c
Line
Count
Source
1
/*
2
 * iLBC storage file format
3
 * Copyright (c) 2012 Martin Storsjo
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 "config_components.h"
23
24
#include "avformat.h"
25
#include "avio_internal.h"
26
#include "demux.h"
27
#include "internal.h"
28
#include "mux.h"
29
#include "rawenc.h"
30
31
static const char mode20_header[] = "#!iLBC20\n";
32
static const char mode30_header[] = "#!iLBC30\n";
33
34
static int ilbc_write_header(AVFormatContext *s)
35
0
{
36
0
    AVIOContext *pb = s->pb;
37
0
    AVCodecParameters *par = s->streams[0]->codecpar;
38
0
39
0
    if (par->block_align == 50) {
40
0
        avio_write(pb, mode30_header, sizeof(mode30_header) - 1);
41
0
    } else if (par->block_align == 38) {
42
0
        avio_write(pb, mode20_header, sizeof(mode20_header) - 1);
43
0
    } else {
44
0
        av_log(s, AV_LOG_ERROR, "Unsupported mode\n");
45
0
        return AVERROR(EINVAL);
46
0
    }
47
0
    return 0;
48
0
}
49
50
static int ilbc_probe(const AVProbeData *p)
51
958k
{
52
    // Only check for "#!iLBC" which matches both formats
53
958k
    if (!memcmp(p->buf, mode20_header, 6))
54
314
        return AVPROBE_SCORE_MAX;
55
958k
    else
56
958k
        return 0;
57
958k
}
58
59
static int ilbc_read_header(AVFormatContext *s)
60
924
{
61
924
    AVIOContext *pb = s->pb;
62
924
    AVStream *st;
63
924
    uint8_t header[9];
64
924
    int ret;
65
66
924
    ret = ffio_read_size(pb, header, 9);
67
924
    if (ret < 0)
68
125
        return ret;
69
70
799
    st = avformat_new_stream(s, NULL);
71
799
    if (!st)
72
0
        return AVERROR(ENOMEM);
73
799
    st->codecpar->codec_id = AV_CODEC_ID_ILBC;
74
799
    st->codecpar->sample_rate = 8000;
75
799
    st->codecpar->ch_layout = (AVChannelLayout)AV_CHANNEL_LAYOUT_MONO;
76
799
    st->codecpar->codec_type = AVMEDIA_TYPE_AUDIO;
77
799
    st->start_time = 0;
78
799
    avpriv_set_pts_info(st, 64, 1, st->codecpar->sample_rate);
79
799
    if (!memcmp(header, mode20_header, sizeof(mode20_header) - 1)) {
80
556
        st->codecpar->block_align = 38;
81
556
        st->codecpar->bit_rate = 15200;
82
556
    } else if (!memcmp(header, mode30_header, sizeof(mode30_header) - 1)) {
83
104
        st->codecpar->block_align = 50;
84
104
        st->codecpar->bit_rate = 13333;
85
139
    } else {
86
139
        av_log(s, AV_LOG_ERROR, "Unrecognized iLBC file header\n");
87
139
        return AVERROR_INVALIDDATA;
88
139
    }
89
90
660
    return 0;
91
799
}
92
93
static int ilbc_read_packet(AVFormatContext *s,
94
                          AVPacket *pkt)
95
180k
{
96
180k
    AVCodecParameters *par = s->streams[0]->codecpar;
97
180k
    int ret;
98
99
180k
    if ((ret = av_get_packet(s->pb, pkt, par->block_align)) != par->block_align)
100
856
        return ret < 0 ? ret : AVERROR_INVALIDDATA;
101
102
179k
    pkt->stream_index = 0;
103
179k
    pkt->duration = par->block_align == 38 ? 160 : 240;
104
105
179k
    return 0;
106
180k
}
107
108
const FFInputFormat ff_ilbc_demuxer = {
109
    .p.name       = "ilbc",
110
    .p.long_name  = NULL_IF_CONFIG_SMALL("iLBC storage"),
111
    .p.flags      = AVFMT_GENERIC_INDEX,
112
    .read_probe   = ilbc_probe,
113
    .read_header  = ilbc_read_header,
114
    .read_packet  = ilbc_read_packet,
115
};
116
117
#if CONFIG_ILBC_MUXER
118
const FFOutputFormat ff_ilbc_muxer = {
119
    .p.name         = "ilbc",
120
    .p.long_name    = NULL_IF_CONFIG_SMALL("iLBC storage"),
121
    .p.mime_type    = "audio/iLBC",
122
    .p.extensions   = "lbc",
123
    .p.video_codec    = AV_CODEC_ID_NONE,
124
    .p.audio_codec  = AV_CODEC_ID_ILBC,
125
    .p.subtitle_codec = AV_CODEC_ID_NONE,
126
    .p.flags        = AVFMT_NOTIMESTAMPS,
127
    .flags_internal   = FF_OFMT_FLAG_MAX_ONE_OF_EACH |
128
                        FF_OFMT_FLAG_ONLY_DEFAULT_CODECS,
129
    .write_header = ilbc_write_header,
130
    .write_packet = ff_raw_write_packet,
131
};
132
#endif