🛠️🐜 Antkeeper superbuild with dependencies included https://antkeeper.com
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

126 lines
5.0 KiB

  1. #include "config.h"
  2. #include "mixer.h"
  3. #include <cmath>
  4. #include "alnumbers.h"
  5. #include "devformat.h"
  6. #include "device.h"
  7. #include "mixer/defs.h"
  8. struct CTag;
  9. MixerFunc MixSamples{Mix_<CTag>};
  10. std::array<float,MaxAmbiChannels> CalcAmbiCoeffs(const float y, const float z, const float x,
  11. const float spread)
  12. {
  13. std::array<float,MaxAmbiChannels> coeffs;
  14. /* Zeroth-order */
  15. coeffs[0] = 1.0f; /* ACN 0 = 1 */
  16. /* First-order */
  17. coeffs[1] = al::numbers::sqrt3_v<float> * y; /* ACN 1 = sqrt(3) * Y */
  18. coeffs[2] = al::numbers::sqrt3_v<float> * z; /* ACN 2 = sqrt(3) * Z */
  19. coeffs[3] = al::numbers::sqrt3_v<float> * x; /* ACN 3 = sqrt(3) * X */
  20. /* Second-order */
  21. const float xx{x*x}, yy{y*y}, zz{z*z}, xy{x*y}, yz{y*z}, xz{x*z};
  22. coeffs[4] = 3.872983346f * xy; /* ACN 4 = sqrt(15) * X * Y */
  23. coeffs[5] = 3.872983346f * yz; /* ACN 5 = sqrt(15) * Y * Z */
  24. coeffs[6] = 1.118033989f * (3.0f*zz - 1.0f); /* ACN 6 = sqrt(5)/2 * (3*Z*Z - 1) */
  25. coeffs[7] = 3.872983346f * xz; /* ACN 7 = sqrt(15) * X * Z */
  26. coeffs[8] = 1.936491673f * (xx - yy); /* ACN 8 = sqrt(15)/2 * (X*X - Y*Y) */
  27. /* Third-order */
  28. coeffs[9] = 2.091650066f * (y*(3.0f*xx - yy)); /* ACN 9 = sqrt(35/8) * Y * (3*X*X - Y*Y) */
  29. coeffs[10] = 10.246950766f * (z*xy); /* ACN 10 = sqrt(105) * Z * X * Y */
  30. coeffs[11] = 1.620185175f * (y*(5.0f*zz - 1.0f)); /* ACN 11 = sqrt(21/8) * Y * (5*Z*Z - 1) */
  31. coeffs[12] = 1.322875656f * (z*(5.0f*zz - 3.0f)); /* ACN 12 = sqrt(7)/2 * Z * (5*Z*Z - 3) */
  32. coeffs[13] = 1.620185175f * (x*(5.0f*zz - 1.0f)); /* ACN 13 = sqrt(21/8) * X * (5*Z*Z - 1) */
  33. coeffs[14] = 5.123475383f * (z*(xx - yy)); /* ACN 14 = sqrt(105)/2 * Z * (X*X - Y*Y) */
  34. coeffs[15] = 2.091650066f * (x*(xx - 3.0f*yy)); /* ACN 15 = sqrt(35/8) * X * (X*X - 3*Y*Y) */
  35. /* Fourth-order */
  36. /* ACN 16 = sqrt(35)*3/2 * X * Y * (X*X - Y*Y) */
  37. /* ACN 17 = sqrt(35/2)*3/2 * (3*X*X - Y*Y) * Y * Z */
  38. /* ACN 18 = sqrt(5)*3/2 * X * Y * (7*Z*Z - 1) */
  39. /* ACN 19 = sqrt(5/2)*3/2 * Y * Z * (7*Z*Z - 3) */
  40. /* ACN 20 = 3/8 * (35*Z*Z*Z*Z - 30*Z*Z + 3) */
  41. /* ACN 21 = sqrt(5/2)*3/2 * X * Z * (7*Z*Z - 3) */
  42. /* ACN 22 = sqrt(5)*3/4 * (X*X - Y*Y) * (7*Z*Z - 1) */
  43. /* ACN 23 = sqrt(35/2)*3/2 * (X*X - 3*Y*Y) * X * Z */
  44. /* ACN 24 = sqrt(35)*3/8 * (X*X*X*X - 6*X*X*Y*Y + Y*Y*Y*Y) */
  45. if(spread > 0.0f)
  46. {
  47. /* Implement the spread by using a spherical source that subtends the
  48. * angle spread. See:
  49. * http://www.ppsloan.org/publications/StupidSH36.pdf - Appendix A3
  50. *
  51. * When adjusted for N3D normalization instead of SN3D, these
  52. * calculations are:
  53. *
  54. * ZH0 = -sqrt(pi) * (-1+ca);
  55. * ZH1 = 0.5*sqrt(pi) * sa*sa;
  56. * ZH2 = -0.5*sqrt(pi) * ca*(-1+ca)*(ca+1);
  57. * ZH3 = -0.125*sqrt(pi) * (-1+ca)*(ca+1)*(5*ca*ca - 1);
  58. * ZH4 = -0.125*sqrt(pi) * ca*(-1+ca)*(ca+1)*(7*ca*ca - 3);
  59. * ZH5 = -0.0625*sqrt(pi) * (-1+ca)*(ca+1)*(21*ca*ca*ca*ca - 14*ca*ca + 1);
  60. *
  61. * The gain of the source is compensated for size, so that the
  62. * loudness doesn't depend on the spread. Thus:
  63. *
  64. * ZH0 = 1.0f;
  65. * ZH1 = 0.5f * (ca+1.0f);
  66. * ZH2 = 0.5f * (ca+1.0f)*ca;
  67. * ZH3 = 0.125f * (ca+1.0f)*(5.0f*ca*ca - 1.0f);
  68. * ZH4 = 0.125f * (ca+1.0f)*(7.0f*ca*ca - 3.0f)*ca;
  69. * ZH5 = 0.0625f * (ca+1.0f)*(21.0f*ca*ca*ca*ca - 14.0f*ca*ca + 1.0f);
  70. */
  71. const float ca{std::cos(spread * 0.5f)};
  72. /* Increase the source volume by up to +3dB for a full spread. */
  73. const float scale{std::sqrt(1.0f + al::numbers::inv_pi_v<float>/2.0f*spread)};
  74. const float ZH0_norm{scale};
  75. const float ZH1_norm{scale * 0.5f * (ca+1.f)};
  76. const float ZH2_norm{scale * 0.5f * (ca+1.f)*ca};
  77. const float ZH3_norm{scale * 0.125f * (ca+1.f)*(5.f*ca*ca-1.f)};
  78. /* Zeroth-order */
  79. coeffs[0] *= ZH0_norm;
  80. /* First-order */
  81. coeffs[1] *= ZH1_norm;
  82. coeffs[2] *= ZH1_norm;
  83. coeffs[3] *= ZH1_norm;
  84. /* Second-order */
  85. coeffs[4] *= ZH2_norm;
  86. coeffs[5] *= ZH2_norm;
  87. coeffs[6] *= ZH2_norm;
  88. coeffs[7] *= ZH2_norm;
  89. coeffs[8] *= ZH2_norm;
  90. /* Third-order */
  91. coeffs[9] *= ZH3_norm;
  92. coeffs[10] *= ZH3_norm;
  93. coeffs[11] *= ZH3_norm;
  94. coeffs[12] *= ZH3_norm;
  95. coeffs[13] *= ZH3_norm;
  96. coeffs[14] *= ZH3_norm;
  97. coeffs[15] *= ZH3_norm;
  98. }
  99. return coeffs;
  100. }
  101. void ComputePanGains(const MixParams *mix, const float*RESTRICT coeffs, const float ingain,
  102. const al::span<float,MAX_OUTPUT_CHANNELS> gains)
  103. {
  104. auto ambimap = mix->AmbiMap.cbegin();
  105. auto iter = std::transform(ambimap, ambimap+mix->Buffer.size(), gains.begin(),
  106. [coeffs,ingain](const BFChannelConfig &chanmap) noexcept -> float
  107. { return chanmap.Scale * coeffs[chanmap.Index] * ingain; }
  108. );
  109. std::fill(iter, gains.end(), 0.0f);
  110. }