Serial.c 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350
  1. /*******************************************************************************************
  2. * File Name: Serial.c
  3. * Function Describe:device for serial
  4. * Relate Module:
  5. * Explain:
  6. 接收采用中断方式
  7. UART1: 连接尾插,用于与PC相连 另外,printf映射到UART1
  8. UART2: 连接GPS
  9. UART3: 连接Modem
  10. 全局变量g_ucUARTSel 调试选择控制,决定了3个串口的转发关系,其值通过以下3个宏控分别代表:
  11. COM_SEL_MCU 正常使用不转发,尾插用于MCU打印LOG或 MCU软件升级
  12. COM_SEL_GPS UART1<-->UART2,尾插用于GPS LOG打印
  13. COM_SEL_MODEM UART1<-->UART3,尾插用于MODEM AT momand 或模块软件升级
  14. * Writer: ShiLiangWen
  15. * Date: 2015.1.20
  16. ********************************************************************************************/
  17. #include "Serial.h"
  18. #include "stm32f10x_dma.h"
  19. #include "includes.h"
  20. #define USART1_DR_Base 0x40013804
  21. #define USART2_DR_Base 0x40004404
  22. unsigned char g_ucUARTSel=0;//0--MCU 1--Modem 2--GPS
  23. //UART1 Trace
  24. unsigned char RxBuffer1[UART1_RX_BUFFER_SIZE];
  25. unsigned char TxBuffer1[UART1_TX_BUFFER_SIZE];
  26. unsigned char g_ucUart1Received=0;//接受到一包数据标志位
  27. unsigned char g_ucUart1Sending=0;//仍有数据待发送标志
  28. unsigned short rx1_ct=0,rx1_len=0;
  29. unsigned short tx1_ct=0,tx1_len=0;
  30. /********************************************************************
  31. *Uart1DMAInit
  32. *串口1 DMA设置
  33. *串口1 发送必须用通道4,接收必须用通道5
  34. *********************************************************************/
  35. void Uart1DMAInit(void)
  36. {
  37. NVIC_InitTypeDef NVIC_InitStructure;
  38. DMA_InitTypeDef DMA_InitStructure;
  39. //启动DMA时钟
  40. RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE);
  41. //DMA 通道4用于UART1 TX
  42. DMA_DeInit(DMA1_Channel4);
  43. DMA_InitStructure.DMA_PeripheralBaseAddr = (u32)(&USART1->DR);//外设地址
  44. DMA_InitStructure.DMA_MemoryBaseAddr = (uint32_t)TxBuffer1;//内存地址
  45. DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralDST;//DMA传输方向单向
  46. DMA_InitStructure.DMA_BufferSize = UART1_TX_BUFFER_SIZE; //设置DMA在传输时缓冲区的长度
  47. DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable; //设置DMA的外设递增模式
  48. DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable; //设置DMA的内存递增模式
  49. DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_Byte;//外设数据字长
  50. DMA_InitStructure.DMA_MemoryDataSize = DMA_PeripheralDataSize_Byte;//内存数据字长
  51. DMA_InitStructure.DMA_Mode = DMA_Mode_Normal; //设置DMA的传输模式
  52. DMA_InitStructure.DMA_Priority = DMA_Priority_Low; //设置DMA的优先级别
  53. DMA_InitStructure.DMA_M2M = DMA_M2M_Disable;//设置DMA的2个memory中的变量互相访问
  54. DMA_Init(DMA1_Channel4,&DMA_InitStructure);
  55. //DMA中断设置
  56. NVIC_InitStructure.NVIC_IRQChannel = DMA1_Channel4_IRQn;
  57. NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 3;
  58. NVIC_InitStructure.NVIC_IRQChannelSubPriority = 2;
  59. NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
  60. NVIC_Init(&NVIC_InitStructure);
  61. DMA_ITConfig(DMA1_Channel4,DMA_IT_TC,ENABLE);//中断设置
  62. DMA_Cmd(DMA1_Channel4, DISABLE); //等到需要发送的时候再使能
  63. //DMA 通道5用于UART1 RX
  64. DMA_DeInit(DMA1_Channel5);
  65. DMA_InitStructure.DMA_PeripheralBaseAddr = (u32)(&USART1->DR);//外设地址
  66. DMA_InitStructure.DMA_MemoryBaseAddr = (uint32_t)RxBuffer1;//内存地址
  67. DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC;//DMA传输方向单向
  68. DMA_InitStructure.DMA_BufferSize = UART1_RX_BUFFER_SIZE; //设置DMA在传输时缓冲区的长度
  69. DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable; //设置DMA的外设递增模式
  70. DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable; //设置DMA的内存递增模式
  71. DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_Byte;//外设数据字长
  72. DMA_InitStructure.DMA_MemoryDataSize = DMA_PeripheralDataSize_Byte;//内存数据字长
  73. DMA_InitStructure.DMA_Mode = DMA_Mode_Normal; //设置DMA的传输模式
  74. DMA_InitStructure.DMA_Priority = DMA_Priority_High; //设置DMA的优先级别
  75. DMA_InitStructure.DMA_M2M = DMA_M2M_Disable;//设置DMA的2个memory中的变量互相访问
  76. DMA_Init(DMA1_Channel5,&DMA_InitStructure);
  77. DMA_Cmd(DMA1_Channel5, ENABLE); //使能通道
  78. }
  79. /********************************************************************************
  80. *Uart1Send
  81. *串口1启动发送
  82. *********************************************************************************/
  83. /*
  84. void Uart1Send(char *txbuf,int len)
  85. {
  86. int i;
  87. unsigned char data;
  88. unsigned char last_data=~UART_TRANSFORM;
  89. //Send Head
  90. USART_SendData(USART1, UART_HEAD);
  91. while (USART_GetFlagStatus(USART1, USART_FLAG_TC) == RESET);
  92. for(i=0;i<len;i++){
  93. data= txbuf[i];
  94. if(UART_TRANSFORM == data){
  95. USART_SendData(USART1,UART_TRANSFORM );
  96. while (USART_GetFlagStatus(USART1, USART_FLAG_TC) == RESET);
  97. USART_SendData(USART1,0x01 );
  98. while (USART_GetFlagStatus(USART1, USART_FLAG_TC) == RESET);
  99. }else if(UART_HEAD == data){
  100. USART_SendData(USART1,UART_TRANSFORM);
  101. while (USART_GetFlagStatus(USART1, USART_FLAG_TC) == RESET);
  102. USART_SendData(USART1,0x02 );
  103. while (USART_GetFlagStatus(USART1, USART_FLAG_TC) == RESET);
  104. }else if(UART_END == data){
  105. USART_SendData(USART1,UART_TRANSFORM);
  106. while (USART_GetFlagStatus(USART1, USART_FLAG_TC) == RESET);
  107. USART_SendData(USART1,0x03 );
  108. while (USART_GetFlagStatus(USART1, USART_FLAG_TC) == RESET);
  109. }else{
  110. USART_SendData(USART1,data);
  111. while (USART_GetFlagStatus(USART1, USART_FLAG_TC) == RESET);
  112. }
  113. }
  114. //Send End
  115. USART_SendData(USART1, UART_END);
  116. while (USART_GetFlagStatus(USART1, USART_FLAG_TC) == RESET);
  117. }
  118. */
  119. void Uart1Send(char *txbuf,int len)
  120. {
  121. int i;
  122. for(i=0;i<len;i++){
  123. USART_SendData(USART1,txbuf[i]);
  124. while (USART_GetFlagStatus(USART1, USART_FLAG_TC) == RESET);
  125. }
  126. }
  127. /*********************************************************************************************
  128. *UART1RxTxISRHandler
  129. *串口1中断处理函数
  130. 通讯协议
  131. RxBuffer1接收缓冲区
  132. sutMsg.Uart1Recv为1表明接受到一包数据,需要应用层处理数据后将其清零以便接受新的数据包
  133. 每包数据均以UART_HEAD开头,UART_END结尾。如解决数据包内容刚好有与UART_HEAD或UART_END相等的值,则需要转义
  134. 转义字符为UART_TRANSFORM,当收到UART_TRANSFORM开头时将根据紧挨其后的字符决定转义
  135. 如UART_HEAD=0x7a UART_END=0x7b UART_TRANSFORM=0x7c
  136. 如拟发送数据为: 01 02 7a 03 04 7b 05 06 7c 08 09 经过转换后的数据包为:
  137. 01 02 7c 02 03 04 7c 03 05 06 7c 01 08 09
  138. 加上包头包尾后:
  139. 7a 01 02 7c 02 03 04 7c 03 05 06 7c 01 08 09 7b
  140. **********************************************************************************************
  141. void UART1RxTxISRHandler(void)
  142. {
  143. __IO unsigned char uart_data;
  144. __IO static unsigned char last_data;
  145. //----接收中断--------
  146. if(USART_GetITStatus(USART1, USART_IT_RXNE) != RESET)
  147. {
  148. uart_data = USART_ReceiveData(USART1);
  149. if(sutMsg.Uart1Recv){//上一批没处理,不接受新数据
  150. USART_ClearITPendingBit(USART1, USART_IT_RXNE); //清除中断标志
  151. rx1_ct=0;
  152. return;
  153. }
  154. if(uart_data==UART_HEAD){//包头
  155. RxBuffer1[0]=UART_HEAD;
  156. rx1_ct=1;
  157. }else if(rx1_ct>=(UART1_RX_BUFFER_SIZE-1)){//超出接受缓冲区,丢掉此包数据,复位重新接收
  158. rx1_ct=0;
  159. }else if(rx1_ct>0){
  160. if(uart_data==UART_END){//包尾
  161. RxBuffer1[rx1_ct++]=uart_data;
  162. rx1_len=rx1_ct;
  163. rx1_ct=0;
  164. sutMsg.Uart1Recv=1;
  165. }else if(uart_data==UART_TRANSFORM){//转义字符,忽略
  166. }else{
  167. if(last_data==UART_TRANSFORM){
  168. if(0x01==uart_data)RxBuffer1[rx1_ct++]=UART_TRANSFORM;
  169. else if(0x02==uart_data)RxBuffer1[rx1_ct++]=UART_HEAD;
  170. else if(0x03==uart_data)RxBuffer1[rx1_ct++]=UART_END;
  171. }else{
  172. RxBuffer1[rx1_ct++]=uart_data;
  173. }
  174. }
  175. }
  176. last_data=uart_data;
  177. USART_ClearITPendingBit(USART1, USART_IT_RXNE); //清除中断标志
  178. }
  179. }
  180. */
  181. /*********************************************************************************************
  182. *UART1RxTxISRHandler
  183. *串口1中断处理函数
  184. **********************************************************************************************/
  185. void UART1RxTxISRHandler(void)
  186. {
  187. __IO unsigned long temp;
  188. //接收空闲中断
  189. if(USART_GetITStatus(USART1, USART_IT_IDLE) != RESET){
  190. temp = USART1->SR;//先读SR,再读DR
  191. temp = USART1->DR;//方可清USART_IT_IDLE标志
  192. temp = UART1_RX_BUFFER_SIZE - DMA1_Channel5->CNDTR;
  193. DMA_Cmd(DMA1_Channel5,DISABLE);
  194. //处理数据
  195. // if(RxBuffer1[0]=='P' && RxBuffer1[1]=='C' && RxBuffer1[2]=='T' && RxBuffer1[3]=='A'){
  196. if(sutMsg.Uart1Recv){//上一批没处理,不接受新数据
  197. }else{
  198. //数据拷贝
  199. memcpy(IapRxBuf,RxBuffer1,temp);
  200. IapRxLen=temp;
  201. sutMsg.Uart1Recv=1;
  202. }
  203. // }
  204. //重设DMA传输数据长度
  205. DMA1_Channel5->CNDTR = UART1_RX_BUFFER_SIZE;
  206. //打开DMA
  207. DMA_Cmd(DMA1_Channel5,ENABLE);
  208. }
  209. }
  210. /*******************************************************************************
  211. Uart1Init
  212. Use for Trace
  213. *******************************************************************************/
  214. void Uart1Init(void)
  215. {
  216. NVIC_InitTypeDef NVIC_InitStructure;
  217. GPIO_InitTypeDef GPIO_InitStructure;
  218. USART_InitTypeDef USART_InitStructure;
  219. RCC_APB2PeriphClockCmd( RCC_APB2Periph_GPIOA | RCC_APB2Periph_USART1,ENABLE);
  220. /*
  221. * USART1_TX -> PA9 , USART1_RX -> PA10
  222. */
  223. GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9;
  224. GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;
  225. GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
  226. GPIO_Init(GPIOA, &GPIO_InitStructure);
  227. GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10;
  228. GPIO_InitStructure.GPIO_Mode =GPIO_Mode_IN_FLOATING;// GPIO_Mode_IPU;//GPIO_Mode_IN_FLOATING;
  229. GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
  230. GPIO_Init(GPIOA, &GPIO_InitStructure);
  231. USART_InitStructure.USART_BaudRate =115200;// 115200;
  232. USART_InitStructure.USART_WordLength = USART_WordLength_8b;
  233. USART_InitStructure.USART_StopBits = USART_StopBits_1;
  234. USART_InitStructure.USART_Parity = USART_Parity_No;
  235. USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None;
  236. USART_InitStructure.USART_Mode = USART_Mode_Rx | USART_Mode_Tx;
  237. USART_Init(USART1, &USART_InitStructure);
  238. USART_Cmd(USART1, ENABLE);
  239. /* Configure the NVIC Preemption Priority Bits */
  240. NVIC_PriorityGroupConfig(NVIC_PriorityGroup_0);
  241. /* Enable the USART1 Interrupt */
  242. NVIC_InitStructure.NVIC_IRQChannel = USART1_IRQn;
  243. NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0;
  244. NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0;
  245. NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
  246. NVIC_Init(&NVIC_InitStructure);
  247. USART_ClearFlag(USART1,USART_FLAG_TC);
  248. USART_ITConfig(USART1, USART_IT_TXE, DISABLE); //ENABLE DISABLE
  249. USART_ITConfig(USART1, USART_IT_RXNE, DISABLE); //ENABLE DISABLE
  250. USART_ITConfig(USART1, USART_IT_IDLE, ENABLE); //USART_IT_IDLE USART_IT_RXNE
  251. USART_DMACmd(USART1,USART_DMAReq_Rx,ENABLE); //使能USART1的接收DMA请求
  252. USART_DMACmd(USART1,USART_DMAReq_Tx,DISABLE); //使能USART1的发送DMA请求
  253. Uart1DMAInit();
  254. }
  255. //调用函数:
  256. //my_func(data, "%s:%d\n", str, n);
  257. /*
  258. void IapTrace(char* format, ...)
  259. {
  260. static char TraceBuffer[512];
  261. va_list argptr;
  262. int cnt;
  263. va_start(argptr, format);
  264. cnt = vsprintf(TraceBuffer, format, argptr);
  265. va_end(argptr);
  266. if(cnt>152){
  267. printf("TraceBuffer Error!\r\n");
  268. return;
  269. }
  270. DelayTick(10);
  271. printf("[IAP]");
  272. printf(TraceBuffer);
  273. DelayTick(10);
  274. }
  275. */
  276. /***********************************************************************************
  277. *
  278. ************************************************************************************/
  279. void IapTrace(char *buf)
  280. {
  281. char *p=buf;
  282. DelayTick(10);
  283. printf("[IAP]");
  284. while(0!=*p){
  285. USART_SendData(USART1, *p);
  286. while (USART_GetFlagStatus(USART1, USART_FLAG_TC) == RESET);
  287. p++;
  288. }
  289. DelayTick(10);
  290. }
  291. void SlwTrace(char *buf)
  292. {
  293. char *p=buf;
  294. while(0!=*p){
  295. USART_SendData(USART1, *p);
  296. while (USART_GetFlagStatus(USART1, USART_FLAG_TC) == RESET);
  297. p++;
  298. }
  299. }
  300. #ifdef __GNUC__
  301. /* With GCC/RAISONANCE, small printf (option LD Linker->Libraries->Small printf
  302. set to 'Yes') calls __io_putchar() */
  303. #define PUTCHAR_PROTOTYPE int __io_putchar(int ch)
  304. #else
  305. #define PUTCHAR_PROTOTYPE int fputc(int ch, FILE *f)
  306. #endif /* __GNUC__ */
  307. /**
  308. * @brief Retargets the C library printf function to the USART.
  309. * @param None
  310. * @retval None
  311. */
  312. PUTCHAR_PROTOTYPE
  313. {
  314. /* Place your implementation of fputc here */
  315. /* e.g. write a character to the USART */
  316. USART_SendData(USART1, (uint8_t) ch);
  317. /* Loop until the end of transmission */
  318. while (USART_GetFlagStatus(USART1, USART_FLAG_TC) == RESET)
  319. {}
  320. return ch;
  321. }