正点原子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);
}

编译后报错…\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);   
		}
	}
}

中断服务函数代码如下:

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
} 

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

  • 6
    点赞
  • 51
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
正点原子stm32f407探索者示波器程序是一种基于正点原子stm32f407开发板的示波器程序。stm32f407开发板是一种高性能的单片机开发板,搭载了stm32f407芯片,具有丰富的外设资源和强大的计算能力。 示波器是一种用于观测电信号波形的仪器,可以帮助工程师进行电路调试和故障排查。正点原子stm32f407探索者示波器程序利用stm32f407开发板的强大功能,实现了示波器的基本功能,包括信号采集、显示和控制等。 在正点原子stm32f407探索者示波器程序中,首先需要进行信号采集。stm32f407开发板通过内置的ADC模块,可以将外部信号转化为数字信号进行处理。开发者可以根据具体需求设置采样率和采样精度,以获取更准确的波形数据。 采集到的信号数据会经过处理和存储,然后通过显示模块进行展示。正点原子stm32f407探索者示波器程序可以通过UART、LCD屏幕等方式将波形数据进行实时显示,以便工程师进行观察和分析。 除了基本的信号采集和显示功能,正点原子stm32f407探索者示波器程序还可以添加更多的功能模块,如触发器、频谱分析等,以满足不同的应用需求。 总之,正点原子stm32f407探索者示波器程序是一款功能强大的示波器软件,可以在stm32f407开发板上运行,帮助工程师进行电路调试和波形分析。通过这个程序,开发者可以利用stm32f407开发板的高性能,快速定位和解决电路问题,提高开发效率。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值