api-hal-irda.c 5.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163
  1. #include "api-hal-irda.h"
  2. #include <cmsis_os2.h>
  3. #include <api-hal-interrupt.h>
  4. #include <api-hal-resources.h>
  5. #include <stdint.h>
  6. #include <stm32wbxx_ll_tim.h>
  7. #include <stm32wbxx_ll_gpio.h>
  8. #include <stdio.h>
  9. #include <furi.h>
  10. #include <main.h>
  11. #include <api-hal-pwm.h>
  12. static struct{
  13. ApiHalIrdaCaptureCallback capture_callback;
  14. void *capture_context;
  15. ApiHalIrdaTimeoutCallback timeout_callback;
  16. void *timeout_context;
  17. } timer_irda;
  18. typedef enum{
  19. TimerIRQSourceCCI1,
  20. TimerIRQSourceCCI2,
  21. } TimerIRQSource;
  22. static void api_hal_irda_handle_timeout(void) {
  23. /* Timers CNT register starts to counting from 0 to ARR, but it is
  24. * reseted when Channel 1 catches interrupt. It is not reseted by
  25. * channel 2, though, so we have to distract it's values (see TimerIRQSourceCCI1 ISR).
  26. * This can cause false timeout: when time is over, but we started
  27. * receiving new signal few microseconds ago, because CNT register
  28. * is reseted once per period, not per sample. */
  29. if (LL_GPIO_IsInputPinSet(gpio_irda_rx.port, gpio_irda_rx.pin) == 0)
  30. return;
  31. if (timer_irda.timeout_callback)
  32. timer_irda.timeout_callback(timer_irda.timeout_context);
  33. }
  34. /* High pin level is a Space state of IRDA signal. Invert level for further processing. */
  35. static void api_hal_irda_handle_capture(TimerIRQSource source) {
  36. uint32_t duration = 0;
  37. bool level = 0;
  38. switch (source) {
  39. case TimerIRQSourceCCI1:
  40. duration = LL_TIM_IC_GetCaptureCH1(TIM2) - LL_TIM_IC_GetCaptureCH2(TIM2);
  41. level = 1;
  42. break;
  43. case TimerIRQSourceCCI2:
  44. duration = LL_TIM_IC_GetCaptureCH2(TIM2);
  45. level = 0;
  46. break;
  47. default:
  48. furi_check(0);
  49. }
  50. if (timer_irda.capture_callback)
  51. timer_irda.capture_callback(timer_irda.capture_context, level, duration);
  52. }
  53. static void api_hal_irda_isr() {
  54. if(LL_TIM_IsActiveFlag_CC3(TIM2)) {
  55. LL_TIM_ClearFlag_CC3(TIM2);
  56. api_hal_irda_handle_timeout();
  57. }
  58. if(LL_TIM_IsActiveFlag_CC1(TIM2)) {
  59. LL_TIM_ClearFlag_CC1(TIM2);
  60. if(READ_BIT(TIM2->CCMR1, TIM_CCMR1_CC1S)) {
  61. // input capture
  62. api_hal_irda_handle_capture(TimerIRQSourceCCI1);
  63. }
  64. }
  65. if(LL_TIM_IsActiveFlag_CC2(TIM2)) {
  66. LL_TIM_ClearFlag_CC2(TIM2);
  67. if(READ_BIT(TIM2->CCMR1, TIM_CCMR1_CC2S)) {
  68. // input capture
  69. api_hal_irda_handle_capture(TimerIRQSourceCCI2);
  70. }
  71. }
  72. }
  73. void api_hal_irda_rx_irq_init(void) {
  74. LL_APB1_GRP1_EnableClock(LL_APB1_GRP1_PERIPH_TIM2);
  75. LL_AHB2_GRP1_EnableClock(LL_AHB2_GRP1_PERIPH_GPIOA);
  76. hal_gpio_init_ex(&gpio_irda_rx, GpioModeAltFunctionPushPull, GpioPullNo, GpioSpeedLow, GpioAltFn1TIM2);
  77. LL_TIM_InitTypeDef TIM_InitStruct = {0};
  78. TIM_InitStruct.Prescaler = 64 - 1;
  79. TIM_InitStruct.CounterMode = LL_TIM_COUNTERMODE_UP;
  80. TIM_InitStruct.Autoreload = 0x7FFFFFFE;
  81. TIM_InitStruct.ClockDivision = LL_TIM_CLOCKDIVISION_DIV1;
  82. LL_TIM_Init(TIM2, &TIM_InitStruct);
  83. LL_TIM_SetClockSource(TIM2, LL_TIM_CLOCKSOURCE_INTERNAL);
  84. LL_TIM_DisableARRPreload(TIM2);
  85. LL_TIM_SetTriggerInput(TIM2, LL_TIM_TS_TI1FP1);
  86. LL_TIM_SetSlaveMode(TIM2, LL_TIM_SLAVEMODE_RESET);
  87. LL_TIM_CC_DisableChannel(TIM2, LL_TIM_CHANNEL_CH2);
  88. LL_TIM_IC_SetFilter(TIM2, LL_TIM_CHANNEL_CH2, LL_TIM_IC_FILTER_FDIV1);
  89. LL_TIM_IC_SetPolarity(TIM2, LL_TIM_CHANNEL_CH2, LL_TIM_IC_POLARITY_FALLING);
  90. LL_TIM_DisableIT_TRIG(TIM2);
  91. LL_TIM_DisableDMAReq_TRIG(TIM2);
  92. LL_TIM_SetTriggerOutput(TIM2, LL_TIM_TRGO_RESET);
  93. LL_TIM_EnableMasterSlaveMode(TIM2);
  94. LL_TIM_IC_SetActiveInput(TIM2, LL_TIM_CHANNEL_CH1, LL_TIM_ACTIVEINPUT_DIRECTTI);
  95. LL_TIM_IC_SetPrescaler(TIM2, LL_TIM_CHANNEL_CH1, LL_TIM_ICPSC_DIV1);
  96. LL_TIM_IC_SetFilter(TIM2, LL_TIM_CHANNEL_CH1, LL_TIM_IC_FILTER_FDIV1);
  97. LL_TIM_IC_SetPolarity(TIM2, LL_TIM_CHANNEL_CH1, LL_TIM_IC_POLARITY_RISING);
  98. LL_TIM_IC_SetActiveInput(TIM2, LL_TIM_CHANNEL_CH2, LL_TIM_ACTIVEINPUT_INDIRECTTI);
  99. LL_TIM_IC_SetPrescaler(TIM2, LL_TIM_CHANNEL_CH2, LL_TIM_ICPSC_DIV1);
  100. LL_TIM_EnableIT_CC1(TIM2);
  101. LL_TIM_EnableIT_CC2(TIM2);
  102. LL_TIM_CC_EnableChannel(TIM2, LL_TIM_CHANNEL_CH1);
  103. LL_TIM_CC_EnableChannel(TIM2, LL_TIM_CHANNEL_CH2);
  104. api_hal_interrupt_set_timer_isr(TIM2, api_hal_irda_isr);
  105. LL_TIM_SetCounter(TIM2, 0);
  106. LL_TIM_EnableCounter(TIM2);
  107. NVIC_SetPriority(TIM2_IRQn, NVIC_EncodePriority(NVIC_GetPriorityGrouping(),5, 0));
  108. NVIC_EnableIRQ(TIM2_IRQn);
  109. }
  110. void api_hal_irda_rx_irq_deinit(void) {
  111. LL_TIM_DeInit(TIM2);
  112. api_hal_interrupt_set_timer_isr(TIM2, NULL);
  113. }
  114. void api_hal_irda_rx_timeout_irq_init(uint32_t timeout_ms) {
  115. LL_TIM_OC_SetCompareCH3(TIM2, timeout_ms * 1000);
  116. LL_TIM_OC_SetMode(TIM2, LL_TIM_CHANNEL_CH3, LL_TIM_OCMODE_ACTIVE);
  117. LL_TIM_CC_EnableChannel(TIM2, LL_TIM_CHANNEL_CH3);
  118. LL_TIM_EnableIT_CC3(TIM2);
  119. }
  120. bool api_hal_irda_rx_irq_is_busy(void) {
  121. return (LL_TIM_IsEnabledIT_CC1(TIM2) || LL_TIM_IsEnabledIT_CC2(TIM2));
  122. }
  123. void api_hal_irda_rx_irq_set_callback(ApiHalIrdaCaptureCallback callback, void *ctx) {
  124. timer_irda.capture_callback = callback;
  125. timer_irda.capture_context = ctx;
  126. }
  127. void api_hal_irda_rx_timeout_irq_set_callback(ApiHalIrdaTimeoutCallback callback, void *ctx) {
  128. timer_irda.timeout_callback = callback;
  129. timer_irda.timeout_context = ctx;
  130. }
  131. void api_hal_irda_pwm_set(float value, float freq) {
  132. hal_pwmn_set(value, freq, &IRDA_TX_TIM, IRDA_TX_CH);
  133. }
  134. void api_hal_irda_pwm_stop() {
  135. hal_pwmn_stop(&IRDA_TX_TIM, IRDA_TX_CH);
  136. }