fshifter.cpp 8.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262
  1. #include "config.h"
  2. #include <optional>
  3. #include <stdexcept>
  4. #include "AL/al.h"
  5. #include "AL/efx.h"
  6. #include "alc/context.h"
  7. #include "alnumeric.h"
  8. #include "effects.h"
  9. #if ALSOFT_EAX
  10. #include <cassert>
  11. #include "al/eax/effect.h"
  12. #include "al/eax/exception.h"
  13. #include "al/eax/utils.h"
  14. #endif // ALSOFT_EAX
  15. namespace {
  16. constexpr std::optional<FShifterDirection> DirectionFromEmum(ALenum value) noexcept
  17. {
  18. switch(value)
  19. {
  20. case AL_FREQUENCY_SHIFTER_DIRECTION_DOWN: return FShifterDirection::Down;
  21. case AL_FREQUENCY_SHIFTER_DIRECTION_UP: return FShifterDirection::Up;
  22. case AL_FREQUENCY_SHIFTER_DIRECTION_OFF: return FShifterDirection::Off;
  23. }
  24. return std::nullopt;
  25. }
  26. constexpr ALenum EnumFromDirection(FShifterDirection dir)
  27. {
  28. switch(dir)
  29. {
  30. case FShifterDirection::Down: return AL_FREQUENCY_SHIFTER_DIRECTION_DOWN;
  31. case FShifterDirection::Up: return AL_FREQUENCY_SHIFTER_DIRECTION_UP;
  32. case FShifterDirection::Off: return AL_FREQUENCY_SHIFTER_DIRECTION_OFF;
  33. }
  34. throw std::runtime_error{fmt::format("Invalid direction: {}", int{al::to_underlying(dir)})};
  35. }
  36. constexpr EffectProps genDefaultProps() noexcept
  37. {
  38. FshifterProps props{};
  39. props.Frequency = AL_FREQUENCY_SHIFTER_DEFAULT_FREQUENCY;
  40. props.LeftDirection = DirectionFromEmum(AL_FREQUENCY_SHIFTER_DEFAULT_LEFT_DIRECTION).value();
  41. props.RightDirection = DirectionFromEmum(AL_FREQUENCY_SHIFTER_DEFAULT_RIGHT_DIRECTION).value();
  42. return props;
  43. }
  44. } // namespace
  45. const EffectProps FshifterEffectProps{genDefaultProps()};
  46. void FshifterEffectHandler::SetParami(ALCcontext *context, FshifterProps &props, ALenum param, int val)
  47. {
  48. switch(param)
  49. {
  50. case AL_FREQUENCY_SHIFTER_LEFT_DIRECTION:
  51. if(auto diropt = DirectionFromEmum(val))
  52. props.LeftDirection = *diropt;
  53. else
  54. context->throw_error(AL_INVALID_VALUE,
  55. "Unsupported frequency shifter left direction: {:#04x}", as_unsigned(val));
  56. return;
  57. case AL_FREQUENCY_SHIFTER_RIGHT_DIRECTION:
  58. if(auto diropt = DirectionFromEmum(val))
  59. props.RightDirection = *diropt;
  60. else
  61. context->throw_error(AL_INVALID_VALUE,
  62. "Unsupported frequency shifter right direction: {:#04x}", as_unsigned(val));
  63. return;
  64. }
  65. context->throw_error(AL_INVALID_ENUM, "Invalid frequency shifter integer property {:#04x}",
  66. as_unsigned(param));
  67. }
  68. void FshifterEffectHandler::SetParamiv(ALCcontext *context, FshifterProps &props, ALenum param, const int *vals)
  69. { SetParami(context, props, param, *vals); }
  70. void FshifterEffectHandler::SetParamf(ALCcontext *context, FshifterProps &props, ALenum param, float val)
  71. {
  72. switch(param)
  73. {
  74. case AL_FREQUENCY_SHIFTER_FREQUENCY:
  75. if(!(val >= AL_FREQUENCY_SHIFTER_MIN_FREQUENCY && val <= AL_FREQUENCY_SHIFTER_MAX_FREQUENCY))
  76. context->throw_error(AL_INVALID_VALUE, "Frequency shifter frequency out of range");
  77. props.Frequency = val;
  78. return;
  79. }
  80. context->throw_error(AL_INVALID_ENUM, "Invalid frequency shifter float property {:#04x}",
  81. as_unsigned(param));
  82. }
  83. void FshifterEffectHandler::SetParamfv(ALCcontext *context, FshifterProps &props, ALenum param, const float *vals)
  84. { SetParamf(context, props, param, *vals); }
  85. void FshifterEffectHandler::GetParami(ALCcontext *context, const FshifterProps &props, ALenum param, int *val)
  86. {
  87. switch(param)
  88. {
  89. case AL_FREQUENCY_SHIFTER_LEFT_DIRECTION:
  90. *val = EnumFromDirection(props.LeftDirection);
  91. return;
  92. case AL_FREQUENCY_SHIFTER_RIGHT_DIRECTION:
  93. *val = EnumFromDirection(props.RightDirection);
  94. return;
  95. }
  96. context->throw_error(AL_INVALID_ENUM, "Invalid frequency shifter integer property {:#04x}",
  97. as_unsigned(param));
  98. }
  99. void FshifterEffectHandler::GetParamiv(ALCcontext *context, const FshifterProps &props, ALenum param, int *vals)
  100. { GetParami(context, props, param, vals); }
  101. void FshifterEffectHandler::GetParamf(ALCcontext *context, const FshifterProps &props, ALenum param, float *val)
  102. {
  103. switch(param)
  104. {
  105. case AL_FREQUENCY_SHIFTER_FREQUENCY: *val = props.Frequency; return;
  106. }
  107. context->throw_error(AL_INVALID_ENUM, "Invalid frequency shifter float property {:#04x}",
  108. as_unsigned(param));
  109. }
  110. void FshifterEffectHandler::GetParamfv(ALCcontext *context, const FshifterProps &props, ALenum param, float *vals)
  111. { GetParamf(context, props, param, vals); }
  112. #if ALSOFT_EAX
  113. namespace {
  114. using FrequencyShifterCommitter = EaxCommitter<EaxFrequencyShifterCommitter>;
  115. struct FrequencyValidator {
  116. void operator()(float flFrequency) const
  117. {
  118. eax_validate_range<FrequencyShifterCommitter::Exception>(
  119. "Frequency",
  120. flFrequency,
  121. EAXFREQUENCYSHIFTER_MINFREQUENCY,
  122. EAXFREQUENCYSHIFTER_MAXFREQUENCY);
  123. }
  124. }; // FrequencyValidator
  125. struct LeftDirectionValidator {
  126. void operator()(unsigned long ulLeftDirection) const
  127. {
  128. eax_validate_range<FrequencyShifterCommitter::Exception>(
  129. "Left Direction",
  130. ulLeftDirection,
  131. EAXFREQUENCYSHIFTER_MINLEFTDIRECTION,
  132. EAXFREQUENCYSHIFTER_MAXLEFTDIRECTION);
  133. }
  134. }; // LeftDirectionValidator
  135. struct RightDirectionValidator {
  136. void operator()(unsigned long ulRightDirection) const
  137. {
  138. eax_validate_range<FrequencyShifterCommitter::Exception>(
  139. "Right Direction",
  140. ulRightDirection,
  141. EAXFREQUENCYSHIFTER_MINRIGHTDIRECTION,
  142. EAXFREQUENCYSHIFTER_MAXRIGHTDIRECTION);
  143. }
  144. }; // RightDirectionValidator
  145. struct AllValidator {
  146. void operator()(const EAXFREQUENCYSHIFTERPROPERTIES& all) const
  147. {
  148. FrequencyValidator{}(all.flFrequency);
  149. LeftDirectionValidator{}(all.ulLeftDirection);
  150. RightDirectionValidator{}(all.ulRightDirection);
  151. }
  152. }; // AllValidator
  153. } // namespace
  154. template<>
  155. struct FrequencyShifterCommitter::Exception : public EaxException {
  156. explicit Exception(const char *message) : EaxException{"EAX_FREQUENCY_SHIFTER_EFFECT", message}
  157. { }
  158. };
  159. template<>
  160. [[noreturn]] void FrequencyShifterCommitter::fail(const char *message)
  161. {
  162. throw Exception{message};
  163. }
  164. bool EaxFrequencyShifterCommitter::commit(const EAXFREQUENCYSHIFTERPROPERTIES &props)
  165. {
  166. if(auto *cur = std::get_if<EAXFREQUENCYSHIFTERPROPERTIES>(&mEaxProps); cur && *cur == props)
  167. return false;
  168. mEaxProps = props;
  169. auto get_direction = [](unsigned long dir) noexcept
  170. {
  171. if(dir == EAX_FREQUENCYSHIFTER_DOWN)
  172. return FShifterDirection::Down;
  173. if(dir == EAX_FREQUENCYSHIFTER_UP)
  174. return FShifterDirection::Up;
  175. return FShifterDirection::Off;
  176. };
  177. mAlProps = [&]{
  178. FshifterProps ret{};
  179. ret.Frequency = props.flFrequency;
  180. ret.LeftDirection = get_direction(props.ulLeftDirection);
  181. ret.RightDirection = get_direction(props.ulRightDirection);
  182. return ret;
  183. }();
  184. return true;
  185. }
  186. void EaxFrequencyShifterCommitter::SetDefaults(EaxEffectProps &props)
  187. {
  188. static constexpr EAXFREQUENCYSHIFTERPROPERTIES defprops{[]
  189. {
  190. EAXFREQUENCYSHIFTERPROPERTIES ret{};
  191. ret.flFrequency = EAXFREQUENCYSHIFTER_DEFAULTFREQUENCY;
  192. ret.ulLeftDirection = EAXFREQUENCYSHIFTER_DEFAULTLEFTDIRECTION;
  193. ret.ulRightDirection = EAXFREQUENCYSHIFTER_DEFAULTRIGHTDIRECTION;
  194. return ret;
  195. }()};
  196. props = defprops;
  197. }
  198. void EaxFrequencyShifterCommitter::Get(const EaxCall &call, const EAXFREQUENCYSHIFTERPROPERTIES &props)
  199. {
  200. switch(call.get_property_id())
  201. {
  202. case EAXFREQUENCYSHIFTER_NONE: break;
  203. case EAXFREQUENCYSHIFTER_ALLPARAMETERS: call.set_value<Exception>(props); break;
  204. case EAXFREQUENCYSHIFTER_FREQUENCY: call.set_value<Exception>(props.flFrequency); break;
  205. case EAXFREQUENCYSHIFTER_LEFTDIRECTION: call.set_value<Exception>(props.ulLeftDirection); break;
  206. case EAXFREQUENCYSHIFTER_RIGHTDIRECTION: call.set_value<Exception>(props.ulRightDirection); break;
  207. default: fail_unknown_property_id();
  208. }
  209. }
  210. void EaxFrequencyShifterCommitter::Set(const EaxCall &call, EAXFREQUENCYSHIFTERPROPERTIES &props)
  211. {
  212. switch(call.get_property_id())
  213. {
  214. case EAXFREQUENCYSHIFTER_NONE: break;
  215. case EAXFREQUENCYSHIFTER_ALLPARAMETERS: defer<AllValidator>(call, props); break;
  216. case EAXFREQUENCYSHIFTER_FREQUENCY: defer<FrequencyValidator>(call, props.flFrequency); break;
  217. case EAXFREQUENCYSHIFTER_LEFTDIRECTION: defer<LeftDirectionValidator>(call, props.ulLeftDirection); break;
  218. case EAXFREQUENCYSHIFTER_RIGHTDIRECTION: defer<RightDirectionValidator>(call, props.ulRightDirection); break;
  219. default: fail_unknown_property_id();
  220. }
  221. }
  222. #endif // ALSOFT_EAX