当前位置: 首页 > article >正文

基于STM32F103的USART的原理及应用(一)(实现手机BLE和MCU进行通信)

一,查阅数据手册,找到对应的IO口和外设总线并配置好外设源文件

想了解USART的具体原理的小伙伴请进传送门:(总结)STM32中USART原理及应用(PC、BLE、ESP8266通信实现)-CSDN博客

二,打开外设时钟

三,配置GPIO引脚为复用模式

四,重映射IO引脚为串口功能(不是必须的,像我用的是PA9 和 PA10,就可以不用,因为默认有串口功能)

五,配置UART1参数(波特率,数据位,停止位,校验位,设置无流控,设置全双工)(跟STM32F407代码一样)

六,配置USART的中断(选择中断通道,抢占优先级,响应优先级,打开中断通道)(跟STM32F407代码一样)

七,选择USART1的中断源,即当接收到数据时触发中断

八,打开串口

九,编写USART中断服务函数

十,完整代码展示

/*
  ******************************************************************************
  * @file    main.c 
  * @author  
  * @version V1.0
  * @date    2024/12/30
  * @brief   本代码配置USART实现BLE和MCU进行串口通信,实现通过手机蓝牙来控制小灯
             的开关
  ******************************************************************************
*/

#include "stm32f10x.h"
#include "delay_us.h"
#include "delay_ms.h"
#include "string.h"

/* Private typedef   用于记录用户自定义的一些数据类型的别名-------------------*/


/* Private define    用于记录用户自定义的类型,比如结构体、共用体、枚举-------*/


/* Private macro     用于记录用户自定义的宏定义-------------------------------*/


/* Private variables 用于记录用户自定义的全局变量-----------------------------*/
uint8_t  u1_rxbuf[512] = {0}; //作为UART1的接收缓冲区
uint32_t u1_rxcnt = 0;        //作为UART1的接收计数器


/* Private function prototypes 用于记录用户自定义的函数声明-------------------*/



/* Private functions 用于记录用户自定义的函数原型-----------------------------*/


/**
  * @brief  该函数初始化IO引脚,用于LED的点亮
  * @param  None
  * @retval None
  */
void LED_Config(void)
{
	GPIO_InitTypeDef GPIO_InitStructure;

	//打开时钟
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOC, ENABLE);

	//配置引脚参数
	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_14 ;
	GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;
	GPIO_Init(GPIOC, &GPIO_InitStructure);
	
	//输出高电平(点亮小灯)
	GPIO_SetBits(GPIOC, GPIO_Pin_14);
}


/**
  * @brief  该函数初始化USART1,用于BLE通信
  * @param  None
  * @retval None
  */
void USART1_Config(u32 baud)
{
	USART_InitTypeDef USART_InitStructure;
	NVIC_InitTypeDef  NVIC_InitStructure;
	GPIO_InitTypeDef  GPIO_InitStructure;

	//打开GPIOA的时钟
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);

	//打开USART2的时钟
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1, ENABLE);

	//重映射IO引脚为串口功能
	GPIO_PinRemapConfig(GPIO_Remap_USART1, ENABLE);


	//配置GPIO的引脚
	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9;
	GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;
	GPIO_Init(GPIOA, &GPIO_InitStructure);

	//配置UART1的参数    最常用的格式: 1bit停止位  8bit数据位  No校验位  9600bps
	USART_InitStructure.USART_BaudRate = baud;											//波特率
	USART_InitStructure.USART_WordLength = USART_WordLength_8b;							//数据位
	USART_InitStructure.USART_StopBits = USART_StopBits_1;								//停止位
	USART_InitStructure.USART_Parity = USART_Parity_No;									//校验位
	USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None; 	//无流控
	USART_InitStructure.USART_Mode = USART_Mode_Rx | USART_Mode_Tx;     				//全双工
	USART_Init(USART1, &USART_InitStructure);

	//配置USART的中断
	NVIC_InitStructure.NVIC_IRQChannel = USART1_IRQn;
	NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0;
	NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0;
	NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
	NVIC_Init(&NVIC_InitStructure);

	//选择UART1的中断源   接收到数据则触发中断请求
	USART_ITConfig(USART1, USART_IT_RXNE, ENABLE);

	//打开UART1串口
	USART_Cmd(USART1, ENABLE);
	
}



/**
  * @brief  程序的入口
  * @param  None
  * @retval None
  */
int main(void)
{
	LED_Config();
	USART1_Config(9600);
	
  
	while (1)
	{
		//判断UART2是否接收到数据  假设接收到 "led_on",则让LED点亮
		if( u1_rxcnt > 0 && strstr((char *)u1_rxbuf,"led_on"))
		{
			
			GPIO_SetBits(GPIOC,GPIO_Pin_14); //输出高电平	
			u1_rxcnt = 0; //计数器复位
			memset((char *)u1_rxbuf,0,512); //清空数组
		}
		
		//判断UART2是否接收到数据  假设接收到 "led_off",则让LED熄灭
		if( u1_rxcnt > 0 && strstr((char *)u1_rxbuf,"led_off"))
		{
			
			GPIO_ResetBits(GPIOC,GPIO_Pin_14); //输出低电平	
			u1_rxcnt = 0; //计数器复位
			memset((char *)u1_rxbuf,0,512); //清空数组
		}


	}
}


/**
* @brief  This function handles USRAT1 interrupt request.
* @param  None
* @retval None
*/
void USART1_IRQHandler(void)
{
 
  //判断是否接收到数据
  if (USART_GetITStatus(USART1, USART_IT_RXNE) == SET)
  {   
      //把串口收到的字节存储到变量data中
      u1_rxbuf[u1_rxcnt++] = USART_ReceiveData(USART1); 
			if( u1_rxcnt >= 512 )
			{
				u1_rxcnt = 0;
			}
		  
  }
}





/********************** (C) COPYRIGHT Your Name xxxx@126.com***END OF FILE****/

http://www.kler.cn/a/465734.html

相关文章:

  • weblogic安装 12.2.1.4.0集群
  • python.exe无法找到程序入口 无法定位程序输入点(Anaconda Prompt报错)
  • 对一个双向链表,从尾部遍历找到第一个值为x的点,将node p插入这个点之前,如果找不到,则插在末尾。使用C语言实现
  • RabbitMq的Java项目实践
  • Backend - C# 的日志 NLog日志
  • Linux Snipaste 截图闪屏/闪烁
  • 探索Wiki:开源知识管理平台及其私有化部署
  • 手机租赁平台开发实用指南与市场趋势分析
  • 探索 Android Instant Apps:InstantAppInfo 的深入解析与架构设计
  • C++中关于异常的简单分析
  • C# 设计模式概况
  • Python爬虫入门(1)
  • 【Patroni官方文档】介绍与目录
  • 【谷歌开发者月刊】十二月精彩资讯回顾,探索科技新可能
  • 【C++】穿越时光隧道,拾贝史海遗珍,轻启C++入门之钥,解锁程序之奥秘(首卷)
  • 随机种子定不死找bug
  • python 字符串算法
  • CTFshow—远程命令执行
  • 区块链方向学习路线
  • 音视频-----RTSP协议 音视频编解码
  • 国产数据库AntDB插件pg_profile安装说明
  • xdoj isbn号码
  • echarts 饼图超过9种颜色就重复了,如何自定义颜色
  • 【vue项目中漏洞修复】
  • Spring源码分析之事件机制——观察者模式(一)
  • 硬件-射频-PCB-常见天线分类-ESP32实例