2016-03-15 05:08:05 -07:00
|
|
|
|
|
|
|
#include "config.h"
|
|
|
|
|
|
|
|
#include "bformatdec.h"
|
|
|
|
#include "ambdec.h"
|
|
|
|
#include "alu.h"
|
|
|
|
|
2017-02-28 19:44:34 -08:00
|
|
|
#include "bool.h"
|
2016-03-23 12:53:36 -07:00
|
|
|
#include "threads.h"
|
2016-03-29 00:44:58 -07:00
|
|
|
#include "almalloc.h"
|
2016-03-23 12:53:36 -07:00
|
|
|
|
2016-03-15 05:08:05 -07:00
|
|
|
|
2018-01-13 09:14:46 -08:00
|
|
|
void bandsplit_init(BandSplitter *splitter, ALfloat f0norm)
|
2016-03-15 05:08:05 -07:00
|
|
|
{
|
2018-01-13 09:14:46 -08:00
|
|
|
ALfloat w = f0norm * F_TAU;
|
2016-03-15 05:08:05 -07:00
|
|
|
ALfloat cw = cosf(w);
|
|
|
|
if(cw > FLT_EPSILON)
|
|
|
|
splitter->coeff = (sinf(w) - 1.0f) / cw;
|
|
|
|
else
|
|
|
|
splitter->coeff = cw * -0.5f;
|
|
|
|
|
|
|
|
splitter->lp_z1 = 0.0f;
|
|
|
|
splitter->lp_z2 = 0.0f;
|
|
|
|
splitter->hp_z1 = 0.0f;
|
|
|
|
}
|
|
|
|
|
2016-08-21 03:05:42 -07:00
|
|
|
void bandsplit_clear(BandSplitter *splitter)
|
|
|
|
{
|
|
|
|
splitter->lp_z1 = 0.0f;
|
|
|
|
splitter->lp_z2 = 0.0f;
|
|
|
|
splitter->hp_z1 = 0.0f;
|
|
|
|
}
|
|
|
|
|
|
|
|
void bandsplit_process(BandSplitter *splitter, ALfloat *restrict hpout, ALfloat *restrict lpout,
|
2017-01-16 08:59:08 -08:00
|
|
|
const ALfloat *input, ALsizei count)
|
2016-03-15 05:08:05 -07:00
|
|
|
{
|
2018-03-29 11:18:16 -07:00
|
|
|
ALfloat lp_coeff, hp_coeff, lp_y, hp_y, d;
|
|
|
|
ALfloat lp_z1, lp_z2, hp_z1;
|
2017-01-16 08:59:08 -08:00
|
|
|
ALsizei i;
|
2016-03-15 05:08:05 -07:00
|
|
|
|
2018-03-29 11:18:16 -07:00
|
|
|
hp_coeff = splitter->coeff;
|
|
|
|
lp_coeff = splitter->coeff*0.5f + 0.5f;
|
|
|
|
lp_z1 = splitter->lp_z1;
|
|
|
|
lp_z2 = splitter->lp_z2;
|
|
|
|
hp_z1 = splitter->hp_z1;
|
2016-03-15 05:08:05 -07:00
|
|
|
for(i = 0;i < count;i++)
|
|
|
|
{
|
2018-03-29 11:18:16 -07:00
|
|
|
ALfloat in = input[i];
|
2016-03-15 05:08:05 -07:00
|
|
|
|
2018-03-29 11:18:16 -07:00
|
|
|
/* Low-pass sample processing. */
|
|
|
|
d = (in - lp_z1) * lp_coeff;
|
|
|
|
lp_y = lp_z1 + d;
|
|
|
|
lp_z1 = lp_y + d;
|
2016-03-15 05:08:05 -07:00
|
|
|
|
2018-03-29 11:18:16 -07:00
|
|
|
d = (lp_y - lp_z2) * lp_coeff;
|
|
|
|
lp_y = lp_z2 + d;
|
|
|
|
lp_z2 = lp_y + d;
|
2016-03-15 05:08:05 -07:00
|
|
|
|
2018-03-29 11:18:16 -07:00
|
|
|
lpout[i] = lp_y;
|
2016-03-15 05:08:05 -07:00
|
|
|
|
2018-03-29 11:18:16 -07:00
|
|
|
/* All-pass sample processing. */
|
|
|
|
d = in - hp_coeff*hp_z1;
|
|
|
|
hp_y = hp_z1 + hp_coeff*d;
|
|
|
|
hp_z1 = d;
|
2016-03-15 05:08:05 -07:00
|
|
|
|
2018-03-29 11:18:16 -07:00
|
|
|
/* High-pass generated from removing low-passed output. */
|
|
|
|
hpout[i] = hp_y - lp_y;
|
2016-03-15 05:08:05 -07:00
|
|
|
}
|
2018-03-29 11:18:16 -07:00
|
|
|
splitter->lp_z1 = lp_z1;
|
|
|
|
splitter->lp_z2 = lp_z2;
|
|
|
|
splitter->hp_z1 = hp_z1;
|
2016-03-15 05:08:05 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-01-13 09:14:46 -08:00
|
|
|
void splitterap_init(SplitterAllpass *splitter, ALfloat f0norm)
|
2017-07-19 02:48:01 -07:00
|
|
|
{
|
2018-01-13 09:14:46 -08:00
|
|
|
ALfloat w = f0norm * F_TAU;
|
2017-07-19 02:48:01 -07:00
|
|
|
ALfloat cw = cosf(w);
|
|
|
|
if(cw > FLT_EPSILON)
|
|
|
|
splitter->coeff = (sinf(w) - 1.0f) / cw;
|
|
|
|
else
|
|
|
|
splitter->coeff = cw * -0.5f;
|
|
|
|
|
|
|
|
splitter->z1 = 0.0f;
|
|
|
|
}
|
|
|
|
|
|
|
|
void splitterap_clear(SplitterAllpass *splitter)
|
|
|
|
{
|
|
|
|
splitter->z1 = 0.0f;
|
|
|
|
}
|
|
|
|
|
|
|
|
void splitterap_process(SplitterAllpass *splitter, ALfloat *restrict samples, ALsizei count)
|
|
|
|
{
|
|
|
|
ALfloat coeff, d, x;
|
|
|
|
ALfloat z1;
|
|
|
|
ALsizei i;
|
|
|
|
|
|
|
|
coeff = splitter->coeff;
|
|
|
|
z1 = splitter->z1;
|
|
|
|
for(i = 0;i < count;i++)
|
|
|
|
{
|
|
|
|
x = samples[i];
|
|
|
|
|
|
|
|
d = x - coeff*z1;
|
|
|
|
x = z1 + coeff*d;
|
|
|
|
z1 = d;
|
|
|
|
|
|
|
|
samples[i] = x;
|
|
|
|
}
|
|
|
|
splitter->z1 = z1;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-02-13 13:47:35 -08:00
|
|
|
/* NOTE: These are scale factors as applied to Ambisonics content. Decoder
|
|
|
|
* coefficients should be divided by these values to get proper N3D scalings.
|
|
|
|
*/
|
|
|
|
const ALfloat N3D2N3DScale[MAX_AMBI_COEFFS] = {
|
2016-03-15 05:08:05 -07:00
|
|
|
1.0f, 1.0f, 1.0f, 1.0f, 1.0f, 1.0f, 1.0f, 1.0f,
|
|
|
|
1.0f, 1.0f, 1.0f, 1.0f, 1.0f, 1.0f, 1.0f, 1.0f
|
|
|
|
};
|
2018-02-13 13:47:35 -08:00
|
|
|
const ALfloat SN3D2N3DScale[MAX_AMBI_COEFFS] = {
|
2016-03-15 05:08:05 -07:00
|
|
|
1.000000000f, /* ACN 0 (W), sqrt(1) */
|
|
|
|
1.732050808f, /* ACN 1 (Y), sqrt(3) */
|
|
|
|
1.732050808f, /* ACN 2 (Z), sqrt(3) */
|
|
|
|
1.732050808f, /* ACN 3 (X), sqrt(3) */
|
|
|
|
2.236067978f, /* ACN 4 (V), sqrt(5) */
|
|
|
|
2.236067978f, /* ACN 5 (T), sqrt(5) */
|
|
|
|
2.236067978f, /* ACN 6 (R), sqrt(5) */
|
|
|
|
2.236067978f, /* ACN 7 (S), sqrt(5) */
|
|
|
|
2.236067978f, /* ACN 8 (U), sqrt(5) */
|
|
|
|
2.645751311f, /* ACN 9 (Q), sqrt(7) */
|
|
|
|
2.645751311f, /* ACN 10 (O), sqrt(7) */
|
|
|
|
2.645751311f, /* ACN 11 (M), sqrt(7) */
|
|
|
|
2.645751311f, /* ACN 12 (K), sqrt(7) */
|
|
|
|
2.645751311f, /* ACN 13 (L), sqrt(7) */
|
|
|
|
2.645751311f, /* ACN 14 (N), sqrt(7) */
|
|
|
|
2.645751311f, /* ACN 15 (P), sqrt(7) */
|
|
|
|
};
|
2018-02-13 13:47:35 -08:00
|
|
|
const ALfloat FuMa2N3DScale[MAX_AMBI_COEFFS] = {
|
2016-03-15 05:08:05 -07:00
|
|
|
1.414213562f, /* ACN 0 (W), sqrt(2) */
|
|
|
|
1.732050808f, /* ACN 1 (Y), sqrt(3) */
|
|
|
|
1.732050808f, /* ACN 2 (Z), sqrt(3) */
|
|
|
|
1.732050808f, /* ACN 3 (X), sqrt(3) */
|
|
|
|
1.936491673f, /* ACN 4 (V), sqrt(15)/2 */
|
|
|
|
1.936491673f, /* ACN 5 (T), sqrt(15)/2 */
|
|
|
|
2.236067978f, /* ACN 6 (R), sqrt(5) */
|
|
|
|
1.936491673f, /* ACN 7 (S), sqrt(15)/2 */
|
|
|
|
1.936491673f, /* ACN 8 (U), sqrt(15)/2 */
|
|
|
|
2.091650066f, /* ACN 9 (Q), sqrt(35/8) */
|
|
|
|
1.972026594f, /* ACN 10 (O), sqrt(35)/3 */
|
|
|
|
2.231093404f, /* ACN 11 (M), sqrt(224/45) */
|
|
|
|
2.645751311f, /* ACN 12 (K), sqrt(7) */
|
|
|
|
2.231093404f, /* ACN 13 (L), sqrt(224/45) */
|
|
|
|
1.972026594f, /* ACN 14 (N), sqrt(35)/3 */
|
|
|
|
2.091650066f, /* ACN 15 (P), sqrt(35/8) */
|
|
|
|
};
|
|
|
|
|
|
|
|
|
2018-02-12 20:58:39 -08:00
|
|
|
#define HF_BAND 0
|
|
|
|
#define LF_BAND 1
|
|
|
|
#define NUM_BANDS 2
|
2016-07-17 00:29:02 -07:00
|
|
|
|
2016-09-06 03:10:38 -07:00
|
|
|
/* These points are in AL coordinates! */
|
|
|
|
static const ALfloat Ambi3DPoints[8][3] = {
|
2016-07-30 09:29:21 -07:00
|
|
|
{ -0.577350269f, 0.577350269f, -0.577350269f },
|
|
|
|
{ 0.577350269f, 0.577350269f, -0.577350269f },
|
|
|
|
{ -0.577350269f, 0.577350269f, 0.577350269f },
|
|
|
|
{ 0.577350269f, 0.577350269f, 0.577350269f },
|
|
|
|
{ -0.577350269f, -0.577350269f, -0.577350269f },
|
|
|
|
{ 0.577350269f, -0.577350269f, -0.577350269f },
|
|
|
|
{ -0.577350269f, -0.577350269f, 0.577350269f },
|
|
|
|
{ 0.577350269f, -0.577350269f, 0.577350269f },
|
|
|
|
};
|
2018-02-12 21:24:58 -08:00
|
|
|
static const ALfloat Ambi3DDecoder[8][MAX_AMBI_COEFFS] = {
|
|
|
|
{ 0.125f, 0.125f, 0.125f, 0.125f },
|
|
|
|
{ 0.125f, -0.125f, 0.125f, 0.125f },
|
|
|
|
{ 0.125f, 0.125f, 0.125f, -0.125f },
|
|
|
|
{ 0.125f, -0.125f, 0.125f, -0.125f },
|
|
|
|
{ 0.125f, 0.125f, -0.125f, 0.125f },
|
|
|
|
{ 0.125f, -0.125f, -0.125f, 0.125f },
|
|
|
|
{ 0.125f, 0.125f, -0.125f, -0.125f },
|
|
|
|
{ 0.125f, -0.125f, -0.125f, -0.125f },
|
|
|
|
};
|
|
|
|
static const ALfloat Ambi3DDecoderHFScale[MAX_AMBI_COEFFS] = {
|
|
|
|
2.0f,
|
|
|
|
1.15470054f, 1.15470054f, 1.15470054f
|
2016-04-09 08:04:56 -07:00
|
|
|
};
|
2016-03-23 12:53:36 -07:00
|
|
|
|
|
|
|
|
2016-07-17 00:29:02 -07:00
|
|
|
/* NOTE: BandSplitter filters are unused with single-band decoding */
|
2016-03-15 05:08:05 -07:00
|
|
|
typedef struct BFormatDec {
|
2018-02-12 20:33:31 -08:00
|
|
|
ALuint Enabled; /* Bitfield of enabled channels. */
|
2016-03-28 14:50:18 -07:00
|
|
|
|
2016-07-17 00:29:02 -07:00
|
|
|
union {
|
2018-02-12 20:58:39 -08:00
|
|
|
alignas(16) ALfloat Dual[MAX_OUTPUT_CHANNELS][NUM_BANDS][MAX_AMBI_COEFFS];
|
2016-07-17 00:29:02 -07:00
|
|
|
alignas(16) ALfloat Single[MAX_OUTPUT_CHANNELS][MAX_AMBI_COEFFS];
|
|
|
|
} Matrix;
|
2016-03-15 05:08:05 -07:00
|
|
|
|
|
|
|
BandSplitter XOver[MAX_AMBI_COEFFS];
|
|
|
|
|
|
|
|
ALfloat (*Samples)[BUFFERSIZE];
|
|
|
|
/* These two alias into Samples */
|
|
|
|
ALfloat (*SamplesHF)[BUFFERSIZE];
|
|
|
|
ALfloat (*SamplesLF)[BUFFERSIZE];
|
|
|
|
|
2016-03-27 13:23:37 -07:00
|
|
|
alignas(16) ALfloat ChannelMix[BUFFERSIZE];
|
2016-03-23 12:53:36 -07:00
|
|
|
|
2016-03-27 13:23:37 -07:00
|
|
|
struct {
|
2017-01-24 19:03:51 -08:00
|
|
|
BandSplitter XOver;
|
2018-02-12 20:58:39 -08:00
|
|
|
ALfloat Gains[NUM_BANDS];
|
2017-01-24 19:03:51 -08:00
|
|
|
} UpSampler[4];
|
2016-03-26 17:33:49 -07:00
|
|
|
|
2017-01-16 09:37:55 -08:00
|
|
|
ALsizei NumChannels;
|
2016-03-15 05:08:05 -07:00
|
|
|
ALboolean DualBand;
|
|
|
|
} BFormatDec;
|
|
|
|
|
|
|
|
BFormatDec *bformatdec_alloc()
|
|
|
|
{
|
|
|
|
return al_calloc(16, sizeof(BFormatDec));
|
|
|
|
}
|
|
|
|
|
2018-02-12 20:48:28 -08:00
|
|
|
void bformatdec_free(BFormatDec **dec)
|
2016-03-15 05:08:05 -07:00
|
|
|
{
|
2018-02-12 22:20:43 -08:00
|
|
|
if(dec && *dec)
|
2016-03-15 05:08:05 -07:00
|
|
|
{
|
2018-02-12 20:48:28 -08:00
|
|
|
al_free((*dec)->Samples);
|
|
|
|
(*dec)->Samples = NULL;
|
|
|
|
(*dec)->SamplesHF = NULL;
|
|
|
|
(*dec)->SamplesLF = NULL;
|
2016-03-15 05:08:05 -07:00
|
|
|
|
2018-02-12 20:48:28 -08:00
|
|
|
al_free(*dec);
|
|
|
|
*dec = NULL;
|
2016-03-15 05:08:05 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-02-19 22:47:59 -08:00
|
|
|
void bformatdec_reset(BFormatDec *dec, const AmbDecConf *conf, ALsizei chancount, ALuint srate, const ALsizei chanmap[MAX_OUTPUT_CHANNELS])
|
2016-03-15 05:08:05 -07:00
|
|
|
{
|
2017-01-29 16:42:02 -08:00
|
|
|
static const ALsizei map2DTo3D[MAX_AMBI2D_COEFFS] = {
|
2016-03-25 19:57:25 -07:00
|
|
|
0, 1, 3, 4, 8, 9, 15
|
|
|
|
};
|
2018-02-13 13:47:35 -08:00
|
|
|
const ALfloat *coeff_scale = N3D2N3DScale;
|
2017-02-28 19:44:34 -08:00
|
|
|
bool periphonic;
|
2017-02-19 22:47:59 -08:00
|
|
|
ALfloat ratio;
|
2017-01-16 09:37:55 -08:00
|
|
|
ALsizei i;
|
2016-03-15 05:08:05 -07:00
|
|
|
|
|
|
|
al_free(dec->Samples);
|
|
|
|
dec->Samples = NULL;
|
|
|
|
dec->SamplesHF = NULL;
|
|
|
|
dec->SamplesLF = NULL;
|
|
|
|
|
|
|
|
dec->NumChannels = chancount;
|
2016-04-09 08:04:56 -07:00
|
|
|
dec->Samples = al_calloc(16, dec->NumChannels*2 * sizeof(dec->Samples[0]));
|
2016-03-15 05:08:05 -07:00
|
|
|
dec->SamplesHF = dec->Samples;
|
|
|
|
dec->SamplesLF = dec->SamplesHF + dec->NumChannels;
|
|
|
|
|
2018-02-12 20:33:31 -08:00
|
|
|
dec->Enabled = 0;
|
2016-03-28 14:50:18 -07:00
|
|
|
for(i = 0;i < conf->NumSpeakers;i++)
|
2018-02-12 20:33:31 -08:00
|
|
|
dec->Enabled |= 1 << chanmap[i];
|
2016-03-28 14:50:18 -07:00
|
|
|
|
2016-03-15 05:08:05 -07:00
|
|
|
if(conf->CoeffScale == ADS_SN3D)
|
|
|
|
coeff_scale = SN3D2N3DScale;
|
|
|
|
else if(conf->CoeffScale == ADS_FuMa)
|
|
|
|
coeff_scale = FuMa2N3DScale;
|
|
|
|
|
2017-01-24 19:03:51 -08:00
|
|
|
memset(dec->UpSampler, 0, sizeof(dec->UpSampler));
|
2016-04-09 08:04:56 -07:00
|
|
|
ratio = 400.0f / (ALfloat)srate;
|
|
|
|
for(i = 0;i < 4;i++)
|
2017-01-24 19:03:51 -08:00
|
|
|
bandsplit_init(&dec->UpSampler[i].XOver, ratio);
|
2016-06-01 05:30:06 -07:00
|
|
|
if((conf->ChanMask&AMBI_PERIPHONIC_MASK))
|
2016-03-23 12:53:36 -07:00
|
|
|
{
|
2017-02-28 19:44:34 -08:00
|
|
|
periphonic = true;
|
2017-01-24 19:03:51 -08:00
|
|
|
|
2018-02-12 20:58:39 -08:00
|
|
|
dec->UpSampler[0].Gains[HF_BAND] = (conf->ChanMask > 0x1ff) ? W_SCALE_3H3P :
|
|
|
|
(conf->ChanMask > 0xf) ? W_SCALE_2H2P : 1.0f;
|
|
|
|
dec->UpSampler[0].Gains[LF_BAND] = 1.0f;
|
2017-01-24 19:03:51 -08:00
|
|
|
for(i = 1;i < 4;i++)
|
|
|
|
{
|
2018-02-12 20:58:39 -08:00
|
|
|
dec->UpSampler[i].Gains[HF_BAND] = (conf->ChanMask > 0x1ff) ? XYZ_SCALE_3H3P :
|
|
|
|
(conf->ChanMask > 0xf) ? XYZ_SCALE_2H2P : 1.0f;
|
|
|
|
dec->UpSampler[i].Gains[LF_BAND] = 1.0f;
|
2017-01-24 19:03:51 -08:00
|
|
|
}
|
2016-03-23 12:53:36 -07:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2017-02-28 19:44:34 -08:00
|
|
|
periphonic = false;
|
2017-01-24 19:03:51 -08:00
|
|
|
|
2018-02-12 20:58:39 -08:00
|
|
|
dec->UpSampler[0].Gains[HF_BAND] = (conf->ChanMask > 0x1ff) ? W_SCALE_3H0P :
|
|
|
|
(conf->ChanMask > 0xf) ? W_SCALE_2H0P : 1.0f;
|
|
|
|
dec->UpSampler[0].Gains[LF_BAND] = 1.0f;
|
2017-02-19 17:45:27 -08:00
|
|
|
for(i = 1;i < 3;i++)
|
2017-01-24 19:03:51 -08:00
|
|
|
{
|
2018-02-12 20:58:39 -08:00
|
|
|
dec->UpSampler[i].Gains[HF_BAND] = (conf->ChanMask > 0x1ff) ? XYZ_SCALE_3H0P :
|
|
|
|
(conf->ChanMask > 0xf) ? XYZ_SCALE_2H0P : 1.0f;
|
|
|
|
dec->UpSampler[i].Gains[LF_BAND] = 1.0f;
|
2017-01-24 19:03:51 -08:00
|
|
|
}
|
2018-02-12 20:58:39 -08:00
|
|
|
dec->UpSampler[3].Gains[HF_BAND] = 0.0f;
|
|
|
|
dec->UpSampler[3].Gains[LF_BAND] = 0.0f;
|
2016-03-23 12:53:36 -07:00
|
|
|
}
|
|
|
|
|
2016-07-17 00:29:02 -07:00
|
|
|
memset(&dec->Matrix, 0, sizeof(dec->Matrix));
|
2016-03-15 05:08:05 -07:00
|
|
|
if(conf->FreqBands == 1)
|
|
|
|
{
|
|
|
|
dec->DualBand = AL_FALSE;
|
2016-07-17 00:29:02 -07:00
|
|
|
for(i = 0;i < conf->NumSpeakers;i++)
|
|
|
|
{
|
2017-01-29 16:42:02 -08:00
|
|
|
ALsizei chan = chanmap[i];
|
2016-07-17 00:29:02 -07:00
|
|
|
ALfloat gain;
|
2017-01-29 16:42:02 -08:00
|
|
|
ALsizei j, k;
|
2016-07-17 00:29:02 -07:00
|
|
|
|
2017-02-28 19:44:34 -08:00
|
|
|
if(!periphonic)
|
2016-07-17 00:29:02 -07:00
|
|
|
{
|
2016-07-17 00:46:18 -07:00
|
|
|
for(j = 0,k = 0;j < MAX_AMBI2D_COEFFS;j++)
|
2016-07-17 00:29:02 -07:00
|
|
|
{
|
2017-01-29 16:42:02 -08:00
|
|
|
ALsizei l = map2DTo3D[j];
|
2016-07-17 00:29:02 -07:00
|
|
|
if(j == 0) gain = conf->HFOrderGain[0];
|
|
|
|
else if(j == 1) gain = conf->HFOrderGain[1];
|
|
|
|
else if(j == 3) gain = conf->HFOrderGain[2];
|
|
|
|
else if(j == 5) gain = conf->HFOrderGain[3];
|
|
|
|
if((conf->ChanMask&(1<<l)))
|
|
|
|
dec->Matrix.Single[chan][j] = conf->HFMatrix[i][k++] / coeff_scale[l] *
|
2017-02-19 22:47:59 -08:00
|
|
|
gain;
|
2016-07-17 00:29:02 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
for(j = 0,k = 0;j < MAX_AMBI_COEFFS;j++)
|
|
|
|
{
|
|
|
|
if(j == 0) gain = conf->HFOrderGain[0];
|
|
|
|
else if(j == 1) gain = conf->HFOrderGain[1];
|
|
|
|
else if(j == 4) gain = conf->HFOrderGain[2];
|
|
|
|
else if(j == 9) gain = conf->HFOrderGain[3];
|
|
|
|
if((conf->ChanMask&(1<<j)))
|
|
|
|
dec->Matrix.Single[chan][j] = conf->HFMatrix[i][k++] / coeff_scale[j] *
|
2017-02-19 22:47:59 -08:00
|
|
|
gain;
|
2016-07-17 00:29:02 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2016-03-15 05:08:05 -07:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
dec->DualBand = AL_TRUE;
|
|
|
|
|
|
|
|
ratio = conf->XOverFreq / (ALfloat)srate;
|
|
|
|
for(i = 0;i < MAX_AMBI_COEFFS;i++)
|
|
|
|
bandsplit_init(&dec->XOver[i], ratio);
|
|
|
|
|
|
|
|
ratio = powf(10.0f, conf->XOverRatio / 40.0f);
|
|
|
|
for(i = 0;i < conf->NumSpeakers;i++)
|
|
|
|
{
|
2017-01-29 16:42:02 -08:00
|
|
|
ALsizei chan = chanmap[i];
|
2016-03-22 14:42:53 -07:00
|
|
|
ALfloat gain;
|
2017-01-29 16:42:02 -08:00
|
|
|
ALsizei j, k;
|
2016-03-15 05:08:05 -07:00
|
|
|
|
2017-02-28 19:44:34 -08:00
|
|
|
if(!periphonic)
|
2016-03-15 05:08:05 -07:00
|
|
|
{
|
2016-07-17 00:46:18 -07:00
|
|
|
for(j = 0,k = 0;j < MAX_AMBI2D_COEFFS;j++)
|
2016-07-17 00:29:02 -07:00
|
|
|
{
|
2017-01-29 16:42:02 -08:00
|
|
|
ALsizei l = map2DTo3D[j];
|
2016-07-17 00:29:02 -07:00
|
|
|
if(j == 0) gain = conf->HFOrderGain[0] * ratio;
|
|
|
|
else if(j == 1) gain = conf->HFOrderGain[1] * ratio;
|
|
|
|
else if(j == 3) gain = conf->HFOrderGain[2] * ratio;
|
|
|
|
else if(j == 5) gain = conf->HFOrderGain[3] * ratio;
|
|
|
|
if((conf->ChanMask&(1<<l)))
|
2018-02-12 20:58:39 -08:00
|
|
|
dec->Matrix.Dual[chan][HF_BAND][j] = conf->HFMatrix[i][k++] /
|
|
|
|
coeff_scale[l] * gain;
|
2016-07-17 00:29:02 -07:00
|
|
|
}
|
2016-07-17 00:46:18 -07:00
|
|
|
for(j = 0,k = 0;j < MAX_AMBI2D_COEFFS;j++)
|
2016-03-25 19:57:25 -07:00
|
|
|
{
|
2017-01-29 16:42:02 -08:00
|
|
|
ALsizei l = map2DTo3D[j];
|
2016-03-25 19:57:25 -07:00
|
|
|
if(j == 0) gain = conf->LFOrderGain[0] / ratio;
|
|
|
|
else if(j == 1) gain = conf->LFOrderGain[1] / ratio;
|
|
|
|
else if(j == 3) gain = conf->LFOrderGain[2] / ratio;
|
|
|
|
else if(j == 5) gain = conf->LFOrderGain[3] / ratio;
|
|
|
|
if((conf->ChanMask&(1<<l)))
|
2018-02-12 20:58:39 -08:00
|
|
|
dec->Matrix.Dual[chan][LF_BAND][j] = conf->LFMatrix[i][k++] /
|
|
|
|
coeff_scale[l] * gain;
|
2016-03-25 19:57:25 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2016-07-17 00:29:02 -07:00
|
|
|
for(j = 0,k = 0;j < MAX_AMBI_COEFFS;j++)
|
|
|
|
{
|
|
|
|
if(j == 0) gain = conf->HFOrderGain[0] * ratio;
|
|
|
|
else if(j == 1) gain = conf->HFOrderGain[1] * ratio;
|
|
|
|
else if(j == 4) gain = conf->HFOrderGain[2] * ratio;
|
|
|
|
else if(j == 9) gain = conf->HFOrderGain[3] * ratio;
|
|
|
|
if((conf->ChanMask&(1<<j)))
|
2018-02-12 20:58:39 -08:00
|
|
|
dec->Matrix.Dual[chan][HF_BAND][j] = conf->HFMatrix[i][k++] /
|
|
|
|
coeff_scale[j] * gain;
|
2016-07-17 00:29:02 -07:00
|
|
|
}
|
|
|
|
for(j = 0,k = 0;j < MAX_AMBI_COEFFS;j++)
|
2016-03-25 19:57:25 -07:00
|
|
|
{
|
|
|
|
if(j == 0) gain = conf->LFOrderGain[0] / ratio;
|
|
|
|
else if(j == 1) gain = conf->LFOrderGain[1] / ratio;
|
|
|
|
else if(j == 4) gain = conf->LFOrderGain[2] / ratio;
|
|
|
|
else if(j == 9) gain = conf->LFOrderGain[3] / ratio;
|
|
|
|
if((conf->ChanMask&(1<<j)))
|
2018-02-12 20:58:39 -08:00
|
|
|
dec->Matrix.Dual[chan][LF_BAND][j] = conf->LFMatrix[i][k++] /
|
|
|
|
coeff_scale[j] * gain;
|
2016-03-25 19:57:25 -07:00
|
|
|
}
|
2016-03-15 05:08:05 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-01-16 09:37:55 -08:00
|
|
|
void bformatdec_process(struct BFormatDec *dec, ALfloat (*restrict OutBuffer)[BUFFERSIZE], ALsizei OutChannels, const ALfloat (*restrict InSamples)[BUFFERSIZE], ALsizei SamplesToDo)
|
2016-03-15 05:08:05 -07:00
|
|
|
{
|
2017-01-16 09:37:55 -08:00
|
|
|
ALsizei chan, i;
|
2016-03-15 05:08:05 -07:00
|
|
|
|
2017-02-20 16:57:25 -08:00
|
|
|
OutBuffer = ASSUME_ALIGNED(OutBuffer, 16);
|
2016-03-15 05:08:05 -07:00
|
|
|
if(dec->DualBand)
|
|
|
|
{
|
|
|
|
for(i = 0;i < dec->NumChannels;i++)
|
|
|
|
bandsplit_process(&dec->XOver[i], dec->SamplesHF[i], dec->SamplesLF[i],
|
|
|
|
InSamples[i], SamplesToDo);
|
|
|
|
|
|
|
|
for(chan = 0;chan < OutChannels;chan++)
|
|
|
|
{
|
2018-02-12 20:33:31 -08:00
|
|
|
if(!(dec->Enabled&(1<<chan)))
|
2016-03-28 14:50:18 -07:00
|
|
|
continue;
|
|
|
|
|
2016-03-26 17:33:49 -07:00
|
|
|
memset(dec->ChannelMix, 0, SamplesToDo*sizeof(ALfloat));
|
2018-02-12 20:58:39 -08:00
|
|
|
MixRowSamples(dec->ChannelMix, dec->Matrix.Dual[chan][HF_BAND],
|
2018-01-14 08:51:03 -08:00
|
|
|
dec->SamplesHF, dec->NumChannels, 0, SamplesToDo
|
2016-07-17 00:29:02 -07:00
|
|
|
);
|
2018-02-12 20:58:39 -08:00
|
|
|
MixRowSamples(dec->ChannelMix, dec->Matrix.Dual[chan][LF_BAND],
|
2018-01-14 08:51:03 -08:00
|
|
|
dec->SamplesLF, dec->NumChannels, 0, SamplesToDo
|
2016-07-17 00:29:02 -07:00
|
|
|
);
|
2016-03-26 17:33:49 -07:00
|
|
|
|
2017-02-19 22:47:59 -08:00
|
|
|
for(i = 0;i < SamplesToDo;i++)
|
2016-03-26 18:06:36 -07:00
|
|
|
OutBuffer[chan][i] += dec->ChannelMix[i];
|
2016-03-15 05:08:05 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
for(chan = 0;chan < OutChannels;chan++)
|
2016-03-26 17:33:49 -07:00
|
|
|
{
|
2018-02-12 20:33:31 -08:00
|
|
|
if(!(dec->Enabled&(1<<chan)))
|
2016-03-28 14:50:18 -07:00
|
|
|
continue;
|
|
|
|
|
2016-03-26 17:33:49 -07:00
|
|
|
memset(dec->ChannelMix, 0, SamplesToDo*sizeof(ALfloat));
|
2018-01-16 12:18:59 -08:00
|
|
|
MixRowSamples(dec->ChannelMix, dec->Matrix.Single[chan], InSamples,
|
|
|
|
dec->NumChannels, 0, SamplesToDo);
|
2016-03-26 17:33:49 -07:00
|
|
|
|
2017-02-19 22:47:59 -08:00
|
|
|
for(i = 0;i < SamplesToDo;i++)
|
2016-03-26 18:06:36 -07:00
|
|
|
OutBuffer[chan][i] += dec->ChannelMix[i];
|
2016-03-26 17:33:49 -07:00
|
|
|
}
|
2016-03-15 05:08:05 -07:00
|
|
|
}
|
|
|
|
}
|
2016-03-23 12:53:36 -07:00
|
|
|
|
|
|
|
|
2017-01-16 09:37:55 -08:00
|
|
|
void bformatdec_upSample(struct BFormatDec *dec, ALfloat (*restrict OutBuffer)[BUFFERSIZE], const ALfloat (*restrict InSamples)[BUFFERSIZE], ALsizei InChannels, ALsizei SamplesToDo)
|
2016-03-23 12:53:36 -07:00
|
|
|
{
|
2017-01-24 19:03:51 -08:00
|
|
|
ALsizei i;
|
2016-04-09 08:04:56 -07:00
|
|
|
|
2017-01-24 19:03:51 -08:00
|
|
|
/* This up-sampler leverages the differences observed in dual-band second-
|
|
|
|
* and third-order decoder matrices compared to first-order. For the same
|
|
|
|
* output channel configuration, the low-frequency matrix has identical
|
|
|
|
* coefficients in the shared input channels, while the high-frequency
|
|
|
|
* matrix has extra scalars applied to the W channel and X/Y/Z channels.
|
|
|
|
* Mixing the first-order content into the higher-order stream with the
|
|
|
|
* appropriate counter-scales applied to the HF response results in the
|
|
|
|
* subsequent higher-order decode generating the same response as a first-
|
|
|
|
* order decode.
|
2016-03-23 12:53:36 -07:00
|
|
|
*/
|
2016-09-01 07:08:52 -07:00
|
|
|
for(i = 0;i < InChannels;i++)
|
2016-03-23 12:53:36 -07:00
|
|
|
{
|
2016-08-30 04:21:57 -07:00
|
|
|
/* First, split the first-order components into low and high frequency
|
|
|
|
* bands.
|
|
|
|
*/
|
2017-01-24 19:03:51 -08:00
|
|
|
bandsplit_process(&dec->UpSampler[i].XOver,
|
2018-02-12 20:58:39 -08:00
|
|
|
dec->Samples[HF_BAND], dec->Samples[LF_BAND],
|
2016-09-01 07:08:52 -07:00
|
|
|
InSamples[i], SamplesToDo
|
2016-08-30 04:21:57 -07:00
|
|
|
);
|
|
|
|
|
|
|
|
/* Now write each band to the output. */
|
2018-01-16 12:18:59 -08:00
|
|
|
MixRowSamples(OutBuffer[i], dec->UpSampler[i].Gains,
|
2018-02-12 20:58:39 -08:00
|
|
|
dec->Samples, NUM_BANDS, 0, SamplesToDo
|
2017-01-24 19:03:51 -08:00
|
|
|
);
|
2016-03-23 12:53:36 -07:00
|
|
|
}
|
|
|
|
}
|
2016-07-30 09:29:21 -07:00
|
|
|
|
|
|
|
|
2017-02-19 22:47:59 -08:00
|
|
|
#define INVALID_UPSAMPLE_INDEX INT_MAX
|
|
|
|
|
2017-01-24 19:03:51 -08:00
|
|
|
static ALsizei GetACNIndex(const BFChannelConfig *chans, ALsizei numchans, ALsizei acn)
|
|
|
|
{
|
|
|
|
ALsizei i;
|
|
|
|
for(i = 0;i < numchans;i++)
|
|
|
|
{
|
|
|
|
if(chans[i].Index == acn)
|
|
|
|
return i;
|
|
|
|
}
|
|
|
|
return INVALID_UPSAMPLE_INDEX;
|
|
|
|
}
|
|
|
|
#define GetChannelForACN(b, a) GetACNIndex((b).Ambi.Map, (b).NumChannels, (a))
|
|
|
|
|
2016-07-30 09:29:21 -07:00
|
|
|
typedef struct AmbiUpsampler {
|
2018-02-12 20:58:39 -08:00
|
|
|
alignas(16) ALfloat Samples[NUM_BANDS][BUFFERSIZE];
|
2016-07-30 09:29:21 -07:00
|
|
|
|
|
|
|
BandSplitter XOver[4];
|
|
|
|
|
2018-02-12 20:58:39 -08:00
|
|
|
ALfloat Gains[4][MAX_OUTPUT_CHANNELS][NUM_BANDS];
|
2016-07-30 09:29:21 -07:00
|
|
|
} AmbiUpsampler;
|
|
|
|
|
|
|
|
AmbiUpsampler *ambiup_alloc()
|
|
|
|
{
|
|
|
|
return al_calloc(16, sizeof(AmbiUpsampler));
|
|
|
|
}
|
|
|
|
|
2018-02-12 20:48:28 -08:00
|
|
|
void ambiup_free(struct AmbiUpsampler **ambiup)
|
2016-07-30 09:29:21 -07:00
|
|
|
{
|
2018-02-12 20:48:28 -08:00
|
|
|
if(ambiup)
|
|
|
|
{
|
|
|
|
al_free(*ambiup);
|
|
|
|
*ambiup = NULL;
|
|
|
|
}
|
2016-07-30 09:29:21 -07:00
|
|
|
}
|
|
|
|
|
2018-02-13 03:03:26 -08:00
|
|
|
void ambiup_reset(struct AmbiUpsampler *ambiup, const ALCdevice *device, ALfloat w_scale, ALfloat xyz_scale)
|
2016-07-30 09:29:21 -07:00
|
|
|
{
|
|
|
|
ALfloat ratio;
|
2018-01-11 10:55:35 -08:00
|
|
|
ALsizei i;
|
2016-07-30 09:29:21 -07:00
|
|
|
|
|
|
|
ratio = 400.0f / (ALfloat)device->Frequency;
|
|
|
|
for(i = 0;i < 4;i++)
|
|
|
|
bandsplit_init(&ambiup->XOver[i], ratio);
|
|
|
|
|
2016-08-30 04:21:57 -07:00
|
|
|
memset(ambiup->Gains, 0, sizeof(ambiup->Gains));
|
2017-01-24 19:03:51 -08:00
|
|
|
if(device->Dry.CoeffCount > 0)
|
|
|
|
{
|
|
|
|
ALfloat encgains[8][MAX_OUTPUT_CHANNELS];
|
|
|
|
ALsizei j;
|
|
|
|
size_t k;
|
|
|
|
|
2018-01-11 10:55:35 -08:00
|
|
|
for(k = 0;k < COUNTOF(Ambi3DPoints);k++)
|
2017-01-24 19:03:51 -08:00
|
|
|
{
|
|
|
|
ALfloat coeffs[MAX_AMBI_COEFFS] = { 0.0f };
|
2018-01-11 10:55:35 -08:00
|
|
|
CalcDirectionCoeffs(Ambi3DPoints[k], 0.0f, coeffs);
|
|
|
|
ComputeDryPanGains(&device->Dry, coeffs, 1.0f, encgains[k]);
|
2017-01-24 19:03:51 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Combine the matrices that do the in->virt and virt->out conversions
|
|
|
|
* so we get a single in->out conversion. NOTE: the Encoder matrix
|
|
|
|
* (encgains) and output are transposed, so the input channels line up
|
|
|
|
* with the rows and the output channels line up with the columns.
|
|
|
|
*/
|
|
|
|
for(i = 0;i < 4;i++)
|
|
|
|
{
|
|
|
|
for(j = 0;j < device->Dry.NumChannels;j++)
|
|
|
|
{
|
2018-02-12 21:24:58 -08:00
|
|
|
ALfloat gain=0.0f;
|
2017-01-24 19:03:51 -08:00
|
|
|
for(k = 0;k < COUNTOF(Ambi3DDecoder);k++)
|
2018-02-12 21:24:58 -08:00
|
|
|
gain += Ambi3DDecoder[k][i] * encgains[k][j];
|
|
|
|
ambiup->Gains[i][j][HF_BAND] = gain * Ambi3DDecoderHFScale[i];
|
|
|
|
ambiup->Gains[i][j][LF_BAND] = gain;
|
2017-01-24 19:03:51 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
for(i = 0;i < 4;i++)
|
|
|
|
{
|
|
|
|
ALsizei index = GetChannelForACN(device->Dry, i);
|
|
|
|
if(index != INVALID_UPSAMPLE_INDEX)
|
|
|
|
{
|
|
|
|
ALfloat scale = device->Dry.Ambi.Map[index].Scale;
|
2018-02-12 20:58:39 -08:00
|
|
|
ambiup->Gains[i][index][HF_BAND] = scale * ((i==0) ? w_scale : xyz_scale);
|
|
|
|
ambiup->Gains[i][index][LF_BAND] = scale;
|
2017-01-24 19:03:51 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2016-07-30 09:29:21 -07:00
|
|
|
}
|
|
|
|
|
2017-01-16 09:37:55 -08:00
|
|
|
void ambiup_process(struct AmbiUpsampler *ambiup, ALfloat (*restrict OutBuffer)[BUFFERSIZE], ALsizei OutChannels, const ALfloat (*restrict InSamples)[BUFFERSIZE], ALsizei SamplesToDo)
|
2016-07-30 09:29:21 -07:00
|
|
|
{
|
2017-01-16 09:37:55 -08:00
|
|
|
ALsizei i, j;
|
2016-07-30 09:29:21 -07:00
|
|
|
|
2016-09-01 07:08:52 -07:00
|
|
|
for(i = 0;i < 4;i++)
|
2016-07-30 09:29:21 -07:00
|
|
|
{
|
2016-09-01 07:08:52 -07:00
|
|
|
bandsplit_process(&ambiup->XOver[i],
|
2018-02-12 20:58:39 -08:00
|
|
|
ambiup->Samples[HF_BAND], ambiup->Samples[LF_BAND],
|
2016-09-01 07:08:52 -07:00
|
|
|
InSamples[i], SamplesToDo
|
2016-08-30 04:21:57 -07:00
|
|
|
);
|
2016-07-30 09:29:21 -07:00
|
|
|
|
|
|
|
for(j = 0;j < OutChannels;j++)
|
2018-01-16 12:18:59 -08:00
|
|
|
MixRowSamples(OutBuffer[j], ambiup->Gains[i][j],
|
2018-02-12 20:58:39 -08:00
|
|
|
ambiup->Samples, NUM_BANDS, 0, SamplesToDo
|
2016-10-04 16:25:43 -07:00
|
|
|
);
|
2016-07-30 09:29:21 -07:00
|
|
|
}
|
|
|
|
}
|