stm32串口代码详解

版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
刚开始写 很乱 将持续更新,欢迎在评论区发言我好改进!!!
参考博客:https://www.cnblogs.com/brianblog/p/7101960.html
https://www.cnblogs.com/121792730applllo/p/5155368.html
https://wenku.baidu.com/view/e9691420b7360b4c2e3f64d9.html

串口

***Mian.c***
#include "led.h"
#include "delay.h"
#include "key.h"
#include "sys.h"
#include "usart.h"

 int main(void)
 {		
 	u16 t;  				
 	//u8是unsigned char,u16是unsigned short,u32是unsigned long。
 	//u8,u16,u32都是C语言数据类型,分别代表8位,16位,32位长度的数据类型,一个字节是8位,所以u8是1个字节,u16是2个字节,u32是4个字节。
	u16 len;	
	u16 times=0; 			//定义并初始化times
	delay_init();	    	 //延时函数初始化	  
	NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);
	 //设置NVIC中断分组2:2位抢占优先级,2位响应优先级

/****
## 
 - 这里不详细讲解,在下一节外部中断中会解释外部中断优先级

***


//NVIC_PriorityGroup_0 先占优先级(抢占优先级) 0 位
//从优先级(响应优先级) 4 位
![在这里插入图片描述](https://img-blog.csdnimg.cn/20191107132326235.bmp?
x-oss-
process=image/watermark,type_ZmFuZ3poZW5naGVpdGk,shadow_10,text_aHR0cHM6
Ly9ibG9nLmNzZG4ubmV0L3dlaXhpbl80NTgwMDU1MA==,size_16,color_FFFFFF,t_70)

//NVIC_PriorityGroup_1 先占优先级 1 位
//从优先级 3 位
//NVIC_PriorityGroup_2 先占优先级 2 位
//从优先级 2 位
//NVIC_PriorityGroup_3 先占优先级 3 位
//从优先级 1 位
//NVIC_PriorityGroup_4 先占优先级 4 位
//从优先级 0 位
*/
	uart_init(115200);	 
	/*//串口初始化为115200
	串口初始化函数
	使用需要添加#include "usart.h"
	*/
 	LED_Init();			     //LED端口初始化  //LED IO初始化
	KEY_Init();          //初始化与按键连接的硬件接口 //按键初始化函数
 	while(1)
	{
		if(USART_RX_STA&0x8000)
		/*
		原子定义了一个16位数:USART_RX_STA为0000 0000 0000 0000,第十六位为0则串口数据没有接收完,为1则接收完了(中断里有判断),而0x8000=1000 0000 0000 0000,所以
USART_RX_STA&0x8000有两种可能:
第一种1××× ×××× ×××× ××××&1000 0000 0000 0000=1000 0000 0000 0000
第二种0××× ×××× ×××× ××××&1000 0000 0000 0000=0000 0000 0000 0000
由此可以判断USART_RX_STA第16位是否为0
*/
			{					   
			len=USART_RX_STA&0x3fff;//得到此次接收到的数据长度
			/*
			USART_RX_STA是一个变量可以是8位的,也可以是16位的。以16位为例它的含义:// bit15,接收完成标志/ / bit14,接收到0x0d
// bit13〜0,接收到的有效字节数
和0x3fff与运算就是低14位的运算,因为每接收到一个有效字节会自动加1
			*/
			printf("\r\n您发送的消息为:\r\n\r\n");
			for(t=0;t<len;t++)
			{
				USART_SendData(USART1, USART_RX_BUF[t]);
				//向串口1发送数据
				/*
				USART_SendData()
				功能描述 通过外设 USARTx 发送单个数据
                输入参数 1 USARTx:x 可以是 1,2 或者 3,来选择 USART 外设
				*/
				while(USART_GetFlagStatus(USART1,USART_FLAG_TC)!=SET);//等待发送结束
				/***USART_GetFlagStatus:检查指定的 USART 标志位设置与否
				输入参数 1 USARTx:x 可以是 1,2 或者 3,来选择 USART 外设
				输入参数 2 USART_FLAG:待检查的 USART 标志位
				参阅 Section:USART_FLAG 查阅更多该参数允许取值范围
				*/
			}
			printf("\r\n\r\n");//插入换行
			USART_RX_STA=0;//设置标志位
		}else
		{
			times++;//时间计数++
			if(times%5000==0)
			{
				printf("\r\n精英STM32开发板 串口实验\r\n");
				printf("正点原子@ALIENTEK\r\n\r\n");
			}
			if(times%200==0)printf("请输入数据,以回车键结束\n");  
			if(times%30==0)LED0=!LED0;//闪烁LED,提示系统正在运行.
			delay_ms(10);   //防抖
		}
	}	 
 }
 
***Led.h***
#ifndef __LED_H      //先测试LED是否被宏定义过
#define __LED_H	     //如果LED H没有被宏定义过,定义LED,LED别名为H
#include "sys.h"
#define LED0 PBout(5)// PB5
#define LED1 PEout(5)// PE5	
void LED_Init(void); //初始化			    
#endif

***Led.c***
#include "led.h"   
//初始化PB5和PE5为输出口.并使能这两个口的时钟		    
//LED IO初始化
void LED_Init(void)
{
 GPIO_InitTypeDef  GPIO_InitStructure;
 RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB|RCC_APB2Periph_GPIOE, ENABLE);	 
 //使能PB,PE端口时钟	
 GPIO_InitStructure.GPIO_Pin = GPIO_Pin_5;				 //LED0-->PB.5 端口配置
 GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP; 		 //推挽输出
 GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;		 //IO口速度为50MHz
 
 GPIO_Init(GPIOB, &GPIO_InitStructure);					 //根据设定参数初始化GPIOB.5
 GPIO_SetBits(GPIOB,GPIO_Pin_5);						 //PB.5 输出高
 GPIO_InitStructure.GPIO_Pin = GPIO_Pin_5;	    		 //LED1-->PE.5 端口配置, 推挽输出
 GPIO_Init(GPIOE, &GPIO_InitStructure);	  				 //推挽输出 ,IO口速度为50MHz
 GPIO_SetBits(GPIOE,GPIO_Pin_5); 						 //PE.5 输出高 
}

***Key.h***
#ifndef __KEY_H
#define __KEY_H	 
#include "sys.h"
//#define KEY0 PEin(4)   	//PE4
//#define KEY1 PEin(3)	//PE3 
//#define WK_UP PAin(0)	//PA0  WK_UP

#define KEY0  GPIO_ReadInputDataBit(GPIOE,GPIO_Pin_4)//读取按键0
#define KEY1  GPIO_ReadInputDataBit(GPIOE,GPIO_Pin_3)//读取按键1
#define WK_UP   GPIO_ReadInputDataBit(GPIOA,GPIO_Pin_0)//读取按键3(WK_UP) 

#define KEY0_PRES 	1	//KEY0按下
#define KEY1_PRES	  2	//KEY1按下
#define WKUP_PRES   3	//KEY_UP按下(即WK_UP/KEY_UP)

void KEY_Init(void);//IO初始化
u8 KEY_Scan(u8);  	//按键扫描函数					    
#endif

***Key.c***
#include "stm32f10x.h"
#include "key.h"
#include "sys.h" 
#include "delay.h"
								    
//按键初始化函数
void KEY_Init(void) //IO初始化
{ 
 	GPIO_InitTypeDef GPIO_InitStructure;
 
 	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA|RCC_APB2Periph_GPIOE,ENABLE);
 	//使能PORTA,PORTE时钟
 	/*函数原形 
 	void RCC_APB2PeriphClockCmd(u32 RCC_APB2Periph,FunctionalState NewState)
 	输入参数 1 RCC_APB2Periph: 门控 APB2 外设时钟
    参阅 Section:RCC_APB2Periph 查阅更多该参数允许取值范围
    输入参数 2 NewState:指定外设时钟的新状态
    这个参数可以取:ENABLE 或者 DISABLE 
 	*/

	GPIO_InitStructure.GPIO_Pin  = GPIO_Pin_4|GPIO_Pin_3;//KEY0-KEY1
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPU; //设置成上拉输入
 	GPIO_Init(GPIOE, &GPIO_InitStructure);//初始化GPIOE4,3

	//初始化 WK_UP-->GPIOA.0	  下拉输入
	GPIO_InitStructure.GPIO_Pin  = GPIO_Pin_0;
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPD; //PA0设置成输入,默认下拉	  
	GPIO_Init(GPIOA, &GPIO_InitStructure);//初始化GPIOA.0

}
//按键处理函数
//返回按键值
//mode:0,不支持连续按;1,支持连续按;
//0,没有任何按键按下
//1,KEY0按下
//2,KEY1按下
//3,KEY3按下 WK_UP
//注意此函数有响应优先级,KEY0>KEY1>KEY_UP!!
u8 KEY_Scan(u8 mode)
{	 
	static u8 key_up=1;//按键按松开标志
	if(mode)key_up=1;  //支持连按		  
	if(key_up&&(KEY0==0||KEY1==0||WK_UP==1))
	{
		delay_ms(10);//去抖动 
		key_up=0;
		if(KEY0==0)return KEY0_PRES;
		else if(KEY1==0)return KEY1_PRES;
		else if(WK_UP==1)return WKUP_PRES;
	}else if(KEY0==1&&KEY1==1&&WK_UP==0)key_up=1; 	    
 	return 0;// 无按键按下
}

这里是引用

你可能感兴趣的:(stm32)