540 lines
18 KiB
C
540 lines
18 KiB
C
|
////////////////////////////////////////////////////////////////////////////
|
||
|
// **** DSDPACK **** //
|
||
|
// Lossless DSD (Direct Stream Digital) Audio Compressor //
|
||
|
// Copyright (c) 2013 - 2016 David Bryant. //
|
||
|
// All Rights Reserved. //
|
||
|
// Distributed under the BSD Software License (see license.txt) //
|
||
|
////////////////////////////////////////////////////////////////////////////
|
||
|
|
||
|
// unpack_dsd.c
|
||
|
|
||
|
// This module actually handles the uncompression of the DSD audio data.
|
||
|
|
||
|
#include <stdlib.h>
|
||
|
#include <string.h>
|
||
|
#include <math.h>
|
||
|
|
||
|
#include "wavpack_local.h"
|
||
|
|
||
|
///////////////////////////// executable code ////////////////////////////////
|
||
|
|
||
|
// This function initialzes the main range-encoded data for DSD audio samples
|
||
|
|
||
|
static int init_dsd_block_fast (WavpackStream *wps, WavpackMetadata *wpmd);
|
||
|
static int init_dsd_block_high (WavpackStream *wps, WavpackMetadata *wpmd);
|
||
|
static int decode_fast (WavpackStream *wps, int32_t *output, int sample_count);
|
||
|
static int decode_high (WavpackStream *wps, int32_t *output, int sample_count);
|
||
|
|
||
|
int init_dsd_block (WavpackContext *wpc, WavpackMetadata *wpmd)
|
||
|
{
|
||
|
WavpackStream *wps = wpc->streams [wpc->current_stream];
|
||
|
|
||
|
if (wpmd->byte_length < 2)
|
||
|
return FALSE;
|
||
|
|
||
|
wps->dsd.byteptr = wpmd->data;
|
||
|
wps->dsd.endptr = wps->dsd.byteptr + wpmd->byte_length;
|
||
|
wpc->dsd_multiplier = 1 << *wps->dsd.byteptr++;
|
||
|
wps->dsd.mode = *wps->dsd.byteptr++;
|
||
|
|
||
|
if (!wps->dsd.mode) {
|
||
|
if (wps->dsd.endptr - wps->dsd.byteptr != wps->wphdr.block_samples * (wps->wphdr.flags & MONO_DATA ? 1 : 2)) {
|
||
|
return FALSE;
|
||
|
}
|
||
|
|
||
|
wps->dsd.ready = 1;
|
||
|
return TRUE;
|
||
|
}
|
||
|
|
||
|
if (wps->dsd.mode == 1)
|
||
|
return init_dsd_block_fast (wps, wpmd);
|
||
|
else if (wps->dsd.mode == 2)
|
||
|
return init_dsd_block_high (wps, wpmd);
|
||
|
else
|
||
|
return FALSE;
|
||
|
}
|
||
|
|
||
|
int32_t unpack_dsd_samples (WavpackContext *wpc, int32_t *buffer, uint32_t sample_count)
|
||
|
{
|
||
|
WavpackStream *wps = wpc->streams [wpc->current_stream];
|
||
|
uint32_t flags = wps->wphdr.flags, crc = wps->crc;
|
||
|
|
||
|
// don't attempt to decode past the end of the block, but watch out for overflow!
|
||
|
|
||
|
if (wps->sample_index + sample_count > GET_BLOCK_INDEX (wps->wphdr) + wps->wphdr.block_samples &&
|
||
|
GET_BLOCK_INDEX (wps->wphdr) + wps->wphdr.block_samples - wps->sample_index < sample_count)
|
||
|
sample_count = (uint32_t) (GET_BLOCK_INDEX (wps->wphdr) + wps->wphdr.block_samples - wps->sample_index);
|
||
|
|
||
|
if (GET_BLOCK_INDEX (wps->wphdr) > wps->sample_index || wps->wphdr.block_samples < sample_count)
|
||
|
wps->mute_error = TRUE;
|
||
|
|
||
|
if (!wps->mute_error) {
|
||
|
if (!wps->dsd.mode) {
|
||
|
int total_samples = sample_count * ((flags & MONO_DATA) ? 1 : 2);
|
||
|
int32_t *bptr = buffer;
|
||
|
|
||
|
if (wps->dsd.endptr - wps->dsd.byteptr < total_samples)
|
||
|
total_samples = (int)(wps->dsd.endptr - wps->dsd.byteptr);
|
||
|
|
||
|
while (total_samples--)
|
||
|
wps->crc += (wps->crc << 1) + (*bptr++ = *wps->dsd.byteptr++);
|
||
|
}
|
||
|
else if (wps->dsd.mode == 1) {
|
||
|
if (!decode_fast (wps, buffer, sample_count))
|
||
|
wps->mute_error = TRUE;
|
||
|
}
|
||
|
else if (!decode_high (wps, buffer, sample_count))
|
||
|
wps->mute_error = TRUE;
|
||
|
}
|
||
|
|
||
|
if (wps->mute_error) {
|
||
|
int samples_to_null;
|
||
|
if (wpc->reduced_channels == 1 || wpc->config.num_channels == 1 || (flags & MONO_FLAG))
|
||
|
samples_to_null = sample_count;
|
||
|
else
|
||
|
samples_to_null = sample_count * 2;
|
||
|
|
||
|
while (samples_to_null--)
|
||
|
*buffer++ = 0x55;
|
||
|
|
||
|
wps->sample_index += sample_count;
|
||
|
return sample_count;
|
||
|
}
|
||
|
|
||
|
if (flags & FALSE_STEREO) {
|
||
|
int32_t *dptr = buffer + sample_count * 2;
|
||
|
int32_t *sptr = buffer + sample_count;
|
||
|
int32_t c = sample_count;
|
||
|
|
||
|
while (c--) {
|
||
|
*--dptr = *--sptr;
|
||
|
*--dptr = *sptr;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
wps->sample_index += sample_count;
|
||
|
|
||
|
return sample_count;
|
||
|
}
|
||
|
|
||
|
/*------------------------------------------------------------------------------------------------------------------------*/
|
||
|
|
||
|
// #define DSD_BYTE_READY(low,high) (((low) >> 24) == ((high) >> 24))
|
||
|
// #define DSD_BYTE_READY(low,high) (!(((low) ^ (high)) >> 24))
|
||
|
#define DSD_BYTE_READY(low,high) (!(((low) ^ (high)) & 0xff000000))
|
||
|
#define MAX_HISTORY_BITS 5
|
||
|
|
||
|
static int init_dsd_block_fast (WavpackStream *wps, WavpackMetadata *wpmd)
|
||
|
{
|
||
|
unsigned char history_bits, max_probability;
|
||
|
int total_summed_probabilities = 0, i;
|
||
|
|
||
|
if (wps->dsd.byteptr == wps->dsd.endptr)
|
||
|
return FALSE;
|
||
|
|
||
|
history_bits = *wps->dsd.byteptr++;
|
||
|
|
||
|
if (wps->dsd.byteptr == wps->dsd.endptr || history_bits > MAX_HISTORY_BITS)
|
||
|
return FALSE;
|
||
|
|
||
|
wps->dsd.history_bins = 1 << history_bits;
|
||
|
|
||
|
wps->dsd.value_lookup = malloc (sizeof (*wps->dsd.value_lookup) * wps->dsd.history_bins);
|
||
|
memset (wps->dsd.value_lookup, 0, sizeof (*wps->dsd.value_lookup) * wps->dsd.history_bins);
|
||
|
wps->dsd.summed_probabilities = malloc (sizeof (*wps->dsd.summed_probabilities) * wps->dsd.history_bins);
|
||
|
wps->dsd.probabilities = malloc (sizeof (*wps->dsd.probabilities) * wps->dsd.history_bins);
|
||
|
|
||
|
max_probability = *wps->dsd.byteptr++;
|
||
|
|
||
|
if (max_probability < 0xff) {
|
||
|
unsigned char *outptr = (unsigned char *) wps->dsd.probabilities;
|
||
|
unsigned char *outend = outptr + sizeof (*wps->dsd.probabilities) * wps->dsd.history_bins;
|
||
|
|
||
|
while (outptr < outend && wps->dsd.byteptr < wps->dsd.endptr) {
|
||
|
int code = *wps->dsd.byteptr++;
|
||
|
|
||
|
if (code > max_probability) {
|
||
|
int zcount = code - max_probability;
|
||
|
|
||
|
while (outptr < outend && zcount--)
|
||
|
*outptr++ = 0;
|
||
|
}
|
||
|
else if (code)
|
||
|
*outptr++ = code;
|
||
|
else
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
if (outptr < outend || (wps->dsd.byteptr < wps->dsd.endptr && *wps->dsd.byteptr++))
|
||
|
return FALSE;
|
||
|
}
|
||
|
else if (wps->dsd.endptr - wps->dsd.byteptr > (int) sizeof (*wps->dsd.probabilities) * wps->dsd.history_bins) {
|
||
|
memcpy (wps->dsd.probabilities, wps->dsd.byteptr, sizeof (*wps->dsd.probabilities) * wps->dsd.history_bins);
|
||
|
wps->dsd.byteptr += sizeof (*wps->dsd.probabilities) * wps->dsd.history_bins;
|
||
|
}
|
||
|
else
|
||
|
return FALSE;
|
||
|
|
||
|
for (wps->dsd.p0 = 0; wps->dsd.p0 < wps->dsd.history_bins; ++wps->dsd.p0) {
|
||
|
int32_t sum_values;
|
||
|
unsigned char *vp;
|
||
|
|
||
|
for (sum_values = i = 0; i < 256; ++i)
|
||
|
wps->dsd.summed_probabilities [wps->dsd.p0] [i] = sum_values += wps->dsd.probabilities [wps->dsd.p0] [i];
|
||
|
|
||
|
if (sum_values) {
|
||
|
total_summed_probabilities += sum_values;
|
||
|
vp = wps->dsd.value_lookup [wps->dsd.p0] = malloc (sum_values);
|
||
|
|
||
|
for (i = 0; i < 256; i++) {
|
||
|
int c = wps->dsd.probabilities [wps->dsd.p0] [i];
|
||
|
|
||
|
while (c--)
|
||
|
*vp++ = i;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if (wps->dsd.endptr - wps->dsd.byteptr < 4 || total_summed_probabilities > wps->dsd.history_bins * 1280)
|
||
|
return FALSE;
|
||
|
|
||
|
for (i = 4; i--;)
|
||
|
wps->dsd.value = (wps->dsd.value << 8) | *wps->dsd.byteptr++;
|
||
|
|
||
|
wps->dsd.p0 = wps->dsd.p1 = 0;
|
||
|
wps->dsd.low = 0; wps->dsd.high = 0xffffffff;
|
||
|
wps->dsd.ready = 1;
|
||
|
|
||
|
return TRUE;
|
||
|
}
|
||
|
|
||
|
static int decode_fast (WavpackStream *wps, int32_t *output, int sample_count)
|
||
|
{
|
||
|
int total_samples = sample_count;
|
||
|
|
||
|
if (!(wps->wphdr.flags & MONO_DATA))
|
||
|
total_samples *= 2;
|
||
|
|
||
|
while (total_samples--) {
|
||
|
int mult, index, code, i;
|
||
|
|
||
|
if (!wps->dsd.summed_probabilities [wps->dsd.p0] [255])
|
||
|
return 0;
|
||
|
|
||
|
mult = (wps->dsd.high - wps->dsd.low) / wps->dsd.summed_probabilities [wps->dsd.p0] [255];
|
||
|
|
||
|
if (!mult) {
|
||
|
if (wps->dsd.endptr - wps->dsd.byteptr >= 4)
|
||
|
for (i = 4; i--;)
|
||
|
wps->dsd.value = (wps->dsd.value << 8) | *wps->dsd.byteptr++;
|
||
|
|
||
|
wps->dsd.low = 0;
|
||
|
wps->dsd.high = 0xffffffff;
|
||
|
mult = wps->dsd.high / wps->dsd.summed_probabilities [wps->dsd.p0] [255];
|
||
|
|
||
|
if (!mult)
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
index = (wps->dsd.value - wps->dsd.low) / mult;
|
||
|
|
||
|
if (index >= wps->dsd.summed_probabilities [wps->dsd.p0] [255])
|
||
|
return 0;
|
||
|
|
||
|
if ((*output++ = code = wps->dsd.value_lookup [wps->dsd.p0] [index]))
|
||
|
wps->dsd.low += wps->dsd.summed_probabilities [wps->dsd.p0] [code-1] * mult;
|
||
|
|
||
|
wps->dsd.high = wps->dsd.low + wps->dsd.probabilities [wps->dsd.p0] [code] * mult - 1;
|
||
|
wps->crc += (wps->crc << 1) + code;
|
||
|
|
||
|
if (wps->wphdr.flags & MONO_DATA)
|
||
|
wps->dsd.p0 = code & (wps->dsd.history_bins-1);
|
||
|
else {
|
||
|
wps->dsd.p0 = wps->dsd.p1;
|
||
|
wps->dsd.p1 = code & (wps->dsd.history_bins-1);
|
||
|
}
|
||
|
|
||
|
while (DSD_BYTE_READY (wps->dsd.high, wps->dsd.low) && wps->dsd.byteptr < wps->dsd.endptr) {
|
||
|
wps->dsd.value = (wps->dsd.value << 8) | *wps->dsd.byteptr++;
|
||
|
wps->dsd.high = (wps->dsd.high << 8) | 0xff;
|
||
|
wps->dsd.low <<= 8;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return sample_count;
|
||
|
}
|
||
|
|
||
|
/*------------------------------------------------------------------------------------------------------------------------*/
|
||
|
|
||
|
#define PTABLE_BITS 8
|
||
|
#define PTABLE_BINS (1<<PTABLE_BITS)
|
||
|
#define PTABLE_MASK (PTABLE_BINS-1)
|
||
|
|
||
|
#define UP 0x010000fe
|
||
|
#define DOWN 0x00010000
|
||
|
#define DECAY 8
|
||
|
|
||
|
#define PRECISION 24
|
||
|
#define VALUE_ONE (1 << PRECISION)
|
||
|
#define PRECISION_USE 12
|
||
|
|
||
|
#define RATE_S 20
|
||
|
|
||
|
static void init_ptable (int *table, int rate_i, int rate_s)
|
||
|
{
|
||
|
int value = 0x808000, rate = rate_i << 8, c, i;
|
||
|
|
||
|
for (c = (rate + 128) >> 8; c--;)
|
||
|
value += (DOWN - value) >> DECAY;
|
||
|
|
||
|
for (i = 0; i < PTABLE_BINS/2; ++i) {
|
||
|
table [i] = value;
|
||
|
table [PTABLE_BINS-1-i] = 0x100ffff - value;
|
||
|
|
||
|
if (value > 0x010000) {
|
||
|
rate += (rate * rate_s + 128) >> 8;
|
||
|
|
||
|
for (c = (rate + 64) >> 7; c--;)
|
||
|
value += (DOWN - value) >> DECAY;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static int init_dsd_block_high (WavpackStream *wps, WavpackMetadata *wpmd)
|
||
|
{
|
||
|
uint32_t flags = wps->wphdr.flags;
|
||
|
int channel, rate_i, rate_s, i;
|
||
|
|
||
|
if (wps->dsd.endptr - wps->dsd.byteptr < ((flags & MONO_DATA) ? 13 : 20))
|
||
|
return FALSE;
|
||
|
|
||
|
rate_i = *wps->dsd.byteptr++;
|
||
|
rate_s = *wps->dsd.byteptr++;
|
||
|
|
||
|
if (rate_s != RATE_S)
|
||
|
return FALSE;
|
||
|
|
||
|
wps->dsd.ptable = malloc (PTABLE_BINS * sizeof (*wps->dsd.ptable));
|
||
|
init_ptable (wps->dsd.ptable, rate_i, rate_s);
|
||
|
|
||
|
for (channel = 0; channel < ((flags & MONO_DATA) ? 1 : 2); ++channel) {
|
||
|
DSDfilters *sp = wps->dsd.filters + channel;
|
||
|
|
||
|
sp->filter1 = *wps->dsd.byteptr++ << 16;
|
||
|
sp->filter2 = *wps->dsd.byteptr++ << 16;
|
||
|
sp->filter3 = *wps->dsd.byteptr++ << 16;
|
||
|
sp->filter4 = *wps->dsd.byteptr++ << 16;
|
||
|
sp->filter5 = *wps->dsd.byteptr++ << 16;
|
||
|
sp->filter6 = 0;
|
||
|
sp->factor = *wps->dsd.byteptr++ & 0xff;
|
||
|
sp->factor |= (*wps->dsd.byteptr++ << 8) & 0xff00;
|
||
|
sp->factor = (sp->factor << 16) >> 16;
|
||
|
}
|
||
|
|
||
|
wps->dsd.high = 0xffffffff;
|
||
|
wps->dsd.low = 0x0;
|
||
|
|
||
|
for (i = 4; i--;)
|
||
|
wps->dsd.value = (wps->dsd.value << 8) | *wps->dsd.byteptr++;
|
||
|
|
||
|
wps->dsd.ready = 1;
|
||
|
|
||
|
return TRUE;
|
||
|
}
|
||
|
|
||
|
static int decode_high (WavpackStream *wps, int32_t *output, int sample_count)
|
||
|
{
|
||
|
int total_samples = sample_count, channel = 0;
|
||
|
|
||
|
if (!(wps->wphdr.flags & MONO_DATA))
|
||
|
total_samples *= 2;
|
||
|
|
||
|
while (total_samples--) {
|
||
|
DSDfilters *sp = wps->dsd.filters + channel;
|
||
|
int byte = 0, bitcount = 8;
|
||
|
|
||
|
while (bitcount--) {
|
||
|
int value = sp->filter1 - sp->filter5 + sp->filter6 * (sp->factor >> 2);
|
||
|
int index = (value >> (PRECISION - PRECISION_USE)) & PTABLE_MASK;
|
||
|
unsigned int range = wps->dsd.high - wps->dsd.low, split;
|
||
|
int *val = wps->dsd.ptable + index;
|
||
|
|
||
|
split = wps->dsd.low + ((range & 0xff000000) ? (range >> 8) * (*val >> 16) : ((range * (*val >> 16)) >> 8));
|
||
|
value += sp->filter6 << 3;
|
||
|
|
||
|
if (wps->dsd.value <= split) {
|
||
|
wps->dsd.high = split;
|
||
|
byte = (byte << 1) | 1;
|
||
|
*val += (UP - *val) >> DECAY;
|
||
|
sp->filter1 += (VALUE_ONE - sp->filter1) >> 6;
|
||
|
sp->filter2 += (VALUE_ONE - sp->filter2) >> 4;
|
||
|
|
||
|
if ((value ^ (value - (sp->filter6 << 4))) < 0)
|
||
|
sp->factor -= (value >> 31) | 1;
|
||
|
}
|
||
|
else {
|
||
|
wps->dsd.low = split + 1;
|
||
|
byte <<= 1;
|
||
|
*val += (DOWN - *val) >> DECAY;
|
||
|
sp->filter1 -= sp->filter1 >> 6;
|
||
|
sp->filter2 -= sp->filter2 >> 4;
|
||
|
|
||
|
if ((value ^ (value - (sp->filter6 << 4))) < 0)
|
||
|
sp->factor += (value >> 31) | 1;
|
||
|
}
|
||
|
|
||
|
while (DSD_BYTE_READY (wps->dsd.high, wps->dsd.low) && wps->dsd.byteptr < wps->dsd.endptr) {
|
||
|
wps->dsd.value = (wps->dsd.value << 8) | *wps->dsd.byteptr++;
|
||
|
wps->dsd.high = (wps->dsd.high << 8) | 0xff;
|
||
|
wps->dsd.low <<= 8;
|
||
|
}
|
||
|
|
||
|
sp->filter3 += (sp->filter2 - sp->filter3) >> 4;
|
||
|
sp->filter4 += (sp->filter3 - sp->filter4) >> 4;
|
||
|
sp->filter5 += value = (sp->filter4 - sp->filter5) >> 4;
|
||
|
sp->filter6 += (value - sp->filter6) >> 3;
|
||
|
}
|
||
|
|
||
|
wps->crc += (wps->crc << 1) + (*output++ = byte);
|
||
|
|
||
|
if (!(wps->wphdr.flags & MONO_DATA))
|
||
|
channel ^= 1;
|
||
|
}
|
||
|
|
||
|
return sample_count;
|
||
|
}
|
||
|
|
||
|
/*------------------------------------------------------------------------------------------------------------------------*/
|
||
|
|
||
|
// 80 term DSD decimation filter
|
||
|
// < 1 dB down at 20 kHz
|
||
|
// > 108 dB stopband attenuation
|
||
|
|
||
|
static const int32_t decm_filter [] = {
|
||
|
4, 17, 56, 147, 336, 693, 1320, 2359,
|
||
|
4003, 6502, 10170, 15392, 22623, 32389, 45275, 61920,
|
||
|
82994, 109174, 141119, 179431, 224621, 277068, 336983, 404373,
|
||
|
479004, 560384, 647741, 740025, 835917, 933849, 1032042, 1128551,
|
||
|
1221329, 1308290, 1387386, 1456680, 1514425, 1559128, 1589610, 1605059,
|
||
|
1605059, 1589610, 1559128, 1514425, 1456680, 1387386, 1308290, 1221329,
|
||
|
1128551, 1032042, 933849, 835917, 740025, 647741, 560384, 479004,
|
||
|
404373, 336983, 277068, 224621, 179431, 141119, 109174, 82994,
|
||
|
61920, 45275, 32389, 22623, 15392, 10170, 6502, 4003,
|
||
|
2359, 1320, 693, 336, 147, 56, 17, 4,
|
||
|
};
|
||
|
|
||
|
#define NUM_FILTER_TERMS ((int)(sizeof (decm_filter) / sizeof (decm_filter [0])))
|
||
|
#define HISTORY_BYTES ((NUM_FILTER_TERMS+7)/8)
|
||
|
|
||
|
typedef struct {
|
||
|
unsigned char delay [HISTORY_BYTES];
|
||
|
} DecimationChannel;
|
||
|
|
||
|
typedef struct {
|
||
|
int32_t conv_tables [HISTORY_BYTES] [256];
|
||
|
DecimationChannel *chans;
|
||
|
int num_channels;
|
||
|
} DecimationContext;
|
||
|
|
||
|
void *decimate_dsd_init (int num_channels)
|
||
|
{
|
||
|
DecimationContext *context = malloc (sizeof (DecimationContext));
|
||
|
double filter_sum = 0, filter_scale;
|
||
|
int skipped_terms, i, j;
|
||
|
|
||
|
if (!context)
|
||
|
return context;
|
||
|
|
||
|
memset (context, 0, sizeof (*context));
|
||
|
context->num_channels = num_channels;
|
||
|
context->chans = malloc (num_channels * sizeof (DecimationChannel));
|
||
|
|
||
|
if (!context->chans) {
|
||
|
free (context);
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
for (i = 0; i < NUM_FILTER_TERMS; ++i)
|
||
|
filter_sum += decm_filter [i];
|
||
|
|
||
|
filter_scale = ((1 << 23) - 1) / filter_sum * 16.0;
|
||
|
// fprintf (stderr, "convolution, %d terms, %f sum, %f scale\n", NUM_FILTER_TERMS, filter_sum, filter_scale);
|
||
|
|
||
|
for (skipped_terms = i = 0; i < NUM_FILTER_TERMS; ++i) {
|
||
|
int scaled_term = (int) floor (decm_filter [i] * filter_scale + 0.5);
|
||
|
|
||
|
if (scaled_term) {
|
||
|
for (j = 0; j < 256; ++j)
|
||
|
if (j & (0x80 >> (i & 0x7)))
|
||
|
context->conv_tables [i >> 3] [j] += scaled_term;
|
||
|
else
|
||
|
context->conv_tables [i >> 3] [j] -= scaled_term;
|
||
|
}
|
||
|
else
|
||
|
skipped_terms++;
|
||
|
}
|
||
|
|
||
|
// fprintf (stderr, "%d terms skipped\n", skipped_terms);
|
||
|
|
||
|
decimate_dsd_reset (context);
|
||
|
|
||
|
return context;
|
||
|
}
|
||
|
|
||
|
void decimate_dsd_reset (void *decimate_context)
|
||
|
{
|
||
|
DecimationContext *context = (DecimationContext *) decimate_context;
|
||
|
int chan = 0, i;
|
||
|
|
||
|
if (!context)
|
||
|
return;
|
||
|
|
||
|
for (chan = 0; chan < context->num_channels; ++chan)
|
||
|
for (i = 0; i < HISTORY_BYTES; ++i)
|
||
|
context->chans [chan].delay [i] = 0x55;
|
||
|
}
|
||
|
|
||
|
void decimate_dsd_run (void *decimate_context, int32_t *samples, int num_samples)
|
||
|
{
|
||
|
DecimationContext *context = (DecimationContext *) decimate_context;
|
||
|
int chan = 0;
|
||
|
|
||
|
if (!context)
|
||
|
return;
|
||
|
|
||
|
while (num_samples) {
|
||
|
DecimationChannel *sp = context->chans + chan;
|
||
|
int sum = 0;
|
||
|
|
||
|
sum += context->conv_tables [0] [sp->delay [0] = sp->delay [1]];
|
||
|
sum += context->conv_tables [1] [sp->delay [1] = sp->delay [2]];
|
||
|
sum += context->conv_tables [2] [sp->delay [2] = sp->delay [3]];
|
||
|
sum += context->conv_tables [3] [sp->delay [3] = sp->delay [4]];
|
||
|
sum += context->conv_tables [4] [sp->delay [4] = sp->delay [5]];
|
||
|
sum += context->conv_tables [5] [sp->delay [5] = sp->delay [6]];
|
||
|
sum += context->conv_tables [6] [sp->delay [6] = sp->delay [7]];
|
||
|
sum += context->conv_tables [7] [sp->delay [7] = sp->delay [8]];
|
||
|
sum += context->conv_tables [8] [sp->delay [8] = sp->delay [9]];
|
||
|
sum += context->conv_tables [9] [sp->delay [9] = *samples];
|
||
|
*samples++ = sum >> 4;
|
||
|
|
||
|
if (++chan == context->num_channels) {
|
||
|
num_samples--;
|
||
|
chan = 0;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void decimate_dsd_destroy (void *decimate_context)
|
||
|
{
|
||
|
DecimationContext *context = (DecimationContext *) decimate_context;
|
||
|
|
||
|
if (!context)
|
||
|
return;
|
||
|
|
||
|
if (context->chans)
|
||
|
free (context->chans);
|
||
|
|
||
|
free (context);
|
||
|
}
|