| Line | Branch | Exec | Source |
|---|---|---|---|
| 1 | /* | ||
| 2 | * Copyright (c) 2020 Marton Balint | ||
| 3 | * | ||
| 4 | * This file is part of FFmpeg. | ||
| 5 | * | ||
| 6 | * FFmpeg is free software; you can redistribute it and/or | ||
| 7 | * modify it under the terms of the GNU Lesser General Public | ||
| 8 | * License as published by the Free Software Foundation; either | ||
| 9 | * version 2.1 of the License, or (at your option) any later version. | ||
| 10 | * | ||
| 11 | * FFmpeg is distributed in the hope that it will be useful, | ||
| 12 | * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||
| 13 | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU | ||
| 14 | * Lesser General Public License for more details. | ||
| 15 | * | ||
| 16 | * You should have received a copy of the GNU Lesser General Public | ||
| 17 | * License along with FFmpeg; if not, write to the Free Software | ||
| 18 | * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA | ||
| 19 | */ | ||
| 20 | |||
| 21 | #include "bsf.h" | ||
| 22 | #include "bsf_internal.h" | ||
| 23 | #include "libavutil/avassert.h" | ||
| 24 | #include "libavutil/opt.h" | ||
| 25 | |||
| 26 | typedef struct PCMContext { | ||
| 27 | const AVClass *class; | ||
| 28 | |||
| 29 | int nb_out_samples; | ||
| 30 | int pad; | ||
| 31 | AVRational frame_rate; | ||
| 32 | |||
| 33 | AVPacket *in_pkt; | ||
| 34 | AVPacket *out_pkt; | ||
| 35 | int sample_size; | ||
| 36 | int64_t n; | ||
| 37 | } PCMContext; | ||
| 38 | |||
| 39 | 17 | static int init(AVBSFContext *ctx) | |
| 40 | { | ||
| 41 | 17 | PCMContext *s = ctx->priv_data; | |
| 42 | 17 | AVRational sr = av_make_q(ctx->par_in->sample_rate, 1); | |
| 43 | int64_t min_samples; | ||
| 44 | |||
| 45 |
2/4✓ Branch 0 taken 17 times.
✗ Branch 1 not taken.
✗ Branch 2 not taken.
✓ Branch 3 taken 17 times.
|
17 | if (ctx->par_in->ch_layout.nb_channels <= 0 || ctx->par_in->sample_rate <= 0) |
| 46 | ✗ | return AVERROR(EINVAL); | |
| 47 | |||
| 48 | 17 | ctx->time_base_out = av_inv_q(sr); | |
| 49 | 34 | s->sample_size = ctx->par_in->ch_layout.nb_channels * | |
| 50 | 17 | av_get_bits_per_sample(ctx->par_in->codec_id) / 8; | |
| 51 | |||
| 52 |
2/2✓ Branch 0 taken 13 times.
✓ Branch 1 taken 4 times.
|
17 | if (s->frame_rate.num) { |
| 53 | 13 | min_samples = av_rescale_q_rnd(1, sr, s->frame_rate, AV_ROUND_DOWN); | |
| 54 | } else { | ||
| 55 | 4 | min_samples = s->nb_out_samples; | |
| 56 | } | ||
| 57 |
2/4✓ Branch 0 taken 17 times.
✗ Branch 1 not taken.
✗ Branch 2 not taken.
✓ Branch 3 taken 17 times.
|
17 | if (min_samples <= 0 || min_samples > INT_MAX / s->sample_size - 1) |
| 58 | ✗ | return AVERROR(EINVAL); | |
| 59 | |||
| 60 | 17 | s->in_pkt = av_packet_alloc(); | |
| 61 | 17 | s->out_pkt = av_packet_alloc(); | |
| 62 |
2/4✓ Branch 0 taken 17 times.
✗ Branch 1 not taken.
✗ Branch 2 not taken.
✓ Branch 3 taken 17 times.
|
17 | if (!s->in_pkt || !s->out_pkt) |
| 63 | ✗ | return AVERROR(ENOMEM); | |
| 64 | |||
| 65 | 17 | return 0; | |
| 66 | } | ||
| 67 | |||
| 68 | 17 | static void uninit(AVBSFContext *ctx) | |
| 69 | { | ||
| 70 | 17 | PCMContext *s = ctx->priv_data; | |
| 71 | 17 | av_packet_free(&s->in_pkt); | |
| 72 | 17 | av_packet_free(&s->out_pkt); | |
| 73 | 17 | } | |
| 74 | |||
| 75 | ✗ | static void flush(AVBSFContext *ctx) | |
| 76 | { | ||
| 77 | ✗ | PCMContext *s = ctx->priv_data; | |
| 78 | ✗ | av_packet_unref(s->in_pkt); | |
| 79 | ✗ | av_packet_unref(s->out_pkt); | |
| 80 | ✗ | s->n = 0; | |
| 81 | ✗ | } | |
| 82 | |||
| 83 | 342 | static int send_packet(PCMContext *s, int nb_samples, AVPacket *pkt) | |
| 84 | { | ||
| 85 | 342 | pkt->duration = nb_samples; | |
| 86 | 342 | s->n++; | |
| 87 | 342 | return 0; | |
| 88 | } | ||
| 89 | |||
| 90 | 492 | static void drain_packet(AVPacket *pkt, int drain_data, int drain_samples) | |
| 91 | { | ||
| 92 | 492 | pkt->size -= drain_data; | |
| 93 | 492 | pkt->data += drain_data; | |
| 94 |
1/2✓ Branch 0 taken 492 times.
✗ Branch 1 not taken.
|
492 | if (pkt->dts != AV_NOPTS_VALUE) |
| 95 | 492 | pkt->dts += drain_samples; | |
| 96 |
1/2✓ Branch 0 taken 492 times.
✗ Branch 1 not taken.
|
492 | if (pkt->pts != AV_NOPTS_VALUE) |
| 97 | 492 | pkt->pts += drain_samples; | |
| 98 | 492 | } | |
| 99 | |||
| 100 | 566 | static int get_next_nb_samples(AVBSFContext *ctx) | |
| 101 | { | ||
| 102 | 566 | PCMContext *s = ctx->priv_data; | |
| 103 |
2/2✓ Branch 0 taken 420 times.
✓ Branch 1 taken 146 times.
|
566 | if (s->frame_rate.num) { |
| 104 | 420 | AVRational sr = av_make_q(ctx->par_in->sample_rate, 1); | |
| 105 | 420 | return av_rescale_q(s->n + 1, sr, s->frame_rate) - av_rescale_q(s->n, sr, s->frame_rate); | |
| 106 | } else { | ||
| 107 | 146 | return s->nb_out_samples; | |
| 108 | } | ||
| 109 | } | ||
| 110 | |||
| 111 | 5 | static void set_silence(AVCodecParameters *par, uint8_t *buf, size_t size) | |
| 112 | { | ||
| 113 | 5 | int c = 0; | |
| 114 |
1/4✗ Branch 0 not taken.
✗ Branch 1 not taken.
✗ Branch 2 not taken.
✓ Branch 3 taken 5 times.
|
5 | switch (par->codec_id) { |
| 115 | ✗ | case AV_CODEC_ID_PCM_ALAW: c = 0xd5; break; | |
| 116 | ✗ | case AV_CODEC_ID_PCM_MULAW: | |
| 117 | ✗ | case AV_CODEC_ID_PCM_VIDC: c = 0xff; break; | |
| 118 | ✗ | case AV_CODEC_ID_PCM_U8: c = 0x80; break; | |
| 119 | } | ||
| 120 | 5 | memset(buf, c, size); | |
| 121 | 5 | } | |
| 122 | |||
| 123 | 566 | static int rechunk_filter(AVBSFContext *ctx, AVPacket *pkt) | |
| 124 | { | ||
| 125 | 566 | PCMContext *s = ctx->priv_data; | |
| 126 | 566 | int nb_samples = get_next_nb_samples(ctx); | |
| 127 | 566 | int data_size = nb_samples * s->sample_size; | |
| 128 | int ret; | ||
| 129 | |||
| 130 | do { | ||
| 131 |
2/2✓ Branch 0 taken 517 times.
✓ Branch 1 taken 256 times.
|
773 | if (s->in_pkt->size) { |
| 132 |
4/4✓ Branch 0 taken 342 times.
✓ Branch 1 taken 175 times.
✓ Branch 2 taken 143 times.
✓ Branch 3 taken 199 times.
|
698 | if (s->out_pkt->size || s->in_pkt->size < data_size) { |
| 133 | 318 | int drain = FFMIN(s->in_pkt->size, data_size - s->out_pkt->size); | |
| 134 |
2/2✓ Branch 0 taken 143 times.
✓ Branch 1 taken 175 times.
|
318 | if (!s->out_pkt->size) { |
| 135 | 143 | ret = av_new_packet(s->out_pkt, data_size); | |
| 136 |
1/2✗ Branch 0 not taken.
✓ Branch 1 taken 143 times.
|
143 | if (ret < 0) |
| 137 | ✗ | return ret; | |
| 138 | 143 | ret = av_packet_copy_props(s->out_pkt, s->in_pkt); | |
| 139 |
1/2✗ Branch 0 not taken.
✓ Branch 1 taken 143 times.
|
143 | if (ret < 0) { |
| 140 | ✗ | av_packet_unref(s->out_pkt); | |
| 141 | ✗ | return ret; | |
| 142 | } | ||
| 143 | 143 | s->out_pkt->size = 0; | |
| 144 | } | ||
| 145 | 318 | memcpy(s->out_pkt->data + s->out_pkt->size, s->in_pkt->data, drain); | |
| 146 | 318 | s->out_pkt->size += drain; | |
| 147 | 318 | drain_packet(s->in_pkt, drain, drain / s->sample_size); | |
| 148 |
2/2✓ Branch 0 taken 182 times.
✓ Branch 1 taken 136 times.
|
318 | if (!s->in_pkt->size) |
| 149 | 182 | av_packet_unref(s->in_pkt); | |
| 150 |
2/2✓ Branch 0 taken 137 times.
✓ Branch 1 taken 181 times.
|
318 | if (s->out_pkt->size == data_size) { |
| 151 | 137 | av_packet_move_ref(pkt, s->out_pkt); | |
| 152 | 137 | return send_packet(s, nb_samples, pkt); | |
| 153 | } | ||
| 154 |
1/2✗ Branch 0 not taken.
✓ Branch 1 taken 181 times.
|
181 | av_assert0(!s->in_pkt->size); |
| 155 |
2/2✓ Branch 0 taken 174 times.
✓ Branch 1 taken 25 times.
|
199 | } else if (s->in_pkt->size > data_size) { |
| 156 | 174 | ret = av_packet_ref(pkt, s->in_pkt); | |
| 157 |
1/2✗ Branch 0 not taken.
✓ Branch 1 taken 174 times.
|
174 | if (ret < 0) |
| 158 | ✗ | return ret; | |
| 159 | 174 | pkt->size = data_size; | |
| 160 | 174 | drain_packet(s->in_pkt, data_size, nb_samples); | |
| 161 | 174 | return send_packet(s, nb_samples, pkt); | |
| 162 | } else { | ||
| 163 |
1/2✗ Branch 0 not taken.
✓ Branch 1 taken 25 times.
|
25 | av_assert0(s->in_pkt->size == data_size); |
| 164 | 25 | av_packet_move_ref(pkt, s->in_pkt); | |
| 165 | 25 | return send_packet(s, nb_samples, pkt); | |
| 166 | } | ||
| 167 | } else | ||
| 168 | 256 | av_packet_unref(s->in_pkt); | |
| 169 | |||
| 170 | 437 | ret = ff_bsf_get_packet_ref(ctx, s->in_pkt); | |
| 171 |
4/4✓ Branch 0 taken 23 times.
✓ Branch 1 taken 414 times.
✓ Branch 2 taken 6 times.
✓ Branch 3 taken 17 times.
|
437 | if (ret == AVERROR_EOF && s->out_pkt->size) { |
| 172 |
2/2✓ Branch 0 taken 5 times.
✓ Branch 1 taken 1 times.
|
6 | if (s->pad) { |
| 173 | 5 | set_silence(ctx->par_in, s->out_pkt->data + s->out_pkt->size, data_size - s->out_pkt->size); | |
| 174 | 5 | s->out_pkt->size = data_size; | |
| 175 | } else { | ||
| 176 | 1 | nb_samples = s->out_pkt->size / s->sample_size; | |
| 177 | } | ||
| 178 | 6 | av_packet_move_ref(pkt, s->out_pkt); | |
| 179 | 6 | return send_packet(s, nb_samples, pkt); | |
| 180 | } | ||
| 181 |
2/2✓ Branch 0 taken 207 times.
✓ Branch 1 taken 224 times.
|
431 | if (ret >= 0) |
| 182 | 207 | av_packet_rescale_ts(s->in_pkt, ctx->time_base_in, ctx->time_base_out); | |
| 183 |
2/2✓ Branch 0 taken 207 times.
✓ Branch 1 taken 224 times.
|
431 | } while (ret >= 0); |
| 184 | |||
| 185 | 224 | return ret; | |
| 186 | } | ||
| 187 | |||
| 188 | #define OFFSET(x) offsetof(PCMContext, x) | ||
| 189 | #define FLAGS (AV_OPT_FLAG_AUDIO_PARAM | AV_OPT_FLAG_BSF_PARAM) | ||
| 190 | static const AVOption options[] = { | ||
| 191 | { "nb_out_samples", "set the number of per-packet output samples", OFFSET(nb_out_samples), AV_OPT_TYPE_INT, {.i64=1024}, 1, INT_MAX, FLAGS }, | ||
| 192 | { "n", "set the number of per-packet output samples", OFFSET(nb_out_samples), AV_OPT_TYPE_INT, {.i64=1024}, 1, INT_MAX, FLAGS }, | ||
| 193 | { "pad", "pad last packet with zeros", OFFSET(pad), AV_OPT_TYPE_BOOL, {.i64=1} , 0, 1, FLAGS }, | ||
| 194 | { "p", "pad last packet with zeros", OFFSET(pad), AV_OPT_TYPE_BOOL, {.i64=1} , 0, 1, FLAGS }, | ||
| 195 | { "frame_rate", "set number of packets per second", OFFSET(frame_rate), AV_OPT_TYPE_RATIONAL, {.dbl=0}, 0, INT_MAX, FLAGS }, | ||
| 196 | { "r", "set number of packets per second", OFFSET(frame_rate), AV_OPT_TYPE_RATIONAL, {.dbl=0}, 0, INT_MAX, FLAGS }, | ||
| 197 | { NULL }, | ||
| 198 | }; | ||
| 199 | |||
| 200 | static const AVClass pcm_rechunk_class = { | ||
| 201 | .class_name = "pcm_rechunk_bsf", | ||
| 202 | .item_name = av_default_item_name, | ||
| 203 | .option = options, | ||
| 204 | .version = LIBAVUTIL_VERSION_INT, | ||
| 205 | }; | ||
| 206 | |||
| 207 | static const enum AVCodecID codec_ids[] = { | ||
| 208 | AV_CODEC_ID_PCM_ALAW, | ||
| 209 | AV_CODEC_ID_PCM_F16LE, | ||
| 210 | AV_CODEC_ID_PCM_F24LE, | ||
| 211 | AV_CODEC_ID_PCM_F32BE, | ||
| 212 | AV_CODEC_ID_PCM_F32LE, | ||
| 213 | AV_CODEC_ID_PCM_F64BE, | ||
| 214 | AV_CODEC_ID_PCM_F64LE, | ||
| 215 | AV_CODEC_ID_PCM_MULAW, | ||
| 216 | AV_CODEC_ID_PCM_S16BE, | ||
| 217 | AV_CODEC_ID_PCM_S16LE, | ||
| 218 | AV_CODEC_ID_PCM_S24BE, | ||
| 219 | AV_CODEC_ID_PCM_S24DAUD, | ||
| 220 | AV_CODEC_ID_PCM_S24LE, | ||
| 221 | AV_CODEC_ID_PCM_S32BE, | ||
| 222 | AV_CODEC_ID_PCM_S32LE, | ||
| 223 | AV_CODEC_ID_PCM_S64BE, | ||
| 224 | AV_CODEC_ID_PCM_S64LE, | ||
| 225 | AV_CODEC_ID_PCM_S8, | ||
| 226 | AV_CODEC_ID_PCM_SGA, | ||
| 227 | AV_CODEC_ID_PCM_U8, | ||
| 228 | AV_CODEC_ID_PCM_VIDC, | ||
| 229 | AV_CODEC_ID_NONE, | ||
| 230 | }; | ||
| 231 | |||
| 232 | const FFBitStreamFilter ff_pcm_rechunk_bsf = { | ||
| 233 | .p.name = "pcm_rechunk", | ||
| 234 | .p.codec_ids = codec_ids, | ||
| 235 | .p.priv_class = &pcm_rechunk_class, | ||
| 236 | .priv_data_size = sizeof(PCMContext), | ||
| 237 | .filter = rechunk_filter, | ||
| 238 | .init = init, | ||
| 239 | .flush = flush, | ||
| 240 | .close = uninit, | ||
| 241 | }; | ||
| 242 |