From: Paul B Mahol <onemda@gmail.com> To: FFmpeg development discussions and patches <ffmpeg-devel@ffmpeg.org> Subject: Re: [FFmpeg-devel] [PATCH] QOA decoding support Date: Sun, 24 Sep 2023 02:50:12 +0200 Message-ID: <CAPYw7P6xeiYuiCR_j_8Oh6AsEEipML1d9uY27bK96D4_PFPvKg@mail.gmail.com> (raw) In-Reply-To: <CAPYw7P7HjG6Ow+Uv4JJgyhyKiqSsTqBLOePwU9hOw5pF61peiw@mail.gmail.com> [-- Attachment #1: Type: text/plain, Size: 13 bytes --] New patches. [-- Attachment #2: 0002-avformat-add-QOA-demuxer.patch --] [-- Type: text/x-patch, Size: 4429 bytes --] From 63392e60c7106bfcc6dec5fe45a9dd38f4c29e10 Mon Sep 17 00:00:00 2001 From: Paul B Mahol <onemda@gmail.com> Date: Sat, 23 Sep 2023 16:38:35 +0200 Subject: [PATCH 2/2] avformat: add QOA demuxer Signed-off-by: Paul B Mahol <onemda@gmail.com> --- libavformat/Makefile | 1 + libavformat/allformats.c | 1 + libavformat/qoadec.c | 83 ++++++++++++++++++++++++++++++++++++++++ 3 files changed, 85 insertions(+) create mode 100644 libavformat/qoadec.c diff --git a/libavformat/Makefile b/libavformat/Makefile index 329055ccfd..2db83aff81 100644 --- a/libavformat/Makefile +++ b/libavformat/Makefile @@ -484,6 +484,7 @@ OBJS-$(CONFIG_PP_BNK_DEMUXER) += pp_bnk.o OBJS-$(CONFIG_PVA_DEMUXER) += pva.o OBJS-$(CONFIG_PVF_DEMUXER) += pvfdec.o pcm.o OBJS-$(CONFIG_QCP_DEMUXER) += qcp.o +OBJS-$(CONFIG_QOA_DEMUXER) += qoadec.o OBJS-$(CONFIG_R3D_DEMUXER) += r3d.o OBJS-$(CONFIG_RAWVIDEO_DEMUXER) += rawvideodec.o OBJS-$(CONFIG_RAWVIDEO_MUXER) += rawenc.o diff --git a/libavformat/allformats.c b/libavformat/allformats.c index d4b505a5a3..c8bb4e3866 100644 --- a/libavformat/allformats.c +++ b/libavformat/allformats.c @@ -383,6 +383,7 @@ extern const FFOutputFormat ff_psp_muxer; extern const AVInputFormat ff_pva_demuxer; extern const AVInputFormat ff_pvf_demuxer; extern const AVInputFormat ff_qcp_demuxer; +extern const AVInputFormat ff_qoa_demuxer; extern const AVInputFormat ff_r3d_demuxer; extern const AVInputFormat ff_rawvideo_demuxer; extern const FFOutputFormat ff_rawvideo_muxer; diff --git a/libavformat/qoadec.c b/libavformat/qoadec.c new file mode 100644 index 0000000000..228e08cee8 --- /dev/null +++ b/libavformat/qoadec.c @@ -0,0 +1,83 @@ +/* + * QOA demuxer + * + * 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 "avformat.h" +#include "avio_internal.h" +#include "internal.h" +#include "rawdec.h" +#include "libavutil/intreadwrite.h" + +static int qoa_probe(const AVProbeData *p) +{ + if ((p->buf_size < 16) || + (AV_RB32(p->buf) != MKBETAG('q','o','a','f')) || + (AV_RB32(p->buf + 4) == 0) || + (p->buf[8] == 0) || + (AV_RB24(p->buf + 9) == 0) || + (AV_RB16(p->buf + 12) == 0) || + (AV_RB16(p->buf + 14) == 0)) + return 0; + return AVPROBE_SCORE_MAX; +} + +static int qoa_read_header(AVFormatContext *s) +{ + AVIOContext *pb = s->pb; + AVStream *st; + + st = avformat_new_stream(s, NULL); + if (!st) + return AVERROR(ENOMEM); + + avio_skip(pb, 4); + st->codecpar->codec_type = AVMEDIA_TYPE_AUDIO; + st->codecpar->codec_id = AV_CODEC_ID_QOA; + ffstream(st)->need_parsing = AVSTREAM_PARSE_FULL_RAW; + st->duration = avio_rb32(pb); + st->start_time = 0; + + ffio_ensure_seekback(pb, 4); + st->codecpar->ch_layout.nb_channels = avio_r8(pb); + if (st->codecpar->ch_layout.nb_channels == 0) + return AVERROR_INVALIDDATA; + + st->codecpar->sample_rate = avio_rb24(pb); + if (st->codecpar->sample_rate == 0) + return AVERROR_INVALIDDATA; + + avio_seek(pb, -4, SEEK_CUR); + + avpriv_set_pts_info(st, 64, 1, st->codecpar->sample_rate); + + return 0; +} + +const AVInputFormat ff_qoa_demuxer = { + .name = "qoa", + .long_name = NULL_IF_CONFIG_SMALL("raw QOA"), + .read_probe = qoa_probe, + .read_header = qoa_read_header, + .read_packet = ff_raw_read_partial_packet, + .flags = AVFMT_GENERIC_INDEX, + .extensions = "qoa", + .raw_codec_id = AV_CODEC_ID_QOA, + .priv_data_size = sizeof(FFRawDemuxerContext), + .priv_class = &ff_raw_demuxer_class, +}; -- 2.42.0 [-- Attachment #3: 0001-avcodec-add-QOA-decoder-and-parser.patch --] [-- Type: text/x-patch, Size: 13159 bytes --] From a5789be297faa409c00eb62c30afd0965f4d70ea Mon Sep 17 00:00:00 2001 From: Paul B Mahol <onemda@gmail.com> Date: Sat, 23 Sep 2023 16:49:25 +0200 Subject: [PATCH 1/2] avcodec: add QOA decoder and parser Signed-off-by: Paul B Mahol <onemda@gmail.com> --- libavcodec/Makefile | 2 + libavcodec/allcodecs.c | 1 + libavcodec/codec_desc.c | 7 ++ libavcodec/codec_id.h | 1 + libavcodec/parsers.c | 1 + libavcodec/qoa_parser.c | 89 ++++++++++++++++++++ libavcodec/qoadec.c | 175 ++++++++++++++++++++++++++++++++++++++++ 7 files changed, 276 insertions(+) create mode 100644 libavcodec/qoa_parser.c create mode 100644 libavcodec/qoadec.c diff --git a/libavcodec/Makefile b/libavcodec/Makefile index a1eb83f92d..c70d27a5c8 100644 --- a/libavcodec/Makefile +++ b/libavcodec/Makefile @@ -623,6 +623,7 @@ OBJS-$(CONFIG_QCELP_DECODER) += qcelpdec.o \ OBJS-$(CONFIG_QDM2_DECODER) += qdm2.o OBJS-$(CONFIG_QDMC_DECODER) += qdmc.o OBJS-$(CONFIG_QDRAW_DECODER) += qdrw.o +OBJS-$(CONFIG_QOA_DECODER) += qoadec.o OBJS-$(CONFIG_QOI_DECODER) += qoidec.o OBJS-$(CONFIG_QOI_ENCODER) += qoienc.o OBJS-$(CONFIG_QPEG_DECODER) += qpeg.o @@ -1205,6 +1206,7 @@ OBJS-$(CONFIG_OPUS_PARSER) += opus_parser.o opus_parse.o \ vorbis_data.o OBJS-$(CONFIG_PNG_PARSER) += png_parser.o OBJS-$(CONFIG_PNM_PARSER) += pnm_parser.o pnm.o +OBJS-$(CONFIG_QOA_PARSER) += qoa_parser.o OBJS-$(CONFIG_QOI_PARSER) += qoi_parser.o OBJS-$(CONFIG_RV34_PARSER) += rv34_parser.o OBJS-$(CONFIG_SBC_PARSER) += sbc_parser.o diff --git a/libavcodec/allcodecs.c b/libavcodec/allcodecs.c index 4ea6515c6c..86d0972742 100644 --- a/libavcodec/allcodecs.c +++ b/libavcodec/allcodecs.c @@ -523,6 +523,7 @@ extern const FFCodec ff_paf_audio_decoder; extern const FFCodec ff_qcelp_decoder; extern const FFCodec ff_qdm2_decoder; extern const FFCodec ff_qdmc_decoder; +extern const FFCodec ff_qoa_decoder; extern const FFCodec ff_ra_144_encoder; extern const FFCodec ff_ra_144_decoder; extern const FFCodec ff_ra_288_decoder; diff --git a/libavcodec/codec_desc.c b/libavcodec/codec_desc.c index 3ddd2fd3ee..fe838b49cb 100644 --- a/libavcodec/codec_desc.c +++ b/libavcodec/codec_desc.c @@ -3434,6 +3434,13 @@ static const AVCodecDescriptor codec_descriptors[] = { .long_name = NULL_IF_CONFIG_SMALL("Sonarc (Speech Compression)"), .props = AV_CODEC_PROP_INTRA_ONLY | AV_CODEC_PROP_LOSSLESS, }, + { + .id = AV_CODEC_ID_QOA, + .type = AVMEDIA_TYPE_AUDIO, + .name = "qoa", + .long_name = NULL_IF_CONFIG_SMALL("QOA (Quite OK Audio)"), + .props = AV_CODEC_PROP_INTRA_ONLY | AV_CODEC_PROP_LOSSY, + }, /* subtitle codecs */ { diff --git a/libavcodec/codec_id.h b/libavcodec/codec_id.h index 73352f5461..c313536f7b 100644 --- a/libavcodec/codec_id.h +++ b/libavcodec/codec_id.h @@ -546,6 +546,7 @@ enum AVCodecID { AV_CODEC_ID_AC4, AV_CODEC_ID_OSQ, AV_CODEC_ID_SONARC, + AV_CODEC_ID_QOA, /* subtitle codecs */ AV_CODEC_ID_FIRST_SUBTITLE = 0x17000, ///< A dummy ID pointing at the start of subtitle codecs. diff --git a/libavcodec/parsers.c b/libavcodec/parsers.c index 5128009cd4..2ed96120f1 100644 --- a/libavcodec/parsers.c +++ b/libavcodec/parsers.c @@ -65,6 +65,7 @@ extern const AVCodecParser ff_mpegvideo_parser; extern const AVCodecParser ff_opus_parser; extern const AVCodecParser ff_png_parser; extern const AVCodecParser ff_pnm_parser; +extern const AVCodecParser ff_qoa_parser; extern const AVCodecParser ff_qoi_parser; extern const AVCodecParser ff_rv34_parser; extern const AVCodecParser ff_sbc_parser; diff --git a/libavcodec/qoa_parser.c b/libavcodec/qoa_parser.c new file mode 100644 index 0000000000..0e5537a326 --- /dev/null +++ b/libavcodec/qoa_parser.c @@ -0,0 +1,89 @@ +/* + * QOA parser + * + * 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 "parser.h" + +typedef struct QOAParseContext { + ParseContext pc; + int frame_size; + int remaining; + int duration; +} QOAParseContext; + +static int qoa_parse(AVCodecParserContext *s, AVCodecContext *avctx, + const uint8_t **poutbuf, int *poutbuf_size, + const uint8_t *buf, int buf_size) +{ + QOAParseContext *qoa = s->priv_data; + uint64_t state = qoa->pc.state64; + int next = END_NOT_FOUND, i = 0; + + *poutbuf_size = 0; + *poutbuf = NULL; + + if (s->flags & PARSER_FLAG_COMPLETE_FRAMES) { + next = buf_size; + } else { + if (!qoa->frame_size) { + for (; i < buf_size; i++) { + state = (state << 8) | buf[i]; + if (((state & 0xFFFF) > 0 && (state >> 56))) { + qoa->frame_size = state & 0xFFFF; + qoa->duration = (state >> 16) & 0xFFFF; + break; + } + } + } + + if (qoa->frame_size) { + if (!qoa->remaining) + qoa->remaining = qoa->frame_size; + if (qoa->remaining <= buf_size) { + next = qoa->remaining; + qoa->remaining = 0; + qoa->frame_size = 0; + state = 0; + } else { + qoa->remaining -= buf_size; + } + } + + qoa->pc.state64 = state; + if (ff_combine_frame(&qoa->pc, next, &buf, &buf_size) < 0) { + *poutbuf = NULL; + *poutbuf_size = 0; + return buf_size; + } + + s->duration = qoa->duration; + } + + *poutbuf = buf; + *poutbuf_size = buf_size; + + return next; +} + +const AVCodecParser ff_qoa_parser = { + .codec_ids = { AV_CODEC_ID_QOA }, + .priv_data_size = sizeof(QOAParseContext), + .parser_parse = qoa_parse, + .parser_close = ff_parse_close, +}; diff --git a/libavcodec/qoadec.c b/libavcodec/qoadec.c new file mode 100644 index 0000000000..9b2abae833 --- /dev/null +++ b/libavcodec/qoadec.c @@ -0,0 +1,175 @@ +/* + * QOA decoder + * + * 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 "avcodec.h" +#include "codec_internal.h" +#include "decode.h" +#include "get_bits.h" +#include "bytestream.h" +#include "mathops.h" + +#define QOA_SLICE_LEN 20 +#define QOA_LMS_LEN 4 + +typedef struct QOAChannel { + int history[QOA_LMS_LEN]; + int weights[QOA_LMS_LEN]; +} QOAChannel; + +typedef struct QOAContext { + QOAChannel *ch; +} QOAContext; + +static const int16_t qoa_dequant_tab[16][8] = { + { 1, -1, 3, -3, 5, -5, 7, -7}, + { 5, -5, 18, -18, 32, -32, 49, -49}, + { 16, -16, 53, -53, 95, -95, 147, -147}, + { 34, -34, 113, -113, 203, -203, 315, -315}, + { 63, -63, 210, -210, 378, -378, 588, -588}, + { 104, -104, 345, -345, 621, -621, 966, -966}, + { 158, -158, 528, -528, 950, -950, 1477, -1477}, + { 228, -228, 760, -760, 1368, -1368, 2128, -2128}, + { 316, -316, 1053, -1053, 1895, -1895, 2947, -2947}, + { 422, -422, 1405, -1405, 2529, -2529, 3934, -3934}, + { 548, -548, 1828, -1828, 3290, -3290, 5117, -5117}, + { 696, -696, 2320, -2320, 4176, -4176, 6496, -6496}, + { 868, -868, 2893, -2893, 5207, -5207, 8099, -8099}, + {1064, -1064, 3548, -3548, 6386, -6386, 9933, -9933}, + {1286, -1286, 4288, -4288, 7718, -7718, 12005, -12005}, + {1536, -1536, 5120, -5120, 9216, -9216, 14336, -14336}, +}; + +static av_cold int qoa_decode_init(AVCodecContext *avctx) +{ + QOAContext *s = avctx->priv_data; + + avctx->sample_fmt = AV_SAMPLE_FMT_S16; + + s->ch = av_calloc(avctx->ch_layout.nb_channels, sizeof(*s->ch)); + if (!s->ch) + return AVERROR(ENOMEM); + + return 0; +} + +static int qoa_lms_predict(QOAChannel *lms) +{ + int prediction = 0; + for (int i = 0; i < QOA_LMS_LEN; i++) + prediction += lms->weights[i] * lms->history[i]; + return prediction >> 13; +} + +static void qoa_lms_update(QOAChannel *lms, int sample, int residual) +{ + int delta = residual >> 4; + for (int i = 0; i < QOA_LMS_LEN; i++) + lms->weights[i] += lms->history[i] < 0 ? -delta : delta; + for (int i = 0; i < QOA_LMS_LEN-1; i++) + lms->history[i] = lms->history[i+1]; + lms->history[QOA_LMS_LEN-1] = sample; +} + +static int qoa_decode_frame(AVCodecContext *avctx, AVFrame *frame, + int *got_frame_ptr, AVPacket *avpkt) +{ + QOAContext *s = avctx->priv_data; + int ret, frame_size, nb_channels; + GetByteContext gb; + int16_t *samples; + + bytestream2_init(&gb, avpkt->data, avpkt->size); + + nb_channels = bytestream2_get_byte(&gb); + if (avctx->ch_layout.nb_channels != nb_channels) + return AVERROR_INVALIDDATA; + + avctx->sample_rate = bytestream2_get_be24(&gb); + frame->nb_samples = bytestream2_get_be16(&gb); + frame_size = bytestream2_get_be16(&gb); + if (frame_size > avpkt->size) + return AVERROR_INVALIDDATA; + + if (frame_size < 8 + QOA_LMS_LEN * 4 * nb_channels + + 8LL * frame->nb_samples * nb_channels / QOA_SLICE_LEN) + return AVERROR_INVALIDDATA; + + if ((ret = ff_get_buffer(avctx, frame, 0)) < 0) + return ret; + samples = (int16_t *)frame->data[0]; + + for (int ch = 0; ch < nb_channels; ch++) { + QOAChannel *qch = &s->ch[ch]; + + for (int n = 0; n < QOA_LMS_LEN; n++) + qch->history[n] = sign_extend(bytestream2_get_be16u(&gb), 16); + for (int n = 0; n < QOA_LMS_LEN; n++) + qch->weights[n] = sign_extend(bytestream2_get_be16u(&gb), 16); + } + + for (int sample_index = 0; sample_index < frame->nb_samples * nb_channels; + sample_index += QOA_SLICE_LEN) { + for (int ch = 0; ch < nb_channels; ch++) { + QOAChannel *lms = &s->ch[ch]; + uint64_t slice = bytestream2_get_be64u(&gb); + int scalefactor = (slice >> 60) & 0xf; + int slice_start = sample_index * nb_channels + ch; + int slice_end = av_clip(sample_index + QOA_SLICE_LEN, 0, frame->nb_samples) * nb_channels + ch; + + for (int si = slice_start; si < slice_end; si += nb_channels) { + int predicted = qoa_lms_predict(lms); + int quantized = (slice >> 57) & 0x7; + int dequantized = qoa_dequant_tab[scalefactor][quantized]; + int reconstructed = av_clip_int16(predicted + dequantized); + + samples[si] = reconstructed; + slice <<= 3; + + qoa_lms_update(lms, reconstructed, dequantized); + } + } + } + + *got_frame_ptr = 1; + + return avpkt->size; +} + +static av_cold int qoa_decode_end(AVCodecContext *avctx) +{ + QOAContext *s = avctx->priv_data; + av_freep(&s->ch); + return 0; +} + +const FFCodec ff_qoa_decoder = { + .p.name = "qoa", + CODEC_LONG_NAME("QOA (Quite OK Audio)"), + .p.type = AVMEDIA_TYPE_AUDIO, + .p.id = AV_CODEC_ID_QOA, + .priv_data_size = sizeof(QOAContext), + .init = qoa_decode_init, + FF_CODEC_DECODE_CB(qoa_decode_frame), + .close = qoa_decode_end, + .p.capabilities = AV_CODEC_CAP_CHANNEL_CONF | + AV_CODEC_CAP_DR1, + .p.sample_fmts = (const enum AVSampleFormat[]) { AV_SAMPLE_FMT_S16, + AV_SAMPLE_FMT_NONE }, +}; -- 2.42.0 [-- 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 prev parent reply other threads:[~2023-09-24 0:50 UTC|newest] Thread overview: 21+ messages / expand[flat|nested] mbox.gz Atom feed top 2023-09-23 21:39 Paul B Mahol 2023-09-24 0:05 ` Andreas Rheinhardt 2023-09-24 0:48 ` Paul B Mahol 2023-09-27 12:52 ` Andreas Rheinhardt 2023-09-27 12:52 ` Paul B Mahol 2023-09-27 12:57 ` Nicolas George 2023-09-27 13:00 ` Paul B Mahol 2023-09-27 18:37 ` Michael Niedermayer 2023-09-27 19:59 ` Paul B Mahol 2023-10-06 11:54 ` Paul B Mahol 2023-09-27 13:02 ` Paul B Mahol 2023-09-27 12:58 ` Nicolas George 2023-09-27 13:05 ` Andreas Rheinhardt 2023-09-27 13:09 ` Paul B Mahol 2023-09-24 0:50 ` Paul B Mahol [this message] 2023-09-27 12:44 ` Paul B Mahol 2023-09-27 12:54 ` Tomas Härdin 2023-09-27 12:56 ` Paul B Mahol 2023-09-27 12:57 ` Tomas Härdin 2023-09-27 12:59 ` Paul B Mahol 2023-09-27 13:05 ` Tomas Härdin
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=CAPYw7P6xeiYuiCR_j_8Oh6AsEEipML1d9uY27bK96D4_PFPvKg@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