flizzer_tracker_hal.c 4.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139
  1. #include "flizzer_tracker_hal.h"
  2. #include "flizzer_tracker.h"
  3. #define SPEAKER_PWM_TIMER TIM16
  4. #define SAMPLE_RATE_TIMER TIM2
  5. #define TRACKER_ENGINE_TIMER TIM1
  6. #define SPEAKER_PWM_TIMER_CHANNEL LL_TIM_CHANNEL_CH1
  7. #define TIMER_BASE_CLOCK 64000000 /* CPU frequency, 64 MHz */
  8. #define DMA_INSTANCE DMA1, LL_DMA_CHANNEL_1
  9. void sound_engine_dma_isr(void* ctx)
  10. {
  11. SoundEngine* sound_engine = (SoundEngine*)ctx;
  12. // half of transfer
  13. if(LL_DMA_IsActiveFlag_HT1(DMA1))
  14. {
  15. LL_DMA_ClearFlag_HT1(DMA1);
  16. // fill first half of buffer
  17. uint16_t* audio_buffer = sound_engine->audio_buffer;
  18. uint32_t audio_buffer_length = sound_engine->audio_buffer_size / 2;
  19. sound_engine_fill_buffer(sound_engine, audio_buffer, audio_buffer_length);
  20. }
  21. // transfer complete
  22. if(LL_DMA_IsActiveFlag_TC1(DMA1))
  23. {
  24. LL_DMA_ClearFlag_TC1(DMA1);
  25. // fill second half of buffer
  26. uint32_t audio_buffer_length = sound_engine->audio_buffer_size / 2;
  27. uint16_t* audio_buffer = &sound_engine->audio_buffer[audio_buffer_length];
  28. sound_engine_fill_buffer(sound_engine, audio_buffer, audio_buffer_length);
  29. }
  30. }
  31. void sound_engine_PWM_timer_init(bool external_audio_output) //external audio on pin PA6
  32. {
  33. LL_TIM_InitTypeDef TIM_InitStruct = {0};
  34. LL_TIM_OC_InitTypeDef TIM_OC_InitStruct = {0};
  35. TIM_InitStruct.Prescaler = 0;
  36. TIM_InitStruct.Autoreload = 1023; //10-bit PWM resolution at around 60 kHz PWM rate
  37. TIM_InitStruct.CounterMode = LL_TIM_COUNTERMODE_UP;
  38. LL_TIM_Init(SPEAKER_PWM_TIMER, &TIM_InitStruct);
  39. TIM_OC_InitStruct.OCMode = LL_TIM_OCMODE_PWM1;
  40. TIM_OC_InitStruct.OCState = LL_TIM_OCSTATE_ENABLE;
  41. TIM_OC_InitStruct.CompareValue = 0;
  42. LL_TIM_OC_Init(SPEAKER_PWM_TIMER, SPEAKER_PWM_TIMER_CHANNEL, &TIM_OC_InitStruct);
  43. if(external_audio_output)
  44. {
  45. furi_hal_gpio_init_ex(&gpio_ext_pa6, GpioModeAltFunctionPushPull, GpioPullNo, GpioSpeedVeryHigh, GpioAltFn14TIM16);
  46. }
  47. else
  48. {
  49. bool unu = furi_hal_speaker_acquire(1000);
  50. UNUSED(unu);
  51. }
  52. }
  53. void sound_engine_timer_init(uint32_t sample_rate) //external audio on pin PA6
  54. {
  55. LL_TIM_InitTypeDef TIM_InitStruct = {0};
  56. LL_TIM_OC_InitTypeDef TIM_OC_InitStruct = {0};
  57. TIM_InitStruct.Prescaler = 0;
  58. TIM_InitStruct.Autoreload = TIMER_BASE_CLOCK / sample_rate; //to support various sample rates
  59. TIM_InitStruct.CounterMode = LL_TIM_COUNTERMODE_UP;
  60. LL_TIM_Init(SAMPLE_RATE_TIMER, &TIM_InitStruct);
  61. TIM_OC_InitStruct.OCMode = LL_TIM_OCMODE_PWM1;
  62. TIM_OC_InitStruct.OCState = LL_TIM_OCSTATE_ENABLE;
  63. LL_TIM_OC_Init(SAMPLE_RATE_TIMER, SPEAKER_PWM_TIMER_CHANNEL, &TIM_OC_InitStruct);
  64. }
  65. void sound_engine_dma_init(uint32_t address, uint32_t size)
  66. {
  67. uint32_t dma_dst = (uint32_t) & (SPEAKER_PWM_TIMER->CCR1);
  68. LL_DMA_ConfigAddresses(DMA_INSTANCE, address, dma_dst, LL_DMA_DIRECTION_MEMORY_TO_PERIPH);
  69. LL_DMA_SetDataLength(DMA_INSTANCE, size);
  70. LL_DMA_SetPeriphRequest(DMA_INSTANCE, LL_DMAMUX_REQ_TIM2_UP);
  71. LL_DMA_SetDataTransferDirection(DMA_INSTANCE, LL_DMA_DIRECTION_MEMORY_TO_PERIPH);
  72. LL_DMA_SetChannelPriorityLevel(DMA_INSTANCE, LL_DMA_PRIORITY_VERYHIGH);
  73. LL_DMA_SetMode(DMA_INSTANCE, LL_DMA_MODE_CIRCULAR);
  74. LL_DMA_SetPeriphIncMode(DMA_INSTANCE, LL_DMA_PERIPH_NOINCREMENT);
  75. LL_DMA_SetMemoryIncMode(DMA_INSTANCE, LL_DMA_MEMORY_INCREMENT);
  76. LL_DMA_SetPeriphSize(DMA_INSTANCE, LL_DMA_PDATAALIGN_HALFWORD);
  77. LL_DMA_SetMemorySize(DMA_INSTANCE, LL_DMA_MDATAALIGN_HALFWORD);
  78. LL_DMA_EnableIT_TC(DMA_INSTANCE);
  79. LL_DMA_EnableIT_HT(DMA_INSTANCE);
  80. }
  81. void sound_engine_init_hardware(uint32_t sample_rate, bool external_audio_output, uint16_t* audio_buffer, uint32_t audio_buffer_size)
  82. {
  83. sound_engine_dma_init((uint32_t)audio_buffer, audio_buffer_size);
  84. sound_engine_timer_init(sample_rate);
  85. sound_engine_PWM_timer_init(external_audio_output);
  86. }
  87. void sound_engine_dma_start()
  88. {
  89. LL_DMA_EnableChannel(DMA_INSTANCE);
  90. LL_TIM_EnableDMAReq_UPDATE(SAMPLE_RATE_TIMER);
  91. }
  92. void sound_engine_dma_stop()
  93. {
  94. LL_DMA_DisableChannel(DMA_INSTANCE);
  95. }
  96. void sound_engine_start()
  97. {
  98. LL_TIM_EnableAllOutputs(SAMPLE_RATE_TIMER);
  99. LL_TIM_EnableCounter(SAMPLE_RATE_TIMER);
  100. LL_TIM_EnableAllOutputs(SPEAKER_PWM_TIMER);
  101. LL_TIM_EnableCounter(SPEAKER_PWM_TIMER);
  102. sound_engine_dma_start();
  103. }
  104. void sound_engine_stop()
  105. {
  106. LL_TIM_DisableAllOutputs(SAMPLE_RATE_TIMER);
  107. LL_TIM_DisableCounter(SAMPLE_RATE_TIMER);
  108. LL_TIM_DisableAllOutputs(SPEAKER_PWM_TIMER);
  109. LL_TIM_DisableCounter(SPEAKER_PWM_TIMER);
  110. sound_engine_dma_stop();
  111. }