commit
						bbbd216244
					
				| @ -0,0 +1,378 @@ | |||||||
|  | //
 | ||||||
|  | // Created by MightyPork on 2018/01/14.
 | ||||||
|  | //
 | ||||||
|  | 
 | ||||||
|  | #include <stm32f072xb.h> | ||||||
|  | #include "platform.h" | ||||||
|  | #include "irq_dispatcher.h" | ||||||
|  | 
 | ||||||
|  | void irqd_init(void) | ||||||
|  | { | ||||||
|  | //    NVIC_EnableIRQ(WWDG_IRQn);                  /*!< Window WatchDog Interrupt                                       */
 | ||||||
|  | //    NVIC_EnableIRQ(PVD_VDDIO2_IRQn);            /*!< PVD & VDDIO2 Interrupt through EXTI Lines 16 and 31             */
 | ||||||
|  | //    NVIC_EnableIRQ(RTC_IRQn);                   /*!< RTC Interrupt through EXTI Lines 17, 19 and 20                  */
 | ||||||
|  | //    NVIC_EnableIRQ(FLASH_IRQn);                 /*!< FLASH global Interrupt                                          */
 | ||||||
|  | //    NVIC_EnableIRQ(RCC_CRS_IRQn);               /*!< RCC & CRS global Interrupt                                      */
 | ||||||
|  | //    NVIC_EnableIRQ(EXTI0_1_IRQn);               /*!< EXTI Line 0 and 1 Interrupt                                     */
 | ||||||
|  | //    NVIC_EnableIRQ(EXTI2_3_IRQn);               /*!< EXTI Line 2 and 3 Interrupt                                     */
 | ||||||
|  | //    NVIC_EnableIRQ(EXTI4_15_IRQn);              /*!< EXTI Line 4 to 15 Interrupt                                     */
 | ||||||
|  | //    NVIC_EnableIRQ(TSC_IRQn);                   /*!< Touch Sensing Controller Interrupts                             */
 | ||||||
|  |     NVIC_EnableIRQ(DMA1_Channel1_IRQn);         /*!< DMA1 Channel 1 Interrupt                                        */ | ||||||
|  |     NVIC_EnableIRQ(DMA1_Channel2_3_IRQn);       /*!< DMA1 Channel 2 and Channel 3 Interrupt                          */ | ||||||
|  |     NVIC_EnableIRQ(DMA1_Channel4_5_6_7_IRQn);   /*!< DMA1 Channel 4 to Channel 7 Interrupt                           */ | ||||||
|  |     HAL_NVIC_SetPriority(DMA1_Channel1_IRQn, 3, 0); | ||||||
|  |     HAL_NVIC_SetPriority(DMA1_Channel2_3_IRQn, 3, 0); | ||||||
|  |     HAL_NVIC_SetPriority(DMA1_Channel4_5_6_7_IRQn, 3, 0); | ||||||
|  | //    NVIC_EnableIRQ(ADC1_COMP_IRQn);             /*!< ADC1 and COMP interrupts (ADC interrupt combined with EXTI Lines 21 and 22 */
 | ||||||
|  | //    NVIC_EnableIRQ(TIM2_IRQn);                  /*!< TIM2 global Interrupt                                           */
 | ||||||
|  | //    NVIC_EnableIRQ(TIM3_IRQn);                  /*!< TIM3 global Interrupt                                           */
 | ||||||
|  | //    NVIC_EnableIRQ(TIM6_DAC_IRQn);              /*!< TIM6 global and DAC channel underrun error Interrupt            */
 | ||||||
|  | //    NVIC_EnableIRQ(TIM7_IRQn);                  /*!< TIM7 global Interrupt                                           */
 | ||||||
|  | //    NVIC_EnableIRQ(TIM14_IRQn);                 /*!< TIM14 global Interrupt                                          */
 | ||||||
|  | //    NVIC_EnableIRQ(TIM15_IRQn);                 /*!< TIM15 global Interrupt                                          */
 | ||||||
|  | //    NVIC_EnableIRQ(TIM16_IRQn);                 /*!< TIM16 global Interrupt                                          */
 | ||||||
|  | //    NVIC_EnableIRQ(TIM17_IRQn);                 /*!< TIM17 global Interrupt                                          */
 | ||||||
|  | //    NVIC_EnableIRQ(I2C1_IRQn);                  /*!< I2C1 Event Interrupt & EXTI Line23 Interrupt (I2C1 wakeup)      */
 | ||||||
|  | //    NVIC_EnableIRQ(I2C2_IRQn);                  /*!< I2C2 Event Interrupt                                            */
 | ||||||
|  | //    NVIC_EnableIRQ(SPI1_IRQn);                  /*!< SPI1 global Interrupt                                           */
 | ||||||
|  | //    NVIC_EnableIRQ(SPI2_IRQn);                  /*!< SPI2 global Interrupt                                           */
 | ||||||
|  |     NVIC_EnableIRQ(USART1_IRQn);                /*!< USART1 global Interrupt & EXTI Line25 Interrupt (USART1 wakeup) */ | ||||||
|  |     NVIC_EnableIRQ(USART2_IRQn);                /*!< USART2 global Interrupt & EXTI Line26 Interrupt (USART2 wakeup) */ | ||||||
|  |     NVIC_EnableIRQ(USART3_4_IRQn);              /*!< USART3 and USART4 global Interrupt                              */ | ||||||
|  |     HAL_NVIC_SetPriority(USART1_IRQn, 3, 0); | ||||||
|  |     HAL_NVIC_SetPriority(USART2_IRQn, 3, 0); | ||||||
|  |     HAL_NVIC_SetPriority(USART3_4_IRQn, 3, 0); | ||||||
|  | //    NVIC_EnableIRQ(CEC_CAN_IRQn);               /*!< CEC and CAN global Interrupts & EXTI Line27 Interrupt           */
 | ||||||
|  | 
 | ||||||
|  | //    NVIC_EnableIRQ(TIM1_BRK_UP_TRG_COM_IRQn);   /*!< TIM1 Break, Update, Trigger and Commutation Interrupt           */ // - handled by hal msp init
 | ||||||
|  | //    NVIC_EnableIRQ(TIM1_CC_IRQn);               /*!< TIM1 Capture Compare Interrupt                                  */
 | ||||||
|  | //    NVIC_EnableIRQ(USB_IRQn);                   /*!< USB global Interrupt  & EXTI Line18 Interrupt                   */ // - USB IRQ is handled by the USB library
 | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | //void Default_Handler(void)
 | ||||||
|  | //{
 | ||||||
|  | //    warn("Missing IRQHandler, ISPR[0]=0x%p", (void*)NVIC->ISPR[0]);
 | ||||||
|  | //}
 | ||||||
|  | 
 | ||||||
|  | struct cbslot { | ||||||
|  |     IrqCallback callback; | ||||||
|  |     void *arg; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | static struct callbacks_ { | ||||||
|  |     struct cbslot usart1; | ||||||
|  |     struct cbslot usart2; | ||||||
|  |     struct cbslot usart3; | ||||||
|  | #ifdef USART4 | ||||||
|  |     struct cbslot usart4; | ||||||
|  | #endif | ||||||
|  | #ifdef USART5 | ||||||
|  |     struct cbslot usart5; | ||||||
|  | #endif | ||||||
|  |     struct cbslot dma1_1; | ||||||
|  |     struct cbslot dma1_2; | ||||||
|  |     struct cbslot dma1_3; | ||||||
|  |     struct cbslot dma1_4; | ||||||
|  |     struct cbslot dma1_5; | ||||||
|  |     struct cbslot dma1_6; | ||||||
|  |     struct cbslot dma1_7; | ||||||
|  |     struct cbslot dma1_8; | ||||||
|  | 
 | ||||||
|  |     struct cbslot dma2_1; | ||||||
|  |     struct cbslot dma2_2; | ||||||
|  |     struct cbslot dma2_3; | ||||||
|  |     struct cbslot dma2_4; | ||||||
|  |     struct cbslot dma2_5; | ||||||
|  |     struct cbslot dma2_6; | ||||||
|  |     struct cbslot dma2_7; | ||||||
|  |     struct cbslot dma2_8; | ||||||
|  | 
 | ||||||
|  |     // XXX add more callbacks here when needed
 | ||||||
|  | } callbacks = {0}; | ||||||
|  | 
 | ||||||
|  | static struct cbslot *get_slot_for_periph(void *periph) | ||||||
|  | { | ||||||
|  |     struct cbslot *slot = NULL; | ||||||
|  |     // --- USART ---
 | ||||||
|  |     if (periph == USART1) slot = &callbacks.usart1; | ||||||
|  |     else if (periph == USART2) slot = &callbacks.usart2; | ||||||
|  |     else if (periph == USART3) slot = &callbacks.usart3; | ||||||
|  | #ifdef USART4 | ||||||
|  |     else if (periph == USART4) slot = &callbacks.usart4; | ||||||
|  | #endif | ||||||
|  | #ifdef USART5 | ||||||
|  |         else if (periph == USART5) slot = &callbacks.usart5; | ||||||
|  | #endif | ||||||
|  | 
 | ||||||
|  |         // --- DMA1 ---
 | ||||||
|  |     else if (periph == DMA1_Channel1) slot = &callbacks.dma1_1; | ||||||
|  |     else if (periph == DMA1_Channel2) slot = &callbacks.dma1_2; | ||||||
|  |     else if (periph == DMA1_Channel3) slot = &callbacks.dma1_3; | ||||||
|  |     else if (periph == DMA1_Channel4) slot = &callbacks.dma1_4; | ||||||
|  |     else if (periph == DMA1_Channel5) slot = &callbacks.dma1_5; | ||||||
|  |     else if (periph == DMA1_Channel6) slot = &callbacks.dma1_6; | ||||||
|  |     else if (periph == DMA1_Channel7) slot = &callbacks.dma1_7; | ||||||
|  | #ifdef DMA1_Channel8 | ||||||
|  |         else if (periph == DMA1_Channel7) slot = &callbacks.dma1_8; | ||||||
|  | #endif | ||||||
|  | 
 | ||||||
|  |         // --- DMA2 ---
 | ||||||
|  | #ifdef DMA2 | ||||||
|  |         else if (periph == DMA2_Channel1) slot = &callbacks.dma2_1; | ||||||
|  |     else if (periph == DMA2_Channel2) slot = &callbacks.dma2_2; | ||||||
|  |     else if (periph == DMA2_Channel3) slot = &callbacks.dma2_3; | ||||||
|  |     else if (periph == DMA2_Channel4) slot = &callbacks.dma2_4; | ||||||
|  |     else if (periph == DMA2_Channel5) slot = &callbacks.dma2_5; | ||||||
|  |     #ifdef DMA2_Channel6 | ||||||
|  |         else if (periph == DMA2_Channel6) slot = &callbacks.dma2_6; | ||||||
|  |     #endif | ||||||
|  |     #ifdef DMA2_Channel7 | ||||||
|  |         else if (periph == DMA2_Channel7) slot = &callbacks.dma2_7; | ||||||
|  |     #endif | ||||||
|  |     #ifdef DMA2_Channel8 | ||||||
|  |         else if (periph == DMA2_Channel7) slot = &callbacks.dma2_8; | ||||||
|  |     #endif | ||||||
|  | #endif | ||||||
|  |     else { | ||||||
|  |         trap("No IRQ cb slot for periph 0x%p", periph); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     return slot; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void irqd_attach(void *periph, IrqCallback callback, void *arg) | ||||||
|  | { | ||||||
|  |     struct cbslot *slot = get_slot_for_periph(periph); | ||||||
|  |     assert_param(slot->callback == NULL); | ||||||
|  |     slot->callback = callback; | ||||||
|  |     slot->arg = arg; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void irqd_detach(void *periph, IrqCallback callback) | ||||||
|  | { | ||||||
|  |     struct cbslot *slot = get_slot_for_periph(periph); | ||||||
|  |     if (slot->callback == callback) { | ||||||
|  |         slot->callback = NULL; | ||||||
|  |         slot->arg = NULL; | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | #define CALL_IRQ_HANDLER(slot) do { if (slot.callback) slot.callback(slot.arg); } while (0) | ||||||
|  | 
 | ||||||
|  | void USART1_IRQHandler(void) | ||||||
|  | { | ||||||
|  |     CALL_IRQ_HANDLER(callbacks.usart1); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void USART2_IRQHandler(void) | ||||||
|  | { | ||||||
|  |     CALL_IRQ_HANDLER(callbacks.usart2); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | #if 0 | ||||||
|  | static bool usart_check_irq(USART_TypeDef *USARTx) | ||||||
|  | { | ||||||
|  |     uint32_t isrflags   = USARTx->ISR; | ||||||
|  |     uint32_t cr1its     = USARTx->CR1; | ||||||
|  |     uint32_t cr2its     = USARTx->CR2; | ||||||
|  |     uint32_t cr3its     = USARTx->CR3; | ||||||
|  | 
 | ||||||
|  |     if ((cr1its & USART_CR1_RTOIE) && (isrflags & USART_ISR_RTOF)) return true; | ||||||
|  |     if ((cr1its & USART_CR1_RXNEIE) && (isrflags & USART_ISR_RXNE)) return true; | ||||||
|  |     if ((cr1its & USART_CR1_TCIE) && (isrflags & USART_ISR_TC)) return true; | ||||||
|  |     if ((cr1its & USART_CR1_TXEIE) && (isrflags & USART_ISR_TXE)) return true; | ||||||
|  |     if ((cr3its & USART_CR3_EIE) && (isrflags & (USART_ISR_PE | USART_ISR_FE | USART_ISR_ORE | USART_ISR_NE))) return true; | ||||||
|  | 
 | ||||||
|  |     if ((cr1its & USART_CR1_IDLEIE) && (isrflags & USART_ISR_IDLE)) return true; | ||||||
|  |     if ((cr1its & USART_CR1_PEIE) && (isrflags & USART_ISR_PE)) return true; | ||||||
|  |     if ((cr1its & USART_CR1_CMIE) && (isrflags & USART_ISR_CMF)) return true; | ||||||
|  |     if ((cr1its & USART_CR1_EOBIE) && (isrflags & USART_ISR_EOBF)) return true; | ||||||
|  |     if ((cr2its & USART_CR2_LBDIE) && (isrflags & USART_ISR_LBDF)) return true; | ||||||
|  |     if ((cr3its & USART_CR3_CTSIE) && (isrflags & USART_ISR_CTS)) return true; | ||||||
|  |     if ((cr3its & USART_CR3_WUFIE) && (isrflags & USART_ISR_WUF)) return true; | ||||||
|  | 
 | ||||||
|  |     return false; | ||||||
|  | } | ||||||
|  | #endif | ||||||
|  | 
 | ||||||
|  | void USART3_4_IRQHandler(void) | ||||||
|  | { | ||||||
|  |     // we won't check the flags here, both can be pending and it's better to let the handler deal with it
 | ||||||
|  |     CALL_IRQ_HANDLER(callbacks.usart3); | ||||||
|  |     CALL_IRQ_HANDLER(callbacks.usart4); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void DMA1_Channel1_IRQHandler(void) | ||||||
|  | { | ||||||
|  |     CALL_IRQ_HANDLER(callbacks.dma1_1); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void DMA1_Channel2_3_IRQHandler(void) | ||||||
|  | { | ||||||
|  |     if (LL_DMA_IsActiveFlag_GI2(DMA1)) CALL_IRQ_HANDLER(callbacks.dma1_2); | ||||||
|  |     if (LL_DMA_IsActiveFlag_GI3(DMA1)) CALL_IRQ_HANDLER(callbacks.dma1_3); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void DMA1_Channel4_5_6_7_IRQHandler(void) | ||||||
|  | { | ||||||
|  |     if (LL_DMA_IsActiveFlag_GI4(DMA1)) CALL_IRQ_HANDLER(callbacks.dma1_4); | ||||||
|  |     if (LL_DMA_IsActiveFlag_GI5(DMA1)) CALL_IRQ_HANDLER(callbacks.dma1_5); | ||||||
|  |     if (LL_DMA_IsActiveFlag_GI6(DMA1)) CALL_IRQ_HANDLER(callbacks.dma1_6); | ||||||
|  |     if (LL_DMA_IsActiveFlag_GI7(DMA1)) CALL_IRQ_HANDLER(callbacks.dma1_7); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | #if 0 | ||||||
|  | void WWDG_IRQHandler(void) | ||||||
|  | { | ||||||
|  |     //
 | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void PVD_VDDIO2_IRQHandler(void) | ||||||
|  | { | ||||||
|  |     //
 | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void RTC_IRQHandler(void) | ||||||
|  | { | ||||||
|  |     //
 | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void FLASH_IRQHandler(void) | ||||||
|  | { | ||||||
|  |     //
 | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void RCC_CRS_IRQHandler(void) | ||||||
|  | { | ||||||
|  |     //
 | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void EXTI0_1_IRQHandler(void) | ||||||
|  | { | ||||||
|  |     //
 | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void EXTI2_3_IRQHandler(void) | ||||||
|  | { | ||||||
|  |     //
 | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void EXTI4_15_IRQHandler(void) | ||||||
|  | { | ||||||
|  |     //
 | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void TSC_IRQHandler(void) | ||||||
|  | { | ||||||
|  |     //
 | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void ADC1_COMP_IRQHandler(void) | ||||||
|  | { | ||||||
|  |     //
 | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void TIM1_BRK_UP_TRG_COM_IRQHandler(void) | ||||||
|  | { | ||||||
|  |     //
 | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void TIM1_CC_IRQHandler(void) | ||||||
|  | { | ||||||
|  |     //
 | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void TIM2_IRQHandler(void) | ||||||
|  | { | ||||||
|  |     //
 | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void TIM3_IRQHandler(void) | ||||||
|  | { | ||||||
|  |     //
 | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // not on F072
 | ||||||
|  | void TIM4_IRQHandler(void) | ||||||
|  | { | ||||||
|  |     //
 | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // not on F072
 | ||||||
|  | void TIM5_IRQHandler(void) | ||||||
|  | { | ||||||
|  |     //
 | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void TIM6_DAC_IRQHandler(void) | ||||||
|  | { | ||||||
|  |     //
 | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void TIM7_IRQHandler(void) | ||||||
|  | { | ||||||
|  |     //
 | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void TIM8_IRQHandler(void) | ||||||
|  | { | ||||||
|  |     //
 | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void TIM9_IRQHandler(void) | ||||||
|  | { | ||||||
|  |     //
 | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void TIM10_IRQHandler(void) | ||||||
|  | { | ||||||
|  |     //
 | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void TIM11_IRQHandler(void) | ||||||
|  | { | ||||||
|  |     //
 | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void TIM12_IRQHandler(void) | ||||||
|  | { | ||||||
|  |     //
 | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void TIM13_IRQHandler(void) | ||||||
|  | { | ||||||
|  |     //
 | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void TIM14_IRQHandler(void) | ||||||
|  | { | ||||||
|  |     //
 | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void TIM15_IRQHandler(void) | ||||||
|  | { | ||||||
|  |     //
 | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void TIM16_IRQHandler(void) | ||||||
|  | { | ||||||
|  |     //
 | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void TIM17_IRQHandler(void) | ||||||
|  | { | ||||||
|  |     //
 | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void CEC_CAN_IRQHandler(void) | ||||||
|  | { | ||||||
|  |     //
 | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // USB is handled by the USB driver
 | ||||||
|  | 
 | ||||||
|  | //void USB_IRQHandler(void)
 | ||||||
|  | //{
 | ||||||
|  | //    //
 | ||||||
|  | //}
 | ||||||
|  | #endif | ||||||
| @ -0,0 +1,39 @@ | |||||||
|  | //
 | ||||||
|  | // Created by MightyPork on 2018/01/14.
 | ||||||
|  | //
 | ||||||
|  | 
 | ||||||
|  | #ifndef GEX_F072_IRQ_DISPATCHER_H | ||||||
|  | #define GEX_F072_IRQ_DISPATCHER_H | ||||||
|  | 
 | ||||||
|  | /**
 | ||||||
|  |  * Initialize the interrupt dispatcher | ||||||
|  |  */ | ||||||
|  | void irqd_init(void); | ||||||
|  | 
 | ||||||
|  | /** Typedef for a IRQ callback run by the dispatcher */ | ||||||
|  | typedef void (*IrqCallback)(void*); | ||||||
|  | 
 | ||||||
|  | /**
 | ||||||
|  |  * Attach a callback to a IRQ handler. | ||||||
|  |  * Pass NULL to remove the handler. | ||||||
|  |  * | ||||||
|  |  * NOTE: The handler is responsible for clearing any interrupt status flags. | ||||||
|  |  * NOTE: It is not guaranteed that the particular peripheral caused the interrupt when | ||||||
|  |  *       the function is called; some interrupt vectors are shared. Make sure to check the flags. | ||||||
|  |  * | ||||||
|  |  * @param periph - peripheral we're attaching to | ||||||
|  |  * @param callback - callback to fire on IRQ | ||||||
|  |  * @param data - data passed to the callback (user context) | ||||||
|  |  */ | ||||||
|  | void irqd_attach(void *periph, IrqCallback callback, void *data); | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | /**
 | ||||||
|  |  * Remove a callback | ||||||
|  |  * | ||||||
|  |  * @param periph - peripheral we're attaching to | ||||||
|  |  * @param callback - callback to remove, if it doesn't match, do nothing | ||||||
|  |  */ | ||||||
|  | void irqd_detach(void *periph, IrqCallback callback); | ||||||
|  | 
 | ||||||
|  | #endif //GEX_F072_IRQ_DISPATCHER_H
 | ||||||
| @ -0,0 +1,444 @@ | |||||||
|  | //
 | ||||||
|  | // Created by MightyPork on 2018/01/14.
 | ||||||
|  | //
 | ||||||
|  | 
 | ||||||
|  | #include "platform.h" | ||||||
|  | #include "irq_dispatcher.h" | ||||||
|  | #include "unit_base.h" | ||||||
|  | 
 | ||||||
|  | #define UUSART_INTERNAL | ||||||
|  | #include "_internal.h" | ||||||
|  | 
 | ||||||
|  | static void UUSART_DMA_RxHandler(void *arg); | ||||||
|  | static void UUSART_DMA_TxHandler(void *arg); | ||||||
|  | 
 | ||||||
|  | #if UUSART_DEBUG | ||||||
|  | #define dbg_uusart(fmt, ...) dbg(fmt, ##__VA_ARGS__) | ||||||
|  | #else | ||||||
|  | #define dbg_uusart(fmt, ...) | ||||||
|  | #endif | ||||||
|  | 
 | ||||||
|  | error_t UUSART_ClaimDMAs(Unit *unit) | ||||||
|  | { | ||||||
|  |     error_t rv; | ||||||
|  |     assert_param(unit); | ||||||
|  |     struct priv *priv = unit->data; | ||||||
|  |     assert_param(priv); | ||||||
|  | 
 | ||||||
|  |     priv->dma = DMA1; | ||||||
|  | 
 | ||||||
|  |     switch (priv->periph_num) { | ||||||
|  |         /* USART1 */ | ||||||
|  |         case 1: | ||||||
|  |             // TX
 | ||||||
|  |             rv = rsc_claim(unit, R_DMA1_2); | ||||||
|  |             if (rv == E_SUCCESS) { | ||||||
|  |                 LL_SYSCFG_SetRemapDMA_USART(LL_SYSCFG_USART1TX_RMP_DMA1CH2); | ||||||
|  |                 priv->dma_tx = DMA1_Channel2; | ||||||
|  |                 priv->dma_tx_chnum = 2; | ||||||
|  |             } else { | ||||||
|  |                 // try the remap
 | ||||||
|  |                 TRY(rsc_claim(unit, R_DMA1_4)); | ||||||
|  |                 LL_SYSCFG_SetRemapDMA_USART(LL_SYSCFG_USART1TX_RMP_DMA1CH4); | ||||||
|  |                 priv->dma_tx = DMA1_Channel4; | ||||||
|  |                 priv->dma_tx_chnum = 4; | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             // RX
 | ||||||
|  |             rv = rsc_claim(unit, R_DMA1_3); | ||||||
|  |             if (rv == E_SUCCESS) { | ||||||
|  |                 LL_SYSCFG_SetRemapDMA_USART(LL_SYSCFG_USART1RX_RMP_DMA1CH3); | ||||||
|  |                 priv->dma_rx = DMA1_Channel3; | ||||||
|  |                 priv->dma_rx_chnum = 3; | ||||||
|  |             } else { | ||||||
|  |                 // try the remap
 | ||||||
|  |                 TRY(rsc_claim(unit, R_DMA1_5)); | ||||||
|  |                 LL_SYSCFG_SetRemapDMA_USART(LL_SYSCFG_USART1RX_RMP_DMA1CH5); | ||||||
|  |                 priv->dma_rx = DMA1_Channel5; | ||||||
|  |                 priv->dma_rx_chnum = 5; | ||||||
|  |             } | ||||||
|  |             break; | ||||||
|  | 
 | ||||||
|  |         /* USART2 */ | ||||||
|  |         case 2: | ||||||
|  |             // RX,TX
 | ||||||
|  |             rv = rsc_claim_range(unit, R_DMA1_4, R_DMA1_5); | ||||||
|  |             if (rv == E_SUCCESS) { | ||||||
|  |                 LL_SYSCFG_SetRemapDMA_USART(LL_SYSCFG_USART2_RMP_DMA1CH54); | ||||||
|  |                 priv->dma_tx = DMA1_Channel4; | ||||||
|  |                 priv->dma_rx = DMA1_Channel5; | ||||||
|  |                 priv->dma_tx_chnum = 4; | ||||||
|  |                 priv->dma_rx_chnum = 5; | ||||||
|  |             } else { | ||||||
|  |                 // try the remap
 | ||||||
|  |                 TRY(rsc_claim_range(unit, R_DMA1_6, R_DMA1_7)); | ||||||
|  |                 LL_SYSCFG_SetRemapDMA_USART(LL_SYSCFG_USART2_RMP_DMA1CH67); | ||||||
|  |                 priv->dma_tx = DMA1_Channel7; | ||||||
|  |                 priv->dma_rx = DMA1_Channel6; | ||||||
|  |                 priv->dma_tx_chnum = 7; | ||||||
|  |                 priv->dma_rx_chnum = 6; | ||||||
|  |             } | ||||||
|  |             break; | ||||||
|  | 
 | ||||||
|  |         /* USART3 */ | ||||||
|  |         case 3: | ||||||
|  |             // RX,TX
 | ||||||
|  |             rv = rsc_claim_range(unit, R_DMA1_6, R_DMA1_7); | ||||||
|  |             if (rv == E_SUCCESS) { | ||||||
|  |                 LL_SYSCFG_SetRemapDMA_USART(LL_SYSCFG_USART3_RMP_DMA1CH67); | ||||||
|  |                 priv->dma_tx = DMA1_Channel7; | ||||||
|  |                 priv->dma_rx = DMA1_Channel6; | ||||||
|  |                 priv->dma_tx_chnum = 7; | ||||||
|  |                 priv->dma_rx_chnum = 6; | ||||||
|  |             } else { | ||||||
|  |                 // try the remap
 | ||||||
|  |                 TRY(rsc_claim_range(unit, R_DMA1_2, R_DMA1_3)); | ||||||
|  |                 LL_SYSCFG_SetRemapDMA_USART(LL_SYSCFG_USART3_RMP_DMA1CH32); | ||||||
|  |                 priv->dma_tx = DMA1_Channel2; | ||||||
|  |                 priv->dma_rx = DMA1_Channel3; | ||||||
|  |                 priv->dma_tx_chnum = 2; | ||||||
|  |                 priv->dma_rx_chnum = 3; | ||||||
|  |             } | ||||||
|  |             break; | ||||||
|  | 
 | ||||||
|  |         /* USART4 */ | ||||||
|  |         case 4: | ||||||
|  |             // RX,TX
 | ||||||
|  |             TRY(rsc_claim_range(unit, R_DMA1_6, R_DMA1_7)); | ||||||
|  |             priv->dma_tx = DMA1_Channel7; | ||||||
|  |             priv->dma_rx = DMA1_Channel6; | ||||||
|  |             priv->dma_tx_chnum = 7; | ||||||
|  |             priv->dma_rx_chnum = 6; | ||||||
|  |             break; | ||||||
|  | 
 | ||||||
|  |         default: | ||||||
|  |             trap("Missing DMA mapping for USART%d", (int)priv->periph_num); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     dbg_uusart("USART %d - selected DMA ch Tx(%d), Rx(%d)", priv->periph_num, priv->dma_tx_chnum, priv->dma_rx_chnum); | ||||||
|  | 
 | ||||||
|  |     return E_SUCCESS; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | error_t UUSART_SetupDMAs(Unit *unit) | ||||||
|  | { | ||||||
|  |     assert_param(unit); | ||||||
|  |     struct priv *priv = unit->data; | ||||||
|  |     assert_param(priv); | ||||||
|  | 
 | ||||||
|  |     priv->rx_buffer = malloc_ck(UUSART_RXBUF_LEN); | ||||||
|  |     if (NULL == priv->rx_buffer) return E_OUT_OF_MEM; | ||||||
|  | 
 | ||||||
|  |     priv->tx_buffer = malloc_ck(UUSART_TXBUF_LEN); | ||||||
|  |     if (NULL == priv->tx_buffer) return E_OUT_OF_MEM; | ||||||
|  | 
 | ||||||
|  |     // Those must be aligned to a word boundary for the DMAs to work.
 | ||||||
|  |     // Any well-behaved malloc impl should do this correctly.
 | ||||||
|  |     assert_param(((uint32_t)priv->rx_buffer & 3) == 0); | ||||||
|  |     assert_param(((uint32_t)priv->tx_buffer & 3) == 0); | ||||||
|  | 
 | ||||||
|  |     priv->rx_buf_readpos = 0; | ||||||
|  | 
 | ||||||
|  |     LL_DMA_InitTypeDef init; | ||||||
|  | 
 | ||||||
|  |     // Transmit buffer
 | ||||||
|  |     { | ||||||
|  |         LL_DMA_StructInit(&init); | ||||||
|  |         init.Direction = LL_DMA_DIRECTION_MEMORY_TO_PERIPH; | ||||||
|  |         init.Mode = LL_DMA_MODE_NORMAL; | ||||||
|  | 
 | ||||||
|  |         init.PeriphOrM2MSrcAddress = (uint32_t) &priv->periph->TDR; | ||||||
|  |         init.PeriphOrM2MSrcDataSize = LL_DMA_PDATAALIGN_BYTE; | ||||||
|  |         init.PeriphOrM2MSrcIncMode = LL_DMA_PERIPH_NOINCREMENT; | ||||||
|  | 
 | ||||||
|  |         init.MemoryOrM2MDstAddress = (uint32_t) priv->tx_buffer; | ||||||
|  |         init.MemoryOrM2MDstDataSize = LL_DMA_MDATAALIGN_BYTE; | ||||||
|  |         init.MemoryOrM2MDstIncMode = LL_DMA_MEMORY_INCREMENT; | ||||||
|  | 
 | ||||||
|  |         assert_param(SUCCESS == LL_DMA_Init(priv->dma, priv->dma_tx_chnum, &init)); | ||||||
|  | 
 | ||||||
|  |         irqd_attach(priv->dma_tx, UUSART_DMA_TxHandler, unit); | ||||||
|  |         // Interrupt on transfer complete
 | ||||||
|  |         LL_DMA_EnableIT_TC(priv->dma, priv->dma_tx_chnum); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     // Receive buffer
 | ||||||
|  |     { | ||||||
|  |         LL_DMA_StructInit(&init); | ||||||
|  |         init.Direction = LL_DMA_DIRECTION_PERIPH_TO_MEMORY; | ||||||
|  | 
 | ||||||
|  |         init.Mode = LL_DMA_MODE_CIRCULAR; | ||||||
|  |         init.NbData = UUSART_RXBUF_LEN; | ||||||
|  | 
 | ||||||
|  |         init.PeriphOrM2MSrcAddress = (uint32_t) &priv->periph->RDR; | ||||||
|  |         init.PeriphOrM2MSrcDataSize = LL_DMA_PDATAALIGN_BYTE; | ||||||
|  |         init.PeriphOrM2MSrcIncMode = LL_DMA_PERIPH_NOINCREMENT; | ||||||
|  | 
 | ||||||
|  |         init.MemoryOrM2MDstAddress = (uint32_t) priv->rx_buffer; | ||||||
|  |         init.MemoryOrM2MDstDataSize = LL_DMA_MDATAALIGN_BYTE; | ||||||
|  |         init.MemoryOrM2MDstIncMode = LL_DMA_MEMORY_INCREMENT; | ||||||
|  | 
 | ||||||
|  |         assert_param(SUCCESS == LL_DMA_Init(priv->dma, priv->dma_rx_chnum, &init)); | ||||||
|  | 
 | ||||||
|  |         irqd_attach(priv->dma_rx, UUSART_DMA_RxHandler, unit); | ||||||
|  |         // Interrupt on transfer 1/2 and complete
 | ||||||
|  |         // We will capture the first and second half and send it while the other half is being filled.
 | ||||||
|  |         LL_DMA_EnableIT_HT(priv->dma, priv->dma_rx_chnum); | ||||||
|  |         LL_DMA_EnableIT_TC(priv->dma, priv->dma_rx_chnum); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     LL_DMA_EnableChannel(priv->dma, priv->dma_rx_chnum); | ||||||
|  |     LL_DMA_EnableChannel(priv->dma, priv->dma_tx_chnum); | ||||||
|  | 
 | ||||||
|  |     // TODO also set up usart timeout interrupt that grabs whatever is in the DMA buffer and sends it
 | ||||||
|  |     return E_SUCCESS; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /**
 | ||||||
|  |  * Handler for the Rx DMA half or full interrupt | ||||||
|  |  * @param arg - unit instance | ||||||
|  |  */ | ||||||
|  | static void UUSART_DMA_RxHandler(void *arg) | ||||||
|  | { | ||||||
|  |     Unit *unit = arg; | ||||||
|  |     assert_param(unit); | ||||||
|  |     struct priv *priv = unit->data; | ||||||
|  |     assert_param(priv); | ||||||
|  | 
 | ||||||
|  |     uint32_t isrsnapshot = priv->dma->ISR; | ||||||
|  | 
 | ||||||
|  |     if (LL_DMA_IsActiveFlag_G(isrsnapshot, priv->dma_rx_chnum)) { | ||||||
|  |         bool tc = LL_DMA_IsActiveFlag_TC(isrsnapshot, priv->dma_rx_chnum); | ||||||
|  |         bool ht = LL_DMA_IsActiveFlag_HT(isrsnapshot, priv->dma_rx_chnum); | ||||||
|  | 
 | ||||||
|  |         // Here we have to either copy it somewhere else, or notify another thread (queue?)
 | ||||||
|  |         // that the data is ready for reading
 | ||||||
|  | 
 | ||||||
|  |         if (ht) { | ||||||
|  |             uint16_t end = (uint16_t) UUSART_RXBUF_LEN / 2; | ||||||
|  |             UUSART_DMA_HandleRxFromIRQ(unit, end); | ||||||
|  |             LL_DMA_ClearFlag_HT(priv->dma, priv->dma_rx_chnum); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         if (tc) { | ||||||
|  |             uint16_t end = (uint16_t) UUSART_RXBUF_LEN; | ||||||
|  |             UUSART_DMA_HandleRxFromIRQ(unit, end); | ||||||
|  |             LL_DMA_ClearFlag_TC(priv->dma, priv->dma_rx_chnum); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         if (LL_DMA_IsActiveFlag_TE(isrsnapshot, priv->dma_rx_chnum)) { | ||||||
|  |             // this shouldn't happen
 | ||||||
|  |             LL_DMA_ClearFlag_TE(priv->dma, priv->dma_rx_chnum); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /**
 | ||||||
|  |  * Start sending a chunk of data. | ||||||
|  |  * This must be called when the DMA is completed. | ||||||
|  |  * | ||||||
|  |  * @param priv | ||||||
|  |  */ | ||||||
|  | static void UUSART_DMA_TxStart(struct priv *priv) | ||||||
|  | { | ||||||
|  |     priv->tx_dma_busy = true; | ||||||
|  |     assert_param(priv->dma_tx->CNDTR == 0); | ||||||
|  | 
 | ||||||
|  |     dbg_uusart("DMA_TxStart (nr %d, nw %d)", (int)priv->tx_buf_nr, (int)priv->tx_buf_nw); | ||||||
|  | 
 | ||||||
|  |     uint16_t nr = priv->tx_buf_nr; | ||||||
|  |     uint16_t nw = priv->tx_buf_nw; | ||||||
|  | 
 | ||||||
|  |     if (nr == nw) { | ||||||
|  |         dbg_uusart("remain=0,do nothing"); | ||||||
|  |         return; | ||||||
|  |     } // do nothing if we're done
 | ||||||
|  | 
 | ||||||
|  |     uint8_t chunk = priv->tx_buffer[nr++]; | ||||||
|  |     //nr += (uint16_t) (4 - (nr & 0b11));
 | ||||||
|  |     if (chunk == 0) { | ||||||
|  |         // wrap-around
 | ||||||
|  |         chunk = priv->tx_buffer[0]; | ||||||
|  |         nr = 1; | ||||||
|  |         assert_param(nr < nw); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     // nr was advanced by the lpad preamble
 | ||||||
|  |     priv->tx_buf_nr = nr; | ||||||
|  |     priv->tx_buf_chunk = chunk; // will be further moved by 'chunk' bytes when dma completes
 | ||||||
|  | 
 | ||||||
|  |     dbg_uusart("# TX: chunk start %d, len %d", (int)nr, (int)chunk); | ||||||
|  | //#if UUSART_DEBUG
 | ||||||
|  | //    PUTS(">"); PUTSN((char *) (priv->tx_buffer + nr), chunk); PUTS("<");
 | ||||||
|  | //    PUTNL();
 | ||||||
|  | //#endif
 | ||||||
|  | 
 | ||||||
|  |     LL_DMA_DisableChannel(priv->dma, priv->dma_tx_chnum); | ||||||
|  |     { | ||||||
|  |         LL_DMA_ClearFlags(priv->dma, priv->dma_tx_chnum); | ||||||
|  |         LL_DMA_SetMemoryAddress(priv->dma, priv->dma_tx_chnum, (uint32_t) (priv->tx_buffer + nr)); | ||||||
|  |         LL_DMA_SetDataLength(priv->dma, priv->dma_tx_chnum, chunk); | ||||||
|  |         LL_USART_ClearFlag_TC(priv->periph); | ||||||
|  |     } | ||||||
|  |     LL_DMA_EnableChannel(priv->dma, priv->dma_tx_chnum); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | COMPILER_ASSERT(UUSART_TXBUF_LEN <= 256); // more would break the "len tag" algorithm
 | ||||||
|  | 
 | ||||||
|  | /**
 | ||||||
|  |  * Put data on the queue. Only a part may be sent due to a buffer size limit. | ||||||
|  |  * | ||||||
|  |  * @param priv | ||||||
|  |  * @param buffer - buffer to send | ||||||
|  |  * @param len - buffer size | ||||||
|  |  * @return number of bytes that were really written (from the beginning) | ||||||
|  |  */ | ||||||
|  | uint16_t UUSART_DMA_TxQueue(struct priv *priv, const uint8_t *buffer, uint16_t len) | ||||||
|  | { | ||||||
|  |     const uint16_t nr = priv->tx_buf_nr; | ||||||
|  |     uint16_t nw = priv->tx_buf_nw; | ||||||
|  | 
 | ||||||
|  |     // shortcut for checking a completely full buffer
 | ||||||
|  |     if (nw == nr-1 || (nr==0&&nw==UUSART_TXBUF_LEN-1)) { | ||||||
|  |         dbg_uusart("Buffer full, cant queue"); | ||||||
|  |         return 0; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     dbg_uusart("\r\nQueue.."); | ||||||
|  | 
 | ||||||
|  |     uint16_t used = 0; | ||||||
|  |     if (nr == nw) { | ||||||
|  |         used = 0; | ||||||
|  |     } else if (nw > nr) { // simple linear
 | ||||||
|  |         used = (uint16_t) (nw - nr); | ||||||
|  |     } else if (nw < nr) { // wrapped
 | ||||||
|  |         used = (uint16_t) ((UUSART_TXBUF_LEN - nr) + nw); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     dbg_uusart("Trying to send buffer of len %d", (int)len); | ||||||
|  |     uint16_t avail = (const uint16_t) (UUSART_TXBUF_LEN - 1 - used); | ||||||
|  |     dbg_uusart("nr %d, nw %d, used %d, avail %d", (int)nr, (int)nw, (int)used, (int)avail); | ||||||
|  | 
 | ||||||
|  |     // hack to avoid too large chunks (we use 1 byte to store chunk size)
 | ||||||
|  |     if (avail > 255) avail = 255; | ||||||
|  | 
 | ||||||
|  |     uint8_t written = 0; | ||||||
|  | 
 | ||||||
|  |     // this avoids attempting to write if we don't have space
 | ||||||
|  |     if (avail <= 5) { | ||||||
|  |         dbg_uusart("No space (only %d)", (int) avail); | ||||||
|  |         return written; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     int cnt = 0; | ||||||
|  |     while (avail > 0 && written < len) { | ||||||
|  |         assert_param(cnt < 2); // if more than two, we have a bug and it's repeating infinitely
 | ||||||
|  | 
 | ||||||
|  |         cnt++; | ||||||
|  |         // Padding with chunk information (1 byte: length) - for each chunk
 | ||||||
|  |         const uint8_t lpad = 1; | ||||||
|  | 
 | ||||||
|  |         // Chunk can go max to the end of the buffer
 | ||||||
|  |         uint8_t chunk = (uint8_t) MIN((len-written) + lpad, UUSART_TXBUF_LEN - nw); | ||||||
|  |         if (chunk > avail) chunk = (uint8_t) avail; | ||||||
|  | 
 | ||||||
|  |         dbg_uusart("nw %d, raw available chunk %d", (int) nw, (int)chunk); | ||||||
|  |         if (chunk < lpad + 1) { | ||||||
|  |             // write 0 to indicate a wrap-around
 | ||||||
|  |             dbg_uusart("Wrap-around marker at offset %d", (int) nw); | ||||||
|  |             priv->tx_buffer[nw] = 0; | ||||||
|  |             nw = 0; | ||||||
|  |         } | ||||||
|  |         else { | ||||||
|  |             // enough space for a preamble + some data
 | ||||||
|  |             dbg_uusart("Preamble of %d bytes at offset %d", (int) lpad, (int) nw); | ||||||
|  |             priv->tx_buffer[nw] = (uint8_t) (chunk - lpad); | ||||||
|  |             nw += lpad; | ||||||
|  |             uint8_t datachunk = (uint8_t) (chunk - lpad); | ||||||
|  |             dbg_uusart("Datachunk len %d at offset %d", (int) datachunk, (int) nw); | ||||||
|  | //#if UUSART_DEBUG
 | ||||||
|  | //            PUTS("mcpy src >"); PUTSN((char *) (buffer), datachunk); PUTS("<\r\n");
 | ||||||
|  | //#endif
 | ||||||
|  |             memcpy((uint8_t *) (priv->tx_buffer + nw), buffer, datachunk); | ||||||
|  | //#if UUSART_DEBUG
 | ||||||
|  | //            PUTS("mcpy dst >"); PUTSN((char *) (priv->tx_buffer + nw), datachunk); PUTS("<\r\n");
 | ||||||
|  | //#endif
 | ||||||
|  |             buffer += datachunk; | ||||||
|  |             nw += datachunk; | ||||||
|  |             written += datachunk; | ||||||
|  |             if (nw == UUSART_TXBUF_LEN) nw = 0; | ||||||
|  |         } | ||||||
|  |         avail -= chunk; | ||||||
|  |         dbg_uusart(". end of loop, avail is %d", (int)avail); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     { | ||||||
|  |         dbg_uusart("Write done -> nr %d, nw %d", (int) nr, (int) nw); | ||||||
|  | 
 | ||||||
|  |         // FIXME a potential race condition can happen here (but it's unlikely)
 | ||||||
|  | 
 | ||||||
|  |         priv->tx_buf_nw = nw; | ||||||
|  | 
 | ||||||
|  |         if (!priv->tx_dma_busy) { | ||||||
|  |             dbg_uusart("Write done, requesting DMA."); | ||||||
|  |             UUSART_DMA_TxStart(priv); | ||||||
|  |         } | ||||||
|  |         else { | ||||||
|  |             dbg_uusart("DMA in progress, not requesting"); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     return written; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /**
 | ||||||
|  |  * Handler for the Tx DMA - completion interrupt | ||||||
|  |  * @param arg - unit instance | ||||||
|  |  */ | ||||||
|  | static void UUSART_DMA_TxHandler(void *arg) | ||||||
|  | { | ||||||
|  |     Unit *unit = arg; | ||||||
|  |     assert_param(unit); | ||||||
|  |     struct priv *priv = unit->data; | ||||||
|  |     assert_param(priv); | ||||||
|  | 
 | ||||||
|  |     uint32_t isrsnapshot = priv->dma->ISR; | ||||||
|  |     if (LL_DMA_IsActiveFlag_TC(isrsnapshot, priv->dma_tx_chnum)) { | ||||||
|  |         // chunk Tx is finished
 | ||||||
|  |         dbg_uusart("~ DMA tx done, nr %d, nw %d, chunk %d", (int)priv->tx_buf_nr, (int)priv->tx_buf_nw, (int)priv->tx_buf_chunk); | ||||||
|  | 
 | ||||||
|  |         priv->tx_buf_nr += priv->tx_buf_chunk; | ||||||
|  |         if (UUSART_TXBUF_LEN == priv->tx_buf_nr) priv->tx_buf_nr = 0; | ||||||
|  |         priv->tx_buf_chunk = 0; | ||||||
|  | 
 | ||||||
|  |         LL_DMA_ClearFlag_TC(priv->dma, priv->dma_tx_chnum); | ||||||
|  | 
 | ||||||
|  |         // Wait for TC
 | ||||||
|  |         while (!LL_USART_IsActiveFlag_TC(priv->periph)); // TODO timeout
 | ||||||
|  | 
 | ||||||
|  |         // start the next chunk
 | ||||||
|  |         if (priv->tx_buf_nr != priv->tx_buf_nw) { | ||||||
|  |             dbg_uusart("  Asking for more, if any"); | ||||||
|  |             UUSART_DMA_TxStart(priv); | ||||||
|  |         } else { | ||||||
|  |             priv->tx_dma_busy = false; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | void UUSART_DeInitDMAs(Unit *unit) | ||||||
|  | { | ||||||
|  |     assert_param(unit); | ||||||
|  |     struct priv *priv = unit->data; | ||||||
|  |     assert_param(priv); | ||||||
|  | 
 | ||||||
|  |     irqd_detach(priv->dma_tx, UUSART_DMA_RxHandler); | ||||||
|  |     irqd_detach(priv->dma_rx, UUSART_DMA_TxHandler); | ||||||
|  | 
 | ||||||
|  |     LL_DMA_DeInit(priv->dma, priv->dma_rx_chnum); | ||||||
|  |     LL_DMA_DeInit(priv->dma, priv->dma_tx_chnum); | ||||||
|  | 
 | ||||||
|  |     free_ck(priv->rx_buffer); | ||||||
|  |     free_ck(priv->tx_buffer); | ||||||
|  | } | ||||||
| @ -0,0 +1,345 @@ | |||||||
|  | //
 | ||||||
|  | // Created by MightyPork on 2018/01/14.
 | ||||||
|  | //
 | ||||||
|  | #include <platform/irq_dispatcher.h> | ||||||
|  | #include "platform.h" | ||||||
|  | #include "unit_base.h" | ||||||
|  | 
 | ||||||
|  | #define UUSART_INTERNAL | ||||||
|  | #include "_internal.h" | ||||||
|  | 
 | ||||||
|  | extern error_t UUSART_ClaimDMAs(Unit *unit); | ||||||
|  | extern error_t UUSART_SetupDMAs(Unit *unit); | ||||||
|  | extern void UUSART_DeInitDMAs(Unit *unit); | ||||||
|  | 
 | ||||||
|  | /** Allocate data structure and set defaults */ | ||||||
|  | error_t UUSART_preInit(Unit *unit) | ||||||
|  | { | ||||||
|  |     struct priv *priv = unit->data = calloc_ck(1, sizeof(struct priv)); | ||||||
|  |     if (priv == NULL) return E_OUT_OF_MEM; | ||||||
|  | 
 | ||||||
|  |     // some defaults
 | ||||||
|  |     priv->periph_num = 1; | ||||||
|  |     priv->remap = 0; | ||||||
|  | 
 | ||||||
|  |     priv->baudrate = 115200; | ||||||
|  |     priv->parity = 0;         //!< 0-none, 1-odd, 2-even
 | ||||||
|  |     priv->stopbits = 1;       //!< 0-half, 1-one, 2-1.5, 3-two
 | ||||||
|  |     priv->direction = 3; // RXTX
 | ||||||
|  | 
 | ||||||
|  |     priv->hw_flow_control = false; | ||||||
|  |     priv->clock_output = false; | ||||||
|  |     priv->cpol = 0; | ||||||
|  |     priv->cpha = 0; | ||||||
|  |     priv->lsb_first = true; // LSB first is default for UART
 | ||||||
|  |     priv->width = 8; | ||||||
|  | 
 | ||||||
|  |     priv->data_inv = false; | ||||||
|  |     priv->rx_inv = false; | ||||||
|  |     priv->tx_inv = false; | ||||||
|  | 
 | ||||||
|  |     priv->de_output = false; | ||||||
|  |     priv->de_polarity = 1; // active high
 | ||||||
|  |     // this should equal to a half-byte length when oversampling by 16 is used (default)
 | ||||||
|  |     priv->de_assert_time = 8; | ||||||
|  |     priv->de_clear_time = 8; | ||||||
|  | 
 | ||||||
|  |     return E_SUCCESS; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /** Claim the peripheral and assign priv->periph */ | ||||||
|  | static inline error_t UUSART_claimPeriph(Unit *unit) | ||||||
|  | { | ||||||
|  |     struct priv *priv = unit->data; | ||||||
|  | 
 | ||||||
|  |     if (!(priv->periph_num >= 1 && priv->periph_num <= 5)) { | ||||||
|  |         dbg("!! Bad USART periph"); | ||||||
|  |         return E_BAD_CONFIG; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     // assign and claim the peripheral
 | ||||||
|  |     if (priv->periph_num == 1) { | ||||||
|  |         TRY(rsc_claim(unit, R_USART1)); | ||||||
|  |         priv->periph = USART1; | ||||||
|  |     } | ||||||
|  |     else if (priv->periph_num == 2) { | ||||||
|  |         TRY(rsc_claim(unit, R_USART2)); | ||||||
|  |         priv->periph = USART2; | ||||||
|  |     } | ||||||
|  |     else if (priv->periph_num == 3) { | ||||||
|  |         TRY(rsc_claim(unit, R_USART3)); | ||||||
|  |         priv->periph = USART3; | ||||||
|  |     } | ||||||
|  | #if defined(USART4) | ||||||
|  |     else if (priv->periph_num == 4) { | ||||||
|  |         TRY(rsc_claim(unit, R_USART4)); | ||||||
|  |         priv->periph = USART4; | ||||||
|  |     } | ||||||
|  | #endif | ||||||
|  | #if defined(USART5) | ||||||
|  |         else if (priv->periph_num == 5) { | ||||||
|  |         TRY(rsc_claim(unit, R_USART5)); | ||||||
|  |         priv->periph = USART5; | ||||||
|  |     } | ||||||
|  | #endif | ||||||
|  |     else return E_BAD_CONFIG; | ||||||
|  | 
 | ||||||
|  |     TRY(UUSART_ClaimDMAs(unit)); | ||||||
|  | 
 | ||||||
|  |     return E_SUCCESS; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /** Claim and configure GPIOs used */ | ||||||
|  | static inline error_t UUSART_configPins(Unit *unit) | ||||||
|  | { | ||||||
|  |     struct priv *priv = unit->data; | ||||||
|  |     // This is written for F072, other platforms will need adjustments
 | ||||||
|  | 
 | ||||||
|  |     // Configure UART pins (AF)
 | ||||||
|  | 
 | ||||||
|  | #define want_ck_pin(priv) ((priv)->clock_output) | ||||||
|  | #define want_tx_pin(priv) (bool)((priv)->direction & 2) | ||||||
|  | #define want_rx_pin(priv) (bool)((priv)->direction & 1) | ||||||
|  | #define want_cts_pin(priv) ((priv)->hw_flow_control==2 || (priv)->hw_flow_control==3) | ||||||
|  | #define want_rts_pin(priv) ((priv)->de_output || (priv)->hw_flow_control==1 || (priv)->hw_flow_control==3) | ||||||
|  | 
 | ||||||
|  |     /* List of required pins based on the user config */ | ||||||
|  |     bool pins_wanted[5] = { | ||||||
|  |         want_ck_pin(priv), | ||||||
|  |         want_tx_pin(priv), | ||||||
|  |         want_rx_pin(priv), | ||||||
|  |         want_cts_pin(priv), | ||||||
|  |         want_rts_pin(priv) | ||||||
|  |     }; | ||||||
|  | 
 | ||||||
|  | #if GEX_PLAT_F072_DISCOVERY | ||||||
|  | 
 | ||||||
|  |     const struct PinAF *mappings = NULL; | ||||||
|  | 
 | ||||||
|  |     // TODO adjust this, possibly remove / split to individual pin config for ..
 | ||||||
|  |     // the final board
 | ||||||
|  | 
 | ||||||
|  |     const struct PinAF mapping_1_0[5] = { | ||||||
|  |         {'A', 8, LL_GPIO_AF_1}, // CK
 | ||||||
|  |         {'A', 9, LL_GPIO_AF_1}, // TX
 | ||||||
|  |         {'A', 10, LL_GPIO_AF_1}, // RX
 | ||||||
|  |         {'A', 11, LL_GPIO_AF_1}, // CTS - collides with USB
 | ||||||
|  |         {'A', 12, LL_GPIO_AF_1}, // RTS - collides with USB
 | ||||||
|  |     }; | ||||||
|  | 
 | ||||||
|  |     const struct PinAF mapping_1_1[5] = { | ||||||
|  |         {'A', 8, LL_GPIO_AF_1}, // CK*
 | ||||||
|  |         {'B', 6, LL_GPIO_AF_1}, // TX
 | ||||||
|  |         {'B', 7, LL_GPIO_AF_1}, // RX
 | ||||||
|  |         {'A', 11, LL_GPIO_AF_1}, // CTS* - collides with USB
 | ||||||
|  |         {'A', 12, LL_GPIO_AF_1}, // RTS* - collides with USB
 | ||||||
|  |     }; | ||||||
|  | 
 | ||||||
|  |     const struct PinAF mapping_2_0[5] = { | ||||||
|  |         {'A', 4, LL_GPIO_AF_1}, // CK
 | ||||||
|  |         {'A', 2, LL_GPIO_AF_1}, // TX
 | ||||||
|  |         {'A', 3, LL_GPIO_AF_1}, // RX
 | ||||||
|  |         {'A', 0, LL_GPIO_AF_1}, // CTS
 | ||||||
|  |         {'A', 1, LL_GPIO_AF_1}, // RTS
 | ||||||
|  |     }; | ||||||
|  | 
 | ||||||
|  |     const struct PinAF mapping_2_1[5] = { | ||||||
|  |         {'A', 4, LL_GPIO_AF_1}, // CK*
 | ||||||
|  |         {'A', 14, LL_GPIO_AF_1}, // TX
 | ||||||
|  |         {'A', 15, LL_GPIO_AF_1}, // RX
 | ||||||
|  |         {'A', 0, LL_GPIO_AF_1}, // CTS*
 | ||||||
|  |         {'A', 1, LL_GPIO_AF_1}, // RTS*
 | ||||||
|  |     }; | ||||||
|  | 
 | ||||||
|  |     const struct PinAF mapping_3_0[5] = { | ||||||
|  |         {'B', 12, LL_GPIO_AF_4}, // CK
 | ||||||
|  |         {'B', 10, LL_GPIO_AF_4}, // TX
 | ||||||
|  |         {'B', 11, LL_GPIO_AF_4}, // RX
 | ||||||
|  |         {'B', 13, LL_GPIO_AF_4}, // CTS
 | ||||||
|  |         {'B', 14, LL_GPIO_AF_4}, // RTS
 | ||||||
|  |     }; | ||||||
|  | 
 | ||||||
|  |     const struct PinAF mapping_4_0[5] = { | ||||||
|  |         {'C', 12, LL_GPIO_AF_0}, // CK
 | ||||||
|  |         {'A', 0, LL_GPIO_AF_4}, // TX
 | ||||||
|  |         {'A', 1, LL_GPIO_AF_4}, // RX
 | ||||||
|  |         {'B', 7, LL_GPIO_AF_4}, // CTS
 | ||||||
|  |         {'A', 15, LL_GPIO_AF_4}, // RTS
 | ||||||
|  |     }; | ||||||
|  | 
 | ||||||
|  |     const struct PinAF mapping_4_1[5] = { | ||||||
|  |         {'C', 12, LL_GPIO_AF_0}, // CK*
 | ||||||
|  |         {'C', 10, LL_GPIO_AF_0}, // TX
 | ||||||
|  |         {'C', 11, LL_GPIO_AF_0}, // RX
 | ||||||
|  |         {'B', 7,  LL_GPIO_AF_4}, // CTS*
 | ||||||
|  |         {'A', 15, LL_GPIO_AF_4}, // RTS*
 | ||||||
|  |     }; | ||||||
|  | 
 | ||||||
|  |     if (priv->periph_num == 1) { | ||||||
|  |         // USART1
 | ||||||
|  |         if (priv->remap == 0) mappings = &mapping_1_0[0]; | ||||||
|  |         else if (priv->remap == 1) mappings = &mapping_1_1[0]; | ||||||
|  |         else return E_BAD_CONFIG; | ||||||
|  |     } | ||||||
|  |     else if (priv->periph_num == 2) { | ||||||
|  |         // USART2
 | ||||||
|  |         if (priv->remap == 0) mappings = &mapping_2_0[0]; | ||||||
|  |         else if (priv->remap == 1) mappings = &mapping_2_1[0]; | ||||||
|  |         else return E_BAD_CONFIG; | ||||||
|  |     } | ||||||
|  |     else if (priv->periph_num == 3) { | ||||||
|  |         // USART3
 | ||||||
|  |         if (priv->remap == 0) mappings = &mapping_3_0[0]; | ||||||
|  |         else return E_BAD_CONFIG; | ||||||
|  |     } | ||||||
|  |     else if (priv->periph_num == 4) { | ||||||
|  |         // USART3
 | ||||||
|  |         if (priv->remap == 0) mappings = &mapping_4_0[0]; | ||||||
|  |         else if (priv->remap == 1) mappings = &mapping_4_1[0]; | ||||||
|  |         else return E_BAD_CONFIG; | ||||||
|  |     } | ||||||
|  |     else return E_BAD_CONFIG; | ||||||
|  | 
 | ||||||
|  |     // Apply mappings based on the 'wanted' table
 | ||||||
|  |     for (int i = 0; i < 5; i++) { | ||||||
|  |         if (pins_wanted[i]) { | ||||||
|  |             if (mappings[i].port == 0) return E_BAD_CONFIG; | ||||||
|  |             TRY(rsc_claim_pin(unit, mappings[i].port, mappings[i].pin)); | ||||||
|  |             hw_configure_gpio_af(mappings[i].port, mappings[i].pin, mappings[i].af); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  | #elif GEX_PLAT_F103_BLUEPILL | ||||||
|  |     #error "NO IMPL" | ||||||
|  | #elif GEX_PLAT_F303_DISCOVERY | ||||||
|  |     #error "NO IMPL" | ||||||
|  | #elif GEX_PLAT_F407_DISCOVERY | ||||||
|  |     #error "NO IMPL" | ||||||
|  | #else | ||||||
|  |     #error "BAD PLATFORM!" | ||||||
|  | #endif | ||||||
|  | 
 | ||||||
|  |     return E_SUCCESS; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /** Finalize unit set-up */ | ||||||
|  | error_t UUSART_init(Unit *unit) | ||||||
|  | { | ||||||
|  |     struct priv *priv = unit->data; | ||||||
|  | 
 | ||||||
|  |     TRY(UUSART_claimPeriph(unit)); | ||||||
|  |     TRY(UUSART_configPins(unit)); | ||||||
|  | 
 | ||||||
|  |     // --- Configure the peripheral ---
 | ||||||
|  | 
 | ||||||
|  |     // Enable clock for the peripheral used
 | ||||||
|  |     hw_periph_clock_enable(priv->periph); | ||||||
|  | 
 | ||||||
|  |     LL_USART_Disable(priv->periph); | ||||||
|  |     { | ||||||
|  |         LL_USART_DeInit(priv->periph); | ||||||
|  |         LL_USART_SetBaudRate(priv->periph, PLAT_APB1_HZ, LL_USART_OVERSAMPLING_16, priv->baudrate); | ||||||
|  | 
 | ||||||
|  |         LL_USART_SetParity(priv->periph, | ||||||
|  |                            priv->parity == 0 ? LL_USART_PARITY_NONE : | ||||||
|  |                            priv->parity == 1 ? LL_USART_PARITY_ODD | ||||||
|  |                                              : LL_USART_PARITY_EVEN); | ||||||
|  | 
 | ||||||
|  |         LL_USART_SetStopBitsLength(priv->periph, | ||||||
|  |                                    priv->stopbits == 0 ? LL_USART_STOPBITS_0_5 : | ||||||
|  |                                    priv->stopbits == 1 ? LL_USART_STOPBITS_1 : | ||||||
|  |                                    priv->stopbits == 2 ? LL_USART_STOPBITS_1_5 | ||||||
|  |                                                        : LL_USART_STOPBITS_2); | ||||||
|  | 
 | ||||||
|  |         LL_USART_SetTransferDirection(priv->periph, | ||||||
|  |                                       priv->direction == 1 ? LL_USART_DIRECTION_RX : | ||||||
|  |                                       priv->direction == 2 ? LL_USART_DIRECTION_TX | ||||||
|  |                                                            : LL_USART_DIRECTION_TX_RX); | ||||||
|  | 
 | ||||||
|  |         LL_USART_SetHWFlowCtrl(priv->periph, | ||||||
|  |                                priv->hw_flow_control == 0 ? LL_USART_HWCONTROL_NONE : | ||||||
|  |                                priv->hw_flow_control == 1 ? LL_USART_HWCONTROL_RTS : | ||||||
|  |                                priv->hw_flow_control == 2 ? LL_USART_HWCONTROL_CTS | ||||||
|  |                                                           : LL_USART_HWCONTROL_RTS_CTS); | ||||||
|  | 
 | ||||||
|  |         LL_USART_ConfigClock(priv->periph, | ||||||
|  |                              priv->cpha ? LL_USART_PHASE_2EDGE : LL_USART_PHASE_1EDGE, | ||||||
|  |                              priv->cpol ? LL_USART_POLARITY_HIGH : LL_USART_POLARITY_LOW, | ||||||
|  |                              true); // clock on last bit - TODO configurable?
 | ||||||
|  | 
 | ||||||
|  |         if (priv->clock_output) | ||||||
|  |             LL_USART_EnableSCLKOutput(priv->periph); | ||||||
|  |         else | ||||||
|  |             LL_USART_DisableSCLKOutput(priv->periph); | ||||||
|  | 
 | ||||||
|  |         LL_USART_SetTransferBitOrder(priv->periph, | ||||||
|  |                                      priv->lsb_first ? LL_USART_BITORDER_LSBFIRST | ||||||
|  |                                                      : LL_USART_BITORDER_MSBFIRST); | ||||||
|  | 
 | ||||||
|  |         LL_USART_SetDataWidth(priv->periph, | ||||||
|  |                               priv->width == 7 ? LL_USART_DATAWIDTH_7B : | ||||||
|  |                               priv->width == 8 ? LL_USART_DATAWIDTH_8B | ||||||
|  |                                                : LL_USART_DATAWIDTH_9B); | ||||||
|  | 
 | ||||||
|  |         LL_USART_SetBinaryDataLogic(priv->periph, | ||||||
|  |                                     priv->data_inv ? LL_USART_BINARY_LOGIC_NEGATIVE | ||||||
|  |                                                    : LL_USART_BINARY_LOGIC_POSITIVE); | ||||||
|  | 
 | ||||||
|  |         LL_USART_SetRXPinLevel(priv->periph, priv->rx_inv ? LL_USART_RXPIN_LEVEL_INVERTED | ||||||
|  |                                                           : LL_USART_RXPIN_LEVEL_STANDARD); | ||||||
|  | 
 | ||||||
|  |         LL_USART_SetTXPinLevel(priv->periph, priv->tx_inv ? LL_USART_TXPIN_LEVEL_INVERTED | ||||||
|  |                                                           : LL_USART_TXPIN_LEVEL_STANDARD); | ||||||
|  | 
 | ||||||
|  |         if (priv->de_output) | ||||||
|  |             LL_USART_EnableDEMode(priv->periph); | ||||||
|  |         else | ||||||
|  |             LL_USART_DisableDEMode(priv->periph); | ||||||
|  | 
 | ||||||
|  |         LL_USART_SetDESignalPolarity(priv->periph, | ||||||
|  |                                      priv->de_polarity ? LL_USART_DE_POLARITY_HIGH | ||||||
|  |                                                        : LL_USART_DE_POLARITY_LOW); | ||||||
|  | 
 | ||||||
|  |         LL_USART_SetDEAssertionTime(priv->periph, priv->de_assert_time); | ||||||
|  |         LL_USART_SetDEDeassertionTime(priv->periph, priv->de_clear_time); | ||||||
|  | 
 | ||||||
|  |         // Prepare for DMA
 | ||||||
|  |         LL_USART_ClearFlag_TC(priv->periph); | ||||||
|  |         LL_USART_EnableDMAReq_RX(priv->periph); | ||||||
|  |         LL_USART_EnableDMAReq_TX(priv->periph); | ||||||
|  |     } | ||||||
|  |     LL_USART_Enable(priv->periph); | ||||||
|  | 
 | ||||||
|  |     // modifies some usart registers that can't be modified when enabled
 | ||||||
|  |     TRY(UUSART_SetupDMAs(unit)); | ||||||
|  | 
 | ||||||
|  |     // timeout based on the baudrate
 | ||||||
|  |     unit->tick_interval = (uint16_t) ((50 * 1000) / priv->baudrate); // receive timeout (ms)
 | ||||||
|  |     if (unit->tick_interval < 5) unit->tick_interval = 5; | ||||||
|  | 
 | ||||||
|  |     return E_SUCCESS; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /** Tear down the unit */ | ||||||
|  | void UUSART_deInit(Unit *unit) | ||||||
|  | { | ||||||
|  |     struct priv *priv = unit->data; | ||||||
|  | 
 | ||||||
|  |     // de-init the pins & peripheral only if inited correctly
 | ||||||
|  |     if (unit->status == E_SUCCESS) { | ||||||
|  |         assert_param(priv->periph); | ||||||
|  |         LL_USART_DeInit(priv->periph); | ||||||
|  | 
 | ||||||
|  |         // Disable clock
 | ||||||
|  |         hw_periph_clock_disable(priv->periph); | ||||||
|  | 
 | ||||||
|  |         UUSART_DeInitDMAs(unit); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     // Release all resources
 | ||||||
|  |     rsc_teardown(unit); | ||||||
|  | 
 | ||||||
|  |     // Free memory
 | ||||||
|  |     free_ck(unit->data); | ||||||
|  |     unit->data = NULL; | ||||||
|  | } | ||||||
| @ -0,0 +1,110 @@ | |||||||
|  | //
 | ||||||
|  | // Created by MightyPork on 2018/01/14.
 | ||||||
|  | //
 | ||||||
|  | 
 | ||||||
|  | #ifndef GEX_F072_UUSART_INTERNAL_H | ||||||
|  | #define GEX_F072_UUSART_INTERNAL_H | ||||||
|  | 
 | ||||||
|  | #ifndef UUSART_INTERNAL | ||||||
|  | #error "Bad include" | ||||||
|  | #endif | ||||||
|  | 
 | ||||||
|  | #include "platform.h" | ||||||
|  | 
 | ||||||
|  | #define UUSART_RXBUF_LEN 64 | ||||||
|  | #define UUSART_TXBUF_LEN 128 | ||||||
|  | 
 | ||||||
|  | /** Private data structure */ | ||||||
|  | struct priv { | ||||||
|  |     uint8_t periph_num; //!< 1-6
 | ||||||
|  |     uint8_t remap;      //!< UART remap option
 | ||||||
|  | 
 | ||||||
|  |     uint32_t baudrate;  //!< baud rate
 | ||||||
|  |     uint8_t parity;     //!< 0-none, 1-odd, 2-even
 | ||||||
|  |     uint8_t stopbits;   //!< 0-half, 1-one, 2-one-and-half, 3-two (halves - 1)
 | ||||||
|  |     uint8_t direction;  //!< 1-RX, 2-TX, 3-RXTX
 | ||||||
|  | 
 | ||||||
|  |     uint8_t hw_flow_control; //!< HW flow control 0-none, 1-RTC, 2-CTS, 3-full
 | ||||||
|  |     bool clock_output;  //!< Output serial clock
 | ||||||
|  |     bool cpol;          //!< clock CPOL setting
 | ||||||
|  |     bool cpha;          //!< clock CPHA setting
 | ||||||
|  |     bool lsb_first;     //!< bit order
 | ||||||
|  |     uint8_t width;      //!< word width - 7, 8, 9 (this includes parity)
 | ||||||
|  | 
 | ||||||
|  |     bool data_inv;  //!< Invert data bytes
 | ||||||
|  |     bool rx_inv;    //!< Invert the RX pin levels
 | ||||||
|  |     bool tx_inv;    //!< Invert the TX pin levels
 | ||||||
|  | 
 | ||||||
|  |     bool de_output; //!< Generate the Driver Enable signal for RS485
 | ||||||
|  |     bool de_polarity; //!< DE active level
 | ||||||
|  |     uint8_t de_assert_time;   //!< Time to assert the DE signal before transmit
 | ||||||
|  |     uint8_t de_clear_time; //!< Time to clear the DE signal after transmit
 | ||||||
|  | 
 | ||||||
|  |     USART_TypeDef *periph; | ||||||
|  | 
 | ||||||
|  |     DMA_TypeDef *dma; | ||||||
|  |     DMA_Channel_TypeDef *dma_rx; | ||||||
|  |     DMA_Channel_TypeDef *dma_tx; | ||||||
|  |     uint8_t dma_rx_chnum; | ||||||
|  |     uint8_t dma_tx_chnum; | ||||||
|  | 
 | ||||||
|  |     // DMA stuff
 | ||||||
|  |     volatile uint8_t *rx_buffer; | ||||||
|  |     volatile uint16_t rx_buf_readpos; | ||||||
|  | 
 | ||||||
|  |     volatile uint8_t *tx_buffer; | ||||||
|  |     volatile uint16_t tx_buf_nr; | ||||||
|  |     volatile uint16_t tx_buf_nw; | ||||||
|  |     volatile uint16_t tx_buf_chunk; | ||||||
|  |     volatile bool tx_dma_busy; | ||||||
|  | 
 | ||||||
|  |     volatile uint16_t rx_last_dmapos; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /** Allocate data structure and set defaults */ | ||||||
|  | error_t UUSART_preInit(Unit *unit); | ||||||
|  | // ------------------------------------------------------------------------
 | ||||||
|  | /** Load from a binary buffer stored in Flash */ | ||||||
|  | void UUSART_loadBinary(Unit *unit, PayloadParser *pp); | ||||||
|  | /** Write to a binary buffer for storing in Flash */ | ||||||
|  | void UUSART_writeBinary(Unit *unit, PayloadBuilder *pb); | ||||||
|  | // ------------------------------------------------------------------------
 | ||||||
|  | /** Parse a key-value pair from the INI file */ | ||||||
|  | error_t UUSART_loadIni(Unit *unit, const char *key, const char *value); | ||||||
|  | /** Generate INI file section for the unit */ | ||||||
|  | void UUSART_writeIni(Unit *unit, IniWriter *iw); | ||||||
|  | // ------------------------------------------------------------------------
 | ||||||
|  | /** Tear down the unit */ | ||||||
|  | void UUSART_deInit(Unit *unit); | ||||||
|  | /** Finalize unit set-up */ | ||||||
|  | error_t UUSART_init(Unit *unit); | ||||||
|  | 
 | ||||||
|  | /**
 | ||||||
|  |  * Handle received data (we're inside the IRQ) | ||||||
|  |  * | ||||||
|  |  * @param unit - handled unit | ||||||
|  |  * @param endpos - end position in the buffer | ||||||
|  |  */ | ||||||
|  | void UUSART_DMA_HandleRxFromIRQ(Unit *unit, uint16_t endpos); | ||||||
|  | 
 | ||||||
|  | /**
 | ||||||
|  |  * Put data on the queue. Only a part may be sent due to a buffer size limit. | ||||||
|  |  * | ||||||
|  |  * @param priv | ||||||
|  |  * @param buffer - buffer to send | ||||||
|  |  * @param len - buffer size | ||||||
|  |  * @return number of bytes that were really written (from the beginning) | ||||||
|  |  */ | ||||||
|  | uint16_t UUSART_DMA_TxQueue(struct priv *priv, const uint8_t *buffer, uint16_t len); | ||||||
|  | 
 | ||||||
|  | /**
 | ||||||
|  |  * Handle rx timeout, grab what is received and send it immediately. | ||||||
|  |  * | ||||||
|  |  * @param unit | ||||||
|  |  */ | ||||||
|  | void UUSART_DMA_HandleRxTimeout(Unit *unit); | ||||||
|  | 
 | ||||||
|  | // ------------------------------------------------------------------------
 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | #endif //GEX_F072_UUSART_INTERNAL_H
 | ||||||
| @ -0,0 +1,74 @@ | |||||||
|  | //
 | ||||||
|  | // Created by MightyPork on 2018/01/14.
 | ||||||
|  | //
 | ||||||
|  | 
 | ||||||
|  | #include "platform.h" | ||||||
|  | #include "unit_base.h" | ||||||
|  | 
 | ||||||
|  | #define UUSART_INTERNAL | ||||||
|  | #include "_internal.h" | ||||||
|  | 
 | ||||||
|  | /** Load from a binary buffer stored in Flash */ | ||||||
|  | void UUSART_loadBinary(Unit *unit, PayloadParser *pp) | ||||||
|  | { | ||||||
|  |     struct priv *priv = unit->data; | ||||||
|  | 
 | ||||||
|  |     uint8_t version = pp_u8(pp); | ||||||
|  |     (void)version; | ||||||
|  | 
 | ||||||
|  |     priv->periph_num = pp_u8(pp); | ||||||
|  |     priv->remap      = pp_u8(pp); | ||||||
|  | 
 | ||||||
|  |     priv->baudrate   = pp_u32(pp); | ||||||
|  |     priv->parity     = pp_u8(pp); | ||||||
|  |     priv->stopbits   = pp_u8(pp); | ||||||
|  |     priv->direction  = pp_u8(pp); | ||||||
|  | 
 | ||||||
|  |     priv->hw_flow_control = pp_u8(pp); | ||||||
|  |     priv->clock_output    = pp_bool(pp); | ||||||
|  |     priv->cpol            = pp_bool(pp); | ||||||
|  |     priv->cpha            = pp_bool(pp); | ||||||
|  |     priv->lsb_first       = pp_bool(pp); | ||||||
|  |     priv->width           = pp_u8(pp); | ||||||
|  | 
 | ||||||
|  |     priv->data_inv = pp_bool(pp); | ||||||
|  |     priv->rx_inv = pp_bool(pp); | ||||||
|  |     priv->tx_inv = pp_bool(pp); | ||||||
|  | 
 | ||||||
|  |     priv->de_output = pp_bool(pp); | ||||||
|  |     priv->de_polarity = pp_bool(pp); | ||||||
|  |     priv->de_assert_time = pp_u8(pp); | ||||||
|  |     priv->de_clear_time = pp_u8(pp); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /** Write to a binary buffer for storing in Flash */ | ||||||
|  | void UUSART_writeBinary(Unit *unit, PayloadBuilder *pb) | ||||||
|  | { | ||||||
|  |     struct priv *priv = unit->data; | ||||||
|  | 
 | ||||||
|  |     pb_u8(pb, 0); // version
 | ||||||
|  | 
 | ||||||
|  |     pb_u8(pb, priv->periph_num); | ||||||
|  |     pb_u8(pb, priv->remap); | ||||||
|  | 
 | ||||||
|  |     pb_u32(pb, priv->baudrate); | ||||||
|  |     pb_u8(pb, priv->parity); | ||||||
|  |     pb_u8(pb, priv->stopbits); | ||||||
|  |     pb_u8(pb, priv->direction); | ||||||
|  | 
 | ||||||
|  |     pb_u8(pb, priv->hw_flow_control); | ||||||
|  |     pb_bool(pb, priv->clock_output); | ||||||
|  |     pb_bool(pb, priv->cpol); | ||||||
|  |     pb_bool(pb, priv->cpha); | ||||||
|  |     pb_bool(pb, priv->lsb_first); | ||||||
|  |     pb_u8(pb, priv->width); | ||||||
|  | 
 | ||||||
|  |     pb_bool(pb, priv->data_inv); | ||||||
|  |     pb_bool(pb, priv->rx_inv); | ||||||
|  |     pb_bool(pb, priv->tx_inv); | ||||||
|  | 
 | ||||||
|  |     pb_bool(pb, priv->de_output); | ||||||
|  |     pb_bool(pb, priv->de_polarity); | ||||||
|  |     pb_u8(pb, priv->de_assert_time); | ||||||
|  |     pb_u8(pb, priv->de_clear_time); | ||||||
|  | } | ||||||
| @ -0,0 +1,168 @@ | |||||||
|  | //
 | ||||||
|  | // Created by MightyPork on 2018/01/14.
 | ||||||
|  | //
 | ||||||
|  | 
 | ||||||
|  | #include "platform.h" | ||||||
|  | #include "unit_base.h" | ||||||
|  | #include "avrlibc.h" | ||||||
|  | 
 | ||||||
|  | #define UUSART_INTERNAL | ||||||
|  | #include "_internal.h" | ||||||
|  | 
 | ||||||
|  | /** Parse a key-value pair from the INI file */ | ||||||
|  | error_t UUSART_loadIni(Unit *unit, const char *key, const char *value) | ||||||
|  | { | ||||||
|  |     bool suc = true; | ||||||
|  |     struct priv *priv = unit->data; | ||||||
|  | 
 | ||||||
|  |     if (streq(key, "device")) { | ||||||
|  |         priv->periph_num = (uint8_t) avr_atoi(value); | ||||||
|  |     } | ||||||
|  |     else if (streq(key, "remap")) { | ||||||
|  |         priv->remap = (uint8_t) avr_atoi(value); | ||||||
|  |     } | ||||||
|  |     else if (streq(key, "baud-rate")) { | ||||||
|  |         priv->baudrate = (uint32_t ) avr_atoi(value); | ||||||
|  |     } | ||||||
|  |     else if (streq(key, "parity")) { | ||||||
|  |         priv->parity = (uint8_t) str_parse_3(value, | ||||||
|  |                                              "NONE", 0, | ||||||
|  |                                              "ODD", 1, | ||||||
|  |                                              "EVEN", 2, &suc); | ||||||
|  |     } | ||||||
|  |     else if (streq(key, "stop-bits")) { | ||||||
|  |         priv->stopbits = (uint8_t) str_parse_4(value, | ||||||
|  |                                                "0.5", 0, | ||||||
|  |                                                "1", 1, | ||||||
|  |                                                "1.5", 2, | ||||||
|  |                                                "2", 3, &suc); | ||||||
|  |     } | ||||||
|  |     else if (streq(key, "direction")) { | ||||||
|  |         priv->direction = (uint8_t) str_parse_3(value, | ||||||
|  |                                                 "RX", 1, | ||||||
|  |                                                 "TX", 2, | ||||||
|  |                                                 "RXTX", 3, &suc); | ||||||
|  |     } | ||||||
|  |     else if (streq(key, "hw-flow-control")) { | ||||||
|  |         priv->hw_flow_control = (uint8_t) str_parse_4(value, | ||||||
|  |                                                       "NONE", 0, | ||||||
|  |                                                       "RTS", 1, | ||||||
|  |                                                       "CTS", 2, | ||||||
|  |                                                       "FULL", 3, &suc); | ||||||
|  |     } | ||||||
|  |     else if (streq(key, "word-width")) { | ||||||
|  |         priv->width = (uint8_t ) avr_atoi(value); | ||||||
|  |     } | ||||||
|  |     else if (streq(key, "first-bit")) { | ||||||
|  |         priv->lsb_first = (bool)str_parse_2(value, "MSB", 0, "LSB", 1, &suc); | ||||||
|  |     } | ||||||
|  |     else if (streq(key, "clock-output")) { | ||||||
|  |         priv->clock_output = str_parse_yn(value, &suc); | ||||||
|  |     } | ||||||
|  |     else if (streq(key, "cpol")) { | ||||||
|  |         priv->cpol = (bool) avr_atoi(value); | ||||||
|  |     } | ||||||
|  |     else if (streq(key, "cpha")) { | ||||||
|  |         priv->cpha = (bool) avr_atoi(value); | ||||||
|  |     } | ||||||
|  |     else if (streq(key, "de-output")) { | ||||||
|  |         priv->de_output = str_parse_yn(value, &suc); | ||||||
|  |     } | ||||||
|  |     else if (streq(key, "de-polarity")) { | ||||||
|  |         priv->de_polarity = (bool) avr_atoi(value); | ||||||
|  |     } | ||||||
|  |     else if (streq(key, "de-assert-time")) { | ||||||
|  |         priv->de_assert_time = (uint8_t) avr_atoi(value); | ||||||
|  |     } | ||||||
|  |     else if (streq(key, "de-clear-time")) { | ||||||
|  |         priv->de_clear_time = (uint8_t) avr_atoi(value); | ||||||
|  |     } | ||||||
|  |     else { | ||||||
|  |         return E_BAD_KEY; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     if (!suc) return E_BAD_VALUE; | ||||||
|  |     return E_SUCCESS; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /** Generate INI file section for the unit */ | ||||||
|  | void UUSART_writeIni(Unit *unit, IniWriter *iw) | ||||||
|  | { | ||||||
|  |     struct priv *priv = unit->data; | ||||||
|  | 
 | ||||||
|  |     iw_comment(iw, "Peripheral number (UARTx 1-4)"); | ||||||
|  |     iw_entry(iw, "device", "%d", (int)priv->periph_num); | ||||||
|  | 
 | ||||||
|  |     iw_comment(iw, "Pin mappings (TX,RX,CK,CTS,RTS/DE)"); | ||||||
|  | #if GEX_PLAT_F072_DISCOVERY | ||||||
|  |     iw_comment(iw, " USART1: (0) A9,A10,A8,A11,A12   (1) B6,B7,A8,A11,A12"); | ||||||
|  |     iw_comment(iw, " USART2: (0) A2,A3,A4,A0,A1      (1) A14,A15,A4,A0,A1"); | ||||||
|  |     iw_comment(iw, " USART3: (0) B10,B11,B12,B13,B14"); | ||||||
|  |     iw_comment(iw, " USART4: (0) A0,A1,C12,B7,A15    (1) C10,C11,C12,B7,A15"); | ||||||
|  | #elif GEX_PLAT_F103_BLUEPILL | ||||||
|  |     #error "NO IMPL" | ||||||
|  | #elif GEX_PLAT_F303_DISCOVERY | ||||||
|  |     #error "NO IMPL" | ||||||
|  | #elif GEX_PLAT_F407_DISCOVERY | ||||||
|  |     #error "NO IMPL" | ||||||
|  | #else | ||||||
|  |     #error "BAD PLATFORM!" | ||||||
|  | #endif | ||||||
|  |     iw_entry(iw, "remap", "%d", (int)priv->remap); | ||||||
|  | 
 | ||||||
|  |     iw_cmt_newline(iw); | ||||||
|  |     iw_comment(iw, "Baud rate in bps (eg. 9600, 115200)"); // TODO examples/range
 | ||||||
|  |     iw_entry(iw, "baud-rate", "%d", (int)priv->baudrate); | ||||||
|  | 
 | ||||||
|  |     iw_comment(iw, "Parity type (NONE, ODD, EVEN)"); | ||||||
|  |     iw_entry(iw, "parity", "%s", str_3(priv->parity, | ||||||
|  |                                        0, "NONE", | ||||||
|  |                                        1, "ODD", | ||||||
|  |                                        2, "EVEN")); | ||||||
|  | 
 | ||||||
|  |     iw_comment(iw, "Number of stop bits (0.5, 1, 1.5, 2)"); | ||||||
|  |     iw_entry(iw, "stop-bits", "%s", str_4(priv->stopbits, | ||||||
|  |                                           0, "0.5", | ||||||
|  |                                           1, "1", | ||||||
|  |                                           2, "1.5", | ||||||
|  |                                           3, "2")); | ||||||
|  | 
 | ||||||
|  |     iw_comment(iw, "Bit order (LSB or MSB first)"); | ||||||
|  |     iw_entry(iw, "first-bit", str_2((uint32_t)priv->lsb_first, | ||||||
|  |                                     0, "MSB", | ||||||
|  |                                     1, "LSB")); | ||||||
|  | 
 | ||||||
|  |     iw_comment(iw, "Word width (7,8,9) - including parity bit if used"); | ||||||
|  |     iw_entry(iw, "word-width", "%d", (int)priv->width); | ||||||
|  | 
 | ||||||
|  |     iw_comment(iw, "Enabled lines (RX,TX,RXTX)"); | ||||||
|  |     iw_entry(iw, "direction", str_3(priv->direction, | ||||||
|  |                                     1, "RX", | ||||||
|  |                                     2, "TX", | ||||||
|  |                                     3, "RXTX")); | ||||||
|  | 
 | ||||||
|  |     iw_comment(iw, "Hardware flow control (NONE, RTS, CTS, FULL)"); | ||||||
|  |     iw_entry(iw, "hw-flow-control", "%s", str_4(priv->hw_flow_control, | ||||||
|  |                                                 0, "NONE", | ||||||
|  |                                                 1, "RTS", | ||||||
|  |                                                 2, "CTS", | ||||||
|  |                                                 3, "FULL")); | ||||||
|  | 
 | ||||||
|  |     iw_cmt_newline(iw); | ||||||
|  |     iw_comment(iw, "Generate serial clock (Y,N)"); | ||||||
|  |     iw_entry(iw, "clock-output", str_yn(priv->clock_output)); | ||||||
|  |     iw_comment(iw, "Output clock polarity: 0,1 (clock idle level)"); | ||||||
|  |     iw_entry(iw, "cpol", "%d", (int)priv->cpol); | ||||||
|  |     iw_comment(iw, "Output clock phase: 0,1 (active edge, 0-first, 1-second)"); | ||||||
|  |     iw_entry(iw, "cpha", "%d", (int)priv->cpha); | ||||||
|  | 
 | ||||||
|  |     iw_cmt_newline(iw); | ||||||
|  |     iw_comment(iw, "Generate RS485 Driver Enable signal (Y,N) - uses RTS pin"); | ||||||
|  |     iw_entry(iw, "de-output", str_yn(priv->de_output)); | ||||||
|  |     iw_comment(iw, "DE active level: 0,1"); | ||||||
|  |     iw_entry(iw, "de-polarity", "%d", (int)(priv->de_polarity)); | ||||||
|  |     iw_comment(iw, "DE assert time (0-31)"); | ||||||
|  |     iw_entry(iw, "de-assert-time", "%d", (int)(priv->de_assert_time)); | ||||||
|  |     iw_comment(iw, "DE clear time (0-31)"); | ||||||
|  |     iw_entry(iw, "de-clear-time", "%d", (int)(priv->de_clear_time)); | ||||||
|  | } | ||||||
| @ -0,0 +1,178 @@ | |||||||
|  | //
 | ||||||
|  | // Created by MightyPork on 2018/01/02.
 | ||||||
|  | //
 | ||||||
|  | 
 | ||||||
|  | #include <stm32f072xb.h> | ||||||
|  | #include "platform.h" | ||||||
|  | #include "comm/messages.h" | ||||||
|  | #include "unit_base.h" | ||||||
|  | #include "unit_usart.h" | ||||||
|  | #include "tasks/task_msg.h" | ||||||
|  | 
 | ||||||
|  | #define UUSART_INTERNAL | ||||||
|  | #include "_internal.h" | ||||||
|  | 
 | ||||||
|  | static void UUSART_SendReceivedDataToMaster(Job *job) | ||||||
|  | { | ||||||
|  |     Unit *unit = job->data1; | ||||||
|  |     struct priv *priv = unit->data; | ||||||
|  | 
 | ||||||
|  |     uint32_t readpos = job->d32; | ||||||
|  |     uint32_t count = job->len; | ||||||
|  | 
 | ||||||
|  |     // Debug: print to debug port
 | ||||||
|  | //    PUTS("Job rx >");
 | ||||||
|  | //    PUTSN((char *) priv->rx_buffer + readpos, (uint16_t) count);
 | ||||||
|  | //    PUTS("<\r\n");
 | ||||||
|  | 
 | ||||||
|  |     // Debug: Write out
 | ||||||
|  | //    UU_USART_Write(unit, (const uint8_t *) (priv->rx_buffer + readpos), count);
 | ||||||
|  | 
 | ||||||
|  |     // TODO modify TF to allow writing in multiple chunks to avoid this useless buffer copying
 | ||||||
|  |     PayloadBuilder pb = pb_start(unit_tmp512, UNIT_TMP_LEN, NULL); | ||||||
|  |     pb_u8(&pb, unit->callsign); | ||||||
|  |     pb_u8(&pb, 0x00); // report type "Data received"
 | ||||||
|  |     pb_buf(&pb, (uint8_t *) (priv->rx_buffer + readpos), count); | ||||||
|  |     assert_param(pb.ok); | ||||||
|  |     com_send_pb(MSG_UNIT_REPORT, &pb); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /**
 | ||||||
|  |  * Handle received data (we're inside the IRQ) | ||||||
|  |  * | ||||||
|  |  * @param unit - handled unit | ||||||
|  |  * @param endpos - end position in the buffer | ||||||
|  |  */ | ||||||
|  | void UUSART_DMA_HandleRxFromIRQ(Unit *unit, uint16_t endpos) | ||||||
|  | { | ||||||
|  |     assert_param(unit); | ||||||
|  |     struct priv *priv = unit->data; | ||||||
|  |     assert_param(priv); | ||||||
|  | 
 | ||||||
|  |     uint16_t readpos = priv->rx_buf_readpos; | ||||||
|  |     assert_param(endpos > readpos); | ||||||
|  | 
 | ||||||
|  |     uint16_t count = (endpos - readpos); | ||||||
|  | 
 | ||||||
|  |     // We defer it to the job queue
 | ||||||
|  |     // FIXME this can starve the shared queue if full duplex is used, we need a second higher priority queue for those report jobs
 | ||||||
|  |     Job j = { | ||||||
|  |         .data1 = unit, | ||||||
|  |         .d32 = priv->rx_buf_readpos, | ||||||
|  |         .len = count, | ||||||
|  |         .cb = UUSART_SendReceivedDataToMaster | ||||||
|  |     }; | ||||||
|  |     scheduleJob(&j); | ||||||
|  | 
 | ||||||
|  |     // Move the read cursor, wrap around if needed
 | ||||||
|  |     if (endpos == UUSART_RXBUF_LEN) endpos = 0; | ||||||
|  |     priv->rx_buf_readpos = endpos; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void UUSART_Tick(Unit *unit) | ||||||
|  | { | ||||||
|  |     assert_param(unit); | ||||||
|  |     struct priv *priv = unit->data; | ||||||
|  |     assert_param(priv); | ||||||
|  | 
 | ||||||
|  |     if (priv->rx_last_dmapos == priv->dma_rx->CNDTR) { | ||||||
|  |         uint16_t endpos = (uint16_t) (UUSART_RXBUF_LEN - priv->rx_last_dmapos); | ||||||
|  |         if (endpos != priv->rx_buf_readpos) { | ||||||
|  |             dbg("DMA timeout"); | ||||||
|  |             UUSART_DMA_HandleRxFromIRQ(unit, endpos); | ||||||
|  |         } | ||||||
|  |     } else { | ||||||
|  |         priv->rx_last_dmapos = (uint16_t) priv->dma_rx->CNDTR; | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | error_t UU_USART_Write(Unit *unit, const uint8_t *buffer, uint32_t len) | ||||||
|  | { | ||||||
|  |     CHECK_TYPE(unit, &UNIT_USART); | ||||||
|  |     struct priv *priv = unit->data; | ||||||
|  | 
 | ||||||
|  |     uint32_t t_start = HAL_GetTick(); | ||||||
|  |     while (len > 0) { | ||||||
|  |         // this should be long enough even for the slowest bitrates and 512 bytes
 | ||||||
|  |         if (HAL_GetTick() - t_start > 5000) { | ||||||
|  |             return E_HW_TIMEOUT; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         uint16_t chunk = UUSART_DMA_TxQueue(priv, buffer, (uint16_t) len); | ||||||
|  | 
 | ||||||
|  |         buffer += chunk; | ||||||
|  |         len -= chunk; | ||||||
|  | 
 | ||||||
|  |         // We give up control if there's another thread waiting and this isn't the last cycle
 | ||||||
|  |         if (len > 0) { | ||||||
|  |             osThreadYield(); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     return E_SUCCESS; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | error_t UU_USART_WriteSync(Unit *unit, const uint8_t *buffer, uint32_t len) | ||||||
|  | { | ||||||
|  |     CHECK_TYPE(unit, &UNIT_USART); | ||||||
|  |     struct priv *priv = unit->data; | ||||||
|  | 
 | ||||||
|  |     TRY(UU_USART_Write(unit, buffer, len)); | ||||||
|  | 
 | ||||||
|  |     // Now wait for the last DMA to complete
 | ||||||
|  |     uint32_t t_start = HAL_GetTick(); | ||||||
|  |     while (priv->tx_dma_busy) { | ||||||
|  |         if (HAL_GetTick() - t_start > 1000) { | ||||||
|  |             return E_HW_TIMEOUT; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     return E_SUCCESS; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | enum PinCmd_ { | ||||||
|  |     CMD_WRITE = 0, | ||||||
|  |     CMD_WRITE_SYNC = 1, | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /** Handle a request message */ | ||||||
|  | static error_t UUSART_handleRequest(Unit *unit, TF_ID frame_id, uint8_t command, PayloadParser *pp) | ||||||
|  | { | ||||||
|  |     uint32_t len; | ||||||
|  |     const uint8_t *pld; | ||||||
|  |     switch (command) { | ||||||
|  |         case CMD_WRITE: | ||||||
|  |             pld = pp_tail(pp, &len); | ||||||
|  |             TRY(UU_USART_Write(unit, pld, len)); | ||||||
|  |             return E_SUCCESS; | ||||||
|  | 
 | ||||||
|  |         case CMD_WRITE_SYNC: | ||||||
|  |             pld = pp_tail(pp, &len); | ||||||
|  |             TRY(UU_USART_WriteSync(unit, pld, len)); | ||||||
|  |             return E_SUCCESS; | ||||||
|  | 
 | ||||||
|  |         default: | ||||||
|  |             return E_UNKNOWN_COMMAND; | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // ------------------------------------------------------------------------
 | ||||||
|  | 
 | ||||||
|  | /** Unit template */ | ||||||
|  | const UnitDriver UNIT_USART = { | ||||||
|  |     .name = "USART", | ||||||
|  |     .description = "Serial port", | ||||||
|  |     // Settings
 | ||||||
|  |     .preInit = UUSART_preInit, | ||||||
|  |     .cfgLoadBinary = UUSART_loadBinary, | ||||||
|  |     .cfgWriteBinary = UUSART_writeBinary, | ||||||
|  |     .cfgLoadIni = UUSART_loadIni, | ||||||
|  |     .cfgWriteIni = UUSART_writeIni, | ||||||
|  |     // Init
 | ||||||
|  |     .init = UUSART_init, | ||||||
|  |     .deInit = UUSART_deInit, | ||||||
|  |     // Function
 | ||||||
|  |     .updateTick = UUSART_Tick, | ||||||
|  |     .handleRequest = UUSART_handleRequest, | ||||||
|  | }; | ||||||
| @ -0,0 +1,34 @@ | |||||||
|  | //
 | ||||||
|  | // Created by MightyPork on 2018/01/02.
 | ||||||
|  | //
 | ||||||
|  | 
 | ||||||
|  | #ifndef GEX_F072_UNIT_USART_H | ||||||
|  | #define GEX_F072_UNIT_USART_H | ||||||
|  | 
 | ||||||
|  | #include "unit.h" | ||||||
|  | 
 | ||||||
|  | extern const UnitDriver UNIT_USART; | ||||||
|  | 
 | ||||||
|  | /**
 | ||||||
|  |  * Write bytes. This function is asynchronous and does not wait for completion. | ||||||
|  |  * It blocks until there's space in the Tx buffer for the data. | ||||||
|  |  * | ||||||
|  |  * @param unit | ||||||
|  |  * @param buffer - bytes to send | ||||||
|  |  * @param len - number of bytes to send | ||||||
|  |  * @return success | ||||||
|  |  */ | ||||||
|  | error_t UU_USART_Write(Unit *unit, const uint8_t *buffer, uint32_t len); | ||||||
|  | 
 | ||||||
|  | /**
 | ||||||
|  |  * Write bytes. Same like UU_USART_Write(), except it waits for the transmission | ||||||
|  |  * to complete after sending the last data. | ||||||
|  |  * | ||||||
|  |  * @param unit | ||||||
|  |  * @param buffer - bytes to send | ||||||
|  |  * @param len - number of bytes to send | ||||||
|  |  * @return success | ||||||
|  |  */ | ||||||
|  | error_t UU_USART_WriteSync(Unit *unit, const uint8_t *buffer, uint32_t len); | ||||||
|  | 
 | ||||||
|  | #endif //GEX_F072_UNIT_USART_H
 | ||||||
					Loading…
					
					
				
		Reference in new issue