From: Paul B Mahol <onemda@gmail.com>
To: FFmpeg development discussions and patches <ffmpeg-devel@ffmpeg.org>
Subject: [FFmpeg-devel] [PATCH] OSQ lossless audio format support
Date: Thu, 24 Aug 2023 11:52:45 +0200
Message-ID: <CAPYw7P7KFPnK3aBba7N0UCsgegT+N3T28bjH19AYc82wh1vg9g@mail.gmail.com> (raw)
[-- Attachment #1: Type: text/plain, Size: 111 bytes --]
Patches attached.
Stereo decoding have some issues with some predictors so not yet bitexact.
Please comment.
[-- Attachment #2: 0002-avformat-add-OSQ-demuxer.patch --]
[-- Type: text/x-patch, Size: 5451 bytes --]
From a90b4fbd4178d4ef434e1255ed20dddebde0ddb8 Mon Sep 17 00:00:00 2001
From: Paul B Mahol <onemda@gmail.com>
Date: Tue, 27 Jun 2023 19:51:54 +0200
Subject: [PATCH 2/2] avformat: add OSQ demuxer
Signed-off-by: Paul B Mahol <onemda@gmail.com>
---
libavformat/Makefile | 1 +
libavformat/allformats.c | 1 +
libavformat/osq.c | 117 +++++++++++++++++++++++++++++++++++++++
3 files changed, 119 insertions(+)
create mode 100644 libavformat/osq.c
diff --git a/libavformat/Makefile b/libavformat/Makefile
index bd78c206b9..1c8c965a74 100644
--- a/libavformat/Makefile
+++ b/libavformat/Makefile
@@ -433,6 +433,7 @@ OBJS-$(CONFIG_OMA_DEMUXER) += omadec.o pcm.o oma.o
OBJS-$(CONFIG_OMA_MUXER) += omaenc.o rawenc.o oma.o id3v2enc.o
OBJS-$(CONFIG_OPUS_MUXER) += oggenc.o \
vorbiscomment.o
+OBJS-$(CONFIG_OSQ_DEMUXER) += osq.o rawdec.o
OBJS-$(CONFIG_PAF_DEMUXER) += paf.o
OBJS-$(CONFIG_PCM_ALAW_DEMUXER) += pcmdec.o pcm.o
OBJS-$(CONFIG_PCM_ALAW_MUXER) += pcmenc.o rawenc.o
diff --git a/libavformat/allformats.c b/libavformat/allformats.c
index 6324952bd2..f4210e4932 100644
--- a/libavformat/allformats.c
+++ b/libavformat/allformats.c
@@ -331,6 +331,7 @@ extern const FFOutputFormat ff_ogv_muxer;
extern const AVInputFormat ff_oma_demuxer;
extern const FFOutputFormat ff_oma_muxer;
extern const FFOutputFormat ff_opus_muxer;
+extern const AVInputFormat ff_osq_demuxer;
extern const AVInputFormat ff_paf_demuxer;
extern const AVInputFormat ff_pcm_alaw_demuxer;
extern const FFOutputFormat ff_pcm_alaw_muxer;
diff --git a/libavformat/osq.c b/libavformat/osq.c
new file mode 100644
index 0000000000..36ce25313f
--- /dev/null
+++ b/libavformat/osq.c
@@ -0,0 +1,117 @@
+/*
+ * OSQ demuxer
+ * Copyright (c) 2023 Paul B Mahol
+ *
+ * This file is part of FFmpeg.
+ *
+ * FFmpeg is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation; either
+ * version 2.1 of the License, or (at your option) any later version.
+ *
+ * FFmpeg is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
+ * Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public
+ * License along with FFmpeg; if not, write to the Free Software
+ * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
+ */
+
+#include "libavutil/intreadwrite.h"
+#include "avio_internal.h"
+#include "avformat.h"
+#include "demux.h"
+#include "internal.h"
+#include "rawdec.h"
+
+static int osq_probe(const AVProbeData *p)
+{
+ if (AV_RL32(p->buf) != MKTAG('O','S','Q',' '))
+ return 0;
+ if (AV_RL32(p->buf + 4) != 48)
+ return 0;
+ if (AV_RL16(p->buf + 8) != 1)
+ return 0;
+ if (!p->buf[10])
+ return 0;
+ if (!p->buf[11])
+ return 0;
+ if (AV_RL32(p->buf + 12) == 0)
+ return 0;
+ if (AV_RL16(p->buf + 16) == 0)
+ return 0;
+
+ return AVPROBE_SCORE_MAX;
+}
+
+static int osq_read_header(AVFormatContext *s)
+{
+ uint32_t t, size;
+ AVStream *st;
+ int ret;
+
+ t = avio_rl32(s->pb);
+ if (t != MKTAG('O','S','Q',' '))
+ return AVERROR_INVALIDDATA;
+
+ size = avio_rl32(s->pb);
+ if (size != 48)
+ return AVERROR_INVALIDDATA;
+
+ st = avformat_new_stream(s, NULL);
+ if (!st)
+ return AVERROR(ENOMEM);
+ if ((ret = ff_get_extradata(s, st->codecpar, s->pb, size)) < 0)
+ return ret;
+
+ t = avio_rl32(s->pb);
+ if (t != MKTAG('R','I','F','F'))
+ return AVERROR_INVALIDDATA;
+ avio_skip(s->pb, 8);
+
+ t = avio_rl32(s->pb);
+ if (t != MKTAG('f','m','t',' '))
+ return AVERROR_INVALIDDATA;
+ size = avio_rl32(s->pb);
+ avio_skip(s->pb, size);
+
+ t = avio_rl32(s->pb);
+ size = avio_rl32(s->pb);
+ while (t != MKTAG('d','a','t','a')) {
+ avio_skip(s->pb, size);
+
+ t = avio_rl32(s->pb);
+ size = avio_rl32(s->pb);
+ if (avio_feof(s->pb))
+ return AVERROR_INVALIDDATA;
+ }
+
+ st->codecpar->codec_type = AVMEDIA_TYPE_AUDIO;
+ st->codecpar->codec_id = AV_CODEC_ID_OSQ;
+ st->codecpar->sample_rate = AV_RL32(st->codecpar->extradata + 4);
+ if (st->codecpar->sample_rate == 0)
+ return AVERROR_INVALIDDATA;
+ st->codecpar->ch_layout.nb_channels = st->codecpar->extradata[3];
+ if (st->codecpar->ch_layout.nb_channels == 0)
+ return AVERROR_INVALIDDATA;
+ st->start_time = 0;
+ st->duration = AV_RL32(st->codecpar->extradata + 16);
+ avpriv_set_pts_info(st, 64, 1, st->codecpar->sample_rate);
+
+ return 0;
+}
+
+const AVInputFormat ff_osq_demuxer = {
+ .name = "osq",
+ .long_name = NULL_IF_CONFIG_SMALL("raw OSQ"),
+ .read_probe = osq_probe,
+ .read_header = osq_read_header,
+ .read_packet = ff_raw_read_partial_packet,
+ .extensions = "osq",
+ .flags = AVFMT_NOBINSEARCH | AVFMT_NOGENSEARCH | AVFMT_NO_BYTE_SEEK | AVFMT_NOTIMESTAMPS,
+ .raw_codec_id = AV_CODEC_ID_OSQ,
+ .priv_data_size = sizeof(FFRawDemuxerContext),
+ .priv_class = &ff_raw_demuxer_class,
+};
--
2.39.1
[-- Attachment #3: 0001-avcodec-add-OSQ-audio-decoder.patch --]
[-- Type: text/x-patch, Size: 16136 bytes --]
From 988ae6c88f0e5e9f26568825c8daa82a176bb2e1 Mon Sep 17 00:00:00 2001
From: Paul B Mahol <onemda@gmail.com>
Date: Tue, 27 Jun 2023 19:54:25 +0200
Subject: [PATCH 1/2] avcodec: add OSQ audio decoder
Signed-off-by: Paul B Mahol <onemda@gmail.com>
---
libavcodec/Makefile | 1 +
libavcodec/allcodecs.c | 1 +
libavcodec/codec_desc.c | 7 +
libavcodec/codec_id.h | 1 +
libavcodec/osq.c | 435 ++++++++++++++++++++++++++++++++++++++++
5 files changed, 445 insertions(+)
create mode 100644 libavcodec/osq.c
diff --git a/libavcodec/Makefile b/libavcodec/Makefile
index 3c16b51462..f3afda61d4 100644
--- a/libavcodec/Makefile
+++ b/libavcodec/Makefile
@@ -579,6 +579,7 @@ OBJS-$(CONFIG_OPUS_DECODER) += opusdec.o opusdec_celt.o opus_celt.o \
opusdsp.o opus_parse.o opus_rc.o
OBJS-$(CONFIG_OPUS_ENCODER) += opusenc.o opusenc_psy.o opus_celt.o \
opus_pvq.o opus_rc.o opustab.o
+OBJS-$(CONFIG_OSQ_DECODER) += osq.o
OBJS-$(CONFIG_PAF_AUDIO_DECODER) += pafaudio.o
OBJS-$(CONFIG_PAF_VIDEO_DECODER) += pafvideo.o
OBJS-$(CONFIG_PAM_DECODER) += pnmdec.o pnm.o
diff --git a/libavcodec/allcodecs.c b/libavcodec/allcodecs.c
index 8775d15a4f..6e95ca5636 100644
--- a/libavcodec/allcodecs.c
+++ b/libavcodec/allcodecs.c
@@ -516,6 +516,7 @@ extern const FFCodec ff_nellymoser_decoder;
extern const FFCodec ff_on2avc_decoder;
extern const FFCodec ff_opus_encoder;
extern const FFCodec ff_opus_decoder;
+extern const FFCodec ff_osq_decoder;
extern const FFCodec ff_paf_audio_decoder;
extern const FFCodec ff_qcelp_decoder;
extern const FFCodec ff_qdm2_decoder;
diff --git a/libavcodec/codec_desc.c b/libavcodec/codec_desc.c
index 4406dd8318..f556bb94d5 100644
--- a/libavcodec/codec_desc.c
+++ b/libavcodec/codec_desc.c
@@ -3413,6 +3413,13 @@ static const AVCodecDescriptor codec_descriptors[] = {
.long_name = NULL_IF_CONFIG_SMALL("AC-4"),
.props = AV_CODEC_PROP_LOSSY,
},
+ {
+ .id = AV_CODEC_ID_OSQ,
+ .type = AVMEDIA_TYPE_AUDIO,
+ .name = "osq",
+ .long_name = NULL_IF_CONFIG_SMALL("OSQ (Original Sound Quality)"),
+ .props = AV_CODEC_PROP_INTRA_ONLY | AV_CODEC_PROP_LOSSLESS,
+ },
/* subtitle codecs */
{
diff --git a/libavcodec/codec_id.h b/libavcodec/codec_id.h
index a5a0cb8525..29b410b8d3 100644
--- a/libavcodec/codec_id.h
+++ b/libavcodec/codec_id.h
@@ -543,6 +543,7 @@ enum AVCodecID {
AV_CODEC_ID_WAVARC,
AV_CODEC_ID_RKA,
AV_CODEC_ID_AC4,
+ AV_CODEC_ID_OSQ,
/* subtitle codecs */
AV_CODEC_ID_FIRST_SUBTITLE = 0x17000, ///< A dummy ID pointing at the start of subtitle codecs.
diff --git a/libavcodec/osq.c b/libavcodec/osq.c
new file mode 100644
index 0000000000..b6dc5c1bb4
--- /dev/null
+++ b/libavcodec/osq.c
@@ -0,0 +1,435 @@
+/*
+ * OSQ audio decoder
+ * Copyright (c) 2023 Paul B Mahol
+ *
+ * This file is part of FFmpeg.
+ *
+ * FFmpeg is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation; either
+ * version 2.1 of the License, or (at your option) any later version.
+ *
+ * FFmpeg is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
+ * Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public
+ * License along with FFmpeg; if not, write to the Free Software
+ * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
+ */
+
+#define ASSERT_LEVEL 5
+#include "libavutil/avassert.h"
+#include "libavutil/internal.h"
+#include "libavutil/intreadwrite.h"
+#include "avcodec.h"
+#include "codec_internal.h"
+#include "decode.h"
+#include "internal.h"
+#define BITSTREAM_READER_LE
+#include "get_bits.h"
+#include "unary.h"
+
+typedef struct OSQChannel {
+ int prediction;
+ int coding_mode;
+ int residue_parameter;
+ int residue_bits;
+} OSQChannel;
+
+typedef struct OSQContext {
+ GetBitContext gb;
+ OSQChannel ch[2];
+
+ uint8_t *bitstream;
+ int64_t max_framesize;
+ int bitstream_size;
+
+ int frame_samples;
+ int64_t nb_samples;
+
+ int32_t *decode_buffer[2];
+
+ AVPacket *pkt;
+ int pkt_offset;
+} OSQContext;
+
+static av_cold int osq_close(AVCodecContext *avctx)
+{
+ OSQContext *s = avctx->priv_data;
+
+ av_freep(&s->bitstream);
+ s->bitstream_size = 0;
+
+ for (int ch = 0; ch < avctx->ch_layout.nb_channels; ch++)
+ av_freep(&s->decode_buffer[ch]);
+
+ return 0;
+}
+
+static av_cold int osq_init(AVCodecContext *avctx)
+{
+ OSQContext *s = avctx->priv_data;
+
+ if (avctx->extradata_size < 48)
+ return AVERROR(EINVAL);
+
+ if (avctx->extradata[0] != 1) {
+ av_log(avctx, AV_LOG_ERROR, "Unsupported version.\n");
+ return AVERROR_INVALIDDATA;
+ }
+
+ avctx->sample_rate = AV_RL32(avctx->extradata + 4);
+ if (avctx->sample_rate < 1)
+ return AVERROR_INVALIDDATA;
+
+ avctx->ch_layout.nb_channels = avctx->extradata[3];
+ if (avctx->ch_layout.nb_channels < 1)
+ return AVERROR_INVALIDDATA;
+
+ switch (avctx->extradata[2]) {
+ case 8: avctx->sample_fmt = AV_SAMPLE_FMT_U8P; break;
+ case 16: avctx->sample_fmt = AV_SAMPLE_FMT_S16P; break;
+ case 20:
+ case 24:
+ case 28:
+ case 32: avctx->sample_fmt = AV_SAMPLE_FMT_S32P; break;
+ default: return AVERROR_INVALIDDATA;
+ }
+
+ s->nb_samples = AV_RL64(avctx->extradata + 16);
+ s->frame_samples = AV_RL16(avctx->extradata + 8);
+ s->max_framesize = (s->frame_samples * 16 + 1024) * avctx->ch_layout.nb_channels;
+
+ s->bitstream = av_calloc(s->max_framesize + AV_INPUT_BUFFER_PADDING_SIZE, sizeof(*s->bitstream));
+ if (!s->bitstream)
+ return AVERROR(ENOMEM);
+
+ for (int ch = 0; ch < avctx->ch_layout.nb_channels; ch++) {
+ s->decode_buffer[ch] = av_calloc(s->frame_samples + 4,
+ sizeof(*s->decode_buffer[ch]));
+ if (!s->decode_buffer[ch])
+ return AVERROR(ENOMEM);
+ }
+
+ s->pkt = avctx->internal->in_pkt;
+
+ return 0;
+}
+
+static uint32_t get_urice(GetBitContext *gb, int k)
+{
+ uint32_t z, x, b;
+
+ x = get_unary(gb, 1, 512);
+ b = get_bits_long(gb, k & 31);
+ z = b | x << (k & 31);
+
+ return z;
+}
+
+static int32_t get_srice(GetBitContext *gb, int x)
+{
+ int32_t y = get_urice(gb, x);
+ return get_bits1(gb) ? -y : y;
+}
+
+static int osq_channel_parameters(AVCodecContext *avctx, int ch)
+{
+ OSQContext *s = avctx->priv_data;
+ OSQChannel *cb = &s->ch[ch];
+ GetBitContext *gb = &s->gb;
+
+ cb->prediction = get_urice(gb, 5);
+ cb->coding_mode = get_urice(gb, 3);
+ if (cb->prediction >= 15)
+ return AVERROR_INVALIDDATA;
+ if (cb->coding_mode > 0 && cb->coding_mode < 3) {
+ cb->residue_parameter = get_urice(gb, 4);
+ if (!cb->residue_parameter || cb->residue_parameter >= 31)
+ return AVERROR_INVALIDDATA;
+ } else if (cb->coding_mode == 3) {
+ cb->residue_bits = get_urice(gb, 4);
+ if (!cb->residue_bits || cb->residue_bits >= 31)
+ return AVERROR_INVALIDDATA;
+ } else if (cb->coding_mode) {
+ return AVERROR_INVALIDDATA;
+ }
+
+ return 0;
+}
+
+static int osq_decode_block(AVCodecContext *avctx, AVFrame *frame)
+{
+ OSQContext *s = avctx->priv_data;
+ int ret, decorrelate, downsample;
+ GetBitContext *gb = &s->gb;
+
+ skip_bits1(gb);
+ decorrelate = get_bits1(gb);
+ downsample = get_bits1(gb);
+
+ for (int ch = 0; ch < avctx->ch_layout.nb_channels; ch++) {
+ if ((ret = osq_channel_parameters(avctx, ch)) < 0) {
+ av_log(avctx, AV_LOG_ERROR, "invalid channel parameters\n");
+ return ret;
+ }
+ }
+
+ for (int n = 0; n < frame->nb_samples; n++) {
+ for (int ch = 0; ch < avctx->ch_layout.nb_channels; ch++) {
+ OSQChannel *cb = &s->ch[ch];
+ int32_t *dst = s->decode_buffer[ch] + 4;
+
+ if (!cb->coding_mode) {
+ dst[n] = 0;
+ } else if (cb->coding_mode == 3) {
+ dst[n] = get_sbits_long(gb, cb->residue_bits);
+ } else {
+ dst[n] = get_srice(gb, cb->residue_parameter);
+ }
+
+ if (get_bits_left(gb) < 0) {
+ av_log(avctx, AV_LOG_ERROR, "overread!\n");
+ return AVERROR_INVALIDDATA;
+ }
+ }
+ }
+
+ align_get_bits(gb);
+
+#define A (n-1)
+#define B (n-2)
+#define C (n-3)
+#define D (n-4)
+#define P2 (dst[A] * 2LL - dst[B])
+#define P3 ((dst[A] * 1LL - dst[B]) * 3LL + dst[C])
+
+ for (int ch = 0; ch < avctx->ch_layout.nb_channels; ch++) {
+ int32_t pred = 0, *dst = s->decode_buffer[ch] + 4;
+ const int nb_samples = frame->nb_samples;
+ OSQChannel *cb = &s->ch[ch];
+
+ av_log(avctx, AV_LOG_DEBUG, "prediction: %d\n", cb->prediction);
+ switch (cb->prediction) {
+ case 0:
+ break;
+ case 1:
+ for (int n = 0; n < nb_samples; n++)
+ dst[n] += dst[A];
+ break;
+ case 2:
+ for (int n = 0; n < nb_samples; n++) {
+ int32_t p = pred;
+ pred = dst[n] / 2;
+ dst[n] += dst[A] + p;
+ }
+ break;
+ case 3:
+ for (int n = 0; n < nb_samples; n++)
+ dst[n] += P2;
+ break;
+ case 4:
+ for (int n = 0; n < nb_samples; n++) {
+ int32_t p = pred;
+ pred = dst[n] / 2;
+ dst[n] += P2 + p;
+ }
+ break;
+ case 5:
+ for (int n = 0; n < nb_samples; n++)
+ dst[n] += P3;
+ break;
+ case 6:
+ for (int n = 0; n < nb_samples; n++) {
+ int32_t p = pred;
+ pred = dst[n] / 2;
+ dst[n] += P3 + p;
+ }
+ break;
+ case 7:
+ for (int n = 0; n < nb_samples; n++) {
+ int32_t p = pred;
+ pred = dst[n] / 2;
+ dst[n] += (P2 + P3) / 2 + p;
+ }
+ break;
+ case 8:
+ for (int n = 0; n < nb_samples; n++)
+ dst[n] += (P2 * 1LL + P3) / 2;
+ break;
+ case 9:
+ for (int n = 0; n < nb_samples; n++) {
+ int32_t p = pred;
+ pred = dst[n] / 2;
+ dst[n] += (P2 * 2 + P3) / 3 + p;
+ }
+ break;
+ case 10:
+ for (int n = 0; n < nb_samples; n++) {
+ int32_t p = pred;
+ pred = dst[n] / 2;
+ dst[n] += (P2 + P3 * 2) / 3 + p;
+ }
+ break;
+ case 11:
+ for (int n = 0; n < nb_samples; n++)
+ dst[n] += (dst[A] * 1LL + dst[B]) / 2;
+ break;
+ case 12:
+ for (int n = 0; n < nb_samples; n++)
+ dst[n] += dst[B];
+ break;
+ case 13:
+ for (int n = 0; n < nb_samples; n++)
+ dst[n] += (dst[D] * 1LL + dst[B]) / 2;
+ break;
+ case 14:
+ for (int n = 0; n < nb_samples; n++) {
+ int32_t p = pred;
+ pred = dst[n] / 2;
+ dst[n] += (P2 + dst[A]) / 2 + p;
+ }
+ break;
+ default:
+ return AVERROR_INVALIDDATA;
+ }
+
+ memcpy(s->decode_buffer[ch], s->decode_buffer[ch] + frame->nb_samples, 4 * sizeof(*s->decode_buffer[0]));
+
+ if (downsample) {
+ int32_t *dst = s->decode_buffer[ch];
+
+ for (int n = 0; n < frame->nb_samples + 4; n++)
+ dst[n] = (dst[n] / 256) * 256;
+ }
+ }
+
+ if (decorrelate && avctx->ch_layout.nb_channels == 2) {
+ int32_t *l = s->decode_buffer[0] + 4;
+ int32_t *r = s->decode_buffer[1] + 4;
+
+ for (int n = 0; n < frame->nb_samples; n++) {
+ int64_t L = l[n];
+ int64_t R = r[n];
+
+ l[n] = L - R;
+ r[n] = L + R;
+ }
+ }
+
+ switch (avctx->sample_fmt) {
+ case AV_SAMPLE_FMT_U8P:
+ for (int ch = 0; ch < avctx->ch_layout.nb_channels; ch++) {
+ uint8_t *dst = (uint8_t *)frame->extended_data[ch];
+ int32_t *src = s->decode_buffer[ch] + 4;
+
+ for (int n = 0; n < frame->nb_samples; n++)
+ dst[n] = av_clip_uint8(src[n] + 0x80);
+ }
+ break;
+ case AV_SAMPLE_FMT_S16P:
+ for (int ch = 0; ch < avctx->ch_layout.nb_channels; ch++) {
+ int16_t *dst = (int16_t *)frame->extended_data[ch];
+ int32_t *src = s->decode_buffer[ch] + 4;
+
+ for (int n = 0; n < frame->nb_samples; n++)
+ dst[n] = av_clip_int16(src[n]);
+ }
+ break;
+ case AV_SAMPLE_FMT_S32P:
+ for (int ch = 0; ch < avctx->ch_layout.nb_channels; ch++) {
+ int32_t *dst = (int32_t *)frame->extended_data[ch];
+ int32_t *src = s->decode_buffer[ch] + 4;
+
+ for (int n = 0; n < frame->nb_samples; n++)
+ dst[n] = src[n];
+ }
+ break;
+ default:
+ return AVERROR_BUG;
+ }
+
+ return 0;
+}
+
+static int osq_receive_frame(AVCodecContext *avctx, AVFrame *frame)
+{
+ OSQContext *s = avctx->priv_data;
+ GetBitContext *gb = &s->gb;
+ int ret, n;
+
+ while (s->bitstream_size < s->max_framesize) {
+ int size;
+
+ if (!s->pkt->data) {
+ ret = ff_decode_get_packet(avctx, s->pkt);
+ if (ret == AVERROR_EOF && s->bitstream_size > 0)
+ break;
+ if (ret < 0)
+ return ret;
+ }
+
+ size = FFMIN(s->pkt->size - s->pkt_offset, s->max_framesize - s->bitstream_size);
+ memcpy(s->bitstream + s->bitstream_size, s->pkt->data + s->pkt_offset, size);
+ s->bitstream_size += size;
+ s->pkt_offset += size;
+
+ if (s->pkt_offset == s->pkt->size) {
+ av_packet_unref(s->pkt);
+ s->pkt_offset = 0;
+ }
+ }
+
+ frame->nb_samples = FFMIN(s->frame_samples, s->nb_samples);
+ if (frame->nb_samples <= 0)
+ return AVERROR_EOF;
+
+ if ((ret = ff_get_buffer(avctx, frame, 0)) < 0)
+ goto fail;
+
+ if ((ret = init_get_bits8(gb, s->bitstream, s->bitstream_size)) < 0)
+ goto fail;
+
+ if ((ret = osq_decode_block(avctx, frame)) < 0)
+ goto fail;
+
+ s->nb_samples -= frame->nb_samples;
+
+ n = get_bits_count(gb) / 8;
+ if (n > s->bitstream_size) {
+ ret = AVERROR_INVALIDDATA;
+ goto fail;
+ }
+
+ memmove(s->bitstream, &s->bitstream[n], s->bitstream_size - n);
+ s->bitstream_size -= n;
+
+ return 0;
+
+fail:
+ s->bitstream_size = 0;
+ s->pkt_offset = 0;
+ av_packet_unref(s->pkt);
+
+ return ret;
+}
+
+const FFCodec ff_osq_decoder = {
+ .p.name = "osq",
+ CODEC_LONG_NAME("OSQ (Original Sound Quality)"),
+ .p.type = AVMEDIA_TYPE_AUDIO,
+ .p.id = AV_CODEC_ID_OSQ,
+ .priv_data_size = sizeof(OSQContext),
+ .init = osq_init,
+ FF_CODEC_RECEIVE_FRAME_CB(osq_receive_frame),
+ .close = osq_close,
+ .p.capabilities = AV_CODEC_CAP_CHANNEL_CONF |
+ AV_CODEC_CAP_DR1,
+ .caps_internal = FF_CODEC_CAP_INIT_CLEANUP,
+ .p.sample_fmts = (const enum AVSampleFormat[]) { AV_SAMPLE_FMT_U8P,
+ AV_SAMPLE_FMT_S16P,
+ AV_SAMPLE_FMT_S32P,
+ AV_SAMPLE_FMT_NONE },
+};
--
2.39.1
[-- Attachment #4: Type: text/plain, Size: 251 bytes --]
_______________________________________________
ffmpeg-devel mailing list
ffmpeg-devel@ffmpeg.org
https://ffmpeg.org/mailman/listinfo/ffmpeg-devel
To unsubscribe, visit link above, or email
ffmpeg-devel-request@ffmpeg.org with subject "unsubscribe".
next reply other threads:[~2023-08-24 9:45 UTC|newest]
Thread overview: 17+ messages / expand[flat|nested] mbox.gz Atom feed top
2023-08-24 9:52 Paul B Mahol [this message]
2023-08-24 18:06 ` Michael Niedermayer
2023-08-24 19:04 ` Paul B Mahol
2023-08-24 19:54 ` James Almer
2023-08-24 20:09 ` Andreas Rheinhardt
2023-08-24 20:33 ` Paul B Mahol
2023-08-24 21:00 ` James Almer
2023-08-24 21:11 ` Paul B Mahol
2023-08-24 21:51 ` James Almer
2023-08-24 22:06 ` Paul B Mahol
2023-08-25 15:57 ` James Almer
2023-08-25 16:28 ` Paul B Mahol
2023-08-25 16:42 ` James Almer
2023-08-25 17:13 ` Paul B Mahol
2023-08-29 21:25 ` Paul B Mahol
2023-08-29 22:20 ` Andreas Rheinhardt
2023-08-31 17:51 ` Paul B Mahol
Reply instructions:
You may reply publicly to this message via plain-text email
using any one of the following methods:
* Save the following mbox file, import it into your mail client,
and reply-to-all from there: mbox
Avoid top-posting and favor interleaved quoting:
https://en.wikipedia.org/wiki/Posting_style#Interleaved_style
* Reply using the --to, --cc, and --in-reply-to
switches of git-send-email(1):
git send-email \
--in-reply-to=CAPYw7P7KFPnK3aBba7N0UCsgegT+N3T28bjH19AYc82wh1vg9g@mail.gmail.com \
--to=onemda@gmail.com \
--cc=ffmpeg-devel@ffmpeg.org \
/path/to/YOUR_REPLY
https://kernel.org/pub/software/scm/git/docs/git-send-email.html
* If your mail client supports setting the In-Reply-To header
via mailto: links, try the mailto: link
Git Inbox Mirror of the ffmpeg-devel mailing list - see https://ffmpeg.org/mailman/listinfo/ffmpeg-devel
This inbox may be cloned and mirrored by anyone:
git clone --mirror https://master.gitmailbox.com/ffmpegdev/0 ffmpegdev/git/0.git
# If you have public-inbox 1.1+ installed, you may
# initialize and index your mirror using the following commands:
public-inbox-init -V2 ffmpegdev ffmpegdev/ https://master.gitmailbox.com/ffmpegdev \
ffmpegdev@gitmailbox.com
public-inbox-index ffmpegdev
Example config snippet for mirrors.
AGPL code for this site: git clone https://public-inbox.org/public-inbox.git