2010-08-03 00:21:36 -07:00
|
|
|
/**
|
|
|
|
* OpenAL cross platform audio library
|
|
|
|
* Copyright (C) 1999-2010 by authors.
|
|
|
|
* This library is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU Library General Public
|
|
|
|
* License as published by the Free Software Foundation; either
|
|
|
|
* version 2 of the License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This library 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
|
|
|
|
* Library General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU Library General Public
|
|
|
|
* License along with this library; if not, write to the
|
2014-08-18 14:11:03 +02:00
|
|
|
* Free Software Foundation, Inc.,
|
|
|
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
|
2010-08-03 00:21:36 -07:00
|
|
|
* Or go to http://www.gnu.org/copyleft/lgpl.html
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "config.h"
|
|
|
|
|
|
|
|
#include <math.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <ctype.h>
|
|
|
|
#include <assert.h>
|
|
|
|
|
|
|
|
#include "alMain.h"
|
2016-01-28 00:02:46 -08:00
|
|
|
#include "alAuxEffectSlot.h"
|
2010-08-03 00:21:36 -07:00
|
|
|
#include "alu.h"
|
2014-11-10 17:53:42 -08:00
|
|
|
#include "bool.h"
|
2016-03-15 05:08:05 -07:00
|
|
|
#include "ambdec.h"
|
|
|
|
#include "bformatdec.h"
|
2016-04-14 15:25:12 -07:00
|
|
|
#include "uhjfilter.h"
|
|
|
|
#include "bs2b.h"
|
2010-08-03 00:21:36 -07:00
|
|
|
|
|
|
|
|
2017-02-23 16:44:59 -08:00
|
|
|
extern inline void CalcAngleCoeffs(ALfloat azimuth, ALfloat elevation, ALfloat spread, ALfloat coeffs[MAX_AMBI_COEFFS]);
|
2016-01-25 06:11:51 -08:00
|
|
|
|
|
|
|
|
2017-01-16 07:45:07 -08:00
|
|
|
static const ALsizei FuMa2ACN[MAX_AMBI_COEFFS] = {
|
2015-08-28 10:58:30 -07:00
|
|
|
0, /* W */
|
|
|
|
3, /* X */
|
|
|
|
1, /* Y */
|
|
|
|
2, /* Z */
|
|
|
|
6, /* R */
|
|
|
|
7, /* S */
|
|
|
|
5, /* T */
|
|
|
|
8, /* U */
|
|
|
|
4, /* V */
|
|
|
|
12, /* K */
|
|
|
|
13, /* L */
|
|
|
|
11, /* M */
|
|
|
|
14, /* N */
|
|
|
|
10, /* O */
|
|
|
|
15, /* P */
|
|
|
|
9, /* Q */
|
|
|
|
};
|
2017-01-16 07:45:07 -08:00
|
|
|
static const ALsizei ACN2ACN[MAX_AMBI_COEFFS] = {
|
2016-07-31 07:46:38 -07:00
|
|
|
0, 1, 2, 3, 4, 5, 6, 7,
|
|
|
|
8, 9, 10, 11, 12, 13, 14, 15
|
|
|
|
};
|
2015-08-28 10:58:30 -07:00
|
|
|
|
2016-03-15 08:00:03 -07:00
|
|
|
/* NOTE: These are scale factors as applied to Ambisonics content. Decoder
|
|
|
|
* coefficients should be divided by these values to get proper N3D scalings.
|
2015-09-23 15:03:53 -07:00
|
|
|
*/
|
2016-03-15 08:00:03 -07:00
|
|
|
static const ALfloat UnitScale[MAX_AMBI_COEFFS] = {
|
|
|
|
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
|
|
|
|
};
|
|
|
|
static const ALfloat SN3D2N3DScale[MAX_AMBI_COEFFS] = {
|
|
|
|
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) */
|
|
|
|
};
|
2015-09-23 15:03:53 -07:00
|
|
|
static const ALfloat FuMa2N3DScale[MAX_AMBI_COEFFS] = {
|
2015-11-06 10:27:28 -08: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) */
|
2015-08-28 10:58:30 -07:00
|
|
|
};
|
|
|
|
|
|
|
|
|
2016-04-24 21:42:59 -07:00
|
|
|
void CalcDirectionCoeffs(const ALfloat dir[3], ALfloat spread, ALfloat coeffs[MAX_AMBI_COEFFS])
|
Use an ambisonics-based panning method
For mono sources, third-order ambisonics is utilized to generate panning gains.
The general idea is that a panned mono sound can be encoded into b-format
ambisonics as:
w[i] = sample[i] * 0.7071;
x[i] = sample[i] * dir[0];
y[i] = sample[i] * dir[1];
...
and subsequently rendered using:
output[chan][i] = w[i] * w_coeffs[chan] +
x[i] * x_coeffs[chan] +
y[i] * y_coeffs[chan] +
...;
By reordering the math, channel gains can be generated by doing:
gain[chan] = 0.7071 * w_coeffs[chan] +
dir[0] * x_coeffs[chan] +
dir[1] * y_coeffs[chan] +
...;
which then get applied as normal:
output[chan][i] = sample[i] * gain[chan];
One of the reasons to use ambisonics for panning is that it provides arguably
better reproduction for sounds emanating from between two speakers. As well,
this makes it easier to pan in all 3 dimensions, with for instance a "3D7.1" or
8-channel cube speaker configuration by simply providing the necessary
coefficients (this will need some work since some methods still use angle-based
panpot, particularly multi-channel sources).
Unfortunately, the math to reliably generate the coefficients for a given
speaker configuration is too costly to do at run-time. They have to be pre-
generated based on a pre-specified speaker arangement, which means the config
options for tweaking speaker angles are no longer supportable. Eventually I
hope to provide config options for custom coefficients, which can either be
generated and written in manually, or via alsoft-config from user-specified
speaker positions.
The current default set of coefficients were generated using the MATLAB scripts
(compatible with GNU Octave) from the excellent Ambisonic Decoder Toolbox, at
https://bitbucket.org/ambidecodertoolbox/adt/
2014-09-30 07:33:13 -07:00
|
|
|
{
|
|
|
|
/* Convert from OpenAL coords to Ambisonics. */
|
2014-10-02 20:25:30 -07:00
|
|
|
ALfloat x = -dir[2];
|
|
|
|
ALfloat y = -dir[0];
|
|
|
|
ALfloat z = dir[1];
|
|
|
|
|
2015-08-28 10:58:30 -07:00
|
|
|
/* Zeroth-order */
|
2015-09-23 15:03:53 -07:00
|
|
|
coeffs[0] = 1.0f; /* ACN 0 = 1 */
|
2015-08-28 10:58:30 -07:00
|
|
|
/* First-order */
|
2015-11-06 10:27:28 -08:00
|
|
|
coeffs[1] = 1.732050808f * y; /* ACN 1 = sqrt(3) * Y */
|
|
|
|
coeffs[2] = 1.732050808f * z; /* ACN 2 = sqrt(3) * Z */
|
|
|
|
coeffs[3] = 1.732050808f * x; /* ACN 3 = sqrt(3) * X */
|
2015-08-28 10:58:30 -07:00
|
|
|
/* Second-order */
|
2015-11-06 10:27:28 -08:00
|
|
|
coeffs[4] = 3.872983346f * x * y; /* ACN 4 = sqrt(15) * X * Y */
|
|
|
|
coeffs[5] = 3.872983346f * y * z; /* ACN 5 = sqrt(15) * Y * Z */
|
|
|
|
coeffs[6] = 1.118033989f * (3.0f*z*z - 1.0f); /* ACN 6 = sqrt(5)/2 * (3*Z*Z - 1) */
|
|
|
|
coeffs[7] = 3.872983346f * x * z; /* ACN 7 = sqrt(15) * X * Z */
|
|
|
|
coeffs[8] = 1.936491673f * (x*x - y*y); /* ACN 8 = sqrt(15)/2 * (X*X - Y*Y) */
|
2015-08-28 10:58:30 -07:00
|
|
|
/* Third-order */
|
2015-11-06 10:27:28 -08:00
|
|
|
coeffs[9] = 2.091650066f * y * (3.0f*x*x - y*y); /* ACN 9 = sqrt(35/8) * Y * (3*X*X - Y*Y) */
|
|
|
|
coeffs[10] = 10.246950766f * z * x * y; /* ACN 10 = sqrt(105) * Z * X * Y */
|
|
|
|
coeffs[11] = 1.620185175f * y * (5.0f*z*z - 1.0f); /* ACN 11 = sqrt(21/8) * Y * (5*Z*Z - 1) */
|
|
|
|
coeffs[12] = 1.322875656f * z * (5.0f*z*z - 3.0f); /* ACN 12 = sqrt(7)/2 * Z * (5*Z*Z - 3) */
|
|
|
|
coeffs[13] = 1.620185175f * x * (5.0f*z*z - 1.0f); /* ACN 13 = sqrt(21/8) * X * (5*Z*Z - 1) */
|
|
|
|
coeffs[14] = 5.123475383f * z * (x*x - y*y); /* ACN 14 = sqrt(105)/2 * Z * (X*X - Y*Y) */
|
|
|
|
coeffs[15] = 2.091650066f * x * (x*x - 3.0f*y*y); /* ACN 15 = sqrt(35/8) * X * (X*X - 3*Y*Y) */
|
2016-04-24 21:42:59 -07:00
|
|
|
|
|
|
|
if(spread > 0.0f)
|
|
|
|
{
|
|
|
|
/* Implement the spread by using a spherical source that subtends the
|
|
|
|
* angle spread. See:
|
|
|
|
* http://www.ppsloan.org/publications/StupidSH36.pdf - Appendix A3
|
|
|
|
*
|
2016-07-10 22:00:27 -07:00
|
|
|
* When adjusted for N3D normalization instead of SN3D, these
|
|
|
|
* calculations are:
|
|
|
|
*
|
|
|
|
* ZH0 = -sqrt(pi) * (-1+ca);
|
|
|
|
* ZH1 = 0.5*sqrt(pi) * sa*sa;
|
|
|
|
* ZH2 = -0.5*sqrt(pi) * ca*(-1+ca)*(ca+1);
|
|
|
|
* ZH3 = -0.125*sqrt(pi) * (-1+ca)*(ca+1)*(5*ca*ca - 1);
|
|
|
|
* ZH4 = -0.125*sqrt(pi) * ca*(-1+ca)*(ca+1)*(7*ca*ca - 3);
|
|
|
|
* ZH5 = -0.0625*sqrt(pi) * (-1+ca)*(ca+1)*(21*ca*ca*ca*ca - 14*ca*ca + 1);
|
2016-04-24 21:42:59 -07:00
|
|
|
*
|
2016-07-10 22:00:27 -07:00
|
|
|
* The gain of the source is compensated for size, so that the
|
|
|
|
* loundness doesn't depend on the spread. That is, the factors are
|
|
|
|
* scaled so that ZH0 remains 1 regardless of the spread. Thus:
|
2016-04-24 21:42:59 -07:00
|
|
|
*
|
2016-07-10 22:00:27 -07:00
|
|
|
* ZH0 = 1.0f;
|
|
|
|
* ZH1 = 0.5f * (ca+1.0f);
|
|
|
|
* ZH2 = 0.5f * (ca+1.0f)*ca;
|
|
|
|
* ZH3 = 0.125f * (ca+1.0f)*(5.0f*ca*ca - 1.0f);
|
|
|
|
* ZH4 = 0.125f * (ca+1.0f)*(7.0f*ca*ca - 3.0f)*ca;
|
|
|
|
* ZH5 = 0.0625f * (ca+1.0f)*(21.0f*ca*ca*ca*ca - 14.0f*ca*ca + 1.0f);
|
2016-04-24 21:42:59 -07:00
|
|
|
*/
|
|
|
|
ALfloat ca = cosf(spread * 0.5f);
|
|
|
|
|
|
|
|
ALfloat ZH0_norm = 1.0f;
|
|
|
|
ALfloat ZH1_norm = 0.5f * (ca+1.f);
|
|
|
|
ALfloat ZH2_norm = 0.5f * (ca+1.f)*ca;
|
|
|
|
ALfloat ZH3_norm = 0.125f * (ca+1.f)*(5.f*ca*ca-1.f);
|
|
|
|
|
|
|
|
/* Zeroth-order */
|
|
|
|
coeffs[0] *= ZH0_norm;
|
|
|
|
/* First-order */
|
|
|
|
coeffs[1] *= ZH1_norm;
|
|
|
|
coeffs[2] *= ZH1_norm;
|
|
|
|
coeffs[3] *= ZH1_norm;
|
|
|
|
/* Second-order */
|
|
|
|
coeffs[4] *= ZH2_norm;
|
|
|
|
coeffs[5] *= ZH2_norm;
|
|
|
|
coeffs[6] *= ZH2_norm;
|
|
|
|
coeffs[7] *= ZH2_norm;
|
|
|
|
coeffs[8] *= ZH2_norm;
|
|
|
|
/* Third-order */
|
|
|
|
coeffs[9] *= ZH3_norm;
|
|
|
|
coeffs[10] *= ZH3_norm;
|
|
|
|
coeffs[11] *= ZH3_norm;
|
|
|
|
coeffs[12] *= ZH3_norm;
|
|
|
|
coeffs[13] *= ZH3_norm;
|
|
|
|
coeffs[14] *= ZH3_norm;
|
|
|
|
coeffs[15] *= ZH3_norm;
|
|
|
|
}
|
2016-01-25 06:11:51 -08:00
|
|
|
}
|
|
|
|
|
2017-02-23 00:23:16 -08:00
|
|
|
void CalcAnglePairwiseCoeffs(ALfloat azimuth, ALfloat elevation, ALfloat spread, ALfloat coeffs[MAX_AMBI_COEFFS])
|
|
|
|
{
|
|
|
|
ALfloat sign = (azimuth < 0.0f) ? -1.0f : 1.0f;
|
|
|
|
if(!(fabsf(azimuth) > F_PI_2))
|
|
|
|
azimuth = minf(fabsf(azimuth) * F_PI_2 / (F_PI/6.0f), F_PI_2) * sign;
|
|
|
|
CalcAngleCoeffs(azimuth, elevation, spread, coeffs);
|
|
|
|
}
|
|
|
|
|
2016-01-25 06:11:51 -08:00
|
|
|
|
2017-01-18 07:13:23 -08:00
|
|
|
void ComputeAmbientGainsMC(const ChannelConfig *chancoeffs, ALsizei numchans, ALfloat ingain, ALfloat gains[MAX_OUTPUT_CHANNELS])
|
2016-01-25 06:11:51 -08:00
|
|
|
{
|
2017-01-18 07:13:23 -08:00
|
|
|
ALsizei i;
|
2016-01-25 06:11:51 -08:00
|
|
|
|
|
|
|
for(i = 0;i < numchans;i++)
|
2017-01-21 11:54:22 -08:00
|
|
|
gains[i] = chancoeffs[i][0] * 1.414213562f * ingain;
|
2016-01-25 06:11:51 -08:00
|
|
|
for(;i < MAX_OUTPUT_CHANNELS;i++)
|
|
|
|
gains[i] = 0.0f;
|
|
|
|
}
|
|
|
|
|
2017-01-18 07:13:23 -08:00
|
|
|
void ComputeAmbientGainsBF(const BFChannelConfig *chanmap, ALsizei numchans, ALfloat ingain, ALfloat gains[MAX_OUTPUT_CHANNELS])
|
2016-04-15 22:05:47 -07:00
|
|
|
{
|
|
|
|
ALfloat gain = 0.0f;
|
2017-01-18 07:13:23 -08:00
|
|
|
ALsizei i;
|
2016-04-15 22:05:47 -07:00
|
|
|
|
|
|
|
for(i = 0;i < numchans;i++)
|
|
|
|
{
|
|
|
|
if(chanmap[i].Index == 0)
|
|
|
|
gain += chanmap[i].Scale;
|
|
|
|
}
|
|
|
|
gains[0] = gain * 1.414213562f * ingain;
|
|
|
|
for(i = 1;i < MAX_OUTPUT_CHANNELS;i++)
|
|
|
|
gains[i] = 0.0f;
|
|
|
|
}
|
|
|
|
|
2017-01-18 07:13:23 -08:00
|
|
|
void ComputePanningGainsMC(const ChannelConfig *chancoeffs, ALsizei numchans, ALsizei numcoeffs, const ALfloat coeffs[MAX_AMBI_COEFFS], ALfloat ingain, ALfloat gains[MAX_OUTPUT_CHANNELS])
|
2016-01-25 06:11:51 -08:00
|
|
|
{
|
2017-01-18 07:13:23 -08:00
|
|
|
ALsizei i, j;
|
Use an ambisonics-based panning method
For mono sources, third-order ambisonics is utilized to generate panning gains.
The general idea is that a panned mono sound can be encoded into b-format
ambisonics as:
w[i] = sample[i] * 0.7071;
x[i] = sample[i] * dir[0];
y[i] = sample[i] * dir[1];
...
and subsequently rendered using:
output[chan][i] = w[i] * w_coeffs[chan] +
x[i] * x_coeffs[chan] +
y[i] * y_coeffs[chan] +
...;
By reordering the math, channel gains can be generated by doing:
gain[chan] = 0.7071 * w_coeffs[chan] +
dir[0] * x_coeffs[chan] +
dir[1] * y_coeffs[chan] +
...;
which then get applied as normal:
output[chan][i] = sample[i] * gain[chan];
One of the reasons to use ambisonics for panning is that it provides arguably
better reproduction for sounds emanating from between two speakers. As well,
this makes it easier to pan in all 3 dimensions, with for instance a "3D7.1" or
8-channel cube speaker configuration by simply providing the necessary
coefficients (this will need some work since some methods still use angle-based
panpot, particularly multi-channel sources).
Unfortunately, the math to reliably generate the coefficients for a given
speaker configuration is too costly to do at run-time. They have to be pre-
generated based on a pre-specified speaker arangement, which means the config
options for tweaking speaker angles are no longer supportable. Eventually I
hope to provide config options for custom coefficients, which can either be
generated and written in manually, or via alsoft-config from user-specified
speaker positions.
The current default set of coefficients were generated using the MATLAB scripts
(compatible with GNU Octave) from the excellent Ambisonic Decoder Toolbox, at
https://bitbucket.org/ambidecodertoolbox/adt/
2014-09-30 07:33:13 -07:00
|
|
|
|
2016-01-25 06:11:51 -08:00
|
|
|
for(i = 0;i < numchans;i++)
|
Use an ambisonics-based panning method
For mono sources, third-order ambisonics is utilized to generate panning gains.
The general idea is that a panned mono sound can be encoded into b-format
ambisonics as:
w[i] = sample[i] * 0.7071;
x[i] = sample[i] * dir[0];
y[i] = sample[i] * dir[1];
...
and subsequently rendered using:
output[chan][i] = w[i] * w_coeffs[chan] +
x[i] * x_coeffs[chan] +
y[i] * y_coeffs[chan] +
...;
By reordering the math, channel gains can be generated by doing:
gain[chan] = 0.7071 * w_coeffs[chan] +
dir[0] * x_coeffs[chan] +
dir[1] * y_coeffs[chan] +
...;
which then get applied as normal:
output[chan][i] = sample[i] * gain[chan];
One of the reasons to use ambisonics for panning is that it provides arguably
better reproduction for sounds emanating from between two speakers. As well,
this makes it easier to pan in all 3 dimensions, with for instance a "3D7.1" or
8-channel cube speaker configuration by simply providing the necessary
coefficients (this will need some work since some methods still use angle-based
panpot, particularly multi-channel sources).
Unfortunately, the math to reliably generate the coefficients for a given
speaker configuration is too costly to do at run-time. They have to be pre-
generated based on a pre-specified speaker arangement, which means the config
options for tweaking speaker angles are no longer supportable. Eventually I
hope to provide config options for custom coefficients, which can either be
generated and written in manually, or via alsoft-config from user-specified
speaker positions.
The current default set of coefficients were generated using the MATLAB scripts
(compatible with GNU Octave) from the excellent Ambisonic Decoder Toolbox, at
https://bitbucket.org/ambidecodertoolbox/adt/
2014-09-30 07:33:13 -07:00
|
|
|
{
|
2015-08-22 07:23:43 -07:00
|
|
|
float gain = 0.0f;
|
2016-04-15 17:31:04 -07:00
|
|
|
for(j = 0;j < numcoeffs;j++)
|
2016-01-25 06:11:51 -08:00
|
|
|
gain += chancoeffs[i][j]*coeffs[j];
|
2017-02-23 00:23:16 -08:00
|
|
|
gains[i] = clampf(gain, 0.0f, 1.0f) * ingain;
|
2014-11-15 04:11:22 -08:00
|
|
|
}
|
2015-09-24 12:36:02 -07:00
|
|
|
for(;i < MAX_OUTPUT_CHANNELS;i++)
|
|
|
|
gains[i] = 0.0f;
|
2014-11-04 03:33:35 -08:00
|
|
|
}
|
|
|
|
|
2017-01-18 07:13:23 -08:00
|
|
|
void ComputePanningGainsBF(const BFChannelConfig *chanmap, ALsizei numchans, const ALfloat coeffs[MAX_AMBI_COEFFS], ALfloat ingain, ALfloat gains[MAX_OUTPUT_CHANNELS])
|
2016-04-14 21:50:36 -07:00
|
|
|
{
|
2017-01-18 07:13:23 -08:00
|
|
|
ALsizei i;
|
2016-04-14 21:50:36 -07:00
|
|
|
|
|
|
|
for(i = 0;i < numchans;i++)
|
|
|
|
gains[i] = chanmap[i].Scale * coeffs[chanmap[i].Index] * ingain;
|
|
|
|
for(;i < MAX_OUTPUT_CHANNELS;i++)
|
|
|
|
gains[i] = 0.0f;
|
|
|
|
}
|
|
|
|
|
2017-01-18 07:13:23 -08:00
|
|
|
void ComputeFirstOrderGainsMC(const ChannelConfig *chancoeffs, ALsizei numchans, const ALfloat mtx[4], ALfloat ingain, ALfloat gains[MAX_OUTPUT_CHANNELS])
|
2014-10-31 17:18:45 -07:00
|
|
|
{
|
2017-01-18 07:13:23 -08:00
|
|
|
ALsizei i, j;
|
2014-10-31 17:18:45 -07:00
|
|
|
|
2016-01-25 06:11:51 -08:00
|
|
|
for(i = 0;i < numchans;i++)
|
2014-10-31 17:18:45 -07:00
|
|
|
{
|
2015-09-24 12:36:02 -07:00
|
|
|
float gain = 0.0f;
|
2015-08-28 10:58:30 -07:00
|
|
|
for(j = 0;j < 4;j++)
|
2016-01-25 06:11:51 -08:00
|
|
|
gain += chancoeffs[i][j] * mtx[j];
|
2017-02-23 00:23:16 -08:00
|
|
|
gains[i] = clampf(gain, 0.0f, 1.0f) * ingain;
|
2014-10-31 17:18:45 -07:00
|
|
|
}
|
2015-09-24 12:36:02 -07:00
|
|
|
for(;i < MAX_OUTPUT_CHANNELS;i++)
|
|
|
|
gains[i] = 0.0f;
|
2014-10-31 17:18:45 -07:00
|
|
|
}
|
|
|
|
|
2017-01-18 07:13:23 -08:00
|
|
|
void ComputeFirstOrderGainsBF(const BFChannelConfig *chanmap, ALsizei numchans, const ALfloat mtx[4], ALfloat ingain, ALfloat gains[MAX_OUTPUT_CHANNELS])
|
2016-04-14 21:50:36 -07:00
|
|
|
{
|
2017-01-18 07:13:23 -08:00
|
|
|
ALsizei i;
|
2016-04-14 21:50:36 -07:00
|
|
|
|
|
|
|
for(i = 0;i < numchans;i++)
|
|
|
|
gains[i] = chanmap[i].Scale * mtx[chanmap[i].Index] * ingain;
|
|
|
|
for(;i < MAX_OUTPUT_CHANNELS;i++)
|
|
|
|
gains[i] = 0.0f;
|
|
|
|
}
|
|
|
|
|
2012-04-29 05:04:46 -07:00
|
|
|
|
2016-09-06 09:09:25 -07:00
|
|
|
static inline const char *GetLabelFromChannel(enum Channel channel)
|
2014-11-15 00:19:56 -08:00
|
|
|
{
|
|
|
|
switch(channel)
|
|
|
|
{
|
|
|
|
case FrontLeft: return "front-left";
|
|
|
|
case FrontRight: return "front-right";
|
|
|
|
case FrontCenter: return "front-center";
|
|
|
|
case LFE: return "lfe";
|
|
|
|
case BackLeft: return "back-left";
|
|
|
|
case BackRight: return "back-right";
|
|
|
|
case BackCenter: return "back-center";
|
|
|
|
case SideLeft: return "side-left";
|
|
|
|
case SideRight: return "side-right";
|
2014-11-22 04:20:17 -08:00
|
|
|
|
2016-02-20 00:53:01 -08:00
|
|
|
case UpperFrontLeft: return "upper-front-left";
|
|
|
|
case UpperFrontRight: return "upper-front-right";
|
|
|
|
case UpperBackLeft: return "upper-back-left";
|
|
|
|
case UpperBackRight: return "upper-back-right";
|
|
|
|
case LowerFrontLeft: return "lower-front-left";
|
|
|
|
case LowerFrontRight: return "lower-front-right";
|
|
|
|
case LowerBackLeft: return "lower-back-left";
|
|
|
|
case LowerBackRight: return "lower-back-right";
|
|
|
|
|
2016-03-16 06:49:35 -07:00
|
|
|
case Aux0: return "aux-0";
|
|
|
|
case Aux1: return "aux-1";
|
|
|
|
case Aux2: return "aux-2";
|
|
|
|
case Aux3: return "aux-3";
|
2016-03-23 10:39:14 -07:00
|
|
|
case Aux4: return "aux-4";
|
|
|
|
case Aux5: return "aux-5";
|
|
|
|
case Aux6: return "aux-6";
|
|
|
|
case Aux7: return "aux-7";
|
|
|
|
case Aux8: return "aux-8";
|
2016-04-19 18:58:19 -07:00
|
|
|
case Aux9: return "aux-9";
|
|
|
|
case Aux10: return "aux-10";
|
|
|
|
case Aux11: return "aux-11";
|
|
|
|
case Aux12: return "aux-12";
|
|
|
|
case Aux13: return "aux-13";
|
|
|
|
case Aux14: return "aux-14";
|
|
|
|
case Aux15: return "aux-15";
|
2014-11-25 22:20:00 -08:00
|
|
|
|
2014-11-15 00:19:56 -08:00
|
|
|
case InvalidChannel: break;
|
|
|
|
}
|
|
|
|
return "(unknown)";
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-11-05 03:46:00 -08:00
|
|
|
typedef struct ChannelMap {
|
|
|
|
enum Channel ChanName;
|
|
|
|
ChannelConfig Config;
|
|
|
|
} ChannelMap;
|
|
|
|
|
2016-01-27 04:44:21 -08:00
|
|
|
static void SetChannelMap(const enum Channel *devchans, ChannelConfig *ambicoeffs,
|
2017-01-21 11:05:05 -08:00
|
|
|
const ChannelMap *chanmap, size_t count, ALsizei *outcount)
|
2014-11-12 19:26:53 -08:00
|
|
|
{
|
2015-11-06 02:21:24 -08:00
|
|
|
size_t j, k;
|
2017-01-18 07:13:23 -08:00
|
|
|
ALsizei i;
|
2014-11-12 19:26:53 -08:00
|
|
|
|
2016-01-27 04:44:21 -08:00
|
|
|
for(i = 0;i < MAX_OUTPUT_CHANNELS && devchans[i] != InvalidChannel;i++)
|
2014-11-12 19:26:53 -08:00
|
|
|
{
|
2016-01-27 04:44:21 -08:00
|
|
|
if(devchans[i] == LFE)
|
2014-11-12 19:26:53 -08:00
|
|
|
{
|
|
|
|
for(j = 0;j < MAX_AMBI_COEFFS;j++)
|
2016-01-27 04:44:21 -08:00
|
|
|
ambicoeffs[i][j] = 0.0f;
|
2014-11-12 19:26:53 -08:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
for(j = 0;j < count;j++)
|
|
|
|
{
|
2016-01-27 04:44:21 -08:00
|
|
|
if(devchans[i] != chanmap[j].ChanName)
|
|
|
|
continue;
|
|
|
|
|
2016-08-02 09:16:12 -07:00
|
|
|
for(k = 0;k < MAX_AMBI_COEFFS;++k)
|
2017-01-21 11:05:05 -08:00
|
|
|
ambicoeffs[i][k] = chanmap[j].Config[k];
|
2016-01-27 04:44:21 -08:00
|
|
|
break;
|
2014-11-12 19:26:53 -08:00
|
|
|
}
|
|
|
|
if(j == count)
|
2016-01-27 04:44:21 -08:00
|
|
|
ERR("Failed to match %s channel (%u) in channel map\n", GetLabelFromChannel(devchans[i]), i);
|
2014-11-12 19:26:53 -08:00
|
|
|
}
|
2016-01-27 04:44:21 -08:00
|
|
|
*outcount = i;
|
2014-11-12 19:26:53 -08:00
|
|
|
}
|
|
|
|
|
2017-01-29 16:42:02 -08:00
|
|
|
static bool MakeSpeakerMap(ALCdevice *device, const AmbDecConf *conf, ALsizei speakermap[MAX_OUTPUT_CHANNELS])
|
2016-03-15 05:08:05 -07:00
|
|
|
{
|
2017-01-16 09:37:55 -08:00
|
|
|
ALsizei i;
|
2016-03-15 05:08:05 -07:00
|
|
|
|
|
|
|
for(i = 0;i < conf->NumSpeakers;i++)
|
|
|
|
{
|
|
|
|
int c = -1;
|
|
|
|
|
|
|
|
/* NOTE: AmbDec does not define any standard speaker names, however
|
|
|
|
* for this to work we have to by able to find the output channel
|
|
|
|
* the speaker definition corresponds to. Therefore, OpenAL Soft
|
|
|
|
* requires these channel labels to be recognized:
|
|
|
|
*
|
|
|
|
* LF = Front left
|
|
|
|
* RF = Front right
|
|
|
|
* LS = Side left
|
|
|
|
* RS = Side right
|
|
|
|
* LB = Back left
|
|
|
|
* RB = Back right
|
|
|
|
* CE = Front center
|
|
|
|
* CB = Back center
|
|
|
|
*
|
|
|
|
* Additionally, surround51 will acknowledge back speakers for side
|
|
|
|
* channels, and surround51rear will acknowledge side speakers for
|
|
|
|
* back channels, to avoid issues with an ambdec expecting 5.1 to
|
|
|
|
* use the side channels when the device is configured for back,
|
|
|
|
* and vice-versa.
|
|
|
|
*/
|
2017-04-04 06:58:53 -07:00
|
|
|
if(alstr_cmp_cstr(conf->Speakers[i].Name, "LF") == 0)
|
2016-03-15 05:08:05 -07:00
|
|
|
c = GetChannelIdxByName(device->RealOut, FrontLeft);
|
2017-04-04 06:58:53 -07:00
|
|
|
else if(alstr_cmp_cstr(conf->Speakers[i].Name, "RF") == 0)
|
2016-03-15 05:08:05 -07:00
|
|
|
c = GetChannelIdxByName(device->RealOut, FrontRight);
|
2017-04-04 06:58:53 -07:00
|
|
|
else if(alstr_cmp_cstr(conf->Speakers[i].Name, "CE") == 0)
|
2016-03-15 05:08:05 -07:00
|
|
|
c = GetChannelIdxByName(device->RealOut, FrontCenter);
|
2017-04-04 06:58:53 -07:00
|
|
|
else if(alstr_cmp_cstr(conf->Speakers[i].Name, "LS") == 0)
|
2016-03-15 05:08:05 -07:00
|
|
|
{
|
|
|
|
if(device->FmtChans == DevFmtX51Rear)
|
|
|
|
c = GetChannelIdxByName(device->RealOut, BackLeft);
|
|
|
|
else
|
|
|
|
c = GetChannelIdxByName(device->RealOut, SideLeft);
|
|
|
|
}
|
2017-04-04 06:58:53 -07:00
|
|
|
else if(alstr_cmp_cstr(conf->Speakers[i].Name, "RS") == 0)
|
2016-03-15 05:08:05 -07:00
|
|
|
{
|
|
|
|
if(device->FmtChans == DevFmtX51Rear)
|
|
|
|
c = GetChannelIdxByName(device->RealOut, BackRight);
|
|
|
|
else
|
|
|
|
c = GetChannelIdxByName(device->RealOut, SideRight);
|
|
|
|
}
|
2017-04-04 06:58:53 -07:00
|
|
|
else if(alstr_cmp_cstr(conf->Speakers[i].Name, "LB") == 0)
|
2016-03-15 05:08:05 -07:00
|
|
|
{
|
|
|
|
if(device->FmtChans == DevFmtX51)
|
|
|
|
c = GetChannelIdxByName(device->RealOut, SideLeft);
|
|
|
|
else
|
|
|
|
c = GetChannelIdxByName(device->RealOut, BackLeft);
|
|
|
|
}
|
2017-04-04 06:58:53 -07:00
|
|
|
else if(alstr_cmp_cstr(conf->Speakers[i].Name, "RB") == 0)
|
2016-03-15 05:08:05 -07:00
|
|
|
{
|
|
|
|
if(device->FmtChans == DevFmtX51)
|
|
|
|
c = GetChannelIdxByName(device->RealOut, SideRight);
|
|
|
|
else
|
|
|
|
c = GetChannelIdxByName(device->RealOut, BackRight);
|
|
|
|
}
|
2017-04-04 06:58:53 -07:00
|
|
|
else if(alstr_cmp_cstr(conf->Speakers[i].Name, "CB") == 0)
|
2016-03-15 05:08:05 -07:00
|
|
|
c = GetChannelIdxByName(device->RealOut, BackCenter);
|
|
|
|
else
|
|
|
|
{
|
2017-04-04 06:58:53 -07:00
|
|
|
const char *name = alstr_get_cstr(conf->Speakers[i].Name);
|
2016-05-12 23:41:23 -07:00
|
|
|
unsigned int n;
|
|
|
|
char ch;
|
|
|
|
|
|
|
|
if(sscanf(name, "AUX%u%c", &n, &ch) == 1 && n < 16)
|
|
|
|
c = GetChannelIdxByName(device->RealOut, Aux0+n);
|
|
|
|
else
|
|
|
|
{
|
|
|
|
ERR("AmbDec speaker label \"%s\" not recognized\n", name);
|
|
|
|
return false;
|
|
|
|
}
|
2016-03-15 05:08:05 -07:00
|
|
|
}
|
|
|
|
if(c == -1)
|
|
|
|
{
|
|
|
|
ERR("Failed to lookup AmbDec speaker label %s\n",
|
2017-04-04 06:58:53 -07:00
|
|
|
alstr_get_cstr(conf->Speakers[i].Name));
|
2016-03-15 05:08:05 -07:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
speakermap[i] = c;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2016-04-14 10:44:57 -07:00
|
|
|
|
|
|
|
static const ChannelMap MonoCfg[1] = {
|
2017-01-21 11:05:05 -08:00
|
|
|
{ FrontCenter, { 1.0f } },
|
2016-04-14 10:44:57 -07:00
|
|
|
}, StereoCfg[2] = {
|
2017-02-23 00:23:16 -08:00
|
|
|
{ FrontLeft, { 5.00000000e-1f, 2.88675135e-1f, 0.0f, 1.19573156e-1f } },
|
|
|
|
{ FrontRight, { 5.00000000e-1f, -2.88675135e-1f, 0.0f, 1.19573156e-1f } },
|
2016-04-14 10:44:57 -07:00
|
|
|
}, QuadCfg[4] = {
|
2017-01-21 11:05:05 -08:00
|
|
|
{ BackLeft, { 3.53553391e-1f, 2.04124145e-1f, 0.0f, -2.04124145e-1f } },
|
|
|
|
{ FrontLeft, { 3.53553391e-1f, 2.04124145e-1f, 0.0f, 2.04124145e-1f } },
|
|
|
|
{ FrontRight, { 3.53553391e-1f, -2.04124145e-1f, 0.0f, 2.04124145e-1f } },
|
|
|
|
{ BackRight, { 3.53553391e-1f, -2.04124145e-1f, 0.0f, -2.04124145e-1f } },
|
2016-04-14 10:44:57 -07:00
|
|
|
}, X51SideCfg[5] = {
|
2017-01-21 11:05:05 -08:00
|
|
|
{ SideLeft, { 3.33001372e-1f, 1.89085671e-1f, 0.0f, -2.00041334e-1f, -2.12309737e-2f, 0.0f, 0.0f, 0.0f, -1.14573483e-2f } },
|
|
|
|
{ FrontLeft, { 1.47751298e-1f, 1.28994110e-1f, 0.0f, 1.15190495e-1f, 7.44949143e-2f, 0.0f, 0.0f, 0.0f, -6.47739980e-3f } },
|
|
|
|
{ FrontCenter, { 7.73595729e-2f, 0.00000000e+0f, 0.0f, 9.71390298e-2f, 0.00000000e+0f, 0.0f, 0.0f, 0.0f, 5.18625335e-2f } },
|
|
|
|
{ FrontRight, { 1.47751298e-1f, -1.28994110e-1f, 0.0f, 1.15190495e-1f, -7.44949143e-2f, 0.0f, 0.0f, 0.0f, -6.47739980e-3f } },
|
|
|
|
{ SideRight, { 3.33001372e-1f, -1.89085671e-1f, 0.0f, -2.00041334e-1f, 2.12309737e-2f, 0.0f, 0.0f, 0.0f, -1.14573483e-2f } },
|
2016-04-14 10:44:57 -07:00
|
|
|
}, X51RearCfg[5] = {
|
2017-01-21 11:05:05 -08:00
|
|
|
{ BackLeft, { 3.33001372e-1f, 1.89085671e-1f, 0.0f, -2.00041334e-1f, -2.12309737e-2f, 0.0f, 0.0f, 0.0f, -1.14573483e-2f } },
|
|
|
|
{ FrontLeft, { 1.47751298e-1f, 1.28994110e-1f, 0.0f, 1.15190495e-1f, 7.44949143e-2f, 0.0f, 0.0f, 0.0f, -6.47739980e-3f } },
|
|
|
|
{ FrontCenter, { 7.73595729e-2f, 0.00000000e+0f, 0.0f, 9.71390298e-2f, 0.00000000e+0f, 0.0f, 0.0f, 0.0f, 5.18625335e-2f } },
|
|
|
|
{ FrontRight, { 1.47751298e-1f, -1.28994110e-1f, 0.0f, 1.15190495e-1f, -7.44949143e-2f, 0.0f, 0.0f, 0.0f, -6.47739980e-3f } },
|
|
|
|
{ BackRight, { 3.33001372e-1f, -1.89085671e-1f, 0.0f, -2.00041334e-1f, 2.12309737e-2f, 0.0f, 0.0f, 0.0f, -1.14573483e-2f } },
|
2016-04-14 10:44:57 -07:00
|
|
|
}, X61Cfg[6] = {
|
2017-01-21 11:05:05 -08:00
|
|
|
{ SideLeft, { 2.04462744e-1f, 2.17178497e-1f, 0.0f, -4.39990188e-2f, -2.60787329e-2f, 0.0f, 0.0f, 0.0f, -6.87238843e-2f } },
|
|
|
|
{ FrontLeft, { 1.18130342e-1f, 9.34633906e-2f, 0.0f, 1.08553749e-1f, 6.80658795e-2f, 0.0f, 0.0f, 0.0f, 1.08999485e-2f } },
|
|
|
|
{ FrontCenter, { 7.73595729e-2f, 0.00000000e+0f, 0.0f, 9.71390298e-2f, 0.00000000e+0f, 0.0f, 0.0f, 0.0f, 5.18625335e-2f } },
|
|
|
|
{ FrontRight, { 1.18130342e-1f, -9.34633906e-2f, 0.0f, 1.08553749e-1f, -6.80658795e-2f, 0.0f, 0.0f, 0.0f, 1.08999485e-2f } },
|
|
|
|
{ SideRight, { 2.04462744e-1f, -2.17178497e-1f, 0.0f, -4.39990188e-2f, 2.60787329e-2f, 0.0f, 0.0f, 0.0f, -6.87238843e-2f } },
|
|
|
|
{ BackCenter, { 2.50001688e-1f, 0.00000000e+0f, 0.0f, -2.50000094e-1f, 0.00000000e+0f, 0.0f, 0.0f, 0.0f, 6.05133395e-2f } },
|
|
|
|
}, X71Cfg[6] = {
|
|
|
|
{ BackLeft, { 2.04124145e-1f, 1.08880247e-1f, 0.0f, -1.88586120e-1f, -1.29099444e-1f, 0.0f, 0.0f, 0.0f, 7.45355993e-2f, 3.73460789e-2f, 0.0f, 0.0f, 0.0f, 0.0f, 0.0f, 0.00000000e+0f } },
|
|
|
|
{ SideLeft, { 2.04124145e-1f, 2.17760495e-1f, 0.0f, 0.00000000e+0f, 0.00000000e+0f, 0.0f, 0.0f, 0.0f, -1.49071198e-1f, -3.73460789e-2f, 0.0f, 0.0f, 0.0f, 0.0f, 0.0f, 0.00000000e+0f } },
|
|
|
|
{ FrontLeft, { 2.04124145e-1f, 1.08880247e-1f, 0.0f, 1.88586120e-1f, 1.29099444e-1f, 0.0f, 0.0f, 0.0f, 7.45355993e-2f, 3.73460789e-2f, 0.0f, 0.0f, 0.0f, 0.0f, 0.0f, 0.00000000e+0f } },
|
|
|
|
{ FrontRight, { 2.04124145e-1f, -1.08880247e-1f, 0.0f, 1.88586120e-1f, -1.29099444e-1f, 0.0f, 0.0f, 0.0f, 7.45355993e-2f, -3.73460789e-2f, 0.0f, 0.0f, 0.0f, 0.0f, 0.0f, 0.00000000e+0f } },
|
|
|
|
{ SideRight, { 2.04124145e-1f, -2.17760495e-1f, 0.0f, 0.00000000e+0f, 0.00000000e+0f, 0.0f, 0.0f, 0.0f, -1.49071198e-1f, 3.73460789e-2f, 0.0f, 0.0f, 0.0f, 0.0f, 0.0f, 0.00000000e+0f } },
|
|
|
|
{ BackRight, { 2.04124145e-1f, -1.08880247e-1f, 0.0f, -1.88586120e-1f, 1.29099444e-1f, 0.0f, 0.0f, 0.0f, 7.45355993e-2f, -3.73460789e-2f, 0.0f, 0.0f, 0.0f, 0.0f, 0.0f, 0.00000000e+0f } },
|
2016-04-14 10:44:57 -07:00
|
|
|
};
|
|
|
|
|
2017-03-10 04:35:32 -08:00
|
|
|
static void InitNearFieldCtrl(ALCdevice *device, ALfloat ctrl_dist, ALsizei order, bool periphonic)
|
|
|
|
{
|
2017-04-04 06:58:53 -07:00
|
|
|
const char *devname = alstr_get_cstr(device->DeviceName);
|
2017-03-10 04:35:32 -08:00
|
|
|
ALsizei i;
|
|
|
|
|
|
|
|
if(GetConfigValueBool(devname, "decoder", "nfc", 1) && ctrl_dist > 0.0f)
|
|
|
|
{
|
|
|
|
/* NFC is only used when AvgSpeakerDist is greater than 0, and
|
|
|
|
* METERS_PER_UNIT is also greater than 0. In addition, NFC can only be
|
|
|
|
* used when rendering to an ambisonic buffer.
|
|
|
|
*/
|
|
|
|
device->AvgSpeakerDist = ctrl_dist;
|
|
|
|
|
|
|
|
device->Dry.NumChannelsPerOrder[0] = 1;
|
|
|
|
if(periphonic)
|
|
|
|
for(i = 1;i < order+1;i++)
|
|
|
|
device->Dry.NumChannelsPerOrder[i] = (i+1)*(i+1) - i*i;
|
|
|
|
else
|
|
|
|
for(i = 1;i < order+1;i++)
|
|
|
|
device->Dry.NumChannelsPerOrder[i] = (i*2+1) - ((i-1)*2+1);
|
|
|
|
for(;i < MAX_AMBI_ORDER+1;i++)
|
|
|
|
device->Dry.NumChannelsPerOrder[i] = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void InitDistanceComp(ALCdevice *device, const AmbDecConf *conf, const ALsizei speakermap[MAX_OUTPUT_CHANNELS])
|
|
|
|
{
|
2017-04-04 06:58:53 -07:00
|
|
|
const char *devname = alstr_get_cstr(device->DeviceName);
|
2017-03-10 04:35:32 -08:00
|
|
|
ALfloat maxdist = 0.0f;
|
|
|
|
ALsizei total = 0;
|
|
|
|
ALsizei i;
|
|
|
|
|
|
|
|
for(i = 0;i < conf->NumSpeakers;i++)
|
|
|
|
maxdist = maxf(maxdist, conf->Speakers[i].Distance);
|
|
|
|
|
|
|
|
if(GetConfigValueBool(devname, "decoder", "distance-comp", 1) && maxdist > 0.0f)
|
|
|
|
{
|
|
|
|
ALfloat srate = (ALfloat)device->Frequency;
|
|
|
|
for(i = 0;i < conf->NumSpeakers;i++)
|
|
|
|
{
|
|
|
|
ALsizei chan = speakermap[i];
|
|
|
|
ALfloat delay;
|
|
|
|
|
|
|
|
/* Distance compensation only delays in steps of the sample rate.
|
|
|
|
* This is a bit less accurate since the delay time falls to the
|
|
|
|
* nearest sample time, but it's far simpler as it doesn't have to
|
|
|
|
* deal with phase offsets. This means at 48khz, for instance, the
|
|
|
|
* distance delay will be in steps of about 7 millimeters.
|
|
|
|
*/
|
|
|
|
delay = floorf((maxdist-conf->Speakers[i].Distance) / SPEEDOFSOUNDMETRESPERSEC *
|
|
|
|
srate + 0.5f);
|
|
|
|
if(delay >= (ALfloat)MAX_DELAY_LENGTH)
|
|
|
|
ERR("Delay for speaker \"%s\" exceeds buffer length (%f >= %u)\n",
|
2017-04-04 06:58:53 -07:00
|
|
|
alstr_get_cstr(conf->Speakers[i].Name), delay, MAX_DELAY_LENGTH);
|
2017-03-10 04:35:32 -08:00
|
|
|
|
|
|
|
device->ChannelDelay[chan].Length = (ALsizei)clampf(
|
|
|
|
delay, 0.0f, (ALfloat)(MAX_DELAY_LENGTH-1)
|
|
|
|
);
|
|
|
|
device->ChannelDelay[chan].Gain = conf->Speakers[i].Distance / maxdist;
|
|
|
|
TRACE("Channel %u \"%s\" distance compensation: %d samples, %f gain\n", chan,
|
2017-04-04 06:58:53 -07:00
|
|
|
alstr_get_cstr(conf->Speakers[i].Name), device->ChannelDelay[chan].Length,
|
2017-03-10 04:35:32 -08:00
|
|
|
device->ChannelDelay[chan].Gain
|
|
|
|
);
|
|
|
|
|
|
|
|
/* Round up to the next 4th sample, so each channel buffer starts
|
|
|
|
* 16-byte aligned.
|
|
|
|
*/
|
|
|
|
total += RoundUp(device->ChannelDelay[chan].Length, 4);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if(total > 0)
|
|
|
|
{
|
|
|
|
device->ChannelDelay[0].Buffer = al_calloc(16, total * sizeof(ALfloat));
|
|
|
|
for(i = 1;i < MAX_OUTPUT_CHANNELS;i++)
|
|
|
|
{
|
|
|
|
size_t len = RoundUp(device->ChannelDelay[i-1].Length, 4);
|
|
|
|
device->ChannelDelay[i].Buffer = device->ChannelDelay[i-1].Buffer + len;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-04-14 15:25:12 -07:00
|
|
|
static void InitPanning(ALCdevice *device)
|
2016-04-14 10:44:57 -07:00
|
|
|
{
|
|
|
|
const ChannelMap *chanmap = NULL;
|
2017-01-16 07:45:07 -08:00
|
|
|
ALsizei coeffcount = 0;
|
|
|
|
ALsizei count = 0;
|
|
|
|
ALsizei i, j;
|
2016-04-14 10:44:57 -07:00
|
|
|
|
2014-09-10 17:53:01 -07:00
|
|
|
switch(device->FmtChans)
|
2010-08-03 00:21:36 -07:00
|
|
|
{
|
2010-12-04 19:50:00 -08:00
|
|
|
case DevFmtMono:
|
2014-11-05 02:54:11 -08:00
|
|
|
count = COUNTOF(MonoCfg);
|
2014-11-05 03:46:00 -08:00
|
|
|
chanmap = MonoCfg;
|
2016-04-15 17:31:04 -07:00
|
|
|
coeffcount = 1;
|
2010-08-03 00:21:36 -07:00
|
|
|
break;
|
|
|
|
|
2010-12-04 19:50:00 -08:00
|
|
|
case DevFmtStereo:
|
2014-11-05 02:54:11 -08:00
|
|
|
count = COUNTOF(StereoCfg);
|
2014-11-05 03:46:00 -08:00
|
|
|
chanmap = StereoCfg;
|
2016-04-15 17:31:04 -07:00
|
|
|
coeffcount = 4;
|
2010-08-03 00:21:36 -07:00
|
|
|
break;
|
|
|
|
|
2010-12-04 19:50:00 -08:00
|
|
|
case DevFmtQuad:
|
2014-11-05 02:54:11 -08:00
|
|
|
count = COUNTOF(QuadCfg);
|
2014-11-05 03:46:00 -08:00
|
|
|
chanmap = QuadCfg;
|
2017-01-21 12:28:54 -08:00
|
|
|
coeffcount = 4;
|
2010-08-03 00:21:36 -07:00
|
|
|
break;
|
|
|
|
|
2010-12-04 19:50:00 -08:00
|
|
|
case DevFmtX51:
|
2014-11-05 02:54:11 -08:00
|
|
|
count = COUNTOF(X51SideCfg);
|
2014-11-05 03:46:00 -08:00
|
|
|
chanmap = X51SideCfg;
|
2016-04-15 17:31:04 -07:00
|
|
|
coeffcount = 9;
|
2011-05-28 19:35:32 -07:00
|
|
|
break;
|
|
|
|
|
2014-11-07 00:54:16 -08:00
|
|
|
case DevFmtX51Rear:
|
|
|
|
count = COUNTOF(X51RearCfg);
|
|
|
|
chanmap = X51RearCfg;
|
2016-04-15 17:31:04 -07:00
|
|
|
coeffcount = 9;
|
2014-11-07 00:54:16 -08:00
|
|
|
break;
|
|
|
|
|
2010-12-04 19:50:00 -08:00
|
|
|
case DevFmtX61:
|
2014-11-05 02:54:11 -08:00
|
|
|
count = COUNTOF(X61Cfg);
|
2014-11-05 03:46:00 -08:00
|
|
|
chanmap = X61Cfg;
|
2017-01-21 12:28:54 -08:00
|
|
|
coeffcount = 9;
|
2010-08-03 00:21:36 -07:00
|
|
|
break;
|
|
|
|
|
2010-12-04 19:50:00 -08:00
|
|
|
case DevFmtX71:
|
2014-11-05 02:54:11 -08:00
|
|
|
count = COUNTOF(X71Cfg);
|
2014-11-05 03:46:00 -08:00
|
|
|
chanmap = X71Cfg;
|
2016-04-15 17:31:04 -07:00
|
|
|
coeffcount = 16;
|
2010-08-03 00:21:36 -07:00
|
|
|
break;
|
2014-11-25 22:20:00 -08:00
|
|
|
|
2017-04-12 18:26:07 -07:00
|
|
|
case DevFmtAmbi3D:
|
2014-11-25 22:20:00 -08:00
|
|
|
break;
|
2010-08-03 00:21:36 -07:00
|
|
|
}
|
2014-11-12 19:26:53 -08:00
|
|
|
|
2017-04-12 18:26:07 -07:00
|
|
|
if(device->FmtChans == DevFmtAmbi3D)
|
2016-07-29 21:55:43 -07:00
|
|
|
{
|
2017-04-04 06:58:53 -07:00
|
|
|
const char *devname = alstr_get_cstr(device->DeviceName);
|
2017-02-27 16:11:45 -08:00
|
|
|
const ALsizei *acnmap = (device->AmbiLayout == AmbiLayout_FuMa) ? FuMa2ACN : ACN2ACN;
|
|
|
|
const ALfloat *n3dscale = (device->AmbiScale == AmbiNorm_FuMa) ? FuMa2N3DScale :
|
|
|
|
(device->AmbiScale == AmbiNorm_SN3D) ? SN3D2N3DScale :
|
|
|
|
/*(device->AmbiScale == AmbiNorm_N3D) ?*/ UnitScale;
|
2017-03-10 04:35:32 -08:00
|
|
|
ALfloat nfc_delay = 0.0f;
|
2016-07-29 21:55:43 -07:00
|
|
|
|
2017-04-12 18:26:07 -07:00
|
|
|
count = (device->AmbiOrder == 3) ? 16 :
|
|
|
|
(device->AmbiOrder == 2) ? 9 :
|
|
|
|
(device->AmbiOrder == 1) ? 4 : 1;
|
2016-07-29 21:55:43 -07:00
|
|
|
for(i = 0;i < count;i++)
|
|
|
|
{
|
2017-01-16 07:45:07 -08:00
|
|
|
ALsizei acn = acnmap[i];
|
2016-08-02 09:16:12 -07:00
|
|
|
device->Dry.Ambi.Map[i].Scale = 1.0f/n3dscale[acn];
|
2016-07-31 07:46:38 -07:00
|
|
|
device->Dry.Ambi.Map[i].Index = acn;
|
2016-07-29 21:55:43 -07:00
|
|
|
}
|
|
|
|
device->Dry.CoeffCount = 0;
|
|
|
|
device->Dry.NumChannels = count;
|
|
|
|
|
2017-04-12 18:26:07 -07:00
|
|
|
if(device->AmbiOrder < 2)
|
2016-07-29 21:55:43 -07:00
|
|
|
{
|
2016-07-31 07:46:38 -07:00
|
|
|
device->FOAOut.Ambi = device->Dry.Ambi;
|
|
|
|
device->FOAOut.CoeffCount = device->Dry.CoeffCount;
|
2017-02-20 16:57:25 -08:00
|
|
|
device->FOAOut.NumChannels = 0;
|
2016-07-29 21:55:43 -07:00
|
|
|
}
|
2016-07-31 07:46:38 -07:00
|
|
|
else
|
|
|
|
{
|
|
|
|
/* FOA output is always ACN+N3D for higher-order ambisonic output.
|
|
|
|
* The upsampler expects this and will convert it for output.
|
|
|
|
*/
|
|
|
|
memset(&device->FOAOut.Ambi, 0, sizeof(device->FOAOut.Ambi));
|
|
|
|
for(i = 0;i < 4;i++)
|
|
|
|
{
|
|
|
|
device->FOAOut.Ambi.Map[i].Scale = 1.0f;
|
|
|
|
device->FOAOut.Ambi.Map[i].Index = i;
|
|
|
|
}
|
|
|
|
device->FOAOut.CoeffCount = 0;
|
2017-02-20 16:57:25 -08:00
|
|
|
device->FOAOut.NumChannels = 4;
|
2016-07-30 09:29:21 -07:00
|
|
|
|
2016-07-31 07:46:38 -07:00
|
|
|
ambiup_reset(device->AmbiUp, device);
|
|
|
|
}
|
2017-03-10 04:35:32 -08:00
|
|
|
|
2017-05-14 18:50:22 -07:00
|
|
|
if(ConfigValueFloat(devname, "decoder", "nfc-ref-delay", &nfc_delay) && nfc_delay > 0.0f)
|
2017-03-10 04:35:32 -08:00
|
|
|
{
|
|
|
|
nfc_delay = clampf(nfc_delay, 0.001f, 1000.0f);
|
|
|
|
InitNearFieldCtrl(device, nfc_delay * SPEEDOFSOUNDMETRESPERSEC,
|
2017-04-12 18:26:07 -07:00
|
|
|
device->AmbiOrder, true);
|
2017-03-10 04:35:32 -08:00
|
|
|
}
|
2016-07-29 21:55:43 -07:00
|
|
|
}
|
2016-04-15 22:05:47 -07:00
|
|
|
else
|
2016-03-25 23:25:13 -07:00
|
|
|
{
|
2017-01-24 19:03:51 -08:00
|
|
|
ALfloat w_scale, xyz_scale;
|
|
|
|
|
2016-04-16 17:21:31 -07:00
|
|
|
SetChannelMap(device->RealOut.ChannelName, device->Dry.Ambi.Coeffs,
|
2017-01-21 11:05:05 -08:00
|
|
|
chanmap, count, &device->Dry.NumChannels);
|
2016-04-15 22:05:47 -07:00
|
|
|
device->Dry.CoeffCount = coeffcount;
|
|
|
|
|
2017-01-24 19:03:51 -08:00
|
|
|
w_scale = (device->Dry.CoeffCount > 9) ? W_SCALE2D_THIRD :
|
|
|
|
(device->Dry.CoeffCount > 4) ? W_SCALE2D_SECOND : 1.0f;
|
|
|
|
xyz_scale = (device->Dry.CoeffCount > 9) ? XYZ_SCALE2D_THIRD :
|
|
|
|
(device->Dry.CoeffCount > 4) ? XYZ_SCALE2D_SECOND : 1.0f;
|
|
|
|
|
2016-04-15 22:05:47 -07:00
|
|
|
memset(&device->FOAOut.Ambi, 0, sizeof(device->FOAOut.Ambi));
|
2017-01-24 19:03:51 -08:00
|
|
|
for(i = 0;i < device->Dry.NumChannels;i++)
|
2016-04-15 22:05:47 -07:00
|
|
|
{
|
2017-01-24 19:03:51 -08:00
|
|
|
device->FOAOut.Ambi.Coeffs[i][0] = device->Dry.Ambi.Coeffs[i][0] * w_scale;
|
2016-04-15 22:05:47 -07:00
|
|
|
for(j = 1;j < 4;j++)
|
2017-01-24 19:03:51 -08:00
|
|
|
device->FOAOut.Ambi.Coeffs[i][j] = device->Dry.Ambi.Coeffs[i][j] * xyz_scale;
|
2016-04-15 22:05:47 -07:00
|
|
|
}
|
|
|
|
device->FOAOut.CoeffCount = 4;
|
2017-02-20 16:57:25 -08:00
|
|
|
device->FOAOut.NumChannels = 0;
|
2016-03-25 23:25:13 -07:00
|
|
|
}
|
2017-02-20 16:57:25 -08:00
|
|
|
device->RealOut.NumChannels = 0;
|
2010-08-03 00:21:36 -07:00
|
|
|
}
|
2016-01-28 00:02:46 -08:00
|
|
|
|
2017-01-29 16:42:02 -08:00
|
|
|
static void InitCustomPanning(ALCdevice *device, const AmbDecConf *conf, const ALsizei speakermap[MAX_OUTPUT_CHANNELS])
|
2016-04-14 16:42:32 -07:00
|
|
|
{
|
|
|
|
ChannelMap chanmap[MAX_OUTPUT_CHANNELS];
|
|
|
|
const ALfloat *coeff_scale = UnitScale;
|
2017-01-24 19:03:51 -08:00
|
|
|
ALfloat w_scale = 1.0f;
|
|
|
|
ALfloat xyz_scale = 1.0f;
|
2017-01-16 07:45:07 -08:00
|
|
|
ALsizei i, j;
|
2016-04-14 16:42:32 -07:00
|
|
|
|
|
|
|
if(conf->FreqBands != 1)
|
|
|
|
ERR("Basic renderer uses the high-frequency matrix as single-band (xover_freq = %.0fhz)\n",
|
|
|
|
conf->XOverFreq);
|
|
|
|
|
2017-01-24 19:03:51 -08:00
|
|
|
if((conf->ChanMask&AMBI_PERIPHONIC_MASK))
|
|
|
|
{
|
|
|
|
if(conf->ChanMask > 0x1ff)
|
|
|
|
{
|
|
|
|
w_scale = W_SCALE3D_THIRD;
|
|
|
|
xyz_scale = XYZ_SCALE3D_THIRD;
|
|
|
|
}
|
|
|
|
else if(conf->ChanMask > 0xf)
|
|
|
|
{
|
|
|
|
w_scale = W_SCALE3D_SECOND;
|
|
|
|
xyz_scale = XYZ_SCALE3D_SECOND;
|
|
|
|
}
|
|
|
|
}
|
2016-04-14 16:42:32 -07:00
|
|
|
else
|
2017-01-24 19:03:51 -08:00
|
|
|
{
|
|
|
|
if(conf->ChanMask > 0x1ff)
|
|
|
|
{
|
|
|
|
w_scale = W_SCALE2D_THIRD;
|
|
|
|
xyz_scale = XYZ_SCALE2D_THIRD;
|
|
|
|
}
|
|
|
|
else if(conf->ChanMask > 0xf)
|
|
|
|
{
|
|
|
|
w_scale = W_SCALE2D_SECOND;
|
|
|
|
xyz_scale = XYZ_SCALE2D_SECOND;
|
|
|
|
}
|
|
|
|
}
|
2016-04-14 16:42:32 -07:00
|
|
|
|
|
|
|
if(conf->CoeffScale == ADS_SN3D)
|
|
|
|
coeff_scale = SN3D2N3DScale;
|
|
|
|
else if(conf->CoeffScale == ADS_FuMa)
|
|
|
|
coeff_scale = FuMa2N3DScale;
|
|
|
|
|
2017-01-16 09:37:55 -08:00
|
|
|
for(i = 0;i < conf->NumSpeakers;i++)
|
2016-04-14 16:42:32 -07:00
|
|
|
{
|
2017-01-16 07:45:07 -08:00
|
|
|
ALsizei chan = speakermap[i];
|
2016-04-14 16:42:32 -07:00
|
|
|
ALfloat gain;
|
2017-01-16 07:45:07 -08:00
|
|
|
ALsizei k = 0;
|
2016-04-14 16:42:32 -07:00
|
|
|
|
|
|
|
for(j = 0;j < MAX_AMBI_COEFFS;j++)
|
|
|
|
chanmap[i].Config[j] = 0.0f;
|
|
|
|
|
|
|
|
chanmap[i].ChanName = device->RealOut.ChannelName[chan];
|
|
|
|
for(j = 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)))
|
|
|
|
chanmap[i].Config[j] = conf->HFMatrix[i][k++] / coeff_scale[j] * gain;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-04-16 17:21:31 -07:00
|
|
|
SetChannelMap(device->RealOut.ChannelName, device->Dry.Ambi.Coeffs, chanmap,
|
2017-01-21 11:05:05 -08:00
|
|
|
conf->NumSpeakers, &device->Dry.NumChannels);
|
2016-04-15 17:31:04 -07:00
|
|
|
device->Dry.CoeffCount = (conf->ChanMask > 0x1ff) ? 16 :
|
|
|
|
(conf->ChanMask > 0xf) ? 9 : 4;
|
2016-04-14 16:42:32 -07:00
|
|
|
|
2016-04-15 22:05:47 -07:00
|
|
|
memset(&device->FOAOut.Ambi, 0, sizeof(device->FOAOut.Ambi));
|
2017-01-24 19:03:51 -08:00
|
|
|
for(i = 0;i < device->Dry.NumChannels;i++)
|
2016-04-14 16:42:32 -07:00
|
|
|
{
|
2017-01-24 19:03:51 -08:00
|
|
|
device->FOAOut.Ambi.Coeffs[i][0] = device->Dry.Ambi.Coeffs[i][0] * w_scale;
|
2016-04-14 16:42:32 -07:00
|
|
|
for(j = 1;j < 4;j++)
|
2017-01-24 19:03:51 -08:00
|
|
|
device->FOAOut.Ambi.Coeffs[i][j] = device->Dry.Ambi.Coeffs[i][j] * xyz_scale;
|
2016-04-14 16:42:32 -07:00
|
|
|
}
|
2016-04-15 22:05:47 -07:00
|
|
|
device->FOAOut.CoeffCount = 4;
|
2017-02-20 16:57:25 -08:00
|
|
|
device->FOAOut.NumChannels = 0;
|
|
|
|
|
|
|
|
device->RealOut.NumChannels = 0;
|
2017-02-20 09:08:57 -08:00
|
|
|
|
|
|
|
InitDistanceComp(device, conf, speakermap);
|
2016-04-14 16:42:32 -07:00
|
|
|
}
|
|
|
|
|
2017-01-29 16:42:02 -08:00
|
|
|
static void InitHQPanning(ALCdevice *device, const AmbDecConf *conf, const ALsizei speakermap[MAX_OUTPUT_CHANNELS])
|
2016-04-14 16:42:32 -07:00
|
|
|
{
|
2017-03-10 04:35:32 -08:00
|
|
|
ALfloat avg_dist;
|
|
|
|
ALsizei count;
|
|
|
|
ALsizei i;
|
2016-04-14 16:42:32 -07:00
|
|
|
|
2016-06-01 05:30:06 -07:00
|
|
|
if((conf->ChanMask&AMBI_PERIPHONIC_MASK))
|
2016-04-15 22:05:47 -07:00
|
|
|
{
|
2016-04-19 18:58:19 -07:00
|
|
|
count = (conf->ChanMask > 0x1ff) ? 16 :
|
|
|
|
(conf->ChanMask > 0xf) ? 9 : 4;
|
2016-04-15 22:05:47 -07:00
|
|
|
for(i = 0;i < count;i++)
|
|
|
|
{
|
|
|
|
device->Dry.Ambi.Map[i].Scale = 1.0f;
|
|
|
|
device->Dry.Ambi.Map[i].Index = i;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2016-07-17 15:38:39 -07:00
|
|
|
static const int map[MAX_AMBI2D_COEFFS] = { 0, 1, 3, 4, 8, 9, 15 };
|
2016-04-19 13:58:33 -07:00
|
|
|
|
2016-04-19 18:58:19 -07:00
|
|
|
count = (conf->ChanMask > 0x1ff) ? 7 :
|
|
|
|
(conf->ChanMask > 0xf) ? 5 : 3;
|
2016-04-15 22:05:47 -07:00
|
|
|
for(i = 0;i < count;i++)
|
|
|
|
{
|
|
|
|
device->Dry.Ambi.Map[i].Scale = 1.0f;
|
|
|
|
device->Dry.Ambi.Map[i].Index = map[i];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
device->Dry.CoeffCount = 0;
|
|
|
|
device->Dry.NumChannels = count;
|
2016-04-14 16:42:32 -07:00
|
|
|
|
|
|
|
TRACE("Enabling %s-band %s-order%s ambisonic decoder\n",
|
|
|
|
(conf->FreqBands == 1) ? "single" : "dual",
|
|
|
|
(conf->ChanMask > 0xf) ? (conf->ChanMask > 0x1ff) ? "third" : "second" : "first",
|
2016-06-01 05:30:06 -07:00
|
|
|
(conf->ChanMask&AMBI_PERIPHONIC_MASK) ? " periphonic" : ""
|
2016-04-14 16:42:32 -07:00
|
|
|
);
|
2017-02-19 22:47:59 -08:00
|
|
|
bformatdec_reset(device->AmbiDecoder, conf, count, device->Frequency, speakermap);
|
2016-04-14 16:42:32 -07:00
|
|
|
|
2017-02-28 19:44:34 -08:00
|
|
|
if(!(conf->ChanMask > 0xf))
|
2016-04-15 22:05:47 -07:00
|
|
|
{
|
2016-07-06 13:33:40 -07:00
|
|
|
device->FOAOut.Ambi = device->Dry.Ambi;
|
2016-04-15 22:05:47 -07:00
|
|
|
device->FOAOut.CoeffCount = device->Dry.CoeffCount;
|
2017-02-20 16:57:25 -08:00
|
|
|
device->FOAOut.NumChannels = 0;
|
2016-04-15 22:05:47 -07:00
|
|
|
}
|
2016-04-14 16:42:32 -07:00
|
|
|
else
|
|
|
|
{
|
2016-04-15 22:05:47 -07:00
|
|
|
memset(&device->FOAOut.Ambi, 0, sizeof(device->FOAOut.Ambi));
|
2017-02-19 17:45:27 -08:00
|
|
|
if((conf->ChanMask&AMBI_PERIPHONIC_MASK))
|
2017-02-20 16:57:25 -08:00
|
|
|
{
|
|
|
|
count = 4;
|
|
|
|
for(i = 0;i < count;i++)
|
2017-02-19 17:45:27 -08:00
|
|
|
{
|
|
|
|
device->FOAOut.Ambi.Map[i].Scale = 1.0f;
|
|
|
|
device->FOAOut.Ambi.Map[i].Index = i;
|
|
|
|
}
|
2017-02-20 16:57:25 -08:00
|
|
|
}
|
2017-02-19 17:45:27 -08:00
|
|
|
else
|
2016-04-15 22:05:47 -07:00
|
|
|
{
|
2017-02-19 17:45:27 -08:00
|
|
|
static const int map[3] = { 0, 1, 3 };
|
2017-02-20 16:57:25 -08:00
|
|
|
count = 3;
|
|
|
|
for(i = 0;i < count;i++)
|
2017-02-19 17:45:27 -08:00
|
|
|
{
|
|
|
|
device->FOAOut.Ambi.Map[i].Scale = 1.0f;
|
|
|
|
device->FOAOut.Ambi.Map[i].Index = map[i];
|
|
|
|
}
|
2016-04-15 22:05:47 -07:00
|
|
|
}
|
|
|
|
device->FOAOut.CoeffCount = 0;
|
2017-02-20 16:57:25 -08:00
|
|
|
device->FOAOut.NumChannels = count;
|
2016-04-14 16:42:32 -07:00
|
|
|
}
|
2017-02-19 22:47:59 -08:00
|
|
|
|
2017-04-12 18:26:07 -07:00
|
|
|
device->RealOut.NumChannels = ChannelsFromDevFmt(device->FmtChans, device->AmbiOrder);
|
2017-02-20 16:57:25 -08:00
|
|
|
|
2017-03-10 04:35:32 -08:00
|
|
|
avg_dist = 0.0f;
|
|
|
|
for(i = 0;i < conf->NumSpeakers;i++)
|
|
|
|
avg_dist += conf->Speakers[i].Distance;
|
|
|
|
avg_dist /= (ALfloat)conf->NumSpeakers;
|
|
|
|
InitNearFieldCtrl(device, avg_dist,
|
|
|
|
(conf->ChanMask > 0x1ff) ? 3 : (conf->ChanMask > 0xf) ? 2 : 1,
|
|
|
|
!!(conf->ChanMask&AMBI_PERIPHONIC_MASK)
|
|
|
|
);
|
|
|
|
|
2017-02-20 09:08:57 -08:00
|
|
|
InitDistanceComp(device, conf, speakermap);
|
2016-04-14 16:42:32 -07:00
|
|
|
}
|
|
|
|
|
2017-04-18 17:39:10 -07:00
|
|
|
static void InitHrtfPanning(ALCdevice *device)
|
2016-04-14 10:44:57 -07:00
|
|
|
{
|
2017-01-18 19:16:24 -08:00
|
|
|
/* NOTE: azimuth goes clockwise. */
|
|
|
|
static const ALfloat AmbiPoints[][2] = {
|
|
|
|
{ DEG2RAD( 90.0f), DEG2RAD( 0.0f) },
|
|
|
|
{ DEG2RAD( 35.0f), DEG2RAD( -45.0f) },
|
|
|
|
{ DEG2RAD( 35.0f), DEG2RAD( 45.0f) },
|
|
|
|
{ DEG2RAD( 35.0f), DEG2RAD( 135.0f) },
|
|
|
|
{ DEG2RAD( 35.0f), DEG2RAD(-135.0f) },
|
|
|
|
{ DEG2RAD( 0.0f), DEG2RAD( 0.0f) },
|
|
|
|
{ DEG2RAD( 0.0f), DEG2RAD( 90.0f) },
|
|
|
|
{ DEG2RAD( 0.0f), DEG2RAD( 180.0f) },
|
|
|
|
{ DEG2RAD( 0.0f), DEG2RAD( -90.0f) },
|
|
|
|
{ DEG2RAD(-35.0f), DEG2RAD( -45.0f) },
|
|
|
|
{ DEG2RAD(-35.0f), DEG2RAD( 45.0f) },
|
|
|
|
{ DEG2RAD(-35.0f), DEG2RAD( 135.0f) },
|
|
|
|
{ DEG2RAD(-35.0f), DEG2RAD(-135.0f) },
|
|
|
|
{ DEG2RAD(-90.0f), DEG2RAD( 0.0f) },
|
|
|
|
};
|
|
|
|
static const ALfloat AmbiMatrixFOA[][2][MAX_AMBI_COEFFS] = {
|
|
|
|
{ { 1.88982237e-001f, 0.00000000e+000f, 1.90399923e-001f, 0.00000000e+000f }, { 7.14285714e-002f, 0.00000000e+000f, 1.24646009e-001f, 0.00000000e+000f } },
|
|
|
|
{ { 1.88982237e-001f, 1.09057783e-001f, 1.09208910e-001f, 1.09057783e-001f }, { 7.14285714e-002f, 7.13950780e-002f, 7.14940135e-002f, 7.13950780e-002f } },
|
|
|
|
{ { 1.88982237e-001f, -1.09057783e-001f, 1.09208910e-001f, 1.09057783e-001f }, { 7.14285714e-002f, -7.13950780e-002f, 7.14940135e-002f, 7.13950780e-002f } },
|
|
|
|
{ { 1.88982237e-001f, -1.09057783e-001f, 1.09208910e-001f, -1.09057783e-001f }, { 7.14285714e-002f, -7.13950780e-002f, 7.14940135e-002f, -7.13950780e-002f } },
|
|
|
|
{ { 1.88982237e-001f, 1.09057783e-001f, 1.09208910e-001f, -1.09057783e-001f }, { 7.14285714e-002f, 7.13950780e-002f, 7.14940135e-002f, -7.13950780e-002f } },
|
|
|
|
{ { 1.88982237e-001f, 0.00000000e+000f, 0.00000000e+000f, 1.88281281e-001f }, { 7.14285714e-002f, 0.00000000e+000f, 0.00000000e+000f, 1.23259031e-001f } },
|
|
|
|
{ { 1.88982237e-001f, -1.88281281e-001f, 0.00000000e+000f, 0.00000000e+000f }, { 7.14285714e-002f, -1.23259031e-001f, 0.00000000e+000f, 0.00000000e+000f } },
|
|
|
|
{ { 1.88982237e-001f, 0.00000000e+000f, 0.00000000e+000f, -1.88281281e-001f }, { 7.14285714e-002f, 0.00000000e+000f, 0.00000000e+000f, -1.23259031e-001f } },
|
|
|
|
{ { 1.88982237e-001f, 1.88281281e-001f, 0.00000000e+000f, 0.00000000e+000f }, { 7.14285714e-002f, 1.23259031e-001f, 0.00000000e+000f, 0.00000000e+000f } },
|
|
|
|
{ { 1.88982237e-001f, 1.09057783e-001f, -1.09208910e-001f, 1.09057783e-001f }, { 7.14285714e-002f, 7.13950780e-002f, -7.14940135e-002f, 7.13950780e-002f } },
|
|
|
|
{ { 1.88982237e-001f, -1.09057783e-001f, -1.09208910e-001f, 1.09057783e-001f }, { 7.14285714e-002f, -7.13950780e-002f, -7.14940135e-002f, 7.13950780e-002f } },
|
|
|
|
{ { 1.88982237e-001f, -1.09057783e-001f, -1.09208910e-001f, -1.09057783e-001f }, { 7.14285714e-002f, -7.13950780e-002f, -7.14940135e-002f, -7.13950780e-002f } },
|
|
|
|
{ { 1.88982237e-001f, 1.09057783e-001f, -1.09208910e-001f, -1.09057783e-001f }, { 7.14285714e-002f, 7.13950780e-002f, -7.14940135e-002f, -7.13950780e-002f } },
|
|
|
|
{ { 1.88982237e-001f, 0.00000000e+000f, -1.90399923e-001f, 0.00000000e+000f }, { 7.14285714e-002f, 0.00000000e+000f, -1.24646009e-001f, 0.00000000e+000f } }
|
|
|
|
}, AmbiMatrixHOA[][2][MAX_AMBI_COEFFS] = {
|
|
|
|
{ { 1.43315266e-001f, 0.00000000e+000f, 1.90399923e-001f, 0.00000000e+000f, 0.00000000e+000f, 0.00000000e+000f, 1.18020996e-001f, 0.00000000e+000f, 0.00000000e+000f }, { 7.26741039e-002f, 0.00000000e+000f, 1.24646009e-001f, 0.00000000e+000f, 0.00000000e+000f, 0.00000000e+000f, 1.49618920e-001f, 0.00000000e+000f, 0.00000000e+000f } },
|
|
|
|
{ { 1.40852210e-001f, 1.09057783e-001f, 1.09208910e-001f, 1.09057783e-001f, 7.58818830e-002f, 7.66295578e-002f, -3.28314629e-004f, 7.66295578e-002f, 0.00000000e+000f }, { 7.14251066e-002f, 7.13950780e-002f, 7.14940135e-002f, 7.13950780e-002f, 9.61978444e-002f, 9.71456952e-002f, -4.16214759e-004f, 9.71456952e-002f, 0.00000000e+000f } },
|
|
|
|
{ { 1.40852210e-001f, -1.09057783e-001f, 1.09208910e-001f, 1.09057783e-001f, -7.58818830e-002f, -7.66295578e-002f, -3.28314629e-004f, 7.66295578e-002f, 0.00000000e+000f }, { 7.14251066e-002f, -7.13950780e-002f, 7.14940135e-002f, 7.13950780e-002f, -9.61978444e-002f, -9.71456952e-002f, -4.16214759e-004f, 9.71456952e-002f, 0.00000000e+000f } },
|
|
|
|
{ { 1.40852210e-001f, -1.09057783e-001f, 1.09208910e-001f, -1.09057783e-001f, 7.58818830e-002f, -7.66295578e-002f, -3.28314629e-004f, -7.66295578e-002f, 0.00000000e+000f }, { 7.14251066e-002f, -7.13950780e-002f, 7.14940135e-002f, -7.13950780e-002f, 9.61978444e-002f, -9.71456952e-002f, -4.16214759e-004f, -9.71456952e-002f, 0.00000000e+000f } },
|
|
|
|
{ { 1.40852210e-001f, 1.09057783e-001f, 1.09208910e-001f, -1.09057783e-001f, -7.58818830e-002f, 7.66295578e-002f, -3.28314629e-004f, -7.66295578e-002f, 0.00000000e+000f }, { 7.14251066e-002f, 7.13950780e-002f, 7.14940135e-002f, -7.13950780e-002f, -9.61978444e-002f, 9.71456952e-002f, -4.16214759e-004f, -9.71456952e-002f, 0.00000000e+000f } },
|
|
|
|
{ { 1.39644596e-001f, 0.00000000e+000f, 0.00000000e+000f, 1.88281281e-001f, 0.00000000e+000f, 0.00000000e+000f, -5.83538687e-002f, 0.00000000e+000f, 1.01835015e-001f }, { 7.08127349e-002f, 0.00000000e+000f, 0.00000000e+000f, 1.23259031e-001f, 0.00000000e+000f, 0.00000000e+000f, -7.39770307e-002f, 0.00000000e+000f, 1.29099445e-001f } },
|
|
|
|
{ { 1.39644596e-001f, -1.88281281e-001f, 0.00000000e+000f, 0.00000000e+000f, 0.00000000e+000f, 0.00000000e+000f, -5.83538687e-002f, 0.00000000e+000f, -1.01835015e-001f }, { 7.08127349e-002f, -1.23259031e-001f, 0.00000000e+000f, 0.00000000e+000f, 0.00000000e+000f, 0.00000000e+000f, -7.39770307e-002f, 0.00000000e+000f, -1.29099445e-001f } },
|
|
|
|
{ { 1.39644596e-001f, 0.00000000e+000f, 0.00000000e+000f, -1.88281281e-001f, 0.00000000e+000f, 0.00000000e+000f, -5.83538687e-002f, 0.00000000e+000f, 1.01835015e-001f }, { 7.08127349e-002f, 0.00000000e+000f, 0.00000000e+000f, -1.23259031e-001f, 0.00000000e+000f, 0.00000000e+000f, -7.39770307e-002f, 0.00000000e+000f, 1.29099445e-001f } },
|
|
|
|
{ { 1.39644596e-001f, 1.88281281e-001f, 0.00000000e+000f, 0.00000000e+000f, 0.00000000e+000f, 0.00000000e+000f, -5.83538687e-002f, 0.00000000e+000f, -1.01835015e-001f }, { 7.08127349e-002f, 1.23259031e-001f, 0.00000000e+000f, 0.00000000e+000f, 0.00000000e+000f, 0.00000000e+000f, -7.39770307e-002f, 0.00000000e+000f, -1.29099445e-001f } },
|
|
|
|
{ { 1.40852210e-001f, 1.09057783e-001f, -1.09208910e-001f, 1.09057783e-001f, 7.58818830e-002f, -7.66295578e-002f, -3.28314629e-004f, -7.66295578e-002f, 0.00000000e+000f }, { 7.14251066e-002f, 7.13950780e-002f, -7.14940135e-002f, 7.13950780e-002f, 9.61978444e-002f, -9.71456952e-002f, -4.16214759e-004f, -9.71456952e-002f, 0.00000000e+000f } },
|
|
|
|
{ { 1.40852210e-001f, -1.09057783e-001f, -1.09208910e-001f, 1.09057783e-001f, -7.58818830e-002f, 7.66295578e-002f, -3.28314629e-004f, -7.66295578e-002f, 0.00000000e+000f }, { 7.14251066e-002f, -7.13950780e-002f, -7.14940135e-002f, 7.13950780e-002f, -9.61978444e-002f, 9.71456952e-002f, -4.16214759e-004f, -9.71456952e-002f, 0.00000000e+000f } },
|
|
|
|
{ { 1.40852210e-001f, -1.09057783e-001f, -1.09208910e-001f, -1.09057783e-001f, 7.58818830e-002f, 7.66295578e-002f, -3.28314629e-004f, 7.66295578e-002f, 0.00000000e+000f }, { 7.14251066e-002f, -7.13950780e-002f, -7.14940135e-002f, -7.13950780e-002f, 9.61978444e-002f, 9.71456952e-002f, -4.16214759e-004f, 9.71456952e-002f, 0.00000000e+000f } },
|
|
|
|
{ { 1.40852210e-001f, 1.09057783e-001f, -1.09208910e-001f, -1.09057783e-001f, -7.58818830e-002f, -7.66295578e-002f, -3.28314629e-004f, 7.66295578e-002f, 0.00000000e+000f }, { 7.14251066e-002f, 7.13950780e-002f, -7.14940135e-002f, -7.13950780e-002f, -9.61978444e-002f, -9.71456952e-002f, -4.16214759e-004f, 9.71456952e-002f, 0.00000000e+000f } },
|
|
|
|
{ { 1.43315266e-001f, 0.00000000e+000f, -1.90399923e-001f, 0.00000000e+000f, 0.00000000e+000f, 0.00000000e+000f, 1.18020996e-001f, 0.00000000e+000f, 0.00000000e+000f }, { 7.26741039e-002f, 0.00000000e+000f, -1.24646009e-001f, 0.00000000e+000f, 0.00000000e+000f, 0.00000000e+000f, 1.49618920e-001f, 0.00000000e+000f, 0.00000000e+000f } },
|
|
|
|
};
|
2017-04-18 17:39:10 -07:00
|
|
|
const ALfloat (*AmbiMatrix)[2][MAX_AMBI_COEFFS] = device->AmbiUp ? AmbiMatrixHOA :
|
|
|
|
AmbiMatrixFOA;
|
|
|
|
ALsizei count = device->AmbiUp ? 9 : 4;
|
2017-03-10 04:35:32 -08:00
|
|
|
ALsizei i;
|
2016-04-14 10:44:57 -07:00
|
|
|
|
2017-01-18 19:16:24 -08:00
|
|
|
static_assert(COUNTOF(AmbiPoints) <= HRTF_AMBI_MAX_CHANNELS, "HRTF_AMBI_MAX_CHANNELS is too small");
|
2017-01-15 13:57:22 -08:00
|
|
|
|
2017-04-18 14:11:15 -07:00
|
|
|
device->Hrtf = al_calloc(16, FAM_SIZE(DirectHrtfState, Chan, count));
|
2017-03-11 06:20:04 -08:00
|
|
|
|
Decode directly from B-Format to HRTF instead of a cube
Last time this attempted to average the HRIRs according to their contribution
to a given B-Format channel as if they were loudspeakers, as well as averaging
the HRIR delays. The latter part resulted in the loss of the ITD (inter-aural
time delay), a key component of HRTF.
This time, the HRIRs are averaged similar to above, except instead of averaging
the delays, they're applied to the resulting coefficients (for example, a delay
of 8 would apply the HRIR starting at the 8th sample of the target HRIR). This
does roughly double the IR length, as the largest delay is about 35 samples
while the filter is normally 32 samples. However, this is still smaller the
original data set IR (which was 256 samples), it also only needs to be applied
to 4 channels for first-order ambisonics, rather than the 8-channel cube. So
it's doing twice as much work per sample, but only working on half the number
of samples.
Additionally, since the resulting HRIRs no longer rely on an extra delay line,
a more efficient HRTF mixing function can be made that doesn't use one. Such a
function can also avoid the per-sample stepping parameters the original uses.
2016-08-11 23:20:35 -07:00
|
|
|
for(i = 0;i < count;i++)
|
|
|
|
{
|
|
|
|
device->Dry.Ambi.Map[i].Scale = 1.0f;
|
2017-01-18 19:16:24 -08:00
|
|
|
device->Dry.Ambi.Map[i].Index = i;
|
Decode directly from B-Format to HRTF instead of a cube
Last time this attempted to average the HRIRs according to their contribution
to a given B-Format channel as if they were loudspeakers, as well as averaging
the HRIR delays. The latter part resulted in the loss of the ITD (inter-aural
time delay), a key component of HRTF.
This time, the HRIRs are averaged similar to above, except instead of averaging
the delays, they're applied to the resulting coefficients (for example, a delay
of 8 would apply the HRIR starting at the 8th sample of the target HRIR). This
does roughly double the IR length, as the largest delay is about 35 samples
while the filter is normally 32 samples. However, this is still smaller the
original data set IR (which was 256 samples), it also only needs to be applied
to 4 channels for first-order ambisonics, rather than the 8-channel cube. So
it's doing twice as much work per sample, but only working on half the number
of samples.
Additionally, since the resulting HRIRs no longer rely on an extra delay line,
a more efficient HRTF mixing function can be made that doesn't use one. Such a
function can also avoid the per-sample stepping parameters the original uses.
2016-08-11 23:20:35 -07:00
|
|
|
}
|
|
|
|
device->Dry.CoeffCount = 0;
|
|
|
|
device->Dry.NumChannels = count;
|
2016-04-14 10:44:57 -07:00
|
|
|
|
2017-04-18 17:39:10 -07:00
|
|
|
if(device->AmbiUp)
|
2017-01-15 13:57:22 -08:00
|
|
|
{
|
|
|
|
memset(&device->FOAOut.Ambi, 0, sizeof(device->FOAOut.Ambi));
|
|
|
|
for(i = 0;i < 4;i++)
|
|
|
|
{
|
|
|
|
device->FOAOut.Ambi.Map[i].Scale = 1.0f;
|
|
|
|
device->FOAOut.Ambi.Map[i].Index = i;
|
|
|
|
}
|
|
|
|
device->FOAOut.CoeffCount = 0;
|
2017-02-20 16:57:25 -08:00
|
|
|
device->FOAOut.NumChannels = 4;
|
2017-01-15 13:57:22 -08:00
|
|
|
|
|
|
|
ambiup_reset(device->AmbiUp, device);
|
|
|
|
}
|
2017-04-18 17:39:10 -07:00
|
|
|
else
|
|
|
|
{
|
|
|
|
device->FOAOut.Ambi = device->Dry.Ambi;
|
|
|
|
device->FOAOut.CoeffCount = device->Dry.CoeffCount;
|
|
|
|
device->FOAOut.NumChannels = 0;
|
|
|
|
}
|
2016-04-14 10:44:57 -07:00
|
|
|
|
2017-04-12 18:26:07 -07:00
|
|
|
device->RealOut.NumChannels = ChannelsFromDevFmt(device->FmtChans, device->AmbiOrder);
|
2017-02-20 16:57:25 -08:00
|
|
|
|
2017-03-10 10:47:43 -08:00
|
|
|
device->Hrtf->IrSize = BuildBFormatHrtf(device->HrtfHandle,
|
2017-03-11 06:20:04 -08:00
|
|
|
device->Hrtf, device->Dry.NumChannels,
|
2017-01-18 19:16:24 -08:00
|
|
|
AmbiPoints, AmbiMatrix, COUNTOF(AmbiPoints)
|
2016-08-24 00:25:28 -07:00
|
|
|
);
|
Decode directly from B-Format to HRTF instead of a cube
Last time this attempted to average the HRIRs according to their contribution
to a given B-Format channel as if they were loudspeakers, as well as averaging
the HRIR delays. The latter part resulted in the loss of the ITD (inter-aural
time delay), a key component of HRTF.
This time, the HRIRs are averaged similar to above, except instead of averaging
the delays, they're applied to the resulting coefficients (for example, a delay
of 8 would apply the HRIR starting at the 8th sample of the target HRIR). This
does roughly double the IR length, as the largest delay is about 35 samples
while the filter is normally 32 samples. However, this is still smaller the
original data set IR (which was 256 samples), it also only needs to be applied
to 4 channels for first-order ambisonics, rather than the 8-channel cube. So
it's doing twice as much work per sample, but only working on half the number
of samples.
Additionally, since the resulting HRIRs no longer rely on an extra delay line,
a more efficient HRTF mixing function can be made that doesn't use one. Such a
function can also avoid the per-sample stepping parameters the original uses.
2016-08-11 23:20:35 -07:00
|
|
|
|
|
|
|
/* Round up to the nearest multiple of 8 */
|
2017-03-10 10:47:43 -08:00
|
|
|
device->Hrtf->IrSize = (device->Hrtf->IrSize+7)&~7;
|
2016-04-14 10:44:57 -07:00
|
|
|
}
|
|
|
|
|
2016-04-14 15:25:12 -07:00
|
|
|
static void InitUhjPanning(ALCdevice *device)
|
2016-04-14 10:44:57 -07:00
|
|
|
{
|
2017-01-16 07:45:07 -08:00
|
|
|
ALsizei count = 3;
|
|
|
|
ALsizei i;
|
2016-04-14 10:44:57 -07:00
|
|
|
|
2016-04-15 22:05:47 -07:00
|
|
|
for(i = 0;i < count;i++)
|
|
|
|
{
|
2017-01-16 07:45:07 -08:00
|
|
|
ALsizei acn = FuMa2ACN[i];
|
2016-04-15 22:05:47 -07:00
|
|
|
device->Dry.Ambi.Map[i].Scale = 1.0f/FuMa2N3DScale[acn];
|
|
|
|
device->Dry.Ambi.Map[i].Index = acn;
|
|
|
|
}
|
|
|
|
device->Dry.CoeffCount = 0;
|
|
|
|
device->Dry.NumChannels = count;
|
2016-04-14 10:44:57 -07:00
|
|
|
|
2016-07-06 13:33:40 -07:00
|
|
|
device->FOAOut.Ambi = device->Dry.Ambi;
|
2016-04-15 22:05:47 -07:00
|
|
|
device->FOAOut.CoeffCount = device->Dry.CoeffCount;
|
2017-02-20 16:57:25 -08:00
|
|
|
device->FOAOut.NumChannels = 0;
|
|
|
|
|
2017-04-12 18:26:07 -07:00
|
|
|
device->RealOut.NumChannels = ChannelsFromDevFmt(device->FmtChans, device->AmbiOrder);
|
2016-04-14 10:44:57 -07:00
|
|
|
}
|
|
|
|
|
2016-04-14 15:25:12 -07:00
|
|
|
void aluInitRenderer(ALCdevice *device, ALint hrtf_id, enum HrtfRequestMode hrtf_appreq, enum HrtfRequestMode hrtf_userreq)
|
|
|
|
{
|
2017-04-06 13:00:29 -07:00
|
|
|
/* Hold the HRTF the device last used, in case it's used again. */
|
|
|
|
struct Hrtf *old_hrtf = device->HrtfHandle;
|
2016-04-14 15:25:12 -07:00
|
|
|
const char *mode;
|
|
|
|
bool headphones;
|
|
|
|
int bs2blevel;
|
|
|
|
size_t i;
|
|
|
|
|
2017-03-10 10:47:43 -08:00
|
|
|
al_free(device->Hrtf);
|
|
|
|
device->Hrtf = NULL;
|
|
|
|
device->HrtfHandle = NULL;
|
2017-04-04 06:58:53 -07:00
|
|
|
alstr_clear(&device->HrtfName);
|
2016-04-14 15:25:12 -07:00
|
|
|
device->Render_Mode = NormalRender;
|
|
|
|
|
2016-04-15 22:05:47 -07:00
|
|
|
memset(&device->Dry.Ambi, 0, sizeof(device->Dry.Ambi));
|
2016-04-15 17:31:04 -07:00
|
|
|
device->Dry.CoeffCount = 0;
|
|
|
|
device->Dry.NumChannels = 0;
|
2017-03-10 04:35:32 -08:00
|
|
|
for(i = 0;i < MAX_AMBI_ORDER+1;i++)
|
|
|
|
device->Dry.NumChannelsPerOrder[i] = 0;
|
2016-04-15 17:31:04 -07:00
|
|
|
|
2017-03-10 04:35:32 -08:00
|
|
|
device->AvgSpeakerDist = 0.0f;
|
2017-02-19 22:47:59 -08:00
|
|
|
memset(device->ChannelDelay, 0, sizeof(device->ChannelDelay));
|
|
|
|
for(i = 0;i < MAX_OUTPUT_CHANNELS;i++)
|
|
|
|
{
|
|
|
|
device->ChannelDelay[i].Gain = 1.0f;
|
|
|
|
device->ChannelDelay[i].Length = 0;
|
|
|
|
}
|
|
|
|
|
2016-04-14 15:25:12 -07:00
|
|
|
if(device->FmtChans != DevFmtStereo)
|
|
|
|
{
|
2017-01-29 16:42:02 -08:00
|
|
|
ALsizei speakermap[MAX_OUTPUT_CHANNELS];
|
2016-04-16 14:00:22 -07:00
|
|
|
const char *devname, *layout = NULL;
|
|
|
|
AmbDecConf conf, *pconf = NULL;
|
2016-04-14 16:42:32 -07:00
|
|
|
|
2017-04-06 13:00:29 -07:00
|
|
|
if(old_hrtf)
|
|
|
|
Hrtf_DecRef(old_hrtf);
|
|
|
|
old_hrtf = NULL;
|
2016-04-14 15:25:12 -07:00
|
|
|
if(hrtf_appreq == Hrtf_Enable)
|
2017-03-10 10:47:43 -08:00
|
|
|
device->HrtfStatus = ALC_HRTF_UNSUPPORTED_FORMAT_SOFT;
|
2016-04-14 15:25:12 -07:00
|
|
|
|
2016-04-14 16:42:32 -07:00
|
|
|
ambdec_init(&conf);
|
|
|
|
|
2017-04-04 06:58:53 -07:00
|
|
|
devname = alstr_get_cstr(device->DeviceName);
|
2016-04-16 14:00:22 -07:00
|
|
|
switch(device->FmtChans)
|
|
|
|
{
|
|
|
|
case DevFmtQuad: layout = "quad"; break;
|
2017-02-19 22:59:55 -08:00
|
|
|
case DevFmtX51: /* fall-through */
|
|
|
|
case DevFmtX51Rear: layout = "surround51"; break;
|
2016-04-16 14:00:22 -07:00
|
|
|
case DevFmtX61: layout = "surround61"; break;
|
|
|
|
case DevFmtX71: layout = "surround71"; break;
|
2016-07-29 21:55:43 -07:00
|
|
|
/* Mono, Stereo, and Ambisonics output don't use custom decoders. */
|
2016-04-16 14:00:22 -07:00
|
|
|
case DevFmtMono:
|
|
|
|
case DevFmtStereo:
|
2017-04-12 18:26:07 -07:00
|
|
|
case DevFmtAmbi3D:
|
2016-04-16 14:00:22 -07:00
|
|
|
break;
|
|
|
|
}
|
2016-04-14 16:42:32 -07:00
|
|
|
if(layout)
|
2016-04-14 15:25:12 -07:00
|
|
|
{
|
2016-04-14 16:42:32 -07:00
|
|
|
const char *fname;
|
|
|
|
if(ConfigValueStr(devname, "decoder", layout, &fname))
|
|
|
|
{
|
|
|
|
if(!ambdec_load(&conf, fname))
|
|
|
|
ERR("Failed to load layout file %s\n", fname);
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if(conf.ChanMask > 0xffff)
|
|
|
|
ERR("Unsupported channel mask 0x%04x (max 0xffff)\n", conf.ChanMask);
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if(MakeSpeakerMap(device, &conf, speakermap))
|
|
|
|
pconf = &conf;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if(pconf && GetConfigValueBool(devname, "decoder", "hq-mode", 0))
|
|
|
|
{
|
2016-07-30 09:29:21 -07:00
|
|
|
ambiup_free(device->AmbiUp);
|
|
|
|
device->AmbiUp = NULL;
|
2016-04-19 18:58:19 -07:00
|
|
|
if(!device->AmbiDecoder)
|
2016-04-14 15:25:12 -07:00
|
|
|
device->AmbiDecoder = bformatdec_alloc();
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
bformatdec_free(device->AmbiDecoder);
|
|
|
|
device->AmbiDecoder = NULL;
|
2017-04-21 13:26:29 -07:00
|
|
|
if(device->FmtChans == DevFmtAmbi3D && device->AmbiOrder > 1)
|
2016-07-30 09:29:21 -07:00
|
|
|
{
|
|
|
|
if(!device->AmbiUp)
|
|
|
|
device->AmbiUp = ambiup_alloc();
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
ambiup_free(device->AmbiUp);
|
|
|
|
device->AmbiUp = NULL;
|
|
|
|
}
|
2016-04-14 15:25:12 -07:00
|
|
|
}
|
|
|
|
|
2016-04-14 16:42:32 -07:00
|
|
|
if(!pconf)
|
|
|
|
InitPanning(device);
|
|
|
|
else if(device->AmbiDecoder)
|
|
|
|
InitHQPanning(device, pconf, speakermap);
|
|
|
|
else
|
|
|
|
InitCustomPanning(device, pconf, speakermap);
|
|
|
|
|
|
|
|
ambdec_deinit(&conf);
|
2016-04-14 15:25:12 -07:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
bformatdec_free(device->AmbiDecoder);
|
|
|
|
device->AmbiDecoder = NULL;
|
|
|
|
|
|
|
|
headphones = device->IsHeadphones;
|
|
|
|
if(device->Type != Loopback)
|
|
|
|
{
|
|
|
|
const char *mode;
|
2017-04-04 06:58:53 -07:00
|
|
|
if(ConfigValueStr(alstr_get_cstr(device->DeviceName), NULL, "stereo-mode", &mode))
|
2016-04-14 15:25:12 -07:00
|
|
|
{
|
|
|
|
if(strcasecmp(mode, "headphones") == 0)
|
|
|
|
headphones = true;
|
|
|
|
else if(strcasecmp(mode, "speakers") == 0)
|
|
|
|
headphones = false;
|
|
|
|
else if(strcasecmp(mode, "auto") != 0)
|
|
|
|
ERR("Unexpected stereo-mode: %s\n", mode);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if(hrtf_userreq == Hrtf_Default)
|
|
|
|
{
|
2016-04-16 14:00:22 -07:00
|
|
|
bool usehrtf = (headphones && hrtf_appreq != Hrtf_Disable) ||
|
|
|
|
(hrtf_appreq == Hrtf_Enable);
|
|
|
|
if(!usehrtf) goto no_hrtf;
|
|
|
|
|
2017-03-10 10:47:43 -08:00
|
|
|
device->HrtfStatus = ALC_HRTF_ENABLED_SOFT;
|
2016-04-14 15:25:12 -07:00
|
|
|
if(headphones && hrtf_appreq != Hrtf_Disable)
|
2017-03-10 10:47:43 -08:00
|
|
|
device->HrtfStatus = ALC_HRTF_HEADPHONES_DETECTED_SOFT;
|
2016-04-14 15:25:12 -07:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2016-04-16 14:00:22 -07:00
|
|
|
if(hrtf_userreq != Hrtf_Enable)
|
|
|
|
{
|
|
|
|
if(hrtf_appreq == Hrtf_Enable)
|
2017-03-10 10:47:43 -08:00
|
|
|
device->HrtfStatus = ALC_HRTF_DENIED_SOFT;
|
2016-04-16 14:00:22 -07:00
|
|
|
goto no_hrtf;
|
|
|
|
}
|
2017-03-10 10:47:43 -08:00
|
|
|
device->HrtfStatus = ALC_HRTF_REQUIRED_SOFT;
|
2016-04-14 15:25:12 -07:00
|
|
|
}
|
|
|
|
|
2017-03-10 10:47:43 -08:00
|
|
|
if(VECTOR_SIZE(device->HrtfList) == 0)
|
2016-04-14 15:25:12 -07:00
|
|
|
{
|
2017-03-10 10:47:43 -08:00
|
|
|
VECTOR_DEINIT(device->HrtfList);
|
|
|
|
device->HrtfList = EnumerateHrtf(device->DeviceName);
|
2016-04-14 15:25:12 -07:00
|
|
|
}
|
|
|
|
|
2017-03-10 10:47:43 -08:00
|
|
|
if(hrtf_id >= 0 && (size_t)hrtf_id < VECTOR_SIZE(device->HrtfList))
|
2016-04-14 15:25:12 -07:00
|
|
|
{
|
2017-04-05 11:29:58 -07:00
|
|
|
const EnumeratedHrtf *entry = &VECTOR_ELEM(device->HrtfList, hrtf_id);
|
2017-04-06 13:00:29 -07:00
|
|
|
struct Hrtf *hrtf = GetLoadedHrtf(entry->hrtf);
|
2017-04-05 12:27:30 -07:00
|
|
|
if(hrtf && hrtf->sampleRate == device->Frequency)
|
2016-04-14 15:25:12 -07:00
|
|
|
{
|
2017-04-05 11:29:58 -07:00
|
|
|
device->HrtfHandle = hrtf;
|
2017-04-04 06:58:53 -07:00
|
|
|
alstr_copy(&device->HrtfName, entry->name);
|
2016-04-14 15:25:12 -07:00
|
|
|
}
|
2017-04-06 13:00:29 -07:00
|
|
|
else if(hrtf)
|
|
|
|
Hrtf_DecRef(hrtf);
|
|
|
|
}
|
|
|
|
|
2017-03-10 10:47:43 -08:00
|
|
|
for(i = 0;!device->HrtfHandle && i < VECTOR_SIZE(device->HrtfList);i++)
|
2016-04-14 15:25:12 -07:00
|
|
|
{
|
2017-04-05 11:29:58 -07:00
|
|
|
const EnumeratedHrtf *entry = &VECTOR_ELEM(device->HrtfList, i);
|
2017-04-06 13:00:29 -07:00
|
|
|
struct Hrtf *hrtf = GetLoadedHrtf(entry->hrtf);
|
2017-04-05 12:27:30 -07:00
|
|
|
if(hrtf && hrtf->sampleRate == device->Frequency)
|
2016-04-14 15:25:12 -07:00
|
|
|
{
|
2017-04-05 11:29:58 -07:00
|
|
|
device->HrtfHandle = hrtf;
|
2017-04-04 06:58:53 -07:00
|
|
|
alstr_copy(&device->HrtfName, entry->name);
|
2016-04-14 15:25:12 -07:00
|
|
|
}
|
2017-04-06 13:00:29 -07:00
|
|
|
else if(hrtf)
|
|
|
|
Hrtf_DecRef(hrtf);
|
2016-04-14 15:25:12 -07:00
|
|
|
}
|
|
|
|
|
2017-03-10 10:47:43 -08:00
|
|
|
if(device->HrtfHandle)
|
2016-04-14 15:25:12 -07:00
|
|
|
{
|
2017-04-06 13:00:29 -07:00
|
|
|
if(old_hrtf)
|
|
|
|
Hrtf_DecRef(old_hrtf);
|
|
|
|
old_hrtf = NULL;
|
|
|
|
|
2016-04-16 14:00:22 -07:00
|
|
|
device->Render_Mode = HrtfRender;
|
2017-04-04 06:58:53 -07:00
|
|
|
if(ConfigValueStr(alstr_get_cstr(device->DeviceName), NULL, "hrtf-mode", &mode))
|
2016-04-14 15:25:12 -07:00
|
|
|
{
|
|
|
|
if(strcasecmp(mode, "full") == 0)
|
|
|
|
device->Render_Mode = HrtfRender;
|
|
|
|
else if(strcasecmp(mode, "basic") == 0)
|
|
|
|
device->Render_Mode = NormalRender;
|
|
|
|
else
|
|
|
|
ERR("Unexpected hrtf-mode: %s\n", mode);
|
|
|
|
}
|
|
|
|
|
2017-01-15 13:57:22 -08:00
|
|
|
if(device->Render_Mode == HrtfRender)
|
|
|
|
{
|
|
|
|
/* Don't bother with HOA when using full HRTF rendering. Nothing
|
|
|
|
* needs it, and it eases the CPU/memory load.
|
|
|
|
*/
|
|
|
|
ambiup_free(device->AmbiUp);
|
|
|
|
device->AmbiUp = NULL;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if(!device->AmbiUp)
|
|
|
|
device->AmbiUp = ambiup_alloc();
|
|
|
|
}
|
|
|
|
|
2016-11-01 14:07:14 -07:00
|
|
|
TRACE("%s HRTF rendering enabled, using \"%s\"\n",
|
|
|
|
((device->Render_Mode == HrtfRender) ? "Full" : "Basic"),
|
2017-04-04 06:58:53 -07:00
|
|
|
alstr_get_cstr(device->HrtfName)
|
2016-11-01 14:07:14 -07:00
|
|
|
);
|
2017-04-18 17:39:10 -07:00
|
|
|
InitHrtfPanning(device);
|
2016-04-14 15:25:12 -07:00
|
|
|
return;
|
|
|
|
}
|
2017-03-10 10:47:43 -08:00
|
|
|
device->HrtfStatus = ALC_HRTF_UNSUPPORTED_FORMAT_SOFT;
|
2016-04-14 15:25:12 -07:00
|
|
|
|
|
|
|
no_hrtf:
|
2017-04-06 13:00:29 -07:00
|
|
|
if(old_hrtf)
|
|
|
|
Hrtf_DecRef(old_hrtf);
|
|
|
|
old_hrtf = NULL;
|
2016-04-14 15:25:12 -07:00
|
|
|
TRACE("HRTF disabled\n");
|
|
|
|
|
2017-02-22 19:18:01 -08:00
|
|
|
device->Render_Mode = StereoPair;
|
|
|
|
|
2017-01-15 13:57:22 -08:00
|
|
|
ambiup_free(device->AmbiUp);
|
|
|
|
device->AmbiUp = NULL;
|
|
|
|
|
2016-04-14 15:25:12 -07:00
|
|
|
bs2blevel = ((headphones && hrtf_appreq != Hrtf_Disable) ||
|
|
|
|
(hrtf_appreq == Hrtf_Enable)) ? 5 : 0;
|
|
|
|
if(device->Type != Loopback)
|
2017-04-04 06:58:53 -07:00
|
|
|
ConfigValueInt(alstr_get_cstr(device->DeviceName), NULL, "cf_level", &bs2blevel);
|
2016-04-14 15:25:12 -07:00
|
|
|
if(bs2blevel > 0 && bs2blevel <= 6)
|
|
|
|
{
|
|
|
|
device->Bs2b = al_calloc(16, sizeof(*device->Bs2b));
|
|
|
|
bs2b_set_params(device->Bs2b, bs2blevel, device->Frequency);
|
|
|
|
TRACE("BS2B enabled\n");
|
|
|
|
InitPanning(device);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
TRACE("BS2B disabled\n");
|
|
|
|
|
2017-04-04 06:58:53 -07:00
|
|
|
if(ConfigValueStr(alstr_get_cstr(device->DeviceName), NULL, "stereo-encoding", &mode))
|
2016-04-14 15:25:12 -07:00
|
|
|
{
|
2017-02-22 19:18:01 -08:00
|
|
|
if(strcasecmp(mode, "uhj") == 0)
|
|
|
|
device->Render_Mode = NormalRender;
|
|
|
|
else if(strcasecmp(mode, "panpot") != 0)
|
|
|
|
ERR("Unexpected stereo-encoding: %s\n", mode);
|
2016-04-14 15:25:12 -07:00
|
|
|
}
|
|
|
|
if(device->Render_Mode == NormalRender)
|
|
|
|
{
|
|
|
|
device->Uhj_Encoder = al_calloc(16, sizeof(Uhj2Encoder));
|
|
|
|
TRACE("UHJ enabled\n");
|
|
|
|
InitUhjPanning(device);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
TRACE("UHJ disabled\n");
|
|
|
|
InitPanning(device);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-01-28 00:02:46 -08:00
|
|
|
void aluInitEffectPanning(ALeffectslot *slot)
|
|
|
|
{
|
2017-01-16 07:45:07 -08:00
|
|
|
ALsizei i;
|
2016-01-28 00:02:46 -08:00
|
|
|
|
2016-04-14 21:50:36 -07:00
|
|
|
memset(slot->ChanMap, 0, sizeof(slot->ChanMap));
|
2016-01-28 00:02:46 -08:00
|
|
|
slot->NumChannels = 0;
|
|
|
|
|
2016-04-14 21:50:36 -07:00
|
|
|
for(i = 0;i < MAX_EFFECT_CHANNELS;i++)
|
|
|
|
{
|
|
|
|
slot->ChanMap[i].Scale = 1.0f;
|
|
|
|
slot->ChanMap[i].Index = i;
|
|
|
|
}
|
|
|
|
slot->NumChannels = i;
|
2016-01-28 00:02:46 -08:00
|
|
|
}
|