Basic_uart.c 9.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324
  1. /* Copyright Statement:
  2. *
  3. * This software/firmware and related documentation ("AutoChips Software") are
  4. * protected under relevant copyright laws. The information contained herein is
  5. * confidential and proprietary to AutoChips Inc. and/or its licensors. Without
  6. * the prior written permission of AutoChips inc. and/or its licensors, any
  7. * reproduction, modification, use or disclosure of AutoChips Software, and
  8. * information contained herein, in whole or in part, shall be strictly
  9. * prohibited.
  10. *
  11. * AutoChips Inc. (C) 2016. All rights reserved.
  12. *
  13. * BY OPENING THIS FILE, RECEIVER HEREBY UNEQUIVOCALLY ACKNOWLEDGES AND AGREES
  14. * THAT THE SOFTWARE/FIRMWARE AND ITS DOCUMENTATIONS ("AUTOCHIPS SOFTWARE")
  15. * RECEIVED FROM AUTOCHIPS AND/OR ITS REPRESENTATIVES ARE PROVIDED TO RECEIVER
  16. * ON AN "AS-IS" BASIS ONLY. AUTOCHIPS EXPRESSLY DISCLAIMS ANY AND ALL
  17. * WARRANTIES, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE IMPLIED
  18. * WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE OR
  19. * NONINFRINGEMENT. NEITHER DOES AUTOCHIPS PROVIDE ANY WARRANTY WHATSOEVER WITH
  20. * RESPECT TO THE SOFTWARE OF ANY THIRD PARTY WHICH MAY BE USED BY,
  21. * INCORPORATED IN, OR SUPPLIED WITH THE AUTOCHIPS SOFTWARE, AND RECEIVER AGREES
  22. * TO LOOK ONLY TO SUCH THIRD PARTY FOR ANY WARRANTY CLAIM RELATING THERETO.
  23. * RECEIVER EXPRESSLY ACKNOWLEDGES THAT IT IS RECEIVER'S SOLE RESPONSIBILITY TO
  24. * OBTAIN FROM ANY THIRD PARTY ALL PROPER LICENSES CONTAINED IN AUTOCHIPS
  25. * SOFTWARE. AUTOCHIPS SHALL ALSO NOT BE RESPONSIBLE FOR ANY AUTOCHIPS SOFTWARE
  26. * RELEASES MADE TO RECEIVER'S SPECIFICATION OR TO CONFORM TO A PARTICULAR
  27. * STANDARD OR OPEN FORUM. RECEIVER'S SOLE AND EXCLUSIVE REMEDY AND AUTOCHIPS'S
  28. * ENTIRE AND CUMULATIVE LIABILITY WITH RESPECT TO THE AUTOCHIPS SOFTWARE
  29. * RELEASED HEREUNDER WILL BE, AT AUTOCHIPS'S OPTION, TO REVISE OR REPLACE THE
  30. * AUTOCHIPS SOFTWARE AT ISSUE, OR REFUND ANY SOFTWARE LICENSE FEES OR SERVICE
  31. * CHARGE PAID BY RECEIVER TO AUTOCHIPS FOR SUCH AUTOCHIPS SOFTWARE AT ISSUE.
  32. */
  33. #include <string.h>
  34. #include "ac780x_uart.h"
  35. #include "ac780x_gpio.h"
  36. #include "ac780x_timer.h"
  37. /*
  38. * 本例程通过使用DMA以及UART空闲中断,实现UART DMA收发,并在UART空闲中断中处理接收的数据
  39. */
  40. #define PERH_BUS_CLK (24000000U)
  41. #define TIM_CHN2_PRD_1ms (PERH_BUS_CLK / 1000 - 1)
  42. #define UARTx UART0
  43. #define UARTx_IRQn UART0_IRQn
  44. #define RS485CTRL_PORT (GPIOB)
  45. #define RS485CTRL_PIN (GPIO_PIN3)
  46. #define RS485_TX_EN do{GPIO_SetPinLevel(RS485CTRL_PORT, RS485CTRL_PIN, GPIO_LEVEL_HIGH);}while(0)
  47. #define RS485_RX_EN do{GPIO_SetPinLevel(RS485CTRL_PORT, RS485CTRL_PIN, GPIO_LEVEL_LOW);}while(0)
  48. #define RUNLED_PORT (GPIOA)
  49. #define RUNLED_PIN (GPIO_PIN2)
  50. /*RUNLED动作定义.*/
  51. #define RUNLED_ON do{GPIO_SetPinLevel(RUNLED_PORT, RUNLED_PIN, GPIO_LEVEL_HIGH);}while(0)
  52. #define RUNLED_OFF do{GPIO_SetPinLevel(RUNLED_PORT, RUNLED_PIN, GPIO_LEVEL_LOW);}while(0)
  53. #define RUNLED_TOGGLE do{if(GPIO_GetPinLevel(RUNLED_PORT, RUNLED_PIN)){RUNLED_OFF;}else{RUNLED_ON;}}while(0)
  54. #define BaudRate 9600
  55. #define UART_TRANSMIT_DATA_POOL_COUNT 64
  56. #define UART_RECV_DATA_POOL_COUNT 64
  57. ///这里DMA mem设置为32bit模式,所以DMA传输的buffer必须4字节对齐
  58. uint32_t dmaRxBuf[UART_RECV_DATA_POOL_COUNT>>2];
  59. uint32_t dmaTxBuf[UART_TRANSMIT_DATA_POOL_COUNT>>2];
  60. uint8_t *rxBuf = (uint8_t *)dmaRxBuf;
  61. uint8_t *txBuf = (uint8_t *)dmaTxBuf;
  62. uint8_t trans_finished = 1;
  63. uint16_t g_scanKeyTime; /*扫描按键间隔时间*/
  64. uint16_t g_blinkLedTime; /*LED闪烁频率控制时间*/
  65. uint16_t g_blinkLedTgtTime; /*LED目标闪烁频率*/
  66. /**
  67. * UARTTX_DMA_EventCallback
  68. *
  69. * @param[in] device: DMA_Type pointer
  70. * @param[in] wpara: DMA Channel Status
  71. * @param[in] lpara: reserve for user's code
  72. * @return none
  73. *
  74. * @brief uart transmit event callback
  75. */
  76. void UartTxDMAEventCallback(void *device, uint32_t wpara, uint32_t lpara)
  77. {
  78. if (wpara & DMA_CHANNEL_STATUS_FINISH_Msk) //!<DMA finish
  79. {
  80. //RS485_RX_EN;
  81. trans_finished = 100;
  82. }
  83. else if (wpara & DMA_CHANNEL_STATUS_HALF_FINISH_Msk) //!<DMA half finish
  84. {
  85. }
  86. else if (wpara & DMA_CHANNEL_STATUS_TRANS_ERROR_Msk) //!<DMA error
  87. {
  88. }
  89. else
  90. {
  91. //!<never be here
  92. }
  93. }
  94. /**
  95. * UARTRX_DMA_EventCallback
  96. *
  97. * @param[in] device: DMA_Type pointer
  98. * @param[in] wpara: DMA Channel Status
  99. * @param[in] lpara: reserve for user's code
  100. * @return none
  101. *
  102. * @brief uart dma receive handle
  103. */
  104. static void UartRxDMAEventCallback(void *device, uint32_t wpara, uint32_t lpara)
  105. {
  106. /* Check if DMA Channel Receive Complete */
  107. if (wpara & DMA_CHANNEL_STATUS_FINISH_Msk) //!<DMA finish
  108. {
  109. }
  110. else if (wpara & DMA_CHANNEL_STATUS_HALF_FINISH_Msk) //!<DMA half finish
  111. {
  112. }
  113. else if (wpara & DMA_CHANNEL_STATUS_TRANS_ERROR_Msk) //!<DMA error
  114. {
  115. }
  116. else
  117. {
  118. //!<never be here
  119. }
  120. }
  121. /**
  122. * UARTRX_DMA_EventCallback
  123. *
  124. * @param[in] device: UART_Type pointer
  125. * @param[in] wpara: UART lsr0 register
  126. * @param[in] lpara: UART lsr1 register
  127. * @return none
  128. *
  129. * @brief uart dma receive handle
  130. */
  131. static void UartRxIdleCallBack(void *device, uint32_t wpara, uint32_t lpara)
  132. {
  133. /* Check if IDLE interrupt happened */
  134. if (lpara & UART_LSR1_IDLE_Msk)
  135. {
  136. uint16_t recLen;
  137. uint16_t index;
  138. ///接收不足4字节会在DMA fifo中缓存
  139. recLen = DMA0_CHANNEL1->DATA_TRANS_NUM + DMA0_CHANNEL1->FIFO_LEFT_NUM;
  140. ///将DMA fifo数据flush出来,因为mem设置为32bit,所以fifo也是4字节对齐
  141. ///(即使fifo里只有1个字节也会刷出4字节,需要保证缓存冗余不会溢出)
  142. DMA_ChannelFlush(DMA0_CHANNEL1);
  143. for (index=0;index<recLen;index++)
  144. {
  145. txBuf[index] = rxBuf[index];
  146. }
  147. RS485_TX_EN;
  148. trans_finished = 1;
  149. UART_TransmitDMA(UARTx, DMA0_CHANNEL0, txBuf, recLen, UartTxDMAEventCallback);
  150. UART_ReceiveDMA(UARTx, DMA0_CHANNEL1, rxBuf, UART_RECV_DATA_POOL_COUNT, UartRxDMAEventCallback);
  151. }
  152. }
  153. /*************<prototype>**************/
  154. void TIM_CHN2_Callback(void *device, uint32_t wpara, uint32_t lpara);
  155. /**
  156. * @prototype TIMER_PrdInit(void)
  157. *
  158. * @param[in] void
  159. * @return void
  160. *
  161. * @brief 初始化定时器通道2.
  162. */
  163. void TIMER_PrdInit(void)
  164. {
  165. TIMER_ConfigType tmrConfig;
  166. /*清零变量.*/
  167. memset(&tmrConfig, 0x00, sizeof(tmrConfig));
  168. /*配置定时器.*/
  169. tmrConfig.linkModeEn = DISABLE;
  170. tmrConfig.interruptEn = ENABLE;
  171. tmrConfig.periodValue = TIM_CHN2_PRD_1ms;
  172. tmrConfig.timerEn = ENABLE;
  173. tmrConfig.callBack = TIM_CHN2_Callback;
  174. TIMER_Init(TIMER_CHANNEL2, &tmrConfig);
  175. }
  176. /**
  177. * @prototype TIM_CHN2_Callback(void *device, uint32_t wpara, uint32_t lpara)
  178. *
  179. * @param[in] ...
  180. * @return void
  181. *
  182. * @brief 定时器通道2中断回调函数.
  183. */
  184. void TIM_CHN2_Callback(void *device, uint32_t wpara, uint32_t lpara)
  185. {
  186. if (g_scanKeyTime < 0xFFFF)
  187. {
  188. g_scanKeyTime++;
  189. }
  190. if (g_blinkLedTime < 0xFFFF)
  191. {
  192. g_blinkLedTime++;
  193. }
  194. if(trans_finished > 0 ){
  195. trans_finished++;
  196. }
  197. }
  198. /**
  199. * Basic_uart
  200. *
  201. * @param[in] none
  202. * @return none
  203. *
  204. * @brief Basic_uart function
  205. */
  206. void Basic_uart(void)
  207. {
  208. uint8_t i = 0;
  209. UART_ConfigType uartConfig;
  210. memset((void *)&uartConfig, 0, sizeof(UART_ConfigType));
  211. //set pin mux
  212. GPIO_SetFunc(GPIOA, GPIO_PIN7, GPIO_FUN1);
  213. GPIO_SetFunc(GPIOA, GPIO_PIN8, GPIO_FUN1);
  214. //GPIO_SetDir(GPIOA, GPIO_PIN7, GPIO_OUT);
  215. //GPIO_SetPullup(GPIOA, GPIO_PIN7, ENABLE);
  216. //GPIO_SetDir(GPIOA, GPIO_PIN8, GPIO_IN);
  217. GPIO_SetFunc(RS485CTRL_PORT, RS485CTRL_PIN, GPIO_FUN0);
  218. GPIO_SetDir(RS485CTRL_PORT, RS485CTRL_PIN, GPIO_OUT);
  219. GPIO_SetFunc(RUNLED_PORT, RUNLED_PIN, GPIO_FUN0);
  220. GPIO_SetDir(RUNLED_PORT, RUNLED_PIN, GPIO_OUT);
  221. uartConfig.baudrate = BaudRate;
  222. uartConfig.dataBits = UART_WORD_LEN_8BIT;
  223. uartConfig.stopBits = UART_STOP_1BIT;
  224. uartConfig.parity = UART_PARI_NO;
  225. uartConfig.fifoByteEn = ENABLE; ///<must enable fifoByte when use DMA
  226. uartConfig.dmaEn = UART_DMA_TXRX_EN;
  227. uartConfig.callBack = UartRxIdleCallBack; ///<uart2 interrupt callback
  228. UART_Init(UARTx, &uartConfig);
  229. UARTx->IDLE |= UART_IDLE_IDLEIE_Msk | UART_IDLE_ILEN_Msk; ///<enable uart idle interrupt
  230. ///Enable UARTx interrupt
  231. NVIC_SetPriority(UARTx_IRQn, 3);
  232. NVIC_ClearPendingIRQ(UARTx_IRQn);
  233. NVIC_EnableIRQ(UARTx_IRQn);
  234. TIMER_PrdInit();
  235. /* Initializing data in the txBuf */
  236. for (i = 0; i < 10; i++)
  237. {
  238. txBuf[i] = i + 1;
  239. }
  240. RS485_TX_EN;
  241. trans_finished = 1;
  242. UART_TransmitDMA(UARTx, DMA0_CHANNEL0, txBuf, 10, UartTxDMAEventCallback);
  243. UART_ReceiveDMA(UARTx, DMA0_CHANNEL1, rxBuf, UART_RECV_DATA_POOL_COUNT, UartRxDMAEventCallback);
  244. while(1){
  245. if (g_scanKeyTime >= 1000)/*定时周期到,扫描按键动作.*/
  246. {
  247. g_scanKeyTime = 0;
  248. RUNLED_TOGGLE;
  249. }
  250. if(trans_finished >= 120){
  251. trans_finished = 0;
  252. RS485_RX_EN;
  253. }
  254. /*
  255. do{
  256. if(GPIO_GetPinLevel(GPIOA, GPIO_PIN7)){
  257. GPIO_SetPinLevel(GPIOA, GPIO_PIN7, GPIO_LEVEL_LOW);
  258. }else{
  259. GPIO_SetPinLevel(GPIOA, GPIO_PIN7, GPIO_LEVEL_HIGH);
  260. }
  261. }while(0);
  262. */
  263. //RS485_TX_EN;
  264. //UART_TransmitDMA(UARTx, DMA0_CHANNEL0, txBuf, 10, UartTxDMAEventCallback);
  265. }
  266. }