furi_hal_speaker.c 2.1 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071
  1. #include <furi_hal_speaker.h>
  2. #include <furi_hal_gpio.h>
  3. #include <furi_hal_resources.h>
  4. #include <stm32wbxx_ll_tim.h>
  5. #define FURI_HAL_SPEAKER_TIMER TIM16
  6. #define FURI_HAL_SPEAKER_CHANNEL LL_TIM_CHANNEL_CH1
  7. #define FURI_HAL_SPEAKER_PRESCALER 500
  8. #define FURI_HAL_SPEAKER_MAX_VOLUME 60
  9. // #define FURI_HAL_SPEAKER_NEW_VOLUME
  10. void furi_hal_speaker_init() {
  11. FURI_CRITICAL_ENTER();
  12. LL_TIM_DeInit(FURI_HAL_SPEAKER_TIMER);
  13. FURI_CRITICAL_EXIT();
  14. furi_hal_gpio_init_ex(
  15. &gpio_speaker, GpioModeAltFunctionPushPull, GpioPullNo, GpioSpeedLow, GpioAltFn14TIM16);
  16. }
  17. void furi_hal_speaker_start(float frequency, float volume) {
  18. if(volume == 0) {
  19. return;
  20. }
  21. if(volume < 0) volume = 0;
  22. if(volume > 1) volume = 1;
  23. volume = volume * volume * volume;
  24. uint32_t autoreload = (SystemCoreClock / FURI_HAL_SPEAKER_PRESCALER / frequency) - 1;
  25. if(autoreload < 2) {
  26. autoreload = 2;
  27. } else if(autoreload > UINT16_MAX) {
  28. autoreload = UINT16_MAX;
  29. }
  30. LL_TIM_InitTypeDef TIM_InitStruct = {0};
  31. TIM_InitStruct.Prescaler = FURI_HAL_SPEAKER_PRESCALER - 1;
  32. TIM_InitStruct.Autoreload = autoreload;
  33. LL_TIM_Init(FURI_HAL_SPEAKER_TIMER, &TIM_InitStruct);
  34. #ifdef FURI_HAL_SPEAKER_NEW_VOLUME
  35. uint32_t compare_value = volume * FURI_HAL_SPEAKER_MAX_VOLUME;
  36. uint32_t clip_value = volume * TIM_InitStruct.Autoreload / 2;
  37. if(compare_value > clip_value) {
  38. compare_value = clip_value;
  39. }
  40. #else
  41. uint32_t compare_value = volume * autoreload / 2;
  42. #endif
  43. if(compare_value == 0) {
  44. compare_value = 1;
  45. }
  46. LL_TIM_OC_InitTypeDef TIM_OC_InitStruct = {0};
  47. TIM_OC_InitStruct.OCMode = LL_TIM_OCMODE_PWM1;
  48. TIM_OC_InitStruct.OCState = LL_TIM_OCSTATE_ENABLE;
  49. TIM_OC_InitStruct.CompareValue = compare_value;
  50. LL_TIM_OC_Init(FURI_HAL_SPEAKER_TIMER, FURI_HAL_SPEAKER_CHANNEL, &TIM_OC_InitStruct);
  51. LL_TIM_EnableAllOutputs(FURI_HAL_SPEAKER_TIMER);
  52. LL_TIM_EnableCounter(FURI_HAL_SPEAKER_TIMER);
  53. }
  54. void furi_hal_speaker_stop() {
  55. LL_TIM_DisableAllOutputs(FURI_HAL_SPEAKER_TIMER);
  56. LL_TIM_DisableCounter(FURI_HAL_SPEAKER_TIMER);
  57. }