api-hal-irda.c 4.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147
  1. #include "api-hal-interrupt.h"
  2. #include "api-hal-irda.h"
  3. #include <stm32wbxx_ll_tim.h>
  4. #include <stm32wbxx_ll_gpio.h>
  5. #include <stdio.h>
  6. #include <furi.h>
  7. #include "main.h"
  8. #include "api-hal-pwm.h"
  9. static struct{
  10. TimerISRCallback callback;
  11. void *ctx;
  12. } timer_irda;
  13. typedef enum{
  14. TimerIRQSourceCCI1,
  15. TimerIRQSourceCCI2,
  16. } TimerIRQSource;
  17. static void api_hal_irda_handle_capture(TimerIRQSource source)
  18. {
  19. uint32_t duration = 0;
  20. bool level = 0;
  21. switch (source) {
  22. case TimerIRQSourceCCI1:
  23. duration = LL_TIM_OC_GetCompareCH1(TIM2);
  24. LL_TIM_SetCounter(TIM2, 0);
  25. level = 0;
  26. break;
  27. case TimerIRQSourceCCI2:
  28. duration = LL_TIM_OC_GetCompareCH2(TIM2);
  29. LL_TIM_SetCounter(TIM2, 0);
  30. level = 1;
  31. break;
  32. default:
  33. furi_check(0);
  34. }
  35. if (timer_irda.callback)
  36. timer_irda.callback(timer_irda.ctx, level, duration);
  37. }
  38. static void api_hal_irda_isr() {
  39. if(LL_TIM_IsActiveFlag_CC1(TIM2) == 1) {
  40. LL_TIM_ClearFlag_CC1(TIM2);
  41. if(READ_BIT(TIM2->CCMR1, TIM_CCMR1_CC1S)) {
  42. // input capture
  43. api_hal_irda_handle_capture(TimerIRQSourceCCI1);
  44. } else {
  45. // output compare
  46. // HAL_TIM_OC_DelayElapsedCallback(htim);
  47. // HAL_TIM_PWM_PulseFinishedCallback(htim);
  48. }
  49. }
  50. if(LL_TIM_IsActiveFlag_CC2(TIM2) == 1) {
  51. LL_TIM_ClearFlag_CC2(TIM2);
  52. if(READ_BIT(TIM2->CCMR1, TIM_CCMR1_CC2S)) {
  53. // input capture
  54. api_hal_irda_handle_capture(TimerIRQSourceCCI2);
  55. } else {
  56. // output compare
  57. // HAL_TIM_OC_DelayElapsedCallback(htim);
  58. // HAL_TIM_PWM_PulseFinishedCallback(htim);
  59. }
  60. }
  61. }
  62. void api_hal_irda_rx_irq_init(void)
  63. {
  64. LL_TIM_InitTypeDef TIM_InitStruct = {0};
  65. LL_GPIO_InitTypeDef GPIO_InitStruct = {0};
  66. /* Peripheral clock enable */
  67. LL_APB1_GRP1_EnableClock(LL_APB1_GRP1_PERIPH_TIM2);
  68. LL_AHB2_GRP1_EnableClock(LL_AHB2_GRP1_PERIPH_GPIOA);
  69. /**TIM2 GPIO Configuration
  70. PA0 ------> TIM2_CH1
  71. */
  72. GPIO_InitStruct.Pin = LL_GPIO_PIN_0;
  73. GPIO_InitStruct.Mode = LL_GPIO_MODE_ALTERNATE;
  74. GPIO_InitStruct.Speed = LL_GPIO_SPEED_FREQ_LOW;
  75. GPIO_InitStruct.OutputType = LL_GPIO_OUTPUT_PUSHPULL;
  76. GPIO_InitStruct.Pull = LL_GPIO_PULL_NO;
  77. GPIO_InitStruct.Alternate = LL_GPIO_AF_1;
  78. LL_GPIO_Init(GPIOA, &GPIO_InitStruct);
  79. TIM_InitStruct.Prescaler = 64 - 1;
  80. TIM_InitStruct.CounterMode = LL_TIM_COUNTERMODE_UP;
  81. TIM_InitStruct.Autoreload = 0xFFFFFFFF;
  82. TIM_InitStruct.ClockDivision = LL_TIM_CLOCKDIVISION_DIV1;
  83. LL_TIM_Init(TIM2, &TIM_InitStruct);
  84. LL_TIM_SetClockSource(TIM2, LL_TIM_CLOCKSOURCE_INTERNAL);
  85. LL_TIM_EnableARRPreload(TIM2);
  86. LL_TIM_SetTriggerOutput(TIM2, LL_TIM_TRGO_RESET);
  87. LL_TIM_DisableMasterSlaveMode(TIM2);
  88. LL_TIM_IC_SetActiveInput(TIM2, LL_TIM_CHANNEL_CH1, LL_TIM_ACTIVEINPUT_DIRECTTI);
  89. LL_TIM_IC_SetPrescaler(TIM2, LL_TIM_CHANNEL_CH1, LL_TIM_ICPSC_DIV1);
  90. LL_TIM_IC_SetFilter(TIM2, LL_TIM_CHANNEL_CH1, LL_TIM_IC_FILTER_FDIV1);
  91. LL_TIM_IC_SetPolarity(TIM2, LL_TIM_CHANNEL_CH1, LL_TIM_IC_POLARITY_FALLING);
  92. LL_TIM_IC_SetActiveInput(TIM2, LL_TIM_CHANNEL_CH2, LL_TIM_ACTIVEINPUT_INDIRECTTI);
  93. LL_TIM_IC_SetPrescaler(TIM2, LL_TIM_CHANNEL_CH2, LL_TIM_ICPSC_DIV1);
  94. LL_TIM_IC_SetFilter(TIM2, LL_TIM_CHANNEL_CH2, LL_TIM_IC_FILTER_FDIV1);
  95. LL_TIM_IC_SetPolarity(TIM2, LL_TIM_CHANNEL_CH2, LL_TIM_IC_POLARITY_RISING);
  96. LL_TIM_EnableIT_CC1(TIM2);
  97. LL_TIM_EnableIT_CC2(TIM2);
  98. LL_TIM_CC_EnableChannel(TIM2, LL_TIM_CHANNEL_CH1);
  99. LL_TIM_CC_EnableChannel(TIM2, LL_TIM_CHANNEL_CH2);
  100. LL_TIM_SetCounter(TIM2, 0);
  101. LL_TIM_EnableCounter(TIM2);
  102. api_hal_interrupt_set_timer_isr(TIM2, api_hal_irda_isr);
  103. NVIC_SetPriority(TIM2_IRQn, NVIC_EncodePriority(NVIC_GetPriorityGrouping(),5, 0));
  104. NVIC_EnableIRQ(TIM2_IRQn);
  105. }
  106. void api_hal_irda_rx_irq_deinit(void) {
  107. LL_TIM_DeInit(TIM2);
  108. api_hal_interrupt_set_timer_isr(TIM2, NULL);
  109. }
  110. bool api_hal_irda_rx_irq_is_busy(void) {
  111. return (LL_TIM_IsEnabledIT_CC1(TIM2) || LL_TIM_IsEnabledIT_CC2(TIM2));
  112. }
  113. void api_hal_irda_rx_irq_set_callback(TimerISRCallback callback, void *ctx) {
  114. furi_check(callback);
  115. timer_irda.callback = callback;
  116. timer_irda.ctx = ctx;
  117. }
  118. void api_hal_irda_pwm_set(float value, float freq) {
  119. hal_pwmn_set(value, freq, &IRDA_TX_TIM, IRDA_TX_CH);
  120. }
  121. void api_hal_irda_pwm_stop() {
  122. hal_pwmn_stop(&IRDA_TX_TIM, IRDA_TX_CH);
  123. }