1 /*!
2  * \file      uart-board.c
3  *
4  * \brief     Target board UART driver implementation
5  *
6  * \copyright Revised BSD License, see section \ref LICENSE.
7  *
8  * \code
9  *                ______                              _
10  *               / _____)             _              | |
11  *              ( (____  _____ ____ _| |_ _____  ____| |__
12  *               \____ \| ___ |    (_   _) ___ |/ ___)  _ \
13  *               _____) ) ____| | | || |_| ____( (___| | | |
14  *              (______/|_____)_|_|_| \__)_____)\____)_| |_|
15  *              (C)2013-2017 Semtech
16  *
17  * \endcode
18  *
19  * \author    Miguel Luis ( Semtech )
20  *
21  * \author    Gregory Cristian ( Semtech )
22  */
23 #include "stm32l0xx.h"
24 #include "utilities.h"
25 #include "board.h"
26 #include "sysIrqHandlers.h"
27 #include "uart-board.h"
28 
29 /*!
30  * Number of times the UartPutBuffer will try to send the buffer before
31  * returning ERROR
32  */
33 #define TX_BUFFER_RETRY_COUNT                       10
34 
35 static UART_HandleTypeDef UartHandle;
36 uint8_t RxData = 0;
37 uint8_t TxData = 0;
38 
39 extern Uart_t Uart2;
40 
UartMcuInit(Uart_t * obj,UartId_t uartId,PinNames tx,PinNames rx)41 void UartMcuInit( Uart_t *obj, UartId_t uartId, PinNames tx, PinNames rx )
42 {
43     obj->UartId = uartId;
44 
45     if( uartId == UART_USB_CDC )
46     {
47 #if defined( USE_USB_CDC )
48         UartUsbInit( obj, uartId, NC, NC );
49 #endif
50     }
51     else
52     {
53         __HAL_RCC_USART2_FORCE_RESET( );
54         __HAL_RCC_USART2_RELEASE_RESET( );
55         __HAL_RCC_USART2_CLK_ENABLE( );
56 
57         GpioInit( &obj->Tx, tx, PIN_ALTERNATE_FCT, PIN_PUSH_PULL, PIN_PULL_UP, GPIO_AF4_USART2 );
58         GpioInit( &obj->Rx, rx, PIN_ALTERNATE_FCT, PIN_PUSH_PULL, PIN_PULL_UP, GPIO_AF4_USART2 );
59     }
60 }
61 
UartMcuConfig(Uart_t * obj,UartMode_t mode,uint32_t baudrate,WordLength_t wordLength,StopBits_t stopBits,Parity_t parity,FlowCtrl_t flowCtrl)62 void UartMcuConfig( Uart_t *obj, UartMode_t mode, uint32_t baudrate, WordLength_t wordLength, StopBits_t stopBits, Parity_t parity, FlowCtrl_t flowCtrl )
63 {
64     if( obj->UartId == UART_USB_CDC )
65     {
66 #if defined( USE_USB_CDC )
67         UartUsbConfig( obj, mode, baudrate, wordLength, stopBits, parity, flowCtrl );
68 #endif
69     }
70     else
71     {
72         UartHandle.Instance = USART2;
73         UartHandle.Init.BaudRate = baudrate;
74 
75         if( mode == TX_ONLY )
76         {
77             if( obj->FifoTx.Data == NULL )
78             {
79                 assert_param( LMN_STATUS_ERROR );
80             }
81             UartHandle.Init.Mode = UART_MODE_TX;
82         }
83         else if( mode == RX_ONLY )
84         {
85             if( obj->FifoRx.Data == NULL )
86             {
87                 assert_param( LMN_STATUS_ERROR );
88             }
89             UartHandle.Init.Mode = UART_MODE_RX;
90         }
91         else if( mode == RX_TX )
92         {
93             if( ( obj->FifoTx.Data == NULL ) || ( obj->FifoRx.Data == NULL ) )
94             {
95                 assert_param( LMN_STATUS_ERROR );
96             }
97             UartHandle.Init.Mode = UART_MODE_TX_RX;
98         }
99         else
100         {
101             assert_param( LMN_STATUS_ERROR );
102         }
103 
104         if( wordLength == UART_8_BIT )
105         {
106             UartHandle.Init.WordLength = UART_WORDLENGTH_8B;
107         }
108         else if( wordLength == UART_9_BIT )
109         {
110             UartHandle.Init.WordLength = UART_WORDLENGTH_9B;
111         }
112 
113         switch( stopBits )
114         {
115         case UART_2_STOP_BIT:
116             UartHandle.Init.StopBits = UART_STOPBITS_2;
117             break;
118         case UART_1_5_STOP_BIT:
119             UartHandle.Init.StopBits = UART_STOPBITS_1_5;
120             break;
121         case UART_1_STOP_BIT:
122         default:
123             UartHandle.Init.StopBits = UART_STOPBITS_1;
124             break;
125         }
126 
127         if( parity == NO_PARITY )
128         {
129             UartHandle.Init.Parity = UART_PARITY_NONE;
130         }
131         else if( parity == EVEN_PARITY )
132         {
133             UartHandle.Init.Parity = UART_PARITY_EVEN;
134         }
135         else
136         {
137             UartHandle.Init.Parity = UART_PARITY_ODD;
138         }
139 
140         if( flowCtrl == NO_FLOW_CTRL )
141         {
142             UartHandle.Init.HwFlowCtl = UART_HWCONTROL_NONE;
143         }
144         else if( flowCtrl == RTS_FLOW_CTRL )
145         {
146             UartHandle.Init.HwFlowCtl = UART_HWCONTROL_RTS;
147         }
148         else if( flowCtrl == CTS_FLOW_CTRL )
149         {
150             UartHandle.Init.HwFlowCtl = UART_HWCONTROL_CTS;
151         }
152         else if( flowCtrl == RTS_CTS_FLOW_CTRL )
153         {
154             UartHandle.Init.HwFlowCtl = UART_HWCONTROL_RTS_CTS;
155         }
156 
157         UartHandle.Init.OverSampling = UART_OVERSAMPLING_16;
158 
159         if( HAL_UART_Init( &UartHandle ) != HAL_OK )
160         {
161             assert_param( LMN_STATUS_ERROR );
162         }
163 
164         HAL_NVIC_SetPriority( USART2_IRQn, 1, 0 );
165         HAL_NVIC_EnableIRQ( USART2_IRQn );
166 
167         /* Enable the UART Data Register not empty Interrupt */
168         HAL_UART_Receive_IT( &UartHandle, &RxData, 1 );
169     }
170 }
171 
UartMcuDeInit(Uart_t * obj)172 void UartMcuDeInit( Uart_t *obj )
173 {
174     if( obj->UartId == UART_USB_CDC )
175     {
176 #if defined( USE_USB_CDC )
177         UartUsbDeInit( obj );
178 #endif
179     }
180     else
181     {
182         __HAL_RCC_USART2_FORCE_RESET( );
183         __HAL_RCC_USART2_RELEASE_RESET( );
184         __HAL_RCC_USART2_CLK_DISABLE( );
185 
186         GpioInit( &obj->Tx, obj->Tx.pin, PIN_ANALOGIC, PIN_PUSH_PULL, PIN_NO_PULL, 0 );
187         GpioInit( &obj->Rx, obj->Rx.pin, PIN_ANALOGIC, PIN_PUSH_PULL, PIN_NO_PULL, 0 );
188     }
189 }
190 
UartMcuPutChar(Uart_t * obj,uint8_t data)191 uint8_t UartMcuPutChar( Uart_t *obj, uint8_t data )
192 {
193     if( obj->UartId == UART_USB_CDC )
194     {
195 #if defined( USE_USB_CDC )
196         return UartUsbPutChar( obj, data );
197 #else
198         return 255; // Not supported
199 #endif
200     }
201     else
202     {
203         CRITICAL_SECTION_BEGIN( );
204         TxData = data;
205 
206         if( IsFifoFull( &obj->FifoTx ) == false )
207         {
208             FifoPush( &obj->FifoTx, TxData );
209 
210             // Trig UART Tx interrupt to start sending the FIFO contents.
211             __HAL_UART_ENABLE_IT( &UartHandle, UART_IT_TC );
212 
213             CRITICAL_SECTION_END( );
214             return 0; // OK
215         }
216         CRITICAL_SECTION_END( );
217         return 1; // Busy
218     }
219 }
220 
UartMcuGetChar(Uart_t * obj,uint8_t * data)221 uint8_t UartMcuGetChar( Uart_t *obj, uint8_t *data )
222 {
223     if( obj->UartId == UART_USB_CDC )
224     {
225 #if defined( USE_USB_CDC )
226         return UartUsbGetChar( obj, data );
227 #else
228         return 255; // Not supported
229 #endif
230     }
231     else
232     {
233         CRITICAL_SECTION_BEGIN( );
234 
235         if( IsFifoEmpty( &obj->FifoRx ) == false )
236         {
237             *data = FifoPop( &obj->FifoRx );
238             CRITICAL_SECTION_END( );
239             return 0;
240         }
241         CRITICAL_SECTION_END( );
242         return 1;
243     }
244 }
245 
UartMcuPutBuffer(Uart_t * obj,uint8_t * buffer,uint16_t size)246 uint8_t UartMcuPutBuffer( Uart_t *obj, uint8_t *buffer, uint16_t size )
247 {
248     if( obj->UartId == UART_USB_CDC )
249     {
250 #if defined( USE_USB_CDC )
251         return UartUsbPutBuffer( obj, buffer, size );
252 #else
253         return 255; // Not supported
254 #endif
255     }
256     else
257     {
258         uint8_t retryCount;
259         uint16_t i;
260 
261         for( i = 0; i < size; i++ )
262         {
263             retryCount = 0;
264             while( UartPutChar( obj, buffer[i] ) != 0 )
265             {
266                 retryCount++;
267 
268                 // Exit if something goes terribly wrong
269                 if( retryCount > TX_BUFFER_RETRY_COUNT )
270                 {
271                     return 1; // Error
272                 }
273             }
274         }
275         return 0; // OK
276     }
277 }
278 
UartMcuGetBuffer(Uart_t * obj,uint8_t * buffer,uint16_t size,uint16_t * nbReadBytes)279 uint8_t UartMcuGetBuffer( Uart_t *obj, uint8_t *buffer, uint16_t size, uint16_t *nbReadBytes )
280 {
281     uint16_t localSize = 0;
282 
283     while( localSize < size )
284     {
285         if( UartGetChar( obj, buffer + localSize ) == 0 )
286         {
287             localSize++;
288         }
289         else
290         {
291             break;
292         }
293     }
294 
295     *nbReadBytes = localSize;
296 
297     if( localSize == 0 )
298     {
299         return 1; // Empty
300     }
301     return 0; // OK
302 }
303 
HAL_UART_TxCpltCallback(UART_HandleTypeDef * handle)304 void HAL_UART_TxCpltCallback( UART_HandleTypeDef *handle )
305 {
306     if( IsFifoEmpty( &Uart2.FifoTx ) == false )
307     {
308         TxData = FifoPop( &Uart2.FifoTx );
309         //  Write one byte to the transmit data register
310         HAL_UART_Transmit_IT( &UartHandle, &TxData, 1 );
311     }
312 
313     if( Uart2.IrqNotify != NULL )
314     {
315         Uart2.IrqNotify( UART_NOTIFY_TX );
316     }
317 }
318 
HAL_UART_RxCpltCallback(UART_HandleTypeDef * handle)319 void HAL_UART_RxCpltCallback( UART_HandleTypeDef *handle )
320 {
321     if( IsFifoFull( &Uart2.FifoRx ) == false )
322     {
323         // Read one byte from the receive data register
324         FifoPush( &Uart2.FifoRx, RxData );
325     }
326 
327     if( Uart2.IrqNotify != NULL )
328     {
329         Uart2.IrqNotify( UART_NOTIFY_RX );
330     }
331 
332     HAL_UART_Receive_IT( &UartHandle, &RxData, 1 );
333 }
334 
HAL_UART_ErrorCallback(UART_HandleTypeDef * handle)335 void HAL_UART_ErrorCallback( UART_HandleTypeDef *handle )
336 {
337     HAL_UART_Receive_IT( &UartHandle, &RxData, 1 );
338 }
339 
USART2_IRQHandler(void)340 void USART2_IRQHandler( void )
341 {
342     HAL_UART_IRQHandler( &UartHandle );
343 }
344