/src/ffmpeg/libavcodec/dfpwmdec.c
Line | Count | Source |
1 | | /* |
2 | | * DFPWM decoder |
3 | | * Copyright (c) 2022 Jack Bruienne |
4 | | * Copyright (c) 2012, 2016 Ben "GreaseMonkey" Russell |
5 | | * |
6 | | * This file is part of FFmpeg. |
7 | | * |
8 | | * FFmpeg is free software; you can redistribute it and/or |
9 | | * modify it under the terms of the GNU Lesser General Public |
10 | | * License as published by the Free Software Foundation; either |
11 | | * version 2.1 of the License, or (at your option) any later version. |
12 | | * |
13 | | * FFmpeg is distributed in the hope that it will be useful, |
14 | | * but WITHOUT ANY WARRANTY; without even the implied warranty of |
15 | | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU |
16 | | * Lesser General Public License for more details. |
17 | | * |
18 | | * You should have received a copy of the GNU Lesser General Public |
19 | | * License along with FFmpeg; if not, write to the Free Software |
20 | | * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA |
21 | | */ |
22 | | |
23 | | /** |
24 | | * @file |
25 | | * DFPWM1a decoder |
26 | | */ |
27 | | |
28 | | #include "libavutil/internal.h" |
29 | | #include "avcodec.h" |
30 | | #include "codec_id.h" |
31 | | #include "codec_internal.h" |
32 | | #include "decode.h" |
33 | | |
34 | | typedef struct { |
35 | | int fq, q, s, lt; |
36 | | } DFPWMState; |
37 | | |
38 | | // DFPWM codec from https://github.com/ChenThread/dfpwm/blob/master/1a/ |
39 | | // Licensed in the public domain |
40 | | |
41 | | static void au_decompress(DFPWMState *state, int fs, int len, |
42 | | uint8_t *outbuf, const uint8_t *inbuf) |
43 | 194k | { |
44 | 194k | unsigned d; |
45 | 13.4M | for (int i = 0; i < len; i++) { |
46 | | // get bits |
47 | 13.2M | d = *(inbuf++); |
48 | 119M | for (int j = 0; j < 8; j++) { |
49 | 106M | int nq, lq, st, ns, ov; |
50 | | // set target |
51 | 106M | int t = ((d&1) ? 127 : -128); |
52 | 106M | d >>= 1; |
53 | | |
54 | | // adjust charge |
55 | 106M | nq = state->q + ((state->s * (t-state->q) + 512)>>10); |
56 | 106M | if(nq == state->q && nq != t) |
57 | 6.87k | nq += (t == 127 ? 1 : -1); |
58 | 106M | lq = state->q; |
59 | 106M | state->q = nq; |
60 | | |
61 | | // adjust strength |
62 | 106M | st = (t != state->lt ? 0 : 1023); |
63 | 106M | ns = state->s; |
64 | 106M | if(ns != st) |
65 | 35.3M | ns += (st != 0 ? 1 : -1); |
66 | 106M | if(ns < 8) ns = 8; |
67 | 106M | state->s = ns; |
68 | | |
69 | | // FILTER: perform antijerk |
70 | 106M | ov = (t != state->lt ? (nq+lq+1)>>1 : nq); |
71 | | |
72 | | // FILTER: perform LPF |
73 | 106M | state->fq += ((fs*(ov-state->fq) + 0x80)>>8); |
74 | 106M | ov = state->fq; |
75 | | |
76 | | // output sample |
77 | 106M | *(outbuf++) = ov + 128; |
78 | | |
79 | 106M | state->lt = t; |
80 | 106M | } |
81 | 13.2M | } |
82 | 194k | } |
83 | | |
84 | | static av_cold int dfpwm_dec_init(struct AVCodecContext *ctx) |
85 | 547 | { |
86 | 547 | DFPWMState *state = ctx->priv_data; |
87 | | |
88 | 547 | state->fq = 0; |
89 | 547 | state->q = 0; |
90 | 547 | state->s = 0; |
91 | 547 | state->lt = -128; |
92 | | |
93 | 547 | ctx->sample_fmt = AV_SAMPLE_FMT_U8; |
94 | 547 | ctx->bits_per_raw_sample = 8; |
95 | | |
96 | 547 | return 0; |
97 | 547 | } |
98 | | |
99 | | static int dfpwm_dec_frame(struct AVCodecContext *ctx, AVFrame *frame, |
100 | | int *got_frame, struct AVPacket *packet) |
101 | 336k | { |
102 | 336k | DFPWMState *state = ctx->priv_data; |
103 | 336k | int ret; |
104 | | |
105 | 336k | if (packet->size * 8LL % ctx->ch_layout.nb_channels) |
106 | 141k | return AVERROR_PATCHWELCOME; |
107 | | |
108 | 194k | frame->nb_samples = packet->size * 8LL / ctx->ch_layout.nb_channels; |
109 | 194k | if (frame->nb_samples <= 0) { |
110 | 0 | av_log(ctx, AV_LOG_ERROR, "invalid number of samples in packet\n"); |
111 | 0 | return AVERROR_INVALIDDATA; |
112 | 0 | } |
113 | | |
114 | 194k | if ((ret = ff_get_buffer(ctx, frame, 0)) < 0) |
115 | 0 | return ret; |
116 | | |
117 | 194k | au_decompress(state, 140, packet->size, frame->data[0], packet->data); |
118 | | |
119 | 194k | *got_frame = 1; |
120 | 194k | return packet->size; |
121 | 194k | } |
122 | | |
123 | | const FFCodec ff_dfpwm_decoder = { |
124 | | .p.name = "dfpwm", |
125 | | CODEC_LONG_NAME("DFPWM1a audio"), |
126 | | .p.type = AVMEDIA_TYPE_AUDIO, |
127 | | .p.id = AV_CODEC_ID_DFPWM, |
128 | | .priv_data_size = sizeof(DFPWMState), |
129 | | .init = dfpwm_dec_init, |
130 | | FF_CODEC_DECODE_CB(dfpwm_dec_frame), |
131 | | .p.capabilities = AV_CODEC_CAP_DR1, |
132 | | }; |