STM32单片机学习(3) 串口中断通信

发布者:郑大大最新更新时间:2016-06-06 来源: eefocus关键字:STM32  单片机学习  串口中断通信 手机看文章 扫描二维码
随时随地手机看文章
/*
*	
* 软件功能:	 串口实验(软件延时方式)
* 
*/
#include "stm32f10x.h"
#include 
#include "delay.h"

u8 uart1_buf[20];
int count=0;  //字符串长度
int flag=0;	  

void RCC_Configuration(void);
void GPIO_Configuration(void);
void NVIC_Configuration(void);
void USART1_Configuration(void);
void Uart1_PutChar(u8 ch);
void Uart1_PutString(u8* buf , u8 len);
int USART1_ReceivedStrCmp(const char *str);
							
/*
函数: int main(void)
功能: main主函数
参数: 无
返回: 无
/
int main(void)
{
  RCC_Configuration();
  NVIC_Configuration();
  GPIO_Configuration();
  delay_init(72);
  USART1_Configuration();
  GPIO_ResetBits(GPIOA,GPIO_Pin_0);//灭
  while(1)
  {  
	//GPIO_ResetBits(GPIOA,GPIO_Pin_0);//灭
	//delay_ms(500);//延时
	//Uart1_PutString("Hello",5);
	//GPIO_SetBits(GPIOA,GPIO_Pin_0);//亮
   // delay_ms(500);//延时
    /* if(flag==1&&count!=0)
	 {
	 	 Uart1_PutString(uart1_buf,count);
		 count=0;
		 flag=0;
	 } 		 */
  }
}

/*
函数: void RCC_Configuration(void)
功能: 复位和时钟控制 配置
参数: 无
返回: 无
/
void RCC_Configuration(void)
{
  ErrorStatus HSEStartUpStatus;                    //定义外部高速晶体启动状态枚举变量
  RCC_DeInit();                                    //复位RCC外部设备寄存器到默认值
  RCC_HSEConfig(RCC_HSE_ON);                       //打开外部高速晶振
  HSEStartUpStatus = RCC_WaitForHSEStartUp();      //等待外部高速时钟准备好
  if(HSEStartUpStatus == SUCCESS)                  //外部高速时钟已经准别好
  {
    FLASH_PrefetchBufferCmd(FLASH_PrefetchBuffer_Enable); //开启FLASH预读缓冲功能,加速FLASH的读取。所有程序中必须的用法.位置:RCC初始化子函数里面,时钟起振之后
    FLASH_SetLatency(FLASH_Latency_2);                    //flash操作的延时
      	
    RCC_HCLKConfig(RCC_SYSCLK_Div1);               //配置AHB(HCLK)时钟等于==SYSCLK
    RCC_PCLK2Config(RCC_HCLK_Div1);                //配置APB2(PCLK2)钟==AHB时钟
    RCC_PCLK1Config(RCC_HCLK_Div2);                //配置APB1(PCLK1)钟==AHB1/2时钟
         
    RCC_PLLConfig(RCC_PLLSource_HSE_Div1, RCC_PLLMul_9);  //配置PLL时钟 == 外部高速晶体时钟 * 9 = 72MHz
    RCC_PLLCmd(ENABLE);                                   //使能PLL时钟
   
    while(RCC_GetFlagStatus(RCC_FLAG_PLLRDY) == RESET)    //等待PLL时钟就绪
    {
    }
    RCC_SYSCLKConfig(RCC_SYSCLKSource_PLLCLK);            //配置系统时钟 = PLL时钟
    while(RCC_GetSYSCLKSource() != 0x08)                  //检查PLL时钟是否作为系统时钟
    {
    }
  }
  
  RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA | RCC_APB2Periph_USART1 | RCC_APB2Periph_AFIO, ENABLE);  //允许 GPIOA、USART1、AFIO时钟
}

/*
函数: void GPIO_Configuration(void)
功能: GPIO配置
参数: 无
返回: 无
/
void GPIO_Configuration(void)
{
  GPIO_InitTypeDef GPIO_InitStructure;        //定义GPIO初始化结构体

  GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0; 
  GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; 
  GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP; //推挽输出 	 
  GPIO_Init(GPIOA, &GPIO_InitStructure); 	   //PA0输出控制LED灯

  GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9; 
  GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP; //复合推挽输出 	 
  GPIO_Init(GPIOA, &GPIO_InitStructure); 	   //PA9串口输出

  GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10; 
  GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING; //浮空输入
  GPIO_Init(GPIOA, &GPIO_InitStructure); 	   //PA10用于串口读入
}

void NVIC_Configuration(void)	 //中断分组和优先级配置	 详见《STM32的函数说明(中文).pdf》P165
{
	NVIC_InitTypeDef NVIC_InitStructure;

	//NVIC_SetVectorTable(NVIC_VectTab_FLASH, 0x0);  
	// Set the Vector Table base location at 0x08000000 配置中断向量偏移表的.默认是FLASH的,所以你不设置也无关系. RAM调试的时候,必须设置.

	NVIC_PriorityGroupConfig(NVIC_PriorityGroup_0);	 //0组,先占优先级1位,从优先级3位
	NVIC_InitStructure.NVIC_IRQChannel = USART1_IRQn; //USART1 全局中断
	//NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority=0;
	NVIC_InitStructure.NVIC_IRQChannelSubPriority = 1; 	  //中断响应优先级1
	NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;//使能
	NVIC_Init(&NVIC_InitStructure); //初始化配置
}

/*
	函数名:USART1_Configuration
	输  入:
	输  出:
	功能说明:
	初始化串口硬件设备,启用中断
	配置步骤:
	(1)打开GPIO和USART1的时钟
	(2)设置USART1两个管脚GPIO模式
	(3)配置USART1数据格式、波特率等参数
	(4)使能USART1接收中断功能
	(5)最后使能USART1功能
*/
void USART1_Configuration(void)	  //串口配置   详见《STM32的函数说明(中文).pdf》P346
{
	USART_InitTypeDef USART_InitStructure;
	USART_InitStructure.USART_BaudRate=9600;   //波特率为9600
	USART_InitStructure.USART_WordLength=USART_WordLength_8b;  //数据位为8
	USART_InitStructure.USART_StopBits=USART_StopBits_1; //在帧结尾传输 1 个停止位
	USART_InitStructure.USART_Parity=USART_Parity_No; //校验模式:奇偶失能
	USART_InitStructure.USART_HardwareFlowControl=USART_HardwareFlowControl_None; //硬件流控制失能
	USART_InitStructure.USART_Mode=USART_Mode_Tx | USART_Mode_Rx; //USART_Mode 指定了使能或者失能发送和接收模式:发送使能|接收失能
	USART_Init(USART1, &USART_InitStructure);	  //初始化配置

	USART_ITConfig(USART1, USART_IT_RXNE, ENABLE);  //使能或者失能指定的 USART 中断	,此处为接收中断
	USART_Cmd(USART1,ENABLE);	//使能或者失能 USART 外设
	USART_ClearFlag(USART1, USART_FLAG_TC);//清除传输完成标志位,否则可能会丢失第1个字节的数据.USART_FLAG_TC为发送完成标志位
}



/*
                中断服务程序
*/
void USART1_IRQHandler(void)
{	
	u8 dat;
	//u8 uart1_buf[6];
	
    if(USART_GetITStatus(USART1,USART_IT_RXNE)==SET)	//若接收完成
    {
    	
		//GPIO_SetBits(GPIOA,GPIO_Pin_0);//点亮LED	              
    	//delay_ms(500);//延时
		//GPIO_ResetBits(GPIOA,GPIO_Pin_0);//灭 
     
		dat=USART_ReceiveData(USART1);
		//Uart1_PutChar(dat);
		//uart1_buf[count++]=dat;
		
		
		dat = USART_ReceiveData(USART1);
		if(dat!=0x23)	 //!='#'在发送的信息末尾应包含'#'号
		{
			uart1_buf[count++]=dat;
			if(count>19){
			   Uart1_PutString("超出长度![最长20个字符/10个汉字]",32);
			   count=0;
			}
		}else if(count!=0)
			{
				//Uart1_PutString("Received Data:",14);
				//Uart1_PutString(uart1_buf,count);
				if(USART1_ReceivedStrCmp("我爱你")) Uart1_PutString("我也爱你",8);
				if(USART1_ReceivedStrCmp("Hello")) Uart1_PutString("Hi",2);
				count=0;
			}		

		USART_ClearFlag(USART1,USART_IT_RXNE);
    } 

	  //溢出处理-如果发生溢出需要先清除ORE,再读DR寄存器 则可清除不断入中断的问题
        if(USART_GetFlagStatus(USART1,USART_FLAG_ORE)==SET)
        {
                USART_ClearFlag(USART1,USART_FLAG_ORE);        //清除ORE
                USART_ReceiveData(USART1);                                //读DR
        }

}


//发送一个字符
void Uart1_PutChar(u8 ch)
{
    USART_SendData(USART1, (u8) ch);
    while(USART_GetFlagStatus(USART1, USART_FLAG_TXE) == RESET);//等待发送完成
}

//发送一个字符串 Input : buf为发送数据的地址 , len为发送字符的个数
void Uart1_PutString(u8* buf , u8 len)
{   
	u8 i;
    for(i=0;i
				   
关键字:STM32  单片机学习  串口中断通信 引用地址:STM32单片机学习(3) 串口中断通信

上一篇:基于STM32微控制器的电机控制设计
下一篇:ARM学习之定时器Timer0实验

推荐阅读最新更新时间:2024-03-16 14:56

STM32之编码器模式学习
一、编码器模式理论储备 通常为了提高精度我们会选择在上升沿和下降沿都进行计数! 还有一个非常重要的图这里也记录下 其中让人费解的应该是在第二列的相对信号的电平,这里就来详细谈一下吧。 其实也不难理解哈,我们上面也说了通常为了提高精度会在A、B两相的上升沿和下降沿都进行计数,那么对应在一个周期就可以计数四次,计数次数的增加也就意味着精度的提高! 编码器模式下,如果此时处于正转,那么这四次计数应该都是加的。同理,如果是反转,那么这四次计数都是减的。那么问题来了,如何判断正反转呢? 不就是在相对电平的基础上嘛!!! 仔细对照图中,在正转或者反转的情况下,A相对B的电平高低以及上表中的计数方向
[单片机]
<font color='red'>STM32</font>之编码器模式<font color='red'>学习</font>
STM32 CAN应用-过滤器设置
在使用STM32的CAN控制器进行数据收发,当用到位屏蔽模式的时候,就要设置过滤器了,这个关系到是否能够接收到想要的数据。下面针对几种不同情况对CAN过滤器(Filter)进行设置。 CAN_FilterInitStructure.CAN_FilterMode = CAN_FilterMode_IdMask; //标示符屏蔽模式 CAN_FilterInitStructure.CAN_FilterScale = CAN_FilterScale_32bit; 1、对扩展数据帧进行过滤:(只接收扩展数据帧) CAN_FilterInitStructure.CAN_FilterIdHigh =(EXT_ID 13)
[单片机]
STM32上如果不使用外部晶振,OSC_IN和OSC_OUT的接法
如果使用内部RC 振荡器 而不使用外部晶振,请按照下面方法处理: 1)对于100脚或144脚的产品,OSC_IN应接地,OSC_OUT应悬空。 2)对于少于100脚的产品,有2种接法:   2.1)OSC_IN和OSC_OUT分别通过10K 电阻 接地。此方法可提高 EMC 性能。   2.2)分别重映射OSC_IN和OSC_OUT至PD0和PD1,再配置PD0和PD1为推挽输出并输出'0'。此方法可以减小功耗并(相对上面2.1)节省2个外部电阻。 以下是在 ST 英文网站FAQ上抄下的详细解释: 1) In 100 and 144 packages,  the OSC_IN /OSC_Out pins are a de
[单片机]
STM32实战二 新建工程
这一章新建一个工程,主要目的是练习新建C++工程,生成比较干净的代码,后来发现没在太大的意义,直接在原示例中删除文件,然后新建cpp文件即可,也可以把原有main.c的属性变成c++,方法是main.c/右键/Options for file 'main.c' /Properties标签/File Type:/改为C++ Source file,这样更方便。不过下面还是记录一下新建工程的过程,对熟悉环境和配制有一定的帮助。 复制原有流水灯程序到新的文件夹下,打开Keil4,点击新建工程 在对话框中找到下面的文件夹,并新建一个文件夹STM32-AGV,进入STM32-AGV文件夹。 键入工程名ST
[单片机]
<font color='red'>STM32</font>实战二 新建工程
毕业设计| STM32智能婴儿床监控
婴儿床监测主要有以下功能: 床内温度;检测婴儿状态;配置RTC实时时间;电机PWM按键三档调速;自动开关加热器;发送温度到手机终端;把所有要呈现的状态通过OLED屏幕显示。 感兴趣的小伙伴还可以加入更多的功能。 # 硬件元件清单 # 主控为STM32F103C8T6,使用最小系统板: OLED显示屏1个,连接方式:VCC-3.3/5V;GND-GND;SCL-单片机SCL脚;SDA-单片机SDA脚 轻触按键3个,连接方式:对脚连接;一脚接GND;一脚接上拉电阻至电源、及IO口引脚 一路5v继电器模块 1个,连接方式:上面口:VCC-5V;GND-GND;PIN-单片机信号脚下面口:选择常开/常闭连接负载 L298N电机驱
[单片机]
毕业设计| <font color='red'>STM32</font>智能婴儿床监控
浅谈 STM32 硬件I2C的使用 (中断方式 无DMA 无最高优先级)
引子 STM32的硬件I2C很多人都对它望而却步。因为很多电工都说,STM32 硬件 I2C有BUG、不稳定、死机等等……最后都使用GPIO模拟I2C。 的确,模拟I2C好用。但是在我看来在一个72M的Cortex-M3的MCU上这样做非常不妥。一般来说I2C是一种慢速总线,就算工作在400kHz的快速模式上,I2C传送每个字节仍需要至少23us——还没有计算地址、起始信号和结束信号的发送。如果使用GPIO模拟的I2C,这23us的CPU时间都在空转中浪费了,而这23us已经可以做不少的事情了,所以在STM32上I2C还是使用硬件为佳——虽然它多多少少有点缺陷。 这篇文章不是给完全没有接触过STM32 硬件I2C的新手看的,看这篇
[单片机]
浅谈 <font color='red'>STM32</font> 硬件I2C的使用 (<font color='red'>中断</font>方式 无DMA 无最高优先级)
Robomaster-stm32-PWM学习笔记(stm32控制pwm输出)
学习笔记: 脉冲宽度调制(PWM),是英文“Pulse Width Modulation”的缩写,简称脉宽调制,是利用微处理器的数字输出来对模拟电路进行控制的一种非常有效的技术,广泛应用在从测量、通信到功率控制与变换的许多领域中。 将通用 定时器 分为四个部分: 1-选择 时钟 2-时基电路 3-输入捕获 4-输出比较 实践1-pwm- led 闪烁 要求产生周期为200ms,占空比为50%的PWM 信号 来控制led灯。 1.cubemax配置 由原理图可知,led引脚复用为 ti m5 周期为200ms,占空比为50%,Tim5挂在APB1总线上,CLK =
[单片机]
Robomaster-stm32-PWM<font color='red'>学习</font>笔记(<font color='red'>stm32</font>控制pwm输出)
如何在STM32中收发float数据?
实际应用中,我们可能需要两个设备通过串口传输浮点数据: 本篇笔记为了方便演示,使用串口助手模拟其中一个设备,本篇笔记内容如下: 我们创建一个用于管理float类型数据的共用体: union float_data { float f_data; uint8_t by te [4]; }; 数据的流向如: 本次使用串口助手模拟发送设备,省略了第一步,主要看第②、③步。 创建两个共用体变量,用于发送与接收: union float_data rx_float_data, tx_float_data; 收发相关代码: // 公众号: 嵌入式 大杂烩 if(HAL_ UART _Receive(&huart3, rx_float
[单片机]
如何在<font color='red'>STM32</font>中收发float数据?
小广播
添点儿料...
无论热点新闻、行业分析、技术干货……
设计资源 培训 开发板 精华推荐

最新单片机文章
何立民专栏 单片机及嵌入式宝典

北京航空航天大学教授,20余年来致力于单片机与嵌入式系统推广工作。

换一换 更多 相关热搜器件
电子工程世界版权所有 京B2-20211791 京ICP备10001474号-1 电信业务审批[2006]字第258号函 京公网安备 11010802033920号 Copyright © 2005-2024 EEWORLD.com.cn, Inc. All rights reserved