STM32F103RC USART2串口查询方式接收数据

发布者:未来架构师最新更新时间:2019-10-28 来源: eefocus关键字:STM32F103RC  USART2串口  查询方式  接收数据 手机看文章 扫描二维码
随时随地手机看文章

现有一个小需求,使用STM32F1系列单片机做串口2的收发数据的功能,通过PC上的串口调试助手给单片机发一串数据,单片机收到数据后再给PC的串口调试助手发回去。


看似简单的功能,调试了好久才调好,记录一下供大家参考。


STM32使用USART2,对应单片机的PA1控制方向,PA2发送,PA3接收。


代码如下:


z_hardware_usart2.h


#ifndef __STM32F10X_H

#define __STM32F10X_H

#include "stm32f10x.h"

#endif

 

#ifndef __Z_UTIL_TIME_H

#define __Z_UTIL_TIME_H

#include "z_util_time.h"

#endif

 

#ifndef _UART_BUF_LEN_512

#define _UART_BUF_LEN_512 512

#endif

 

#ifndef _UART_BUF_LEN_256

#define _UART_BUF_LEN_256 256

#endif

 

#ifndef _UART_COMM_CH_TIMEOUT_3MS

#define _UART_COMM_CH_TIMEOUT_3MS 3

#endif

 

#ifndef _UART_COMM_CH_TIMEOUT_2MS

#define _UART_COMM_CH_TIMEOUT_2MS 2

#endif

 

#ifndef _UART_COMM_INT_TIMEOUT_300MS

#define _UART_COMM_INT_TIMEOUT_300MS 300

#endif

 

#ifndef _UART_COMM_INT_TIMEOUT_500MS

#define _UART_COMM_INT_TIMEOUT_500MS 500

#endif

 

void init_hardware_usart2(u32 bound);

void func_usart2_send_a_byte(u8 abyte);

u8 func_usart2_recv_a_byte(u8 *recv_data);

void func_usart2_send_bytes(u8 *bytes, u8 bytes_len);

u8 func_usart2_recv_bytes(const u16 BUF_MAX_LEN, u8 *recv_datas, u16 *recv_datas_len, const u16 int_timeout_ms, const u8 ch_timeout_ms);

z_hardware_usart2.c


#ifndef __Z_HARDWARE_USART2_H

#define __Z_HARDWARE_USART2_H

#include "z_hardware_usart2.h"

#endif

 

void init_hardware_usart2(u32 bound)

{

GPIO_InitTypeDef GPIO_InitStructure;

USART_InitTypeDef USART_InitStructure;

RCC_APB1PeriphClockCmd(RCC_APB1Periph_USART2, ENABLE);

  RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA | RCC_APB2Periph_AFIO, ENABLE);

GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;

GPIO_InitStructure.GPIO_Pin = GPIO_Pin_1;

GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;

GPIO_Init(GPIOA, &GPIO_InitStructure);

 

GPIO_InitStructure.GPIO_Pin = GPIO_Pin_2;

GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;

GPIO_Init(GPIOA, &GPIO_InitStructure);

 

GPIO_InitStructure.GPIO_Pin = GPIO_Pin_3;

GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;

GPIO_Init(GPIOA, &GPIO_InitStructure);

USART_InitStructure.USART_BaudRate = bound;

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(USART2, &USART_InitStructure);

USART_ITConfig(USART2, USART_IT_RXNE, DISABLE);

USART_Cmd(USART2, ENABLE);

USART_ClearFlag(USART2, USART_FLAG_TC);

GPIO_ResetBits(GPIOA, GPIO_Pin_1);

}

 

void func_usart2_send_a_byte(u8 abyte)

{

while(USART_GetFlagStatus(USART2, USART_FLAG_TC) == RESET);

USART_SendData(USART2, abyte);

while(USART_GetFlagStatus(USART2, USART_FLAG_TC) == RESET);

}

 

u8 func_usart2_recv_a_byte(u8 *recv_data)

{

if(USART_GetFlagStatus(USART2, USART_FLAG_RXNE) != RESET)

{

*recv_data = USART_ReceiveData(USART2);

USART_ClearFlag(USART2, USART_FLAG_RXNE);

return 0;

}

return 1;

}

 

void func_usart2_send_bytes(u8 *bytes, u8 bytes_len)

{

u8 i;

GPIO_SetBits(GPIOA, GPIO_Pin_1);

for(i = 0; i < bytes_len; i++)

{

func_usart2_send_a_byte(bytes[i]);

}

GPIO_ResetBits(GPIOA, GPIO_Pin_1);

}

 

u8 func_usart2_recv_bytes(const u16 BUF_MAX_LEN, u8 *recv_datas, u16 *recv_datas_len, const u16 int_timeout_ms, const u8 ch_timeout_ms)

{

u8 flag_recv;

u8 tmp_recv;

u16 tmp_int_timeout_cnt = 0;

u8 tmp_ch_timeout_cnt = 0;

u16 len_recv = 0;

u8 *p_recv;

p_recv = recv_datas;

GPIO_ResetBits(GPIOA, GPIO_Pin_1);

for(;func_usart2_recv_a_byte(&tmp_recv) != 0;)

{

tmp_int_timeout_cnt ++;

if(tmp_int_timeout_cnt > int_timeout_ms * 10)

{

return 1;

}

delay_us(100);

}

len_recv ++;

*p_recv = tmp_recv;

p_recv ++;

for(;;)

{

flag_recv = func_usart2_recv_a_byte(&tmp_recv);

if(flag_recv == 0)//data recv

{

tmp_ch_timeout_cnt = 0;

len_recv ++;

*p_recv = tmp_recv;

p_recv ++;

}

else //data not recv

{

tmp_ch_timeout_cnt ++;

delay_us(800);//FIXME 适用9600波特率。波特率不同需要针对性修改

}

if(tmp_ch_timeout_cnt > ch_timeout_ms)

{

*recv_datas_len = len_recv;

return 0;

}

if(len_recv > BUF_MAX_LEN - 1)

{

*recv_datas_len = BUF_MAX_LEN - 1;

return 2;

}

}

}

z_util_time.h


#ifndef __STM32F10X_H

#define __STM32F10X_H

#include "stm32f10x.h"

#endif

 

#define AHB_INPUT 72

 

void delay_us(uint32_t time_us);

void delay_ms(uint32_t time_ms);

z_util_time.c


#ifndef __Z_UTIL_TIME_H

#define __Z_UTIL_TIME_H

#include "z_util_time.h"

#endif

 

void delay_us(uint32_t time_us)

{

SysTick->LOAD = AHB_INPUT * time_us;

SysTick->VAL = 0x00;

SysTick->CTRL = 0x00000005;

for(;!(SysTick->CTRL & 0x00010000););

SysTick->CTRL = 0x00000004;

}

 

void delay_ms(uint32_t time_ms)

{

for(;time_ms-- > 0;)

{

delay_us(1000);

}

}

main.c


#include "stm32f10x.h"

 

#ifndef __Z_HARDWARE_USART2_H

#define __Z_HARDWARE_USART2_H

#include "z_hardware_usart2.h"

#endif

 

#include

 

int main()

{

u8 to_recv[_UART_BUF_LEN_256];

u16 to_recv_len;

init_hardware_usart2(9600);

while (1)

{

if(func_usart2_recv_bytes(_UART_BUF_LEN_256, to_recv, &to_recv_len, _UART_COMM_INT_TIMEOUT_500MS, _UART_COMM_CH_TIMEOUT_3MS) == 0)

{

func_usart2_send_bytes(to_recv, to_recv_len);

}

to_recv_len = 0;

memset(to_recv, 0, sizeof(to_recv));

}

}

效果如下:

经过测试发现,串口调试助手给单片机发送了1W+多次,失误率为0,准确率可以应用。

测试代码,仅供参考。


关键字:STM32F103RC  USART2串口  查询方式  接收数据 引用地址:STM32F103RC USART2串口查询方式接收数据

上一篇:STM32F103 IO口结构和特性
下一篇:STM32F103之Timer2PWM输出

推荐阅读最新更新时间:2024-11-10 03:15

STM32-串口超时判断方式接收未知长度数据
usart.c串口中断处理函数: view plain copy void USART1_IRQHandler(void) { u8 res; if(USART1- SR&(1 5))//接收到数据 { res=USART1- DR; if(USART1_Recv_Len USART1_MAX_RECV_LEN) //还可以接收数据 { TIM3- CNT=0; //计数器清空 if(Rec_Over_Flag==0)TIM3_Set(1); //使能定
[单片机]
STM32的PWM输入模式设置并用DMA接收数据
参考 :STM32输入捕获模式设置并用DMA接收数据 PWM input mode This mode is a particular case of input capture mode. The procedure is the same except:  Two ICx signals are mapped on the same TIx input.  These 2 ICx signals are active on edges with opposite polarity.  One of the two TIxFP signals is selected as trigger input and the
[单片机]
STM32的PWM输入模式设置并用DMA<font color='red'>接收</font><font color='red'>数据</font>
STM32 串口采用DMA方式接收数据
环境: 主机:WINXP 开发环境:MDK4.23 MCU:STM32F103CBT6 说明: 串口可以配置成用DMA的方式接收数据,不过DMA需要定长才能产生接收中断,如何接收可变长度的数据呢? 方法有以下3种: 1.将RX脚与一路时钟外部引脚相连,当串口一帧发完,即可利用此定时器产生超时中断.这个实时性较高,可以做到1个字节实时监测. 2.不改变硬件,开启一个定时器监控DMA接收,如果超时则产生中断.这个实时性不高,因为超时时间必须要大于需要接收帧的时间,精度不好控制. 3.STM32单片机有的串口可以监测总线是否处于空闲,如果空闲则产生中断.可以用它来监测DMA接收是否完毕.这种方式实时性很高. 本文采用第3种方式.在波
[单片机]
STM32 <font color='red'>串口</font>采用DMA<font color='red'>方式</font><font color='red'>接收</font><font color='red'>数据</font>
STM32项目中遇到的一些问题以及解决办法
USART: 1.问题描述:使用中断接收数据时,DR中数值正确,但是RXNE始终是0,没有置位。 solution:经过半天的检查以后,发现在中断函数中使用错了一个函数,应该使用USART_GetFlagStatus(USART1, USART_IT_RXNE)来判断RXNE状态,却错使用了USART_GetITStatus函数。 2.问题描述:进入IDLE中断无法退出 solution:清除IDLE的方式不对,正确的清除方式见《stm32中文参考手册》P540 由软件序列清除该位(先读USART_SR,然后读USART_DR) 注意:IDLE位不会再次被置高直到RXNE位被置起(即又检测到一次空闲总线),
[单片机]
STM32串口接收数据卡死问题解决办法
在使用STM32串口时,有时候串口接受一段时间数据之后便会卡死不在进入中断接收数据。原因是串口中断标志没有清除,导致终端锁死,此时需要判断对应的标志位,重新启动中断。 if((huart1.Instance- CR1 & 0x20)==0) { HAL_UART_Receive_IT(&huart1,bsp_uart1.rx_tmp,1); }
[单片机]
STM32F407 串口编程USART1,USART2,USART3,UART4
串口设置的一般步骤可以总结为如下几个步骤: 1) 串口时钟使能,GPIO时钟使能。 2) 设置引脚复用器映射:调用GPIO_PinAFConfig函数。 3) GPIO初始化设置:要设置模式为复用功能。 4) 串口参数初始化:设置波特率,字长,奇偶校验等参数。 5) 开启中断并且初始化NVIC,使能中断(如果需要开启中断才需要这个步骤)。 6) 使能串口。 7) 编写中断处理函数:函数名格式为USARTxIRQHandler(x对应串口号)。 我们通过USART3简单介绍下这几个与串口基本配置直接相关的几个固件库函数。这些函数和定义主要分布在stm32f4xx_usart.h和stm32f4xx_usart.c文
[单片机]
STM32F407 <font color='red'>串口</font>编程USART1,<font color='red'>USART2</font>,USART3,UART4
STM32—cubeMX+DMA+USART 接收任意长度的数据
前言     之前的一篇文章中我为了可以实现USART接收任意长度的数据,对HAL的库进行了修改,可以实现接收以0x0a结尾的任意长度数据,即认为接收到0x0a时接收结束,见链接:HAL USART接收任意长度。   然而,上述这种方法并不合适,原则上HAL库一般不去修改,不便于其他人移植程序,降低了程序中库的适用性,这是很不好的习惯,所以这种方法并不可取。   后查资料得知STM32中还可以利用DMA的方式实现串口的任意长度数据的接收,故开始学习DMA+串口接收任意长度的数据这种方式。 cubeMX软件配置过程 首先,第一步都是进行时钟树的配置,配置好系统的时钟,不同的芯片配置不同的时钟频率,如图。
[单片机]
STM32—cubeMX+DMA+USART <font color='red'>接收</font>任意长度的<font color='red'>数据</font>
STM32串口接收数据保存到数组——遇到问题及解决方法
遇到问题: PC端发送给stm32的字符串数据,只有最后一个字节被保存在数组的首地址,其他数据都丢失,而且数组中只有首地址有数据。 比如我用PC给stm32发送abcd,返回到PC串口助手的数据只有d,查看数组发现里面数据 。 解决方案: 错误的原因是,在中断服务函数中,每次来数据时,数组地址 t++ 的变量 t 被我定义成了局部变量,导致每次来中断时,t都被赋值为0,才导致只有数组首地址有数据,且之前的数据都被覆盖。 解决办法,把数组地址 t++ 的变量 t 定义为全局变量就ok了
[单片机]
STM32<font color='red'>串口</font><font color='red'>接收</font><font color='red'>数据</font>保存到数组——遇到问题及解决方法
小广播
设计资源 培训 开发板 精华推荐

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

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

换一换 更多 相关热搜器件

 
EEWorld订阅号

 
EEWorld服务号

 
汽车开发圈

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