flizzer_tracker_hal.c 7.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242
  1. #include "flizzer_tracker_hal.h"
  2. #include "flizzer_tracker.h"
  3. void sound_engine_dma_isr(void *ctx)
  4. {
  5. SoundEngine *sound_engine = (SoundEngine *)ctx;
  6. // sound_engine->counter++;
  7. // half of transfer
  8. if (LL_DMA_IsActiveFlag_HT1(DMA1))
  9. {
  10. LL_DMA_ClearFlag_HT1(DMA1);
  11. // fill first half of buffer
  12. uint16_t *audio_buffer = sound_engine->audio_buffer;
  13. uint32_t audio_buffer_length = sound_engine->audio_buffer_size / 2;
  14. sound_engine_fill_buffer(sound_engine, audio_buffer, audio_buffer_length);
  15. }
  16. // transfer complete
  17. if (LL_DMA_IsActiveFlag_TC1(DMA1))
  18. {
  19. LL_DMA_ClearFlag_TC1(DMA1);
  20. // fill second half of buffer
  21. uint32_t audio_buffer_length = sound_engine->audio_buffer_size / 2;
  22. uint16_t *audio_buffer = &sound_engine->audio_buffer[audio_buffer_length];
  23. sound_engine_fill_buffer(sound_engine, audio_buffer, audio_buffer_length);
  24. }
  25. }
  26. void tracker_engine_timer_isr(void *ctx) // the tracker engine interrupt is of higher priority than sound engine one so it can run at the middle of filling the buffer, thus allowing sample-accurate tight effect timing
  27. {
  28. TrackerEngine *tracker_engine = (TrackerEngine *)ctx;
  29. // tracker_engine->counter++;
  30. if (LL_TIM_IsActiveFlag_UPDATE(TRACKER_ENGINE_TIMER))
  31. {
  32. LL_TIM_ClearFlag_UPDATE(TRACKER_ENGINE_TIMER);
  33. tracker_engine_advance_tick(tracker_engine);
  34. }
  35. }
  36. void sound_engine_PWM_timer_init(bool external_audio_output) // external audio on pin PA6
  37. {
  38. LL_TIM_DisableAllOutputs(SPEAKER_PWM_TIMER);
  39. LL_TIM_DisableCounter(SPEAKER_PWM_TIMER);
  40. LL_TIM_InitTypeDef TIM_InitStruct = {0};
  41. LL_TIM_OC_InitTypeDef TIM_OC_InitStruct = {0};
  42. TIM_InitStruct.Prescaler = 0;
  43. TIM_InitStruct.Autoreload = 1023; // 10-bit PWM resolution at around 60 kHz PWM rate
  44. TIM_InitStruct.CounterMode = LL_TIM_COUNTERMODE_UP;
  45. LL_TIM_Init(SPEAKER_PWM_TIMER, &TIM_InitStruct);
  46. TIM_OC_InitStruct.OCMode = LL_TIM_OCMODE_PWM1;
  47. TIM_OC_InitStruct.OCState = LL_TIM_OCSTATE_ENABLE;
  48. TIM_OC_InitStruct.CompareValue = 0;
  49. LL_TIM_OC_Init(SPEAKER_PWM_TIMER, SPEAKER_PWM_TIMER_CHANNEL, &TIM_OC_InitStruct);
  50. if (external_audio_output)
  51. {
  52. furi_hal_gpio_init_ex(&gpio_ext_pa6, GpioModeAltFunctionPushPull, GpioPullNo, GpioSpeedLow, GpioAltFn14TIM16);
  53. if (furi_hal_speaker_is_mine())
  54. {
  55. furi_hal_speaker_release();
  56. }
  57. }
  58. else
  59. {
  60. if (!(furi_hal_speaker_is_mine()))
  61. {
  62. bool unu = furi_hal_speaker_acquire(1000);
  63. UNUSED(unu);
  64. }
  65. furi_hal_gpio_init(&gpio_ext_pa6, GpioModeAnalog, GpioPullNo, GpioSpeedLow);
  66. }
  67. LL_TIM_EnableAllOutputs(SPEAKER_PWM_TIMER);
  68. LL_TIM_EnableCounter(SPEAKER_PWM_TIMER);
  69. }
  70. void sound_engine_set_audio_output(bool external_audio_output)
  71. {
  72. if (external_audio_output)
  73. {
  74. furi_hal_gpio_init_ex(&gpio_ext_pa6, GpioModeAltFunctionPushPull, GpioPullNo, GpioSpeedLow, GpioAltFn14TIM16);
  75. if (furi_hal_speaker_is_mine())
  76. {
  77. furi_hal_speaker_release();
  78. }
  79. }
  80. else
  81. {
  82. if (!(furi_hal_speaker_is_mine()))
  83. {
  84. bool unu = furi_hal_speaker_acquire(1000);
  85. UNUSED(unu);
  86. }
  87. furi_hal_gpio_init(&gpio_ext_pa6, GpioModeAnalog, GpioPullNo, GpioSpeedLow);
  88. }
  89. }
  90. void sound_engine_timer_init(uint32_t sample_rate) // external audio on pin PA6
  91. {
  92. LL_TIM_InitTypeDef TIM_InitStruct = {0};
  93. LL_TIM_OC_InitTypeDef TIM_OC_InitStruct = {0};
  94. TIM_InitStruct.Prescaler = 0;
  95. TIM_InitStruct.Autoreload = TIMER_BASE_CLOCK / sample_rate - 1; // to support various sample rates
  96. TIM_InitStruct.CounterMode = LL_TIM_COUNTERMODE_UP;
  97. LL_TIM_Init(SAMPLE_RATE_TIMER, &TIM_InitStruct);
  98. TIM_OC_InitStruct.OCMode = LL_TIM_OCMODE_PWM1;
  99. TIM_OC_InitStruct.OCState = LL_TIM_OCSTATE_ENABLE;
  100. LL_TIM_OC_Init(SAMPLE_RATE_TIMER, SPEAKER_PWM_TIMER_CHANNEL, &TIM_OC_InitStruct);
  101. LL_TIM_EnableAllOutputs(SAMPLE_RATE_TIMER);
  102. }
  103. void tracker_engine_timer_init(uint8_t rate) // 0-255 hz
  104. {
  105. LL_TIM_InitTypeDef TIM_InitStruct = {0};
  106. LL_TIM_OC_InitTypeDef TIM_OC_InitStruct = {0};
  107. TIM_InitStruct.Prescaler = 0; // using 32-bit timer
  108. TIM_InitStruct.Autoreload = (uint32_t)TIMER_BASE_CLOCK / (uint32_t)rate - 1; // to support various tracker engine rates
  109. TIM_InitStruct.CounterMode = LL_TIM_COUNTERMODE_UP;
  110. LL_TIM_Init(TRACKER_ENGINE_TIMER, &TIM_InitStruct);
  111. TIM_OC_InitStruct.OCMode = LL_TIM_OCMODE_PWM1;
  112. TIM_OC_InitStruct.OCState = LL_TIM_OCSTATE_ENABLE;
  113. LL_TIM_OC_Init(TRACKER_ENGINE_TIMER, SPEAKER_PWM_TIMER_CHANNEL, &TIM_OC_InitStruct);
  114. LL_TIM_EnableIT_UPDATE(TRACKER_ENGINE_TIMER);
  115. }
  116. void tracker_engine_set_rate(uint8_t rate)
  117. {
  118. LL_TIM_InitTypeDef TIM_InitStruct = {0};
  119. TIM_InitStruct.Prescaler = 0; // using 32-bit timer
  120. TIM_InitStruct.Autoreload = (uint32_t)TIMER_BASE_CLOCK / (uint32_t)rate - 1; // to support various tracker engine rates
  121. TIM_InitStruct.CounterMode = LL_TIM_COUNTERMODE_UP;
  122. LL_TIM_Init(TRACKER_ENGINE_TIMER, &TIM_InitStruct);
  123. }
  124. void tracker_engine_init_hardware(uint8_t rate)
  125. {
  126. tracker_engine_timer_init(rate);
  127. }
  128. void sound_engine_dma_init(uint32_t address, uint32_t size)
  129. {
  130. uint32_t dma_dst = (uint32_t) & (SPEAKER_PWM_TIMER->CCR1);
  131. LL_DMA_ConfigAddresses(DMA_INSTANCE, address, dma_dst, LL_DMA_DIRECTION_MEMORY_TO_PERIPH);
  132. LL_DMA_SetDataLength(DMA_INSTANCE, size);
  133. LL_DMA_SetPeriphRequest(DMA_INSTANCE, LL_DMAMUX_REQ_TIM1_UP);
  134. LL_DMA_SetDataTransferDirection(DMA_INSTANCE, LL_DMA_DIRECTION_MEMORY_TO_PERIPH);
  135. LL_DMA_SetChannelPriorityLevel(DMA_INSTANCE, LL_DMA_PRIORITY_VERYHIGH);
  136. LL_DMA_SetMode(DMA_INSTANCE, LL_DMA_MODE_CIRCULAR);
  137. LL_DMA_SetPeriphIncMode(DMA_INSTANCE, LL_DMA_PERIPH_NOINCREMENT);
  138. LL_DMA_SetMemoryIncMode(DMA_INSTANCE, LL_DMA_MEMORY_INCREMENT);
  139. LL_DMA_SetPeriphSize(DMA_INSTANCE, LL_DMA_PDATAALIGN_HALFWORD);
  140. LL_DMA_SetMemorySize(DMA_INSTANCE, LL_DMA_MDATAALIGN_HALFWORD);
  141. LL_DMA_EnableIT_TC(DMA_INSTANCE);
  142. LL_DMA_EnableIT_HT(DMA_INSTANCE);
  143. }
  144. void sound_engine_init_hardware(uint32_t sample_rate, bool external_audio_output, uint16_t *audio_buffer, uint32_t audio_buffer_size)
  145. {
  146. sound_engine_dma_init((uint32_t)audio_buffer, audio_buffer_size);
  147. sound_engine_timer_init(sample_rate);
  148. sound_engine_PWM_timer_init(external_audio_output);
  149. }
  150. void sound_engine_dma_start()
  151. {
  152. LL_DMA_EnableChannel(DMA_INSTANCE);
  153. LL_TIM_EnableDMAReq_UPDATE(SAMPLE_RATE_TIMER);
  154. }
  155. void sound_engine_dma_stop()
  156. {
  157. LL_DMA_DisableChannel(DMA_INSTANCE);
  158. }
  159. void sound_engine_start()
  160. {
  161. LL_TIM_EnableCounter(SAMPLE_RATE_TIMER);
  162. sound_engine_dma_start();
  163. }
  164. void sound_engine_stop()
  165. {
  166. LL_TIM_DisableAllOutputs(SAMPLE_RATE_TIMER);
  167. LL_TIM_DisableCounter(SAMPLE_RATE_TIMER);
  168. sound_engine_dma_stop();
  169. }
  170. void sound_engine_deinit_timer()
  171. {
  172. LL_TIM_DisableAllOutputs(SAMPLE_RATE_TIMER);
  173. LL_TIM_DisableAllOutputs(SPEAKER_PWM_TIMER);
  174. LL_TIM_DisableCounter(SPEAKER_PWM_TIMER);
  175. }
  176. void tracker_engine_start()
  177. {
  178. LL_TIM_EnableAllOutputs(TRACKER_ENGINE_TIMER);
  179. LL_TIM_EnableCounter(TRACKER_ENGINE_TIMER);
  180. }
  181. void tracker_engine_stop()
  182. {
  183. LL_TIM_DisableAllOutputs(TRACKER_ENGINE_TIMER);
  184. LL_TIM_DisableCounter(TRACKER_ENGINE_TIMER);
  185. }
  186. void play()
  187. {
  188. tracker_engine_start();
  189. sound_engine_start();
  190. }
  191. void stop()
  192. {
  193. sound_engine_stop();
  194. tracker_engine_stop();
  195. }