STM32中断方式实现串口通信(标准库)
1)当stm32接收到字符“s”时,停止持续发送“hello windows!”; 当接收到字符“t”时,持续发送“hello windows!”;
2)当stm32接收到字符“stop stm32!”时,停止持续发送“hello windows!”; 当接收到字符“go stm32!”时,持续发送“hello windows!”
实验工具: (1)软件
-
标准库 -
KEIL5:[安装教程](https://blog.csdn.net/zhoushuaiyxlmwan/article/details/127190907? spm=1001.2014.3001.5502) -
mcuisp(或者FlyMcu): mcuisp百度网盘链接提取码:h2xc -
野火多功能调试助手:https://pan.baidu.com/s/14zEjYNlU-2CjgoR1sI5dSg 提取码:rau0
(2)硬件
- STM32F103C8T6的最小核心板
- 杜邦线
- USB转TTL模块
一.串口通信原理以及中断原理
这两部分我在上两篇文章里已经介绍过了,这里就不占用篇幅了 串口通信原理 中断原理
一.问题分析
1.涉及外设
1.串口通信:USART,GPIO 2.中断:GPIO,AFIO,EXTI,NVIC
2.状态机实现
图上的意思是: 设置一个状态标志位S,默认为0,等待包头的出现(这里的包头自己设置一个字符 )。 当检测到接收的一个字符和我们规定的包头字符相符合,就把标志位S置1,并且开始接收后面来的数据,直到读取到某一个字符和我们规定的第一位结束位相符合,则把标志位S置2,并且不再接收数据,只等待字符与第二个结束位相符合,把状态位S置0. 这里是双重结束标志置位,我们也可以设置一个结束标志位,这样就只需要S的值在0或1之间转换,原理不变
二.创建MDK(keil5)项目
1.项目结构
①.USER
存放main.c文件以及标准库的配置文件
②.Hardware
存放串口触发中断,以及利用串口通信的一些函数
③.Delay
存放供我们调用的延时函数,在后面用单片机循环发送消息时会用到
④.其他
都是其他对应的标准库的文件
2.基本设置
三.具体实现
1.配置RCC,把涉及到的外设的时钟全部打开
RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1,ENABLE);
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA,ENABLE);
2.初始化GPIO口
GPIO_InitTypeDef GPIO_InitStruct;
GPIO_InitStruct.GPIO_Mode = GPIO_Mode_AF_PP;
GPIO_InitStruct.GPIO_Pin = GPIO_Pin_9;
GPIO_InitStruct.GPIO_Speed = GPIO_Speed_50MHz;
GPIO_Init(GPIOA, &GPIO_InitStruct);
GPIO_InitStruct.GPIO_Mode = GPIO_Mode_IN_FLOATING;
GPIO_InitStruct.GPIO_Pin = GPIO_Pin_10;
GPIO_InitStruct.GPIO_Speed = GPIO_Speed_50MHz;
GPIO_Init(GPIOA, &GPIO_InitStruct);
3.初始化UART
USART_InitTypeDef USART_InitStruct;
USART_InitStruct.USART_BaudRate = 9600;
USART_InitStruct.USART_HardwareFlowControl =USART_HardwareFlowControl_None;
USART_InitStruct.USART_Mode = USART_Mode_Tx | USART_Mode_Rx;
USART_InitStruct.USART_Parity = USART_Parity_No;
USART_InitStruct.USART_StopBits = USART_StopBits_1;
USART_InitStruct.USART_WordLength = USART_WordLength_8b;
USART_Init(USART1,&USART_InitStruct);
4.开启中断
USART_ITConfig(USART1,USART_IT_RXNE,ENABLE);
5.配置NVIC优先级和分组
NVIC_InitTypeDef NVIC_InitStructure;
NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);
NVIC_InitStructure.NVIC_IRQChannel = USART1_IRQn;
NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 1;
NVIC_InitStructure.NVIC_IRQChannelSubPriority = 1;
NVIC_Init(&NVIC_InitStructure);
6.开启USART外设
USART_Cmd(USART1,ENABLE);
7.中断函数
根据标准库的封装,USART的中断函数名为void USART1_IRQHandler() ,这里由于还有其他函数的调用,直接给出Serial.c文件的完整代码。
#include "stm32f10x.h"
#include "stdio.h"
#include <stdio.h>
#include <string.h>
uint8_t Serial_RxFlag;
char Serial_RxPacket[100];
void Serial_Init(void)
{
GPIO_InitTypeDef GPIO_InitStruct;
USART_InitTypeDef USART_InitStruct;
NVIC_InitTypeDef NVIC_InitStructure;
RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1,ENABLE);
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA,ENABLE);
GPIO_InitStruct.GPIO_Mode = GPIO_Mode_AF_PP;
GPIO_InitStruct.GPIO_Pin = GPIO_Pin_9;
GPIO_InitStruct.GPIO_Speed = GPIO_Speed_50MHz;
GPIO_Init(GPIOA, &GPIO_InitStruct);
GPIO_InitStruct.GPIO_Mode = GPIO_Mode_IN_FLOATING;
GPIO_InitStruct.GPIO_Pin = GPIO_Pin_10;
GPIO_InitStruct.GPIO_Speed = GPIO_Speed_50MHz;
GPIO_Init(GPIOA, &GPIO_InitStruct);
USART_InitStruct.USART_BaudRate = 9600;
USART_InitStruct.USART_HardwareFlowControl =USART_HardwareFlowControl_None;
USART_InitStruct.USART_Mode = USART_Mode_Tx | USART_Mode_Rx;
USART_InitStruct.USART_Parity = USART_Parity_No;
USART_InitStruct.USART_StopBits = USART_StopBits_1;
USART_InitStruct.USART_WordLength = USART_WordLength_8b;
USART_Init(USART1,&USART_InitStruct);
USART_ITConfig(USART1,USART_IT_RXNE,ENABLE);
NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);
NVIC_InitStructure.NVIC_IRQChannel = USART1_IRQn;
NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 1;
NVIC_InitStructure.NVIC_IRQChannelSubPriority = 1;
NVIC_Init(&NVIC_InitStructure);
USART_Cmd(USART1,ENABLE);
}
uint8_t Serial_GetRxFlag(void)
{
if(Serial_RxFlag == 1)
{
Serial_RxFlag = 0;
return 1;
}
return 0;
}
void Serial_SendByte(uint8_t Byte)
{
USART_SendData(USART1,Byte);
while(USART_GetFlagStatus(USART1,USART_FLAG_TXE)==RESET);
}
void Serial_SendString(char *String)
{
uint8_t i;
for(i=0;String[i]!=0;i++)
{
Serial_SendByte(String[i]);
}
}
void USART1_IRQHandler()
{
static u8 pRxpacket = 0;
static u8 Rxstate = 0;
if(USART_GetITStatus(USART1,USART_IT_RXNE)==SET)
{
u8 RxData = USART_ReceiveData(USART1);
if(Rxstate==0)
{
if(RxData=='[')
{
memset(Serial_RxPacket,'\0',sizeof(Serial_RxPacket));
pRxpacket = 0;
Rxstate=1;
}
}
else if(Rxstate==1)
{
if(RxData==']')
{
Rxstate=0;
Serial_RxFlag=1;
}
else
{
Serial_RxPacket[pRxpacket] = RxData;
pRxpacket++;
}
}
USART_ClearITPendingBit(USART1,USART_IT_RXNE);
}
}
- 这里有几个值得注意的点:
- 1.在初始化外设的时候,只要涉及到调用标准库里的宏定义生成一个结构体变量的时候,必须写在最前面,否则编译的时候会报错。例如
GPIO_InitTypeDef GPIO_InitStruct; - 2.用来给数组赋值的数组下标
pRxstate 必须在标志位Rxstate变之前就置0,否则在运行的时候下标不能及时归零。 - 3.起始标识符和结束标识符可以随意设置
8.主函数执行以及判断和响应
#include "stm32f10x.h"
#include <stdio.h>
#include "Serial.h"
#include "Delay.h"
#include <string.h>
int main(void)
{
Serial_Init();
while(1)
{
if(Serial_GetRxFlag()==1)
{
if(strcmp(Serial_RxPacket,"go stm32!")==0)
{
while(1)
{
Serial_SendString("hello windows!\n");
Delay_ms(800);
if(strcmp(Serial_RxPacket,"stop stm32!")==0)
{
break;
}
}
}
}
}
}
- 这里有也几个值得注意的点:
- 1.用来比较字符串的函数
strcmp() 使用的时候必须包含头文件#include <stdio.h> - 2.这里能直接调用
Serial_RxPacket[] 字符数组是因为在Serial.h头文件里把它extern 出来了,而main函数又导入了Serial.h头文件,所以可以直接使用
9.头文件及工具文件
#include "stm32f10x.h"
void Delay_us(uint32_t xus)
{
SysTick->LOAD = 72 * xus;
SysTick->VAL = 0x00;
SysTick->CTRL = 0x00000005;
while(!(SysTick->CTRL & 0x00010000));
SysTick->CTRL = 0x00000004;
}
void Delay_ms(uint32_t xms)
{
while(xms--)
{
Delay_us(1000);
}
}
void Delay_s(uint32_t xs)
{
while(xs--)
{
Delay_ms(1000);
}
}
#ifndef __DELAY_H
#define __DELAY_H
void Delay_us(uint32_t us);
void Delay_ms(uint32_t ms);
void Delay_s(uint32_t s);
#endif
#ifndef _Serial_H
#define _Serial_H
extern char Serial_RxPacket[];
void Serial_Init(void);
void Serial_SendByte(uint8_t Byte);
void Serial_SendString(char *String);
uint8_t Serial_GetRxFlag(void);
#include <stdio.h>
#endif
四.线路连接
这里接线比较简单,只用把PA9(串口发送)与RXD连接,PA10(串口接收端)与TXD连接就好
五.实验结果
生成的Hex文件,在FlyMcu进行烧录后,打开串口助手
六.总结
多次试着使用标准库来完成项目后对外设的理解更深刻了,不过由于C语言的知识掌握不够号,导致在完成的过程中遇到了很多问题,这可能也是标准库比起HAL库比较麻烦的一点。标准库需要我们更熟练的掌握c语言的知识,而且如果不熟练的话会花费很多时间。
|