• 正点原子stm32F407学习笔记5——串口通信实验


    一、串口通信实验1

    上位机给开发板发送数据,开发板将收到的数据发回给上位机
    串口设置的一般步骤可以总结为如下几个步骤:

    1. 串口时钟使能,GPIO 时钟使能。
    2. 设置引脚复用器映射:调用 GPIO_PinAFConfig 函数。
    3. GPIO 初始化设置:要设置模式为复用功能。
    4. 串口参数初始化:设置波特率,字长,奇偶校验等参数。
    5. 开启中断并且初始化 NVIC,使能中断(如果需要开启中断才需要这个步骤)。
    6. 使能串口。
    7. 编写中断处理函数:函数名格式为 USARTxIRQHandler(x 对应串口号)。
    8. 获取相应中断状态
    9. 串口数据发送与接收
      直接在主函数下编写代码如下:
    
    #include "delay.h"
    
    void uart_test()
    {
    	GPIO_InitTypeDef GPIO_InitStructure;
    	USART_InitTypeDef USART_InitStructure;
    	NVIC_InitTypeDef NVIC_InitStructure;
    	
    	//1) 串口时钟和 GPIO 时钟使能
    	RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1,ENABLE);//串口时钟UART1使能,串口是挂载在 APB2 下面的外设
    	RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOA,ENABLE);//GPIO 时钟使能, 串口1 GPIO对应芯片引脚PA9,PA10
    	
    	//2) 设置引脚复用器映射
    	GPIO_PinAFConfig(GPIOA,GPIO_PinSource9,GPIO_AF_USART1);
    	GPIO_PinAFConfig(GPIOA,GPIO_PinSource10,GPIO_AF_USART1);
    	
    	//3) GPIO 端口模式设置:PA9 和 PA10 要设置为复用功能
    	GPIO_InitStructure.GPIO_Pin = (GPIO_Pin_9 | GPIO_Pin_10);//GPIOA9 与 GPIOA10
    	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF;//复用功能
    	GPIO_InitStructure.GPIO_Speed = GPIO_Fast_Speed;//速度 50MHz
    	GPIO_InitStructure.GPIO_OType = GPIO_OType_PP;//推挽复用输出
    	GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_UP;//上拉
    	GPIO_Init(GPIOA,&GPIO_InitStructure); //初始化 PA9,PA10
    	
    	//4) 串口参数初始化:设置波特率,字长,奇偶校验等参数
    	USART_InitStructure.USART_BaudRate = 115200;//波特率设置为115200
    	USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None;//硬件流控制选择无
    	USART_InitStructure.USART_Mode = USART_Mode_Rx | USART_Mode_Tx;//收发模式
    	USART_InitStructure.USART_Parity = USART_Parity_No;//无奇偶校验位
    	USART_InitStructure.USART_StopBits = USART_StopBits_1;//一个停止位
    	USART_InitStructure.USART_WordLength = USART_WordLength_8b;//字长为 8 位数据格式
    	USART_Init(USART1, &USART_InitStructure); //初始化串口1
    	
    	//6) 使能串口
    	USART_Cmd(USART1, ENABLE);  //使能串口1
    	
    	//5) 开启中断并且初始化 NVIC,使能相应中断
    	USART_ITConfig(USART1,USART_IT_RXNE,ENABLE);//开启中断,接收到数据中断
    	
    	NVIC_InitStructure.NVIC_IRQChannel = USART1_IRQn;//串口1中断通道,在顶层头文件stm32f4xx.h中第223行定义
    	NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;//IRQ 通道使能
    	NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 3;//抢占优先级 3
    	NVIC_InitStructure.NVIC_IRQChannelSubPriority = 3;//响应优先级 3
    	NVIC_Init(&NVIC_InitStructure);	//根据指定的参数初始化NVIC寄存器
    }
    
    //7) 中断服务函数(在启动文件startup_stm32f40_41xxx.s中124行已定义好,不可自定义名字)
    void USART1_IRQHandler()
    {
    	//8) 获取相应中断状态
    	if( USART_GetITStatus(USART1, USART_IT_RXNE))//使能了一个中断,判断中断标志位,中断是否发生
    	{
    		u8 res;
    		//9) 串口数据发送与接收
    		res = USART_ReceiveData(USART1);//收到数据后读出来
    		USART_SendData(USART1,res);//读到数据后立刻发送出去
    	}
    }
    
    int main()
    {
    	delay_init(168);  //初始化延时函数
    	NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);//设置系统中断优先级分组2, 实验开启了串口中断,所以我们在系统初始化的时候需要先设置系统的中断优先级分组
    	uart_test();//初始化
    	while(1);
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30
    • 31
    • 32
    • 33
    • 34
    • 35
    • 36
    • 37
    • 38
    • 39
    • 40
    • 41
    • 42
    • 43
    • 44
    • 45
    • 46
    • 47
    • 48
    • 49
    • 50
    • 51
    • 52
    • 53
    • 54
    • 55
    • 56
    • 57
    • 58
    • 59
    • 60
    • 61
    • 62
    • 63
    • 64
    • 65
    • 66
    • 67

    编译后报错…\OBJ\Template.axf: Error: L6200E: Symbol USART1_IRQHandler multiply defined (by usart.o and main.o).
    因为在uart.c中同样定义了中断服务函数,报错重定义,在uart.c中注释掉中断服务函数即可
    编译完成后烧录进开发板,打开XCOM上位机软件,上位机串口的设置与步骤四代码中串口参数初始化设置相同,勾选上发送新行,可以看到上位机发送的数据都被发送回来了
    在这里插入图片描述

    二、串口通信实验2

    直接编译正点原子串口通信实验例程,主要关注main函数和USART1_IRQHandler中断服务函数的逻辑,不太好理解,主要关注下USART_RX_STA这个16位的变量做到类似于标志位的作用与 0x0a换行符0x0d为回车符的处理过程。USART_RX_STA的前13位作为数据长度,第14位收到0x0d后置位0x4000,第15位收到0x0a后置位0x8000后在主函数中发送接收到的数据,这些可以理解为自己定的协议。uart_init()函数与实验1中基本一致。
    主函数代码如下:

    #include "sys.h"
    #include "delay.h"
    #include "usart.h"
    #include "led.h"
    #include "beep.h"
    #include "key.h"
    
    int main(void)
    { 
     
    	u8 t;
    	u8 len;	
    	u16 times=0;  
    	NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);//设置系统中断优先级分组2
    	delay_init(168);		//延时初始化 
    	uart_init(115200);	//串口初始化波特率为115200
    	LED_Init();		  		//初始化与LED连接的硬件接口  
    	while(1)
    	{
    		if(USART_RX_STA&0x8000)//当USART_RX_STA值为0x8000时表示收完上位机数据
    		{					   
    			len=USART_RX_STA&0x3fff;//得到此次接收到的数据长度
    			printf("\r\n您发送的消息为:\r\n");
    			for(t=0;t<len;t++)
    			{
    				USART_SendData(USART1, USART_RX_BUF[t]);         //向串口1发送数据
    				while(USART_GetFlagStatus(USART1,USART_FLAG_TC)!=SET);//等待发送结束	再正点原子库函数文档中这两行是直接操作寄存器的
    			}
    			printf("\r\n\r\n");//插入换行
    			USART_RX_STA=0;//在此处清零后才能继续接收上位机数据不然一直都是0x8000
    		}else//在没有收到数据时打印信息提示系统在运行
    		{
    			times++;
    			if(times%5000==0)
    			{
    				printf("\r\nALIENTEK 探索者STM32F407开发板 串口实验\r\n");
    				printf("正点原子@ALIENTEK\r\n\r\n\r\n");
    			}
    			if(times%200==0)printf("请输入数据,以回车键结束\r\n");  
    			if(times%30==0)LED0=!LED0;//闪烁LED,提示系统正在运行.
    			delay_ms(10);   
    		}
    	}
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30
    • 31
    • 32
    • 33
    • 34
    • 35
    • 36
    • 37
    • 38
    • 39
    • 40
    • 41
    • 42
    • 43
    • 44

    中断服务函数代码如下:

    void USART1_IRQHandler(void)                	//串口1中断服务程序
    {
    	u8 Res;
    #if SYSTEM_SUPPORT_OS 		//如果SYSTEM_SUPPORT_OS为真,则需要支持OS.
    	OSIntEnter();    
    #endif
    	if(USART_GetITStatus(USART1, USART_IT_RXNE) != RESET)  //接收到数据产生中断(接收到的数据必须是0x0d 0x0a结尾)  0x0a为换行符 0x0d为回车符
    	{
    		Res =USART_ReceiveData(USART1);//(USART1->DR);	//读取接收到的数据
    		
    		if((USART_RX_STA&0x8000)==0)//接收未完成
    		{
    			if(USART_RX_STA&0x4000)//接收到了0x0d
    			{
    				if(Res!=0x0a)USART_RX_STA=0;//接收错误,重新开始
    				else USART_RX_STA|=0x8000;	//下一次接受的的数据为0x0a表示接收完成了,将USART_RX_STA赋值为0x8000在主函数中发送数据
    			}
    			else //还没收到0X0D,数据还在一直发
    			{	
    				if(Res==0x0d)USART_RX_STA|=0x4000;//当收到0X0D时将USART_RX_STA值置为0x4000
    				else
    				{
    					USART_RX_BUF[USART_RX_STA&0X3FFF]=Res ;//将每次收到的数据存在数组中,0X3FFF表示前13位是数据长度
    					USART_RX_STA++;//计算数据长度,收到一个数据累加1
    					if(USART_RX_STA>(USART_REC_LEN-1))USART_RX_STA=0;//当接收到的数据大于规定的长度时视为接收数据错误,重新开始接收	USART_REC_LEN为自定义的长度  
    				}		 
    			}
    		}   		 
    	} 
    #if SYSTEM_SUPPORT_OS 	//如果SYSTEM_SUPPORT_OS为真,则需要支持OS.
    	OSIntExit();  											 
    #endif
    } 
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30
    • 31
    • 32
    • 33

    烧录进开发板后通过上位机发送数据如下
    在这里插入图片描述
    要记得勾选发送新行,如果不勾选上位机收不到发到设备的数据,因为没有收到0x0a和0x0d数据就不会发,如果不勾选一直发就会存在数组中直到超过数组长度重新接收。

  • 相关阅读:
    3D感知技术(1)3D数据及其数据表示
    BHQ-3 amine,1661064-89-6可在430nm至730nm范围内猝灭所有普通荧光团
    我们要做怎样的测试/开发程序员,怎样提高自己的价值......
    基于树莓派的freeRDP桌面云终端的简单实现
    FFmpeg入门详解之103:FFmpeg Nginx VLC打造M3U8直播点播
    Jmix 中 REST API 的两种实现
    伦敦金的走势高低的规律
    canvas 基础 和 动图案例
    python 模块 — logging模块、smtplib和email模块
    前端常见安全问题
  • 原文地址:https://blog.csdn.net/weixin_48213877/article/details/127983592