STM32串口中断接收实验

STM32串口中断接收实验的详细说明

准备

材料:STM32F407ZGT6最小系统板,串口1通过跳线帽连接到了CH340上。
在这里插入图片描述

需求:从电脑向板子的串口1发送一个字符串(以回车和换行结尾,字符串末尾两个字符为0x0d和0x0a),板子接收到之后原样返回给电脑。

思路:用串口的接收中断实现。

代码实现

后台执行内容:
当判断接收完毕后,将存放在字符数组里的字符用循环的方式发送出去。

后台时刻检查变量STA的bit15的状态决定要不要发送数据。

大部分时候后台仅仅运行闪灯的那一部分。

int main()
{
	u8 i=0; 
	u16 t=0;
	u16 len=0;
	uint8_t s[10];
	
	SysTick_Init(168);
	NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);  //中断优先级分组 分2组
	USART1_Init(115200);
	LED_Init();
	LED2=0;
	printf("初始化配置完成\r\n");
	
	while(1)
	{
		/*外层先检查15位的状态是0还是1,*/
		if(USART1_RX_STA&0x8000)//如果接收到了0x0a,表明字符串接收结束
		{					   
			len=USART1_RX_STA&0x3fff;//得到此次接收到的数据长度
			for(t=0;t<len;t++)
			{
				USART_SendData(USART1, USART1_RX_BUF[t]);         //向串口1发送数据
				while(USART_GetFlagStatus(USART1,USART_FLAG_TC)!=SET);//等待发送结束
			}
			USART1_RX_STA=0;//将STA清0,以便接收下一个字符串
		}
		/*红灯闪烁指示后台程序的运行情况*/
		LED1=!LED1;
		delay_ms(500);
	}
}

前台执行内容:
用一个16位的变量来计数,同时由于字符串长度不可能达到16383个字符,故变量STA的bit15、bit14永远都是用不到的。

STA计数的作用一是发送数据是一次发送一个字符,要知道发送多少次;二是储存时要按下标存到数组的对应位置。

那么对STA的前两位进行操作时必然会影响到STA整体的值,是否存在问题呢?
不会!发送数据与储存数据均是只提取STA的低14位,在16383之内STA的自增只影响这14位数据。

/*一个16位整数从0自增到16383都不会用到15位与16位,可以按接收的字符情况对前两位进行标记。*/
//bit15,	接收完成标志
//bit14,	接收到0x0d
//bit13~0,	接收到的有效字节数目
//回车换行为0x0d,0x0a
u16 USART1_RX_STA=0;       //接收状态标记,初始16位全部为0

串口每接收一个字符就进入中断一次,对于每一字符来讲都可能有3种情况:有效数据、0x0d、0x0a。先检查15位的状态是0还是1,再检查14位的状态是0还是1,最后对这一次接收的数据进行比对并对相应位进行置1标记或者写入数组操作。

每次接收到字符前台也会检查STA的bit15的状态。

前台的程序执行由硬件层面引发,当串口接收到数据就触发中断,执行中断服务程序里面的内容。这里面会对外界的输入进行检查判断,在合适的时机对一个全局变量USART1_RX_STA进行操作。而后台会时刻检测这个变量。这样就建立起前后台的联系。

/*串口每接收一个字符就进入中断一次*/
void USART1_IRQHandler(void)                	//串口1中断服务程序
{
	u8 r;
	u16 i=0;
	
	if(USART_GetITStatus(USART1, USART_IT_RXNE) != RESET)  //判断中断类型是不是接收中断
	{
		r =USART_ReceiveData(USART1);  //读取接收到的数据
		/*外层先检查bit15的状态是0还是1*/
		if((USART1_RX_STA&0x8000)==0)//如果SAT的bit15是1
		{
			/*中间层检查bit14的状态是0还是1*/
			if(USART1_RX_STA&0x4000)//如果STA的bit14是1
			{
				/*里层判断这一次接收到的字符并执行相应的操作*/
				if(r!=0x0a)//如果这一次接收到的不是0x0a
					USART1_RX_STA=0;//接收错误,将STA重置0
				else USART1_RX_STA|=0x8000;	//否则接收完成,将15位置1(后台检测到这个变量的bit15的变化会立马执行if里的内容,这一句建立了前台与后台的联系)
			}
			else //如果STA14位是0
			{	
				/*里层判断这一次接收到的字符并执行相应的操作*/
				if(r==0x0d)//如果这一次是0x0d
					USART1_RX_STA|=0x4000;//将14位置1
				else//如果这一次也不是0x0d,表明接收到的是有效数据
				{
					USART1_RX_BUF[USART1_RX_STA&0X3FFF]=r;//将本次接收的字符存到数组里,位置由STA的前14位决定
					USART1_RX_STA++;//接收到有效数据的个数
					if(USART1_RX_STA>(USART1_REC_LEN-1))USART1_RX_STA=0;//接收数据错误,重新开始接收	  
				}		 
			}
		}
	} 
} 	

总结

从下午看到晚上才搞懂,花了我这么多时间,算是正式开始单片机之旅了。还是太菜了,万里长征第一步,迈出去就好了。写个博客纪念一下。

  • 8
    点赞
  • 24
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值