经验首页 前端设计 程序设计 Java相关 移动开发 数据库/运维 软件/图像 大数据/云计算 其他经验
当前位置:技术经验 » 程序设计 » C 语言 » 查看文章
UART学习之路(三)基于STM32F103的USART实验
来源:cnblogs  作者:Jaso0n  时间:2018/10/20 15:32:14  对本文有异议

  关于STM32串口的资料可以在RM0008 Reference Manual中找到,有中文版的资料。STM32F103支持5个串口,选取USART1用来实验,其对应的IO口为PA9和PA10。这次的实验基于ALIENTEK的开发板,开发版通过CH340G实现将串口转成USB。因此需要做好一些准备工作。

1.PC端安装Keil v5 MDK开发工具;

2.PC端安装CH340G的驱动;

3.PC端安装ATK XCOM串口收发程序

 

  STM32的串口编程思路:

1.串口时钟设置和复位;

2.选取发射口和接收口的引脚,并设置GPIO端口参数;

3.串口参数的初始化(完成波特率、字长、奇偶校验、收发模式等参数的设置);

4.初始化NVIC(Nested Vectored Interrupt Controller,内嵌向量中断控制器);

5.开启中断和使能串口

 

代码如下:

  1. 1 //main.c:
  2. 2 #include "uart.h"
  3. 3
  4. 4
  5. 5 int main()
  6. 6 {
  7. 7 uart1_init();
  8. 8 while(1)
  9. 9 {
  10. 10 }
  11. 11 }
  1. 1 //USART.c
  2. 2 #include "uart.h"
  3. 3
  4. 4
  5. 5 #define USART1_REC_LEN 256
  6. 6
  7. 7 u8 Uart1_RevBuf_Tail = 0;//接收缓冲区尾部
  8. 8 u8 Uart1_RevBuf[USART1_REC_LEN];//接收缓冲区数组
  9. 9
  10. 10 void uart1_init()
  11. 11 {
  12. 12 //GPIO端口设置
  13. 13 GPIO_InitTypeDef GPIO_InitStructure;
  14. 14 USART_InitTypeDef USART_InitStructure;
  15. 15 NVIC_InitTypeDef NVIC_InitStructure;
  16. 16
  17. 17 RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1|RCC_APB2Periph_GPIOA, ENABLE);
  18. 18 USART_DeInit(USART1);
  19. 19
  20. 20
  21. 21 //USART1端口配置
  22. 22 //UASART_TX PA9
  23. 23 GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9; //PA.9
  24. 24 GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
  25. 25 GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP; //复用推挽输出
  26. 26 GPIO_Init(GPIOA, &GPIO_InitStructure); //初始化PA9
  27. 27 //USART1_RX PA10
  28. 28 GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10;
  29. 29 GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;//浮空输入
  30. 30 GPIO_Init(GPIOA, &GPIO_InitStructure); //初始化PA10
  31. 31
  32. 32 //USART1 初始化设置
  33. 33 USART_InitStructure.USART_BaudRate = 9600;//波特率设置
  34. 34 USART_InitStructure.USART_WordLength = USART_WordLength_8b;//字长为8位数据格式
  35. 35 USART_InitStructure.USART_StopBits = USART_StopBits_1;//一个停止位
  36. 36 USART_InitStructure.USART_Parity = USART_Parity_No;//无奇偶校验位
  37. 37 USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None;//无硬件数据流控制
  38. 38 USART_InitStructure.USART_Mode = USART_Mode_Rx | USART_Mode_Tx; //收发模式
  39. 39 USART_Init(USART1, &USART_InitStructure); //初始化串口1
  40. 40
  41. 41 //Usart1 NVIC 配置
  42. 42 NVIC_InitStructure.NVIC_IRQChannel = USART1_IRQn;//串口1中断通道
  43. 43 NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority=3;//抢占优先级3
  44. 44 NVIC_InitStructure.NVIC_IRQChannelSubPriority =3; //子优先级3
  45. 45 NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE; //IRQ通道使能
  46. 46 NVIC_Init(&NVIC_InitStructure); //根据指定的参数初始化VIC寄存器
  47. 47
  48. 48 USART_Init(USART1, &USART_InitStructure);
  49. 49 USART_ITConfig(USART1, USART_IT_RXNE, ENABLE);//开启相关中断
  50. 50 USART_Cmd(USART1, ENABLE); //使能串口1
  51. 51
  52. 52 }
  53. 53
  54. 54 //串口1中断服务程序
  55. 55 void USART1_IRQHandler(void)
  56. 56 {
  57. 57 if(USART_GetITStatus(USART1, USART_IT_RXNE) != RESET) //接收中断
  58. 58 {
  59. 59
  60. 60 Uart1_RevBuf[Uart1_RevBuf_Tail] = USART_ReceiveData(USART1);//读取接收到的数据,将尾标后移
  61. 61 USART_SendData(USART1,Uart1_RevBuf[Uart1_RevBuf_Tail]);//发送接收到的数据
  62. 62 while (USART_GetFlagStatus(USART1, USART_FLAG_TC) == RESET)
  63. 63 {}
  64. 64 Uart1_RevBuf_Tail++;
  65. 65 if(Uart1_RevBuf_Tail>USART1_REC_LEN-1)
  66. 66 {
  67. 67 Uart1_RevBuf_Tail = 0;
  68. 68 }
  69. 69 }
  70. 70 }

  主函数非常简单,就是调用uart_init()然后等待串口1的接收中断触发。串口1的中断服务函数功能是:当PC端发送据后,将接收到的数据重新发回给PC机。uart_init()的功能是完成串口的配置。在接收数据的时候设置了一个容量位256的数据缓冲区Uart1_RevBuf,用来存放接收到的数据。

程序的运行结果如下:

  分别发送AA,BB,CC后PC端接收到了AA 0D 0A BB 0D 0A CC 0D 0A,0D和0A分别表示回车和换行。说明结果正确。

 

在实际应用中,上位机可以通过多个串口和多个从设备进行通信,因此在串口通信的时候要自行规定一个通信协议。比如由1.头,2.设备号,3.数据长度,4.数据,5.结束位,6.间隔位组成一个数据包。根据协议编写解包函数。解包函数的大致思路就是将接收到的数据一步一步的进行判断,最终完成解出数据的功能。

1.数据包定义:

头:0xAB,设备号:0x01(一号设备),数据长度:0x08(8位数据),数据位:DATA,结束位:0xFF,间隔位:0xFF 0xFF

2.解包函数:

PC机发送一个数据包:AB 01 08 00 01 02 03 04 05 06 07 FF FF FF,解包函数能够将数据00 01 02 03 04 05 06 07取出来并再次发送给PC机。PC机将数据发送给STM32F103,触发接收中断,将数据存入接收缓冲区中,解包函数从缓冲区的头部开始检索,完成数据分析,取出数据。代码如下:

  1. #include "stm32f10x.h"
  2. #include <stdio.h>
  3.  
  4. #define Usart1RecLength 256
  5. u8 Uart1_RevBuf_Tail = 0;
  6. u8 Uart1_RevBuf_Head = 0;
  7. u8 Uart1_RevBuf[Usart1RecLength];
  8. u8 RecState = 0;
  9. u8 TemplateData;
  10. u8 DataLength = 8;
  11. u8 Data[8]={0};
  12. typedef struct
  13. {
  14. u8 StartDataError;
  15. u8 DeviceDataError;
  16. u8 LengthDataError;
  17. u8 StopDataError;
  18. u8 DataReady;
  19. }DataFrameFlag;
  20. DataFrameFlag USART1_FrameFlags;
  21. void USART1_IRQHandler(void)
  22. {
  23. if(USART_GetITStatus(USART1, USART_IT_RXNE) != RESET) //接收中断
  24. {
  25. Uart1_RevBuf[Uart1_RevBuf_Tail] = USART_ReceiveData(USART1); //读取接收到的数据
  26. Uart1_RevBuf_Tail++;
  27. if(Uart1_RevBuf_Tail > Usart1RecLength-1)
  28. {
  29. Uart1_RevBuf_Tail = 0;
  30. }
  31. }
  32. }
  33. void RecDataAnalysis()
  34. {
  35. u8 i = 0;
  36. if(Uart1_RevBuf_Head != Uart1_RevBuf_Tail)//判断是否有数据
  37. {
  38. TemplateData = Uart1_RevBuf[Uart1_RevBuf_Head];//从数据缓冲区取数据
  39. Uart1_RevBuf_Head ++;
  40. if(Uart1_RevBuf_Head > Usart1RecLength-1)
  41. {
  42. Uart1_RevBuf_Head = 0;
  43. }
  44. USART1_FrameFlags.DeviceDataError = 0;
  45. USART1_FrameFlags.StopDataError = 0;
  46. USART1_FrameFlags.LengthDataError = 0;
  47. USART1_FrameFlags.StartDataError = 0;
  48. switch(RecState)
  49. {
  50. case 0:
  51. if(TemplateData == 0xAB)//
  52. {
  53. RecState = 1;
  54. }
  55. else
  56. {
  57. RecState = 0;
  58. USART1_FrameFlags.StartDataError = 1;
  59. }
  60. break;
  61. case 1:
  62. if(TemplateData == 0x01)//设备号
  63. {
  64. RecState = 2;
  65. }
  66. else
  67. {
  68. RecState = 0;
  69. USART1_FrameFlags.DeviceDataError = 1;
  70. }
  71. break;
  72. case 2:
  73. if(TemplateData == 0x08)//数据位
  74. {
  75. RecState = 3;
  76. }
  77. else
  78. {
  79. RecState = 0;
  80. USART1_FrameFlags.LengthDataError = 1;
  81. }
  82. break;
  83. case 3://转存数据
  84. if(DataLength == 0)
  85. {
  86. RecState = 4;
  87. USART1_FrameFlags.DataReady = 1;
  88. }
  89. else if(DataLength != 0)
  90. {
  91. Data[8-DataLength] = TemplateData;
  92. DataLength = DataLength -1;
  93. }
  94. break;
  95. case 4:
  96. if(TemplateData == 0xFF)//尾部
  97. {
  98. RecState = 0;
  99. DataLength = 8;
  100. }
  101. else
  102. {
  103. for(i=0;i < 8;i++)
  104. {
  105. Data[i] = 0;
  106. }
  107. RecState = 0;
  108. DataLength = 8;
  109. USART1_FrameFlags.StopDataError = 1;
  110. USART1_FrameFlags.DataReady = 0;
  111. }
  112. break;
  113. default:
  114. for(i=0;i < 8;i++)
  115. {
  116. Data[i] = 0;
  117. }
  118. RecState = 0;
  119. DataLength = 8;
  120. break;
  121. }
  122. }
  123. }
  124. void Resend()//测试用重发数据函数
  125. {
  126. u8 i = 0;
  127. if(USART1_FrameFlags.DataReady == 1)
  128. {
  129. for(i=0;i<8;i++)
  130. {
  131. USART_SendData(USART1,Data[i]);
  132. while (USART_GetFlagStatus(USART1, USART_FLAG_TC) == RESET);
  133. USART1_FrameFlags.DataReady = 0;
  134. }
  135. }
  136. }

 

函数RecDataAnalysis()完成数据解包,函数Resend()在解包函数准备好数据将数据回发给PC机。结构体DataFrameFlag的作用是当数据出现错误时完成报错,是可选功能,程序中给了一种思路,未做调试。结果如下:

 友情链接:直通硅谷  点职佳  北美留学生论坛

本站QQ群:前端 618073944 | Java 606181507 | Python 626812652 | C/C++ 612253063 | 微信 634508462 | 苹果 692586424 | C#/.net 182808419 | PHP 305140648 | 运维 608723728

W3xue 的所有内容仅供测试,对任何法律问题及风险不承担任何责任。通过使用本站内容随之而来的风险与本站无关。
关于我们  |  意见建议  |  捐助我们  |  报错有奖  |  广告合作、友情链接(目前9元/月)请联系QQ:27243702 沸活量
皖ICP备17017327号-2 皖公网安备34020702000426号