equalizer.cpp 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377
  1. #include "config.h"
  2. #include "AL/al.h"
  3. #include "AL/efx.h"
  4. #include "alc/effects/base.h"
  5. #include "effects.h"
  6. #ifdef ALSOFT_EAX
  7. #include "alnumeric.h"
  8. #include "al/eax/effect.h"
  9. #include "al/eax/exception.h"
  10. #include "al/eax/utils.h"
  11. #endif // ALSOFT_EAX
  12. namespace {
  13. constexpr EffectProps genDefaultProps() noexcept
  14. {
  15. EqualizerProps props{};
  16. props.LowCutoff = AL_EQUALIZER_DEFAULT_LOW_CUTOFF;
  17. props.LowGain = AL_EQUALIZER_DEFAULT_LOW_GAIN;
  18. props.Mid1Center = AL_EQUALIZER_DEFAULT_MID1_CENTER;
  19. props.Mid1Gain = AL_EQUALIZER_DEFAULT_MID1_GAIN;
  20. props.Mid1Width = AL_EQUALIZER_DEFAULT_MID1_WIDTH;
  21. props.Mid2Center = AL_EQUALIZER_DEFAULT_MID2_CENTER;
  22. props.Mid2Gain = AL_EQUALIZER_DEFAULT_MID2_GAIN;
  23. props.Mid2Width = AL_EQUALIZER_DEFAULT_MID2_WIDTH;
  24. props.HighCutoff = AL_EQUALIZER_DEFAULT_HIGH_CUTOFF;
  25. props.HighGain = AL_EQUALIZER_DEFAULT_HIGH_GAIN;
  26. return props;
  27. }
  28. } // namespace
  29. const EffectProps EqualizerEffectProps{genDefaultProps()};
  30. void EqualizerEffectHandler::SetParami(EqualizerProps&, ALenum param, int)
  31. { throw effect_exception{AL_INVALID_ENUM, "Invalid equalizer integer property 0x%04x", param}; }
  32. void EqualizerEffectHandler::SetParamiv(EqualizerProps&, ALenum param, const int*)
  33. {
  34. throw effect_exception{AL_INVALID_ENUM, "Invalid equalizer integer-vector property 0x%04x",
  35. param};
  36. }
  37. void EqualizerEffectHandler::SetParamf(EqualizerProps &props, ALenum param, float val)
  38. {
  39. switch(param)
  40. {
  41. case AL_EQUALIZER_LOW_GAIN:
  42. if(!(val >= AL_EQUALIZER_MIN_LOW_GAIN && val <= AL_EQUALIZER_MAX_LOW_GAIN))
  43. throw effect_exception{AL_INVALID_VALUE, "Equalizer low-band gain out of range"};
  44. props.LowGain = val;
  45. break;
  46. case AL_EQUALIZER_LOW_CUTOFF:
  47. if(!(val >= AL_EQUALIZER_MIN_LOW_CUTOFF && val <= AL_EQUALIZER_MAX_LOW_CUTOFF))
  48. throw effect_exception{AL_INVALID_VALUE, "Equalizer low-band cutoff out of range"};
  49. props.LowCutoff = val;
  50. break;
  51. case AL_EQUALIZER_MID1_GAIN:
  52. if(!(val >= AL_EQUALIZER_MIN_MID1_GAIN && val <= AL_EQUALIZER_MAX_MID1_GAIN))
  53. throw effect_exception{AL_INVALID_VALUE, "Equalizer mid1-band gain out of range"};
  54. props.Mid1Gain = val;
  55. break;
  56. case AL_EQUALIZER_MID1_CENTER:
  57. if(!(val >= AL_EQUALIZER_MIN_MID1_CENTER && val <= AL_EQUALIZER_MAX_MID1_CENTER))
  58. throw effect_exception{AL_INVALID_VALUE, "Equalizer mid1-band center out of range"};
  59. props.Mid1Center = val;
  60. break;
  61. case AL_EQUALIZER_MID1_WIDTH:
  62. if(!(val >= AL_EQUALIZER_MIN_MID1_WIDTH && val <= AL_EQUALIZER_MAX_MID1_WIDTH))
  63. throw effect_exception{AL_INVALID_VALUE, "Equalizer mid1-band width out of range"};
  64. props.Mid1Width = val;
  65. break;
  66. case AL_EQUALIZER_MID2_GAIN:
  67. if(!(val >= AL_EQUALIZER_MIN_MID2_GAIN && val <= AL_EQUALIZER_MAX_MID2_GAIN))
  68. throw effect_exception{AL_INVALID_VALUE, "Equalizer mid2-band gain out of range"};
  69. props.Mid2Gain = val;
  70. break;
  71. case AL_EQUALIZER_MID2_CENTER:
  72. if(!(val >= AL_EQUALIZER_MIN_MID2_CENTER && val <= AL_EQUALIZER_MAX_MID2_CENTER))
  73. throw effect_exception{AL_INVALID_VALUE, "Equalizer mid2-band center out of range"};
  74. props.Mid2Center = val;
  75. break;
  76. case AL_EQUALIZER_MID2_WIDTH:
  77. if(!(val >= AL_EQUALIZER_MIN_MID2_WIDTH && val <= AL_EQUALIZER_MAX_MID2_WIDTH))
  78. throw effect_exception{AL_INVALID_VALUE, "Equalizer mid2-band width out of range"};
  79. props.Mid2Width = val;
  80. break;
  81. case AL_EQUALIZER_HIGH_GAIN:
  82. if(!(val >= AL_EQUALIZER_MIN_HIGH_GAIN && val <= AL_EQUALIZER_MAX_HIGH_GAIN))
  83. throw effect_exception{AL_INVALID_VALUE, "Equalizer high-band gain out of range"};
  84. props.HighGain = val;
  85. break;
  86. case AL_EQUALIZER_HIGH_CUTOFF:
  87. if(!(val >= AL_EQUALIZER_MIN_HIGH_CUTOFF && val <= AL_EQUALIZER_MAX_HIGH_CUTOFF))
  88. throw effect_exception{AL_INVALID_VALUE, "Equalizer high-band cutoff out of range"};
  89. props.HighCutoff = val;
  90. break;
  91. default:
  92. throw effect_exception{AL_INVALID_ENUM, "Invalid equalizer float property 0x%04x", param};
  93. }
  94. }
  95. void EqualizerEffectHandler::SetParamfv(EqualizerProps &props, ALenum param, const float *vals)
  96. { SetParamf(props, param, *vals); }
  97. void EqualizerEffectHandler::GetParami(const EqualizerProps&, ALenum param, int*)
  98. { throw effect_exception{AL_INVALID_ENUM, "Invalid equalizer integer property 0x%04x", param}; }
  99. void EqualizerEffectHandler::GetParamiv(const EqualizerProps&, ALenum param, int*)
  100. {
  101. throw effect_exception{AL_INVALID_ENUM, "Invalid equalizer integer-vector property 0x%04x",
  102. param};
  103. }
  104. void EqualizerEffectHandler::GetParamf(const EqualizerProps &props, ALenum param, float *val)
  105. {
  106. switch(param)
  107. {
  108. case AL_EQUALIZER_LOW_GAIN: *val = props.LowGain; break;
  109. case AL_EQUALIZER_LOW_CUTOFF: *val = props.LowCutoff; break;
  110. case AL_EQUALIZER_MID1_GAIN: *val = props.Mid1Gain; break;
  111. case AL_EQUALIZER_MID1_CENTER: *val = props.Mid1Center; break;
  112. case AL_EQUALIZER_MID1_WIDTH: *val = props.Mid1Width; break;
  113. case AL_EQUALIZER_MID2_GAIN: *val = props.Mid2Gain; break;
  114. case AL_EQUALIZER_MID2_CENTER: *val = props.Mid2Center; break;
  115. case AL_EQUALIZER_MID2_WIDTH: *val = props.Mid2Width; break;
  116. case AL_EQUALIZER_HIGH_GAIN: *val = props.HighGain; break;
  117. case AL_EQUALIZER_HIGH_CUTOFF: *val = props.HighCutoff; break;
  118. default:
  119. throw effect_exception{AL_INVALID_ENUM, "Invalid equalizer float property 0x%04x", param};
  120. }
  121. }
  122. void EqualizerEffectHandler::GetParamfv(const EqualizerProps &props, ALenum param, float *vals)
  123. { GetParamf(props, param, vals); }
  124. #ifdef ALSOFT_EAX
  125. namespace {
  126. using EqualizerCommitter = EaxCommitter<EaxEqualizerCommitter>;
  127. struct LowGainValidator {
  128. void operator()(long lLowGain) const
  129. {
  130. eax_validate_range<EqualizerCommitter::Exception>(
  131. "Low Gain",
  132. lLowGain,
  133. EAXEQUALIZER_MINLOWGAIN,
  134. EAXEQUALIZER_MAXLOWGAIN);
  135. }
  136. }; // LowGainValidator
  137. struct LowCutOffValidator {
  138. void operator()(float flLowCutOff) const
  139. {
  140. eax_validate_range<EqualizerCommitter::Exception>(
  141. "Low Cutoff",
  142. flLowCutOff,
  143. EAXEQUALIZER_MINLOWCUTOFF,
  144. EAXEQUALIZER_MAXLOWCUTOFF);
  145. }
  146. }; // LowCutOffValidator
  147. struct Mid1GainValidator {
  148. void operator()(long lMid1Gain) const
  149. {
  150. eax_validate_range<EqualizerCommitter::Exception>(
  151. "Mid1 Gain",
  152. lMid1Gain,
  153. EAXEQUALIZER_MINMID1GAIN,
  154. EAXEQUALIZER_MAXMID1GAIN);
  155. }
  156. }; // Mid1GainValidator
  157. struct Mid1CenterValidator {
  158. void operator()(float flMid1Center) const
  159. {
  160. eax_validate_range<EqualizerCommitter::Exception>(
  161. "Mid1 Center",
  162. flMid1Center,
  163. EAXEQUALIZER_MINMID1CENTER,
  164. EAXEQUALIZER_MAXMID1CENTER);
  165. }
  166. }; // Mid1CenterValidator
  167. struct Mid1WidthValidator {
  168. void operator()(float flMid1Width) const
  169. {
  170. eax_validate_range<EqualizerCommitter::Exception>(
  171. "Mid1 Width",
  172. flMid1Width,
  173. EAXEQUALIZER_MINMID1WIDTH,
  174. EAXEQUALIZER_MAXMID1WIDTH);
  175. }
  176. }; // Mid1WidthValidator
  177. struct Mid2GainValidator {
  178. void operator()(long lMid2Gain) const
  179. {
  180. eax_validate_range<EqualizerCommitter::Exception>(
  181. "Mid2 Gain",
  182. lMid2Gain,
  183. EAXEQUALIZER_MINMID2GAIN,
  184. EAXEQUALIZER_MAXMID2GAIN);
  185. }
  186. }; // Mid2GainValidator
  187. struct Mid2CenterValidator {
  188. void operator()(float flMid2Center) const
  189. {
  190. eax_validate_range<EqualizerCommitter::Exception>(
  191. "Mid2 Center",
  192. flMid2Center,
  193. EAXEQUALIZER_MINMID2CENTER,
  194. EAXEQUALIZER_MAXMID2CENTER);
  195. }
  196. }; // Mid2CenterValidator
  197. struct Mid2WidthValidator {
  198. void operator()(float flMid2Width) const
  199. {
  200. eax_validate_range<EqualizerCommitter::Exception>(
  201. "Mid2 Width",
  202. flMid2Width,
  203. EAXEQUALIZER_MINMID2WIDTH,
  204. EAXEQUALIZER_MAXMID2WIDTH);
  205. }
  206. }; // Mid2WidthValidator
  207. struct HighGainValidator {
  208. void operator()(long lHighGain) const
  209. {
  210. eax_validate_range<EqualizerCommitter::Exception>(
  211. "High Gain",
  212. lHighGain,
  213. EAXEQUALIZER_MINHIGHGAIN,
  214. EAXEQUALIZER_MAXHIGHGAIN);
  215. }
  216. }; // HighGainValidator
  217. struct HighCutOffValidator {
  218. void operator()(float flHighCutOff) const
  219. {
  220. eax_validate_range<EqualizerCommitter::Exception>(
  221. "High Cutoff",
  222. flHighCutOff,
  223. EAXEQUALIZER_MINHIGHCUTOFF,
  224. EAXEQUALIZER_MAXHIGHCUTOFF);
  225. }
  226. }; // HighCutOffValidator
  227. struct AllValidator {
  228. void operator()(const EAXEQUALIZERPROPERTIES& all) const
  229. {
  230. LowGainValidator{}(all.lLowGain);
  231. LowCutOffValidator{}(all.flLowCutOff);
  232. Mid1GainValidator{}(all.lMid1Gain);
  233. Mid1CenterValidator{}(all.flMid1Center);
  234. Mid1WidthValidator{}(all.flMid1Width);
  235. Mid2GainValidator{}(all.lMid2Gain);
  236. Mid2CenterValidator{}(all.flMid2Center);
  237. Mid2WidthValidator{}(all.flMid2Width);
  238. HighGainValidator{}(all.lHighGain);
  239. HighCutOffValidator{}(all.flHighCutOff);
  240. }
  241. }; // AllValidator
  242. } // namespace
  243. template<>
  244. struct EqualizerCommitter::Exception : public EaxException {
  245. explicit Exception(const char* message) : EaxException{"EAX_EQUALIZER_EFFECT", message}
  246. { }
  247. };
  248. template<>
  249. [[noreturn]] void EqualizerCommitter::fail(const char *message)
  250. {
  251. throw Exception{message};
  252. }
  253. bool EaxEqualizerCommitter::commit(const EAXEQUALIZERPROPERTIES &props)
  254. {
  255. if(auto *cur = std::get_if<EAXEQUALIZERPROPERTIES>(&mEaxProps); cur && *cur == props)
  256. return false;
  257. mEaxProps = props;
  258. mAlProps = [&]{
  259. EqualizerProps ret{};
  260. ret.LowGain = level_mb_to_gain(static_cast<float>(props.lLowGain));
  261. ret.LowCutoff = props.flLowCutOff;
  262. ret.Mid1Gain = level_mb_to_gain(static_cast<float>(props.lMid1Gain));
  263. ret.Mid1Center = props.flMid1Center;
  264. ret.Mid1Width = props.flMid1Width;
  265. ret.Mid2Gain = level_mb_to_gain(static_cast<float>(props.lMid2Gain));
  266. ret.Mid2Center = props.flMid2Center;
  267. ret.Mid2Width = props.flMid2Width;
  268. ret.HighGain = level_mb_to_gain(static_cast<float>(props.lHighGain));
  269. ret.HighCutoff = props.flHighCutOff;
  270. return ret;
  271. }();
  272. return true;
  273. }
  274. void EaxEqualizerCommitter::SetDefaults(EaxEffectProps &props)
  275. {
  276. static constexpr EAXEQUALIZERPROPERTIES defprops{[]
  277. {
  278. EAXEQUALIZERPROPERTIES ret{};
  279. ret.lLowGain = EAXEQUALIZER_DEFAULTLOWGAIN;
  280. ret.flLowCutOff = EAXEQUALIZER_DEFAULTLOWCUTOFF;
  281. ret.lMid1Gain = EAXEQUALIZER_DEFAULTMID1GAIN;
  282. ret.flMid1Center = EAXEQUALIZER_DEFAULTMID1CENTER;
  283. ret.flMid1Width = EAXEQUALIZER_DEFAULTMID1WIDTH;
  284. ret.lMid2Gain = EAXEQUALIZER_DEFAULTMID2GAIN;
  285. ret.flMid2Center = EAXEQUALIZER_DEFAULTMID2CENTER;
  286. ret.flMid2Width = EAXEQUALIZER_DEFAULTMID2WIDTH;
  287. ret.lHighGain = EAXEQUALIZER_DEFAULTHIGHGAIN;
  288. ret.flHighCutOff = EAXEQUALIZER_DEFAULTHIGHCUTOFF;
  289. return ret;
  290. }()};
  291. props = defprops;
  292. }
  293. void EaxEqualizerCommitter::Get(const EaxCall &call, const EAXEQUALIZERPROPERTIES &props)
  294. {
  295. switch(call.get_property_id())
  296. {
  297. case EAXEQUALIZER_NONE: break;
  298. case EAXEQUALIZER_ALLPARAMETERS: call.set_value<Exception>(props); break;
  299. case EAXEQUALIZER_LOWGAIN: call.set_value<Exception>(props.lLowGain); break;
  300. case EAXEQUALIZER_LOWCUTOFF: call.set_value<Exception>(props.flLowCutOff); break;
  301. case EAXEQUALIZER_MID1GAIN: call.set_value<Exception>(props.lMid1Gain); break;
  302. case EAXEQUALIZER_MID1CENTER: call.set_value<Exception>(props.flMid1Center); break;
  303. case EAXEQUALIZER_MID1WIDTH: call.set_value<Exception>(props.flMid1Width); break;
  304. case EAXEQUALIZER_MID2GAIN: call.set_value<Exception>(props.lMid2Gain); break;
  305. case EAXEQUALIZER_MID2CENTER: call.set_value<Exception>(props.flMid2Center); break;
  306. case EAXEQUALIZER_MID2WIDTH: call.set_value<Exception>(props.flMid2Width); break;
  307. case EAXEQUALIZER_HIGHGAIN: call.set_value<Exception>(props.lHighGain); break;
  308. case EAXEQUALIZER_HIGHCUTOFF: call.set_value<Exception>(props.flHighCutOff); break;
  309. default: fail_unknown_property_id();
  310. }
  311. }
  312. void EaxEqualizerCommitter::Set(const EaxCall &call, EAXEQUALIZERPROPERTIES &props)
  313. {
  314. switch(call.get_property_id())
  315. {
  316. case EAXEQUALIZER_NONE: break;
  317. case EAXEQUALIZER_ALLPARAMETERS: defer<AllValidator>(call, props); break;
  318. case EAXEQUALIZER_LOWGAIN: defer<LowGainValidator>(call, props.lLowGain); break;
  319. case EAXEQUALIZER_LOWCUTOFF: defer<LowCutOffValidator>(call, props.flLowCutOff); break;
  320. case EAXEQUALIZER_MID1GAIN: defer<Mid1GainValidator>(call, props.lMid1Gain); break;
  321. case EAXEQUALIZER_MID1CENTER: defer<Mid1CenterValidator>(call, props.flMid1Center); break;
  322. case EAXEQUALIZER_MID1WIDTH: defer<Mid1WidthValidator>(call, props.flMid1Width); break;
  323. case EAXEQUALIZER_MID2GAIN: defer<Mid2GainValidator>(call, props.lMid2Gain); break;
  324. case EAXEQUALIZER_MID2CENTER: defer<Mid2CenterValidator>(call, props.flMid2Center); break;
  325. case EAXEQUALIZER_MID2WIDTH: defer<Mid2WidthValidator>(call, props.flMid2Width); break;
  326. case EAXEQUALIZER_HIGHGAIN: defer<HighGainValidator>(call, props.lHighGain); break;
  327. case EAXEQUALIZER_HIGHCUTOFF: defer<HighCutOffValidator>(call, props.flHighCutOff); break;
  328. default: fail_unknown_property_id();
  329. }
  330. }
  331. #endif // ALSOFT_EAX