2 * OpenAL cross platform audio library
3 * Copyright (C) 1999-2010 by authors.
4 * This library is free software; you can redistribute it and/or
5 * modify it under the terms of the GNU Library General Public
6 * License as published by the Free Software Foundation; either
7 * version 2 of the License, or (at your option) any later version.
9 * This library is distributed in the hope that it will be useful,
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of
11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
12 * Library General Public License for more details.
14 * You should have received a copy of the GNU Library General Public
15 * License along with this library; if not, write to the
16 * Free Software Foundation, Inc.,
17 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
18 * Or go to http://www.gnu.org/copyleft/lgpl.html
40 #include "al/auxeffectslot.h"
44 #include "alnumeric.h"
45 #include "aloptional.h"
51 #include "bformatdec.h"
53 #include "devformat.h"
56 #include "math_defs.h"
57 #include "opthelpers.h"
58 #include "uhjfilter.h"
61 constexpr std::array
<float,MAX_AMBI_CHANNELS
> AmbiScale::FromN3D
;
62 constexpr std::array
<float,MAX_AMBI_CHANNELS
> AmbiScale::FromSN3D
;
63 constexpr std::array
<float,MAX_AMBI_CHANNELS
> AmbiScale::FromFuMa
;
64 constexpr std::array
<uint8_t,MAX_AMBI_CHANNELS
> AmbiIndex::FromFuMa
;
65 constexpr std::array
<uint8_t,MAX_AMBI_CHANNELS
> AmbiIndex::FromACN
;
66 constexpr std::array
<uint8_t,MAX_AMBI2D_CHANNELS
> AmbiIndex::From2D
;
67 constexpr std::array
<uint8_t,MAX_AMBI_CHANNELS
> AmbiIndex::From3D
;
72 using namespace std::placeholders
;
73 using std::chrono::seconds
;
74 using std::chrono::nanoseconds
;
76 inline const char *GetLabelFromChannel(Channel channel
)
80 case FrontLeft
: return "front-left";
81 case FrontRight
: return "front-right";
82 case FrontCenter
: return "front-center";
83 case LFE
: return "lfe";
84 case BackLeft
: return "back-left";
85 case BackRight
: return "back-right";
86 case BackCenter
: return "back-center";
87 case SideLeft
: return "side-left";
88 case SideRight
: return "side-right";
90 case UpperFrontLeft
: return "upper-front-left";
91 case UpperFrontRight
: return "upper-front-right";
92 case UpperBackLeft
: return "upper-back-left";
93 case UpperBackRight
: return "upper-back-right";
94 case LowerFrontLeft
: return "lower-front-left";
95 case LowerFrontRight
: return "lower-front-right";
96 case LowerBackLeft
: return "lower-back-left";
97 case LowerBackRight
: return "lower-back-right";
99 case Aux0
: return "aux-0";
100 case Aux1
: return "aux-1";
101 case Aux2
: return "aux-2";
102 case Aux3
: return "aux-3";
103 case Aux4
: return "aux-4";
104 case Aux5
: return "aux-5";
105 case Aux6
: return "aux-6";
106 case Aux7
: return "aux-7";
107 case Aux8
: return "aux-8";
108 case Aux9
: return "aux-9";
109 case Aux10
: return "aux-10";
110 case Aux11
: return "aux-11";
111 case Aux12
: return "aux-12";
112 case Aux13
: return "aux-13";
113 case Aux14
: return "aux-14";
114 case Aux15
: return "aux-15";
116 case MaxChannels
: break;
122 void AllocChannels(ALCdevice
*device
, const ALuint main_chans
, const ALuint real_chans
)
124 TRACE("Channel config, Main: %u, Real: %u\n", main_chans
, real_chans
);
126 /* Allocate extra channels for any post-filter output. */
127 const ALuint num_chans
{main_chans
+ real_chans
};
129 TRACE("Allocating %u channels, %zu bytes\n", num_chans
,
130 num_chans
*sizeof(device
->MixBuffer
[0]));
131 device
->MixBuffer
.resize(num_chans
);
132 al::span
<FloatBufferLine
> buffer
{device
->MixBuffer
.data(), device
->MixBuffer
.size()};
134 device
->Dry
.Buffer
= buffer
.first(main_chans
);
135 buffer
= buffer
.subspan(main_chans
);
138 device
->RealOut
.Buffer
= buffer
.first(real_chans
);
139 buffer
= buffer
.subspan(real_chans
);
142 device
->RealOut
.Buffer
= device
->Dry
.Buffer
;
148 ALfloat Config
[MAX_AMBI2D_CHANNELS
];
151 bool MakeSpeakerMap(ALCdevice
*device
, const AmbDecConf
*conf
, ALuint (&speakermap
)[MAX_OUTPUT_CHANNELS
])
153 auto map_spkr
= [device
](const AmbDecConf::SpeakerConf
&speaker
) -> ALuint
155 /* NOTE: AmbDec does not define any standard speaker names, however
156 * for this to work we have to by able to find the output channel
157 * the speaker definition corresponds to. Therefore, OpenAL Soft
158 * requires these channel labels to be recognized:
169 * Additionally, surround51 will acknowledge back speakers for side
170 * channels, and surround51rear will acknowledge side speakers for
171 * back channels, to avoid issues with an ambdec expecting 5.1 to
172 * use the side channels when the device is configured for back,
176 if(speaker
.Name
== "LF")
178 else if(speaker
.Name
== "RF")
180 else if(speaker
.Name
== "CE")
182 else if(speaker
.Name
== "LS")
184 if(device
->FmtChans
== DevFmtX51Rear
)
189 else if(speaker
.Name
== "RS")
191 if(device
->FmtChans
== DevFmtX51Rear
)
196 else if(speaker
.Name
== "LB")
198 if(device
->FmtChans
== DevFmtX51
)
203 else if(speaker
.Name
== "RB")
205 if(device
->FmtChans
== DevFmtX51
)
210 else if(speaker
.Name
== "CB")
214 const char *name
{speaker
.Name
.c_str()};
218 if(sscanf(name
, "AUX%u%c", &n
, &c
) == 1 && n
< 16)
219 ch
= static_cast<Channel
>(Aux0
+n
);
222 ERR("AmbDec speaker label \"%s\" not recognized\n", name
);
223 return INVALID_CHANNEL_INDEX
;
226 const ALuint chidx
{GetChannelIdxByName(device
->RealOut
, ch
)};
227 if(chidx
== INVALID_CHANNEL_INDEX
)
228 ERR("Failed to lookup AmbDec speaker label %s\n", speaker
.Name
.c_str());
231 std::transform(conf
->Speakers
.begin(), conf
->Speakers
.end(), std::begin(speakermap
), map_spkr
);
232 /* Return success if no invalid entries are found. */
233 auto spkrmap_end
= std::begin(speakermap
) + conf
->Speakers
.size();
234 return std::find(std::begin(speakermap
), spkrmap_end
, INVALID_CHANNEL_INDEX
) == spkrmap_end
;
238 constexpr ChannelMap MonoCfg
[1] = {
239 { FrontCenter
, { 1.0f
} },
241 { FrontLeft
, { 5.00000000e-1f
, 2.88675135e-1f
, 5.52305643e-2f
} },
242 { FrontRight
, { 5.00000000e-1f
, -2.88675135e-1f
, 5.52305643e-2f
} },
244 { BackLeft
, { 3.53553391e-1f
, 2.04124145e-1f
, -2.04124145e-1f
} },
245 { FrontLeft
, { 3.53553391e-1f
, 2.04124145e-1f
, 2.04124145e-1f
} },
246 { FrontRight
, { 3.53553391e-1f
, -2.04124145e-1f
, 2.04124145e-1f
} },
247 { BackRight
, { 3.53553391e-1f
, -2.04124145e-1f
, -2.04124145e-1f
} },
249 { SideLeft
, { 3.33000782e-1f
, 1.89084803e-1f
, -2.00042375e-1f
, -2.12307769e-2f
, -1.14579885e-2f
} },
250 { FrontLeft
, { 1.88542860e-1f
, 1.27709292e-1f
, 1.66295695e-1f
, 7.30571517e-2f
, 2.10901184e-2f
} },
251 { FrontRight
, { 1.88542860e-1f
, -1.27709292e-1f
, 1.66295695e-1f
, -7.30571517e-2f
, 2.10901184e-2f
} },
252 { SideRight
, { 3.33000782e-1f
, -1.89084803e-1f
, -2.00042375e-1f
, 2.12307769e-2f
, -1.14579885e-2f
} },
254 { BackLeft
, { 3.33000782e-1f
, 1.89084803e-1f
, -2.00042375e-1f
, -2.12307769e-2f
, -1.14579885e-2f
} },
255 { FrontLeft
, { 1.88542860e-1f
, 1.27709292e-1f
, 1.66295695e-1f
, 7.30571517e-2f
, 2.10901184e-2f
} },
256 { FrontRight
, { 1.88542860e-1f
, -1.27709292e-1f
, 1.66295695e-1f
, -7.30571517e-2f
, 2.10901184e-2f
} },
257 { BackRight
, { 3.33000782e-1f
, -1.89084803e-1f
, -2.00042375e-1f
, 2.12307769e-2f
, -1.14579885e-2f
} },
259 { SideLeft
, { 2.04460341e-1f
, 2.17177926e-1f
, -4.39996780e-2f
, -2.60790269e-2f
, -6.87239792e-2f
} },
260 { FrontLeft
, { 1.58923161e-1f
, 9.21772680e-2f
, 1.59658796e-1f
, 6.66278083e-2f
, 3.84686854e-2f
} },
261 { FrontRight
, { 1.58923161e-1f
, -9.21772680e-2f
, 1.59658796e-1f
, -6.66278083e-2f
, 3.84686854e-2f
} },
262 { SideRight
, { 2.04460341e-1f
, -2.17177926e-1f
, -4.39996780e-2f
, 2.60790269e-2f
, -6.87239792e-2f
} },
263 { BackCenter
, { 2.50001688e-1f
, 0.00000000e+0f
, -2.50000094e-1f
, 0.00000000e+0f
, 6.05133395e-2f
} },
265 { BackLeft
, { 2.04124145e-1f
, 1.08880247e-1f
, -1.88586120e-1f
, -1.29099444e-1f
, 7.45355993e-2f
, 3.73460789e-2f
, 0.00000000e+0f
} },
266 { SideLeft
, { 2.04124145e-1f
, 2.17760495e-1f
, 0.00000000e+0f
, 0.00000000e+0f
, -1.49071198e-1f
, -3.73460789e-2f
, 0.00000000e+0f
} },
267 { FrontLeft
, { 2.04124145e-1f
, 1.08880247e-1f
, 1.88586120e-1f
, 1.29099444e-1f
, 7.45355993e-2f
, 3.73460789e-2f
, 0.00000000e+0f
} },
268 { FrontRight
, { 2.04124145e-1f
, -1.08880247e-1f
, 1.88586120e-1f
, -1.29099444e-1f
, 7.45355993e-2f
, -3.73460789e-2f
, 0.00000000e+0f
} },
269 { SideRight
, { 2.04124145e-1f
, -2.17760495e-1f
, 0.00000000e+0f
, 0.00000000e+0f
, -1.49071198e-1f
, 3.73460789e-2f
, 0.00000000e+0f
} },
270 { BackRight
, { 2.04124145e-1f
, -1.08880247e-1f
, -1.88586120e-1f
, 1.29099444e-1f
, 7.45355993e-2f
, -3.73460789e-2f
, 0.00000000e+0f
} },
273 void InitNearFieldCtrl(ALCdevice
*device
, ALfloat ctrl_dist
, ALuint order
,
274 const al::span
<const ALuint
,MAX_AMBI_ORDER
+1> chans_per_order
)
276 /* NFC is only used when AvgSpeakerDist is greater than 0. */
277 const char *devname
{device
->DeviceName
.c_str()};
278 if(!GetConfigValueBool(devname
, "decoder", "nfc", 0) || !(ctrl_dist
> 0.0f
))
281 device
->AvgSpeakerDist
= clampf(ctrl_dist
, 0.1f
, 10.0f
);
282 TRACE("Using near-field reference distance: %.2f meters\n", device
->AvgSpeakerDist
);
284 auto iter
= std::copy(chans_per_order
.begin(), chans_per_order
.begin()+order
+1,
285 std::begin(device
->NumChannelsPerOrder
));
286 std::fill(iter
, std::end(device
->NumChannelsPerOrder
), 0u);
289 void InitDistanceComp(ALCdevice
*device
, const AmbDecConf
*conf
,
290 const ALuint (&speakermap
)[MAX_OUTPUT_CHANNELS
])
292 auto get_max
= std::bind(maxf
, _1
,
293 std::bind(std::mem_fn(&AmbDecConf::SpeakerConf::Distance
), _2
));
294 const ALfloat maxdist
{
295 std::accumulate(conf
->Speakers
.begin(), conf
->Speakers
.end(), float{0.0f
}, get_max
)};
297 const char *devname
{device
->DeviceName
.c_str()};
298 if(!GetConfigValueBool(devname
, "decoder", "distance-comp", 1) || !(maxdist
> 0.0f
))
301 const auto distSampleScale
= static_cast<ALfloat
>(device
->Frequency
)/SPEEDOFSOUNDMETRESPERSEC
;
302 const auto ChanDelay
= device
->ChannelDelay
.as_span();
304 for(size_t i
{0u};i
< conf
->Speakers
.size();i
++)
306 const AmbDecConf::SpeakerConf
&speaker
= conf
->Speakers
[i
];
307 const ALuint chan
{speakermap
[i
]};
309 /* Distance compensation only delays in steps of the sample rate. This
310 * is a bit less accurate since the delay time falls to the nearest
311 * sample time, but it's far simpler as it doesn't have to deal with
312 * phase offsets. This means at 48khz, for instance, the distance delay
313 * will be in steps of about 7 millimeters.
315 ALfloat delay
{std::floor((maxdist
- speaker
.Distance
)*distSampleScale
+ 0.5f
)};
316 if(delay
> ALfloat
{MAX_DELAY_LENGTH
-1})
318 ERR("Delay for speaker \"%s\" exceeds buffer length (%f > %d)\n",
319 speaker
.Name
.c_str(), delay
, MAX_DELAY_LENGTH
-1);
320 delay
= ALfloat
{MAX_DELAY_LENGTH
-1};
323 ChanDelay
[chan
].Length
= static_cast<ALuint
>(delay
);
324 ChanDelay
[chan
].Gain
= speaker
.Distance
/ maxdist
;
325 TRACE("Channel %u \"%s\" distance compensation: %u samples, %f gain\n", chan
,
326 speaker
.Name
.c_str(), ChanDelay
[chan
].Length
, ChanDelay
[chan
].Gain
);
328 /* Round up to the next 4th sample, so each channel buffer starts
331 total
+= RoundUp(ChanDelay
[chan
].Length
, 4);
336 device
->ChannelDelay
.setSampleCount(total
);
337 ChanDelay
[0].Buffer
= device
->ChannelDelay
.getSamples();
338 auto set_bufptr
= [](const DistanceComp::DistData
&last
, const DistanceComp::DistData
&cur
) -> DistanceComp::DistData
340 DistanceComp::DistData ret
{cur
};
341 ret
.Buffer
= last
.Buffer
+ RoundUp(last
.Length
, 4);
344 std::partial_sum(ChanDelay
.begin(), ChanDelay
.end(), ChanDelay
.begin(), set_bufptr
);
349 auto GetAmbiScales(AmbiNorm scaletype
) noexcept
-> const std::array
<float,MAX_AMBI_CHANNELS
>&
351 if(scaletype
== AmbiNorm::FuMa
) return AmbiScale::FromFuMa
;
352 if(scaletype
== AmbiNorm::SN3D
) return AmbiScale::FromSN3D
;
353 return AmbiScale::FromN3D
;
356 auto GetAmbiLayout(AmbiLayout layouttype
) noexcept
-> const std::array
<uint8_t,MAX_AMBI_CHANNELS
>&
358 if(layouttype
== AmbiLayout::FuMa
) return AmbiIndex::FromFuMa
;
359 return AmbiIndex::FromACN
;
363 void InitPanning(ALCdevice
*device
)
365 al::span
<const ChannelMap
> chanmap
;
368 switch(device
->FmtChans
)
386 chanmap
= X51SideCfg
;
391 chanmap
= X51RearCfg
;
409 if(device
->FmtChans
== DevFmtAmbi3D
)
411 const char *devname
{device
->DeviceName
.c_str()};
412 const std::array
<uint8_t,MAX_AMBI_CHANNELS
> &acnmap
= GetAmbiLayout(device
->mAmbiLayout
);
413 const std::array
<float,MAX_AMBI_CHANNELS
> &n3dscale
= GetAmbiScales(device
->mAmbiScale
);
415 /* For DevFmtAmbi3D, the ambisonic order is already set. */
416 const size_t count
{AmbiChannelsFromOrder(device
->mAmbiOrder
)};
417 std::transform(acnmap
.begin(), acnmap
.begin()+count
, std::begin(device
->Dry
.AmbiMap
),
418 [&n3dscale
](const uint8_t &acn
) noexcept
-> BFChannelConfig
419 { return BFChannelConfig
{1.0f
/n3dscale
[acn
], acn
}; }
421 AllocChannels(device
, static_cast<ALuint
>(count
), 0);
423 ALfloat nfc_delay
{ConfigValueFloat(devname
, "decoder", "nfc-ref-delay").value_or(0.0f
)};
426 static constexpr ALuint chans_per_order
[MAX_AMBI_ORDER
+1]{ 1, 3, 5, 7 };
427 InitNearFieldCtrl(device
, nfc_delay
* SPEEDOFSOUNDMETRESPERSEC
, device
->mAmbiOrder
,
433 ChannelDec chancoeffs
[MAX_OUTPUT_CHANNELS
]{};
434 ALuint idxmap
[MAX_OUTPUT_CHANNELS
]{};
435 for(size_t i
{0u};i
< chanmap
.size();++i
)
437 const ALuint idx
{GetChannelIdxByName(device
->RealOut
, chanmap
[i
].ChanName
)};
438 if(idx
== INVALID_CHANNEL_INDEX
)
440 ERR("Failed to find %s channel in device\n",
441 GetLabelFromChannel(chanmap
[i
].ChanName
));
445 std::copy_n(chanmap
[i
].Config
, coeffcount
, chancoeffs
[i
]);
448 /* For non-DevFmtAmbi3D, set the ambisonic order given the mixing
449 * channel count. Built-in speaker decoders are always 2D, so just
450 * reverse that calculation.
452 device
->mAmbiOrder
= (coeffcount
-1) / 2;
454 std::transform(AmbiIndex::From2D
.begin(), AmbiIndex::From2D
.begin()+coeffcount
,
455 std::begin(device
->Dry
.AmbiMap
),
456 [](const uint8_t &index
) noexcept
{ return BFChannelConfig
{1.0f
, index
}; }
458 AllocChannels(device
, coeffcount
, device
->channelsFromFmt());
460 TRACE("Enabling %s-order%s ambisonic decoder\n",
461 (coeffcount
> 5) ? "third" :
462 (coeffcount
> 3) ? "second" : "first",
465 device
->AmbiDecoder
= al::make_unique
<BFormatDec
>(coeffcount
,
466 static_cast<ALsizei
>(chanmap
.size()), chancoeffs
, idxmap
);
470 void InitCustomPanning(ALCdevice
*device
, bool hqdec
, const AmbDecConf
*conf
,
471 const ALuint (&speakermap
)[MAX_OUTPUT_CHANNELS
])
473 static constexpr ALuint chans_per_order2d
[MAX_AMBI_ORDER
+1] = { 1, 2, 2, 2 };
474 static constexpr ALuint chans_per_order3d
[MAX_AMBI_ORDER
+1] = { 1, 3, 5, 7 };
476 if(!hqdec
&& conf
->FreqBands
!= 1)
477 ERR("Basic renderer uses the high-frequency matrix as single-band (xover_freq = %.0fhz)\n",
480 const ALuint order
{(conf
->ChanMask
> AMBI_2ORDER_MASK
) ? 3u :
481 (conf
->ChanMask
> AMBI_1ORDER_MASK
) ? 2u : 1u};
482 device
->mAmbiOrder
= order
;
485 if((conf
->ChanMask
&AMBI_PERIPHONIC_MASK
))
487 count
= static_cast<ALuint
>(AmbiChannelsFromOrder(order
));
488 std::transform(AmbiIndex::From3D
.begin(), AmbiIndex::From3D
.begin()+count
,
489 std::begin(device
->Dry
.AmbiMap
),
490 [](const uint8_t &index
) noexcept
{ return BFChannelConfig
{1.0f
, index
}; }
495 count
= static_cast<ALuint
>(Ambi2DChannelsFromOrder(order
));
496 std::transform(AmbiIndex::From2D
.begin(), AmbiIndex::From2D
.begin()+count
,
497 std::begin(device
->Dry
.AmbiMap
),
498 [](const uint8_t &index
) noexcept
{ return BFChannelConfig
{1.0f
, index
}; }
501 AllocChannels(device
, count
, device
->channelsFromFmt());
503 TRACE("Enabling %s-band %s-order%s ambisonic decoder\n",
504 (!hqdec
|| conf
->FreqBands
== 1) ? "single" : "dual",
505 (conf
->ChanMask
> AMBI_2ORDER_MASK
) ? "third" :
506 (conf
->ChanMask
> AMBI_1ORDER_MASK
) ? "second" : "first",
507 (conf
->ChanMask
&AMBI_PERIPHONIC_MASK
) ? " periphonic" : ""
509 device
->AmbiDecoder
= al::make_unique
<BFormatDec
>(conf
, hqdec
, count
, device
->Frequency
,
512 auto accum_spkr_dist
= std::bind(std::plus
<float>{}, _1
,
513 std::bind(std::mem_fn(&AmbDecConf::SpeakerConf::Distance
), _2
));
514 const ALfloat avg_dist
{
515 std::accumulate(conf
->Speakers
.begin(), conf
->Speakers
.end(), 0.0f
, accum_spkr_dist
) /
516 static_cast<ALfloat
>(conf
->Speakers
.size())};
517 InitNearFieldCtrl(device
, avg_dist
, order
,
518 (conf
->ChanMask
&AMBI_PERIPHONIC_MASK
) ? chans_per_order3d
: chans_per_order2d
);
520 InitDistanceComp(device
, conf
, speakermap
);
523 void InitHrtfPanning(ALCdevice
*device
)
525 static constexpr AngularPoint AmbiPoints
[]{
526 { Deg2Rad( 35.264390f
), Deg2Rad( -45.000000f
) },
527 { Deg2Rad( 35.264390f
), Deg2Rad( 45.000000f
) },
528 { Deg2Rad( 35.264390f
), Deg2Rad( 135.000000f
) },
529 { Deg2Rad( 35.264390f
), Deg2Rad(-135.000000f
) },
530 { Deg2Rad(-35.264390f
), Deg2Rad( -45.000000f
) },
531 { Deg2Rad(-35.264390f
), Deg2Rad( 45.000000f
) },
532 { Deg2Rad(-35.264390f
), Deg2Rad( 135.000000f
) },
533 { Deg2Rad(-35.264390f
), Deg2Rad(-135.000000f
) },
534 { Deg2Rad( 0.000000f
), Deg2Rad( -20.905157f
) },
535 { Deg2Rad( 0.000000f
), Deg2Rad( 20.905157f
) },
536 { Deg2Rad( 0.000000f
), Deg2Rad( 159.094843f
) },
537 { Deg2Rad( 0.000000f
), Deg2Rad(-159.094843f
) },
538 { Deg2Rad( 20.905157f
), Deg2Rad( -90.000000f
) },
539 { Deg2Rad(-20.905157f
), Deg2Rad( -90.000000f
) },
540 { Deg2Rad(-20.905157f
), Deg2Rad( 90.000000f
) },
541 { Deg2Rad( 20.905157f
), Deg2Rad( 90.000000f
) },
542 { Deg2Rad( 69.094843f
), Deg2Rad( 0.000000f
) },
543 { Deg2Rad(-69.094843f
), Deg2Rad( 0.000000f
) },
544 { Deg2Rad(-69.094843f
), Deg2Rad( 180.000000f
) },
545 { Deg2Rad( 69.094843f
), Deg2Rad( 180.000000f
) },
547 static constexpr ALfloat AmbiMatrix
[][MAX_AMBI_CHANNELS
]{
548 { 5.00000000e-02f
, 5.00000000e-02f
, 5.00000000e-02f
, 5.00000000e-02f
, 6.45497224e-02f
, 6.45497224e-02f
, 0.00000000e+00f
, 6.45497224e-02f
, 0.00000000e+00f
, 1.48264644e-02f
, 6.33865691e-02f
, 1.01126676e-01f
, -7.36485380e-02f
, -1.09260065e-02f
, 7.08683387e-02f
, -1.01622099e-01f
},
549 { 5.00000000e-02f
, -5.00000000e-02f
, 5.00000000e-02f
, 5.00000000e-02f
, -6.45497224e-02f
, -6.45497224e-02f
, 0.00000000e+00f
, 6.45497224e-02f
, 0.00000000e+00f
, -1.48264644e-02f
, -6.33865691e-02f
, -1.01126676e-01f
, -7.36485380e-02f
, -1.09260065e-02f
, 7.08683387e-02f
, -1.01622099e-01f
},
550 { 5.00000000e-02f
, -5.00000000e-02f
, 5.00000000e-02f
, -5.00000000e-02f
, 6.45497224e-02f
, -6.45497224e-02f
, 0.00000000e+00f
, -6.45497224e-02f
, 0.00000000e+00f
, -1.48264644e-02f
, 6.33865691e-02f
, -1.01126676e-01f
, -7.36485380e-02f
, 1.09260065e-02f
, 7.08683387e-02f
, 1.01622099e-01f
},
551 { 5.00000000e-02f
, 5.00000000e-02f
, 5.00000000e-02f
, -5.00000000e-02f
, -6.45497224e-02f
, 6.45497224e-02f
, 0.00000000e+00f
, -6.45497224e-02f
, 0.00000000e+00f
, 1.48264644e-02f
, -6.33865691e-02f
, 1.01126676e-01f
, -7.36485380e-02f
, 1.09260065e-02f
, 7.08683387e-02f
, 1.01622099e-01f
},
552 { 5.00000000e-02f
, 5.00000000e-02f
, -5.00000000e-02f
, 5.00000000e-02f
, 6.45497224e-02f
, -6.45497224e-02f
, 0.00000000e+00f
, -6.45497224e-02f
, 0.00000000e+00f
, 1.48264644e-02f
, -6.33865691e-02f
, 1.01126676e-01f
, 7.36485380e-02f
, -1.09260065e-02f
, -7.08683387e-02f
, -1.01622099e-01f
},
553 { 5.00000000e-02f
, -5.00000000e-02f
, -5.00000000e-02f
, 5.00000000e-02f
, -6.45497224e-02f
, 6.45497224e-02f
, 0.00000000e+00f
, -6.45497224e-02f
, 0.00000000e+00f
, -1.48264644e-02f
, 6.33865691e-02f
, -1.01126676e-01f
, 7.36485380e-02f
, -1.09260065e-02f
, -7.08683387e-02f
, -1.01622099e-01f
},
554 { 5.00000000e-02f
, -5.00000000e-02f
, -5.00000000e-02f
, -5.00000000e-02f
, 6.45497224e-02f
, 6.45497224e-02f
, 0.00000000e+00f
, 6.45497224e-02f
, 0.00000000e+00f
, -1.48264644e-02f
, -6.33865691e-02f
, -1.01126676e-01f
, 7.36485380e-02f
, 1.09260065e-02f
, -7.08683387e-02f
, 1.01622099e-01f
},
555 { 5.00000000e-02f
, 5.00000000e-02f
, -5.00000000e-02f
, -5.00000000e-02f
, -6.45497224e-02f
, -6.45497224e-02f
, 0.00000000e+00f
, 6.45497224e-02f
, 0.00000000e+00f
, 1.48264644e-02f
, 6.33865691e-02f
, 1.01126676e-01f
, 7.36485380e-02f
, 1.09260065e-02f
, -7.08683387e-02f
, 1.01622099e-01f
},
556 { 5.00000000e-02f
, 3.09016994e-02f
, 0.00000000e+00f
, 8.09016994e-02f
, 6.45497224e-02f
, 0.00000000e+00f
, -5.59016994e-02f
, 0.00000000e+00f
, 7.21687836e-02f
, 7.76323754e-02f
, 0.00000000e+00f
, -1.49775925e-01f
, 0.00000000e+00f
, -2.95083663e-02f
, 0.00000000e+00f
, 7.76323754e-02f
},
557 { 5.00000000e-02f
, -3.09016994e-02f
, 0.00000000e+00f
, 8.09016994e-02f
, -6.45497224e-02f
, 0.00000000e+00f
, -5.59016994e-02f
, 0.00000000e+00f
, 7.21687836e-02f
, -7.76323754e-02f
, 0.00000000e+00f
, 1.49775925e-01f
, 0.00000000e+00f
, -2.95083663e-02f
, 0.00000000e+00f
, 7.76323754e-02f
},
558 { 5.00000000e-02f
, -3.09016994e-02f
, 0.00000000e+00f
, -8.09016994e-02f
, 6.45497224e-02f
, 0.00000000e+00f
, -5.59016994e-02f
, 0.00000000e+00f
, 7.21687836e-02f
, -7.76323754e-02f
, 0.00000000e+00f
, 1.49775925e-01f
, 0.00000000e+00f
, 2.95083663e-02f
, 0.00000000e+00f
, -7.76323754e-02f
},
559 { 5.00000000e-02f
, 3.09016994e-02f
, 0.00000000e+00f
, -8.09016994e-02f
, -6.45497224e-02f
, 0.00000000e+00f
, -5.59016994e-02f
, 0.00000000e+00f
, 7.21687836e-02f
, 7.76323754e-02f
, 0.00000000e+00f
, -1.49775925e-01f
, 0.00000000e+00f
, 2.95083663e-02f
, 0.00000000e+00f
, -7.76323754e-02f
},
560 { 5.00000000e-02f
, 8.09016994e-02f
, 3.09016994e-02f
, 0.00000000e+00f
, 0.00000000e+00f
, 6.45497224e-02f
, -3.45491503e-02f
, 0.00000000e+00f
, -8.44966837e-02f
, -4.79794466e-02f
, 0.00000000e+00f
, -6.77901327e-02f
, 3.03448665e-02f
, 0.00000000e+00f
, -1.65948192e-01f
, 0.00000000e+00f
},
561 { 5.00000000e-02f
, 8.09016994e-02f
, -3.09016994e-02f
, 0.00000000e+00f
, 0.00000000e+00f
, -6.45497224e-02f
, -3.45491503e-02f
, 0.00000000e+00f
, -8.44966837e-02f
, -4.79794466e-02f
, 0.00000000e+00f
, -6.77901327e-02f
, -3.03448665e-02f
, 0.00000000e+00f
, 1.65948192e-01f
, 0.00000000e+00f
},
562 { 5.00000000e-02f
, -8.09016994e-02f
, -3.09016994e-02f
, 0.00000000e+00f
, 0.00000000e+00f
, 6.45497224e-02f
, -3.45491503e-02f
, 0.00000000e+00f
, -8.44966837e-02f
, 4.79794466e-02f
, 0.00000000e+00f
, 6.77901327e-02f
, -3.03448665e-02f
, 0.00000000e+00f
, 1.65948192e-01f
, 0.00000000e+00f
},
563 { 5.00000000e-02f
, -8.09016994e-02f
, 3.09016994e-02f
, 0.00000000e+00f
, 0.00000000e+00f
, -6.45497224e-02f
, -3.45491503e-02f
, 0.00000000e+00f
, -8.44966837e-02f
, 4.79794466e-02f
, 0.00000000e+00f
, 6.77901327e-02f
, 3.03448665e-02f
, 0.00000000e+00f
, -1.65948192e-01f
, 0.00000000e+00f
},
564 { 5.00000000e-02f
, 0.00000000e+00f
, 8.09016994e-02f
, 3.09016994e-02f
, 0.00000000e+00f
, 0.00000000e+00f
, 9.04508497e-02f
, 6.45497224e-02f
, 1.23279000e-02f
, 0.00000000e+00f
, 0.00000000e+00f
, 0.00000000e+00f
, 7.94438918e-02f
, 1.12611206e-01f
, -2.42115150e-02f
, 1.25611822e-01f
},
565 { 5.00000000e-02f
, 0.00000000e+00f
, -8.09016994e-02f
, 3.09016994e-02f
, 0.00000000e+00f
, 0.00000000e+00f
, 9.04508497e-02f
, -6.45497224e-02f
, 1.23279000e-02f
, 0.00000000e+00f
, 0.00000000e+00f
, 0.00000000e+00f
, -7.94438918e-02f
, 1.12611206e-01f
, 2.42115150e-02f
, 1.25611822e-01f
},
566 { 5.00000000e-02f
, 0.00000000e+00f
, -8.09016994e-02f
, -3.09016994e-02f
, 0.00000000e+00f
, 0.00000000e+00f
, 9.04508497e-02f
, 6.45497224e-02f
, 1.23279000e-02f
, 0.00000000e+00f
, 0.00000000e+00f
, 0.00000000e+00f
, -7.94438918e-02f
, -1.12611206e-01f
, 2.42115150e-02f
, -1.25611822e-01f
},
567 { 5.00000000e-02f
, 0.00000000e+00f
, 8.09016994e-02f
, -3.09016994e-02f
, 0.00000000e+00f
, 0.00000000e+00f
, 9.04508497e-02f
, -6.45497224e-02f
, 1.23279000e-02f
, 0.00000000e+00f
, 0.00000000e+00f
, 0.00000000e+00f
, 7.94438918e-02f
, -1.12611206e-01f
, -2.42115150e-02f
, -1.25611822e-01f
}
569 static constexpr ALfloat AmbiOrderHFGain1O
[MAX_AMBI_ORDER
+1]{
570 3.16227766e+00f
, 1.82574186e+00f
571 }, AmbiOrderHFGain2O
[MAX_AMBI_ORDER
+1]{
572 2.35702260e+00f
, 1.82574186e+00f
, 9.42809042e-01f
573 }, AmbiOrderHFGain3O
[MAX_AMBI_ORDER
+1]{
574 1.86508671e+00f
, 1.60609389e+00f
, 1.14205530e+00f
, 5.68379553e-01f
576 static constexpr ALuint ChansPerOrder
[MAX_AMBI_ORDER
+1]{ 1, 3, 5, 7 };
577 const ALfloat
*AmbiOrderHFGain
{AmbiOrderHFGain1O
};
579 static_assert(al::size(AmbiPoints
) == al::size(AmbiMatrix
), "Ambisonic HRTF mismatch");
581 /* Don't bother with HOA when using full HRTF rendering. Nothing needs it,
582 * and it eases the CPU/memory load.
584 device
->mRenderMode
= HrtfRender
;
585 ALuint ambi_order
{1};
586 if(auto modeopt
= ConfigValueStr(device
->DeviceName
.c_str(), nullptr, "hrtf-mode"))
588 struct HrtfModeEntry
{
593 static constexpr HrtfModeEntry hrtf_modes
[]{
594 { "full", HrtfRender
, 1 },
595 { "ambi1", NormalRender
, 1 },
596 { "ambi2", NormalRender
, 2 },
597 { "ambi3", NormalRender
, 3 },
600 const char *mode
{modeopt
->c_str()};
601 if(al::strcasecmp(mode
, "basic") == 0)
603 ERR("HRTF mode \"%s\" deprecated, substituting \"%s\"\n", mode
, "ambi2");
607 auto match_entry
= [mode
](const HrtfModeEntry
&entry
) -> bool
608 { return al::strcasecmp(mode
, entry
.name
) == 0; };
609 auto iter
= std::find_if(std::begin(hrtf_modes
), std::end(hrtf_modes
), match_entry
);
610 if(iter
== std::end(hrtf_modes
))
611 ERR("Unexpected hrtf-mode: %s\n", mode
);
614 device
->mRenderMode
= iter
->mode
;
615 ambi_order
= iter
->order
;
618 TRACE("%s HRTF rendering enabled, using \"%s\"\n",
619 (device
->mRenderMode
== HrtfRender
) ? "Full" :
620 (ambi_order
>= 3) ? "Third-Order" :
621 (ambi_order
== 2) ? "Second-Order" :
622 (ambi_order
== 1) ? "First-Order" : "Unknown",
623 device
->HrtfName
.c_str());
626 AmbiOrderHFGain
= AmbiOrderHFGain3O
;
627 else if(ambi_order
== 2)
628 AmbiOrderHFGain
= AmbiOrderHFGain2O
;
629 else if(ambi_order
== 1)
630 AmbiOrderHFGain
= AmbiOrderHFGain1O
;
631 device
->mAmbiOrder
= ambi_order
;
633 const size_t count
{AmbiChannelsFromOrder(ambi_order
)};
634 device
->mHrtfState
= DirectHrtfState::Create(count
);
636 std::transform(AmbiIndex::From3D
.begin(), AmbiIndex::From3D
.begin()+count
,
637 std::begin(device
->Dry
.AmbiMap
),
638 [](const uint8_t &index
) noexcept
{ return BFChannelConfig
{1.0f
, index
}; }
640 AllocChannels(device
, static_cast<ALuint
>(count
), device
->channelsFromFmt());
642 BuildBFormatHrtf(device
->mHrtf
, device
->mHrtfState
.get(), AmbiPoints
, AmbiMatrix
,
645 HrtfEntry
*Hrtf
{device
->mHrtf
};
646 InitNearFieldCtrl(device
, Hrtf
->field
[0].distance
, ambi_order
, ChansPerOrder
);
649 void InitUhjPanning(ALCdevice
*device
)
651 /* UHJ is always 2D first-order. */
652 static constexpr size_t count
{Ambi2DChannelsFromOrder(1)};
654 device
->mAmbiOrder
= 1;
656 auto acnmap_end
= AmbiIndex::FromFuMa
.begin() + count
;
657 std::transform(AmbiIndex::FromFuMa
.begin(), acnmap_end
, std::begin(device
->Dry
.AmbiMap
),
658 [](const uint8_t &acn
) noexcept
-> BFChannelConfig
659 { return BFChannelConfig
{1.0f
/AmbiScale::FromFuMa
[acn
], acn
}; }
661 AllocChannels(device
, ALuint
{count
}, device
->channelsFromFmt());
666 void aluInitRenderer(ALCdevice
*device
, ALint hrtf_id
, HrtfRequestMode hrtf_appreq
, HrtfRequestMode hrtf_userreq
)
668 /* Hold the HRTF the device last used, in case it's used again. */
669 HrtfEntry
*old_hrtf
{device
->mHrtf
};
671 device
->mHrtfState
= nullptr;
672 device
->mHrtf
= nullptr;
673 device
->HrtfName
.clear();
674 device
->mRenderMode
= NormalRender
;
676 if(device
->FmtChans
!= DevFmtStereo
)
681 if(hrtf_appreq
== Hrtf_Enable
)
682 device
->HrtfStatus
= ALC_HRTF_UNSUPPORTED_FORMAT_SOFT
;
684 const char *layout
{nullptr};
685 switch(device
->FmtChans
)
687 case DevFmtQuad
: layout
= "quad"; break;
688 case DevFmtX51
: /* fall-through */
689 case DevFmtX51Rear
: layout
= "surround51"; break;
690 case DevFmtX61
: layout
= "surround61"; break;
691 case DevFmtX71
: layout
= "surround71"; break;
692 /* Mono, Stereo, and Ambisonics output don't use custom decoders. */
699 const char *devname
{device
->DeviceName
.c_str()};
700 ALuint speakermap
[MAX_OUTPUT_CHANNELS
];
701 AmbDecConf
*pconf
{nullptr};
705 if(auto decopt
= ConfigValueStr(devname
, "decoder", layout
))
707 if(!conf
.load(decopt
->c_str()))
708 ERR("Failed to load layout file %s\n", decopt
->c_str());
709 else if(conf
.Speakers
.size() > MAX_OUTPUT_CHANNELS
)
710 ERR("Unsupported speaker count %zu (max %d)\n", conf
.Speakers
.size(),
711 MAX_OUTPUT_CHANNELS
);
712 else if(conf
.ChanMask
> AMBI_3ORDER_MASK
)
713 ERR("Unsupported channel mask 0x%04x (max 0x%x)\n", conf
.ChanMask
,
715 else if(MakeSpeakerMap(device
, &conf
, speakermap
))
724 int hqdec
{GetConfigValueBool(devname
, "decoder", "hq-mode", 1)};
725 InitCustomPanning(device
, !!hqdec
, pconf
, speakermap
);
727 if(device
->AmbiDecoder
)
728 device
->PostProcess
= &ALCdevice::ProcessAmbiDec
;
732 bool headphones
{device
->IsHeadphones
!= AL_FALSE
};
733 if(device
->Type
!= Loopback
)
735 if(auto modeopt
= ConfigValueStr(device
->DeviceName
.c_str(), nullptr, "stereo-mode"))
737 const char *mode
{modeopt
->c_str()};
738 if(al::strcasecmp(mode
, "headphones") == 0)
740 else if(al::strcasecmp(mode
, "speakers") == 0)
742 else if(al::strcasecmp(mode
, "auto") != 0)
743 ERR("Unexpected stereo-mode: %s\n", mode
);
747 if(hrtf_userreq
== Hrtf_Default
)
749 bool usehrtf
= (headphones
&& hrtf_appreq
!= Hrtf_Disable
) ||
750 (hrtf_appreq
== Hrtf_Enable
);
751 if(!usehrtf
) goto no_hrtf
;
753 device
->HrtfStatus
= ALC_HRTF_ENABLED_SOFT
;
754 if(headphones
&& hrtf_appreq
!= Hrtf_Disable
)
755 device
->HrtfStatus
= ALC_HRTF_HEADPHONES_DETECTED_SOFT
;
759 if(hrtf_userreq
!= Hrtf_Enable
)
761 if(hrtf_appreq
== Hrtf_Enable
)
762 device
->HrtfStatus
= ALC_HRTF_DENIED_SOFT
;
765 device
->HrtfStatus
= ALC_HRTF_REQUIRED_SOFT
;
768 if(device
->HrtfList
.empty())
769 device
->HrtfList
= EnumerateHrtf(device
->DeviceName
.c_str());
771 if(hrtf_id
>= 0 && static_cast<ALuint
>(hrtf_id
) < device
->HrtfList
.size())
773 const EnumeratedHrtf
&entry
= device
->HrtfList
[static_cast<ALuint
>(hrtf_id
)];
774 HrtfEntry
*hrtf
{GetLoadedHrtf(entry
.hrtf
)};
775 if(hrtf
&& hrtf
->sampleRate
== device
->Frequency
)
777 device
->mHrtf
= hrtf
;
778 device
->HrtfName
= entry
.name
;
786 auto find_hrtf
= [device
](const EnumeratedHrtf
&entry
) -> bool
788 HrtfEntry
*hrtf
{GetLoadedHrtf(entry
.hrtf
)};
789 if(!hrtf
) return false;
790 if(hrtf
->sampleRate
!= device
->Frequency
)
795 device
->mHrtf
= hrtf
;
796 device
->HrtfName
= entry
.name
;
799 std::find_if(device
->HrtfList
.cbegin(), device
->HrtfList
.cend(), find_hrtf
);
808 InitHrtfPanning(device
);
809 device
->PostProcess
= &ALCdevice::ProcessHrtf
;
812 device
->HrtfStatus
= ALC_HRTF_UNSUPPORTED_FORMAT_SOFT
;
819 device
->mRenderMode
= StereoPair
;
821 if(device
->Type
!= Loopback
)
823 if(auto cflevopt
= ConfigValueInt(device
->DeviceName
.c_str(), nullptr, "cf_level"))
825 if(*cflevopt
> 0 && *cflevopt
<= 6)
827 device
->Bs2b
= al::make_unique
<bs2b
>();
828 bs2b_set_params(device
->Bs2b
.get(), *cflevopt
,
829 static_cast<int>(device
->Frequency
));
830 TRACE("BS2B enabled\n");
832 device
->PostProcess
= &ALCdevice::ProcessBs2b
;
838 if(auto encopt
= ConfigValueStr(device
->DeviceName
.c_str(), nullptr, "stereo-encoding"))
840 const char *mode
{encopt
->c_str()};
841 if(al::strcasecmp(mode
, "uhj") == 0)
842 device
->mRenderMode
= NormalRender
;
843 else if(al::strcasecmp(mode
, "panpot") != 0)
844 ERR("Unexpected stereo-encoding: %s\n", mode
);
846 if(device
->mRenderMode
== NormalRender
)
848 device
->Uhj_Encoder
= al::make_unique
<Uhj2Encoder
>();
849 TRACE("UHJ enabled\n");
850 InitUhjPanning(device
);
851 device
->PostProcess
= &ALCdevice::ProcessUhj
;
855 TRACE("Stereo rendering\n");
857 device
->PostProcess
= &ALCdevice::ProcessAmbiDec
;
861 void aluInitEffectPanning(ALeffectslot
*slot
, ALCdevice
*device
)
863 const size_t count
{AmbiChannelsFromOrder(device
->mAmbiOrder
)};
864 slot
->MixBuffer
.resize(count
);
865 slot
->MixBuffer
.shrink_to_fit();
867 auto acnmap_end
= AmbiIndex::From3D
.begin() + count
;
868 auto iter
= std::transform(AmbiIndex::From3D
.begin(), acnmap_end
, slot
->Wet
.AmbiMap
.begin(),
869 [](const uint8_t &acn
) noexcept
-> BFChannelConfig
870 { return BFChannelConfig
{1.0f
, acn
}; }
872 std::fill(iter
, slot
->Wet
.AmbiMap
.end(), BFChannelConfig
{});
873 slot
->Wet
.Buffer
= {slot
->MixBuffer
.data(), slot
->MixBuffer
.size()};
877 void CalcAmbiCoeffs(const float y
, const float z
, const float x
, const float spread
,
878 const al::span
<float,MAX_AMBI_CHANNELS
> coeffs
)
881 coeffs
[0] = 1.0f
; /* ACN 0 = 1 */
883 coeffs
[1] = 1.732050808f
* y
; /* ACN 1 = sqrt(3) * Y */
884 coeffs
[2] = 1.732050808f
* z
; /* ACN 2 = sqrt(3) * Z */
885 coeffs
[3] = 1.732050808f
* x
; /* ACN 3 = sqrt(3) * X */
887 coeffs
[4] = 3.872983346f
* x
* y
; /* ACN 4 = sqrt(15) * X * Y */
888 coeffs
[5] = 3.872983346f
* y
* z
; /* ACN 5 = sqrt(15) * Y * Z */
889 coeffs
[6] = 1.118033989f
* (z
*z
*3.0f
- 1.0f
); /* ACN 6 = sqrt(5)/2 * (3*Z*Z - 1) */
890 coeffs
[7] = 3.872983346f
* x
* z
; /* ACN 7 = sqrt(15) * X * Z */
891 coeffs
[8] = 1.936491673f
* (x
*x
- y
*y
); /* ACN 8 = sqrt(15)/2 * (X*X - Y*Y) */
893 coeffs
[9] = 2.091650066f
* y
* (x
*x
*3.0f
- y
*y
); /* ACN 9 = sqrt(35/8) * Y * (3*X*X - Y*Y) */
894 coeffs
[10] = 10.246950766f
* z
* x
* y
; /* ACN 10 = sqrt(105) * Z * X * Y */
895 coeffs
[11] = 1.620185175f
* y
* (z
*z
*5.0f
- 1.0f
); /* ACN 11 = sqrt(21/8) * Y * (5*Z*Z - 1) */
896 coeffs
[12] = 1.322875656f
* z
* (z
*z
*5.0f
- 3.0f
); /* ACN 12 = sqrt(7)/2 * Z * (5*Z*Z - 3) */
897 coeffs
[13] = 1.620185175f
* x
* (z
*z
*5.0f
- 1.0f
); /* ACN 13 = sqrt(21/8) * X * (5*Z*Z - 1) */
898 coeffs
[14] = 5.123475383f
* z
* (x
*x
- y
*y
); /* ACN 14 = sqrt(105)/2 * Z * (X*X - Y*Y) */
899 coeffs
[15] = 2.091650066f
* x
* (x
*x
- y
*y
*3.0f
); /* ACN 15 = sqrt(35/8) * X * (X*X - 3*Y*Y) */
901 /* ACN 16 = sqrt(35)*3/2 * X * Y * (X*X - Y*Y) */
902 /* ACN 17 = sqrt(35/2)*3/2 * (3*X*X - Y*Y) * Y * Z */
903 /* ACN 18 = sqrt(5)*3/2 * X * Y * (7*Z*Z - 1) */
904 /* ACN 19 = sqrt(5/2)*3/2 * Y * Z * (7*Z*Z - 3) */
905 /* ACN 20 = 3/8 * (35*Z*Z*Z*Z - 30*Z*Z + 3) */
906 /* ACN 21 = sqrt(5/2)*3/2 * X * Z * (7*Z*Z - 3) */
907 /* ACN 22 = sqrt(5)*3/4 * (X*X - Y*Y) * (7*Z*Z - 1) */
908 /* ACN 23 = sqrt(35/2)*3/2 * (X*X - 3*Y*Y) * X * Z */
909 /* ACN 24 = sqrt(35)*3/8 * (X*X*X*X - 6*X*X*Y*Y + Y*Y*Y*Y) */
913 /* Implement the spread by using a spherical source that subtends the
915 * http://www.ppsloan.org/publications/StupidSH36.pdf - Appendix A3
917 * When adjusted for N3D normalization instead of SN3D, these
920 * ZH0 = -sqrt(pi) * (-1+ca);
921 * ZH1 = 0.5*sqrt(pi) * sa*sa;
922 * ZH2 = -0.5*sqrt(pi) * ca*(-1+ca)*(ca+1);
923 * ZH3 = -0.125*sqrt(pi) * (-1+ca)*(ca+1)*(5*ca*ca - 1);
924 * ZH4 = -0.125*sqrt(pi) * ca*(-1+ca)*(ca+1)*(7*ca*ca - 3);
925 * ZH5 = -0.0625*sqrt(pi) * (-1+ca)*(ca+1)*(21*ca*ca*ca*ca - 14*ca*ca + 1);
927 * The gain of the source is compensated for size, so that the
928 * loudness doesn't depend on the spread. Thus:
931 * ZH1 = 0.5f * (ca+1.0f);
932 * ZH2 = 0.5f * (ca+1.0f)*ca;
933 * ZH3 = 0.125f * (ca+1.0f)*(5.0f*ca*ca - 1.0f);
934 * ZH4 = 0.125f * (ca+1.0f)*(7.0f*ca*ca - 3.0f)*ca;
935 * ZH5 = 0.0625f * (ca+1.0f)*(21.0f*ca*ca*ca*ca - 14.0f*ca*ca + 1.0f);
937 const float ca
{std::cos(spread
* 0.5f
)};
938 /* Increase the source volume by up to +3dB for a full spread. */
939 const float scale
{std::sqrt(1.0f
+ spread
/al::MathDefs
<float>::Tau())};
941 const float ZH0_norm
{scale
};
942 const float ZH1_norm
{scale
* 0.5f
* (ca
+1.f
)};
943 const float ZH2_norm
{scale
* 0.5f
* (ca
+1.f
)*ca
};
944 const float ZH3_norm
{scale
* 0.125f
* (ca
+1.f
)*(5.f
*ca
*ca
-1.f
)};
947 coeffs
[0] *= ZH0_norm
;
949 coeffs
[1] *= ZH1_norm
;
950 coeffs
[2] *= ZH1_norm
;
951 coeffs
[3] *= ZH1_norm
;
953 coeffs
[4] *= ZH2_norm
;
954 coeffs
[5] *= ZH2_norm
;
955 coeffs
[6] *= ZH2_norm
;
956 coeffs
[7] *= ZH2_norm
;
957 coeffs
[8] *= ZH2_norm
;
959 coeffs
[9] *= ZH3_norm
;
960 coeffs
[10] *= ZH3_norm
;
961 coeffs
[11] *= ZH3_norm
;
962 coeffs
[12] *= ZH3_norm
;
963 coeffs
[13] *= ZH3_norm
;
964 coeffs
[14] *= ZH3_norm
;
965 coeffs
[15] *= ZH3_norm
;
969 void ComputePanGains(const MixParams
*mix
, const float*RESTRICT coeffs
, const float ingain
,
970 const al::span
<float,MAX_OUTPUT_CHANNELS
> gains
)
972 auto ambimap
= mix
->AmbiMap
.cbegin();
974 auto iter
= std::transform(ambimap
, ambimap
+mix
->Buffer
.size(), gains
.begin(),
975 [coeffs
,ingain
](const BFChannelConfig
&chanmap
) noexcept
-> float
976 { return chanmap
.Scale
* coeffs
[chanmap
.Index
] * ingain
; }
978 std::fill(iter
, gains
.end(), 0.0f
);