flizzer_tracker_hal.c 11 KB

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