forked from len0rd/rockbox
Based on http://src.gnu-darwin.org/ports/multimedia/helixplayer/work/hxplay-1.0.7/audio/resampler/hermite.c Change-Id: Id87565a060aa2383701e7c2f3ea023c7555ad9ef
426 lines
13 KiB
C
Executable file
426 lines
13 KiB
C
Executable file
/***************************************************************************
|
|
* __________ __ ___.
|
|
* Open \______ \ ____ ____ | | _\_ |__ _______ ___
|
|
* Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ /
|
|
* Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < <
|
|
* Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \
|
|
* \/ \/ \/ \/ \/
|
|
* $Id$
|
|
*
|
|
* Copyright (C) 2005 Miika Pekkarinen
|
|
* Copyright (C) 2012 Michael Sevakis
|
|
*
|
|
* This program is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU General Public License
|
|
* as published by the Free Software Foundation; either version 2
|
|
* of the License, or (at your option) any later version.
|
|
*
|
|
* This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
|
|
* KIND, either express or implied.
|
|
*
|
|
****************************************************************************/
|
|
#include "config.h"
|
|
#include "system.h"
|
|
#include "fracmul.h"
|
|
#include "fixedpoint.h"
|
|
#include "dsp_proc_entry.h"
|
|
#include <string.h>
|
|
|
|
/**
|
|
* Linear interpolation resampling that introduces a one sample delay because
|
|
* of our inability to look into the future at the end of a frame.
|
|
*/
|
|
|
|
#define HERMITE 1
|
|
|
|
#if 1 /* Set to '1' to enable debug messages */
|
|
#include <debug.h>
|
|
#else
|
|
#undef DEBUGF
|
|
#define DEBUGF(...)
|
|
#endif
|
|
|
|
#define RESAMPLE_BUF_COUNT 192 /* Per channel, per DSP */
|
|
|
|
/* CODEC_IDX_AUDIO = left and right, CODEC_IDX_VOICE = mono */
|
|
static int32_t resample_out_bufs[3][RESAMPLE_BUF_COUNT] IBSS_ATTR;
|
|
|
|
/* Data for each resampler on each DSP */
|
|
static struct resample_data
|
|
{
|
|
uint32_t delta; /* 00h: Phase delta for each step in s15.16*/
|
|
uint32_t phase; /* 04h: Current phase [pos16|frac16] */
|
|
int32_t last_sample[2]; /* 08h: Last samples for interpolation (L+R) */
|
|
/* 10h */
|
|
int32_t frequency; /* Virtual samplerate */
|
|
struct dsp_config *dsp; /* The DSP for this resampler */
|
|
struct dsp_buffer resample_buf; /* Buffer descriptor for resampled data */
|
|
int32_t *resample_buf_arr[2]; /* Actual output data pointers */
|
|
|
|
/*Hermite Resampler*/
|
|
|
|
int32_t last_samples[6];
|
|
|
|
} resample_data[DSP_COUNT] IBSS_ATTR;
|
|
|
|
/* Actual worker function. Implemented here or in target assembly code. */
|
|
int lin_resample_resample(struct resample_data *data, struct dsp_buffer *src,
|
|
struct dsp_buffer *dst);
|
|
|
|
int hermite_resample_resample(struct resample_data *data, struct dsp_buffer *src,
|
|
struct dsp_buffer *dst);
|
|
|
|
static void lin_resample_flush_data(struct resample_data *data)
|
|
{
|
|
data->phase = 0;
|
|
data->last_sample[0] = 0;
|
|
data->last_sample[1] = 0;
|
|
}
|
|
|
|
static void lin_resample_flush(struct dsp_proc_entry *this)
|
|
{
|
|
struct resample_data *data = (void *)this->data;
|
|
data->resample_buf.remcount = 0;
|
|
lin_resample_flush_data(data);
|
|
}
|
|
|
|
static bool lin_resample_new_delta(struct resample_data *data,
|
|
struct dsp_buffer *buf)
|
|
{
|
|
int32_t frequency = buf->format.frequency; /* virtual samplerate */
|
|
|
|
data->frequency = frequency;
|
|
data->delta = fp_div(frequency, NATIVE_FREQUENCY, 16);
|
|
|
|
if (frequency == NATIVE_FREQUENCY)
|
|
{
|
|
/* NOTE: If fully glitch-free transistions from no resampling to
|
|
resampling are desired, last_sample history should be maintained
|
|
even when not resampling. */
|
|
lin_resample_flush_data(data);
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
|
|
|
|
int hermite_resample_resample(struct resample_data *data, struct dsp_buffer *src,
|
|
struct dsp_buffer *dst)
|
|
{
|
|
int ch = src->format.num_channels - 1;
|
|
uint32_t count = MIN(src->remcount, 0x8000);
|
|
uint32_t delta = data->delta;
|
|
uint32_t phase, pos;
|
|
int32_t *d;
|
|
int x0, x1, x2, x3, frac, acc0;
|
|
//DEBUGF("hermite_resample_resample top\n");
|
|
/* restore state */
|
|
|
|
|
|
//DEBUGF("count: %d delta: %d, phase: %d (%d)\n",count, delta,phase >> 16, phase);
|
|
do
|
|
{
|
|
const int32_t *s = src->p32[ch];
|
|
|
|
d = dst->p32[ch];
|
|
int32_t *dmax = d + dst->bufcount;
|
|
|
|
phase = data->phase;
|
|
pos = phase >> 16;
|
|
pos = MIN(pos, count);
|
|
|
|
int32_t last = pos > 0 ? s[pos - 1] : data->last_sample[ch];
|
|
|
|
if (pos < count)
|
|
{
|
|
while (1)
|
|
{
|
|
|
|
int i = pos;
|
|
if (i < 3) {
|
|
x3 = (i < 3 ? data->last_samples[i+0] : s[i-3]) ;
|
|
x2 = (i < 2 ? data->last_samples[i+1] : s[i-2]) ;
|
|
x1 = (i < 1 ? data->last_samples[i+2] : s[i-1]) ;
|
|
} else {
|
|
x3 = s[i-3] ;
|
|
x2 = s[i-2] ;
|
|
x1 = s[i-1] ;
|
|
}
|
|
x0 = s[i] ;
|
|
//frac = f >> 1;
|
|
frac=(0x0000FFFF&phase) << 15;
|
|
//DEBUGF("pos: %d phase: %d frac: %d\n",pos, phase, frac);
|
|
|
|
/* 4-tap Hermite, using Farrow structure */
|
|
acc0 = (3 * (x2 - x1) + x0 - x3) >> 1;
|
|
acc0 = FRACMUL(acc0, frac);
|
|
acc0 += 2 * x1 + x3 - ((5 * x2 + x0) >> 1);
|
|
acc0 = FRACMUL(acc0, frac);
|
|
acc0 += (x1 - x3) >> 1;
|
|
acc0 = FRACMUL(acc0, frac);
|
|
acc0 += x2;
|
|
|
|
|
|
*d++ = acc0;
|
|
|
|
phase += delta;
|
|
pos = phase >> 16;
|
|
|
|
if (pos >= count || d >= dmax)
|
|
break;
|
|
|
|
// if (pos > 0){
|
|
// /* save delay samples for next time (last_samples[0] = oldest, last_samples[2] = newest) */
|
|
// data->last_samples[ch*3+0] = (pos < 3 ? data->last_samples[pos+0] : s[pos-3]);
|
|
// data->last_samples[ch*3+1] = (pos < 2 ? data->last_samples[pos+1] : s[pos-2]);
|
|
// data->last_samples[ch*3+2] = (pos < 1 ? data->last_samples[pos+2] : s[pos-1]);
|
|
// }
|
|
}
|
|
|
|
//if (pos > 0)
|
|
//{
|
|
pos = MIN(pos, count);
|
|
data->last_samples[ch*3+0] = (pos < 3 ? data->last_samples[pos+0] : s[pos-3]);
|
|
data->last_samples[ch*3+1] = (pos < 2 ? data->last_samples[pos+1] : s[pos-2]);
|
|
data->last_samples[ch*3+2] = (pos < 1 ? data->last_samples[pos+2] : s[pos-1]);
|
|
//}
|
|
}
|
|
|
|
}
|
|
while (--ch >= 0);
|
|
|
|
|
|
/* Wrap phase accumulator back to start of next frame. */
|
|
data->phase = phase - (pos << 16);
|
|
|
|
dst->remcount = d - dst->p32[0];
|
|
return pos;
|
|
}
|
|
|
|
#if !defined(CPU_COLDFIRE) && !defined(CPU_ARM)
|
|
/* Where the real work is done */
|
|
int lin_resample_resample(struct resample_data *data, struct dsp_buffer *src,
|
|
struct dsp_buffer *dst)
|
|
{
|
|
int ch = src->format.num_channels - 1;
|
|
uint32_t count = MIN(src->remcount, 0x8000);
|
|
uint32_t delta = data->delta;
|
|
uint32_t phase, pos;
|
|
int32_t *d;
|
|
DEBUGF("count: %d delta: %d, phase: %d (%d)\n",count, delta,phase >> 16, phase);
|
|
do
|
|
{
|
|
const int32_t *s = src->p32[ch];
|
|
|
|
d = dst->p32[ch];
|
|
int32_t *dmax = d + dst->bufcount;
|
|
|
|
phase = data->phase;
|
|
pos = phase >> 16;
|
|
pos = MIN(pos, count);
|
|
|
|
int32_t last = pos > 0 ? s[pos - 1] : data->last_sample[ch];
|
|
|
|
if (pos < count)
|
|
{
|
|
while (1)
|
|
{
|
|
DEBUGF("phase: %d frac: %d\n", phase, (phase & 0xffff) << 15);
|
|
*d++ = last + FRACMUL((phase & 0xffff) << 15, s[pos] - last);
|
|
phase += delta;
|
|
pos = phase >> 16;
|
|
|
|
if (pos >= count || d >= dmax)
|
|
break;
|
|
|
|
if (pos > 0)
|
|
last = s[pos - 1];
|
|
}
|
|
|
|
if (pos > 0)
|
|
{
|
|
pos = MIN(pos, count);
|
|
last = s[pos - 1];
|
|
}
|
|
}
|
|
|
|
DEBUGF("pos: %d count: %d\n", pos, count);
|
|
data->last_sample[ch] = last;
|
|
}
|
|
while (--ch >= 0);
|
|
|
|
/* Wrap phase accumulator back to start of next frame. */
|
|
data->phase = phase - (pos << 16);
|
|
|
|
dst->remcount = d - dst->p32[0];
|
|
DEBUGF("remcount: %d, pos %d\n", dst->remcount, pos);
|
|
return pos;
|
|
}
|
|
#endif /* CPU */
|
|
|
|
/* Resample count stereo samples or stop when the destination is full.
|
|
* Updates the src buffer and changes to its own output buffer to refer to
|
|
* the resampled data. */
|
|
static void lin_resample_process(struct dsp_proc_entry *this,
|
|
struct dsp_buffer **buf_p)
|
|
{
|
|
struct resample_data *data = (void *)this->data;
|
|
struct dsp_buffer *src = *buf_p;
|
|
struct dsp_buffer *dst = &data->resample_buf;
|
|
|
|
*buf_p = dst;
|
|
|
|
if (dst->remcount > 0)
|
|
return; /* data still remains */
|
|
|
|
dst->remcount = 0;
|
|
dst->p32[0] = data->resample_buf_arr[0];
|
|
dst->p32[1] = data->resample_buf_arr[1];
|
|
|
|
if (src->remcount > 0)
|
|
{
|
|
dst->bufcount = RESAMPLE_BUF_COUNT;
|
|
|
|
#if HERMITE
|
|
int consumed = hermite_resample_resample(data, src, dst);
|
|
#else
|
|
int consumed = lin_resample_resample(data, src, dst);
|
|
#endif
|
|
|
|
|
|
/* Advance src by consumed amount */
|
|
if (consumed > 0)
|
|
dsp_advance_buffer32(src, consumed);
|
|
}
|
|
/* else purged resample_buf */
|
|
|
|
/* Inherit in-place processed mask from source buffer */
|
|
dst->proc_mask = src->proc_mask;
|
|
}
|
|
|
|
/* Finish draining old samples then switch format or shut off */
|
|
static void lin_resample_new_format(struct dsp_proc_entry *this,
|
|
struct dsp_buffer **buf_p)
|
|
{
|
|
struct resample_data *data = (void *)this->data;
|
|
struct dsp_buffer *src = *buf_p;
|
|
struct dsp_buffer *dst = &data->resample_buf;
|
|
|
|
if (dst->remcount > 0)
|
|
{
|
|
*buf_p = dst;
|
|
return; /* data still remains */
|
|
}
|
|
|
|
DSP_PRINT_FORMAT(DSP_PROC_RESAMPLE, DSP_PROC_RESAMPLE, src->format);
|
|
|
|
struct dsp_config *dsp = data->dsp;
|
|
int32_t frequency = data->frequency;
|
|
bool active = dsp_proc_active(dsp, DSP_PROC_RESAMPLE);
|
|
|
|
if (src->format.frequency != frequency)
|
|
{
|
|
DEBUGF(" DSP_PROC_RESAMPLE- new delta\n");
|
|
DEBUGF("hermite_resample_new_delta in\n");
|
|
#if 0
|
|
active = hermite_resample_new_delta(data, src);
|
|
#else
|
|
active = lin_resample_new_delta(data, src);
|
|
#endif
|
|
DEBUGF("hermite_resample_new_delta out\n");
|
|
dsp_proc_activate(dsp, DSP_PROC_RESAMPLE, active);
|
|
}
|
|
|
|
/* Everything after us is NATIVE_FREQUENCY */
|
|
struct sample_format f = src->format;
|
|
f.frequency = NATIVE_FREQUENCY;
|
|
f.codec_frequency = NATIVE_FREQUENCY;
|
|
|
|
if (!active)
|
|
{
|
|
DEBUGF(" DSP_PROC_RESAMPLE- not active\n");
|
|
dst->format = f; /* Keep track */
|
|
return; /* No resampling required */
|
|
}
|
|
|
|
format_change_ack(&src->format);
|
|
|
|
if (EQU_SAMPLE_FORMAT(f, dst->format))
|
|
{
|
|
DEBUGF(" DSP_PROC_RESAMPLE- same dst format\n");
|
|
format_change_ack(&f); /* Nothing changed that matters downstream */
|
|
}
|
|
|
|
dst->format = f;
|
|
dsp_proc_call(this, buf_p, 0);
|
|
}
|
|
|
|
static void lin_resample_init(struct dsp_config *dsp,
|
|
enum dsp_ids dsp_id)
|
|
{
|
|
/* Always enable resampler so that format changes may be monitored and
|
|
* it self-activated when required */
|
|
dsp_proc_enable(dsp, DSP_PROC_RESAMPLE, true);
|
|
|
|
int32_t *lbuf, *rbuf;
|
|
|
|
switch (dsp_id)
|
|
{
|
|
case CODEC_IDX_AUDIO:
|
|
lbuf = resample_out_bufs[0];
|
|
rbuf = resample_out_bufs[1];
|
|
break;
|
|
|
|
case CODEC_IDX_VOICE:
|
|
lbuf = rbuf = resample_out_bufs[2]; /* Always mono */
|
|
break;
|
|
|
|
default:
|
|
/* huh? */
|
|
DEBUGF("DSP_PROC_RESAMPLE- unknown DSP %d\n", (int)dsp_id);
|
|
return;
|
|
}
|
|
|
|
resample_data[dsp_id].resample_buf_arr[0] = lbuf;
|
|
resample_data[dsp_id].resample_buf_arr[1] = rbuf;
|
|
}
|
|
|
|
/* DSP message hook */
|
|
static intptr_t lin_resample_configure(struct dsp_proc_entry *this,
|
|
struct dsp_config *dsp,
|
|
unsigned int setting,
|
|
intptr_t value)
|
|
{
|
|
switch (setting)
|
|
{
|
|
case DSP_INIT:
|
|
lin_resample_init(dsp, (enum dsp_ids)value);
|
|
break;
|
|
|
|
case DSP_FLUSH:
|
|
lin_resample_flush(this);
|
|
break;
|
|
|
|
case DSP_PROC_INIT:
|
|
this->data = (intptr_t)&resample_data[dsp_get_id(dsp)];
|
|
this->ip_mask = 0; /* Not in-place */
|
|
this->process[0] = lin_resample_process;
|
|
this->process[1] = lin_resample_new_format;
|
|
((struct resample_data *)this->data)->dsp = dsp;
|
|
break;
|
|
|
|
case DSP_PROC_CLOSE:
|
|
/* This stage should be enabled at all times */
|
|
DEBUGF("DSP_PROC_RESAMPLE- Error: Closing!\n");
|
|
break;
|
|
}
|
|
|
|
return 1;
|
|
(void)value;
|
|
}
|
|
|
|
/* Database entry */
|
|
DSP_PROC_DB_ENTRY(RESAMPLE,
|
|
lin_resample_configure);
|