stm32之USART/UART应用实例(详细)

  • 硬件:STM32F103VCT6
  • 开发工具:Keil uVision4
  • 下载调试工具:ARM仿真器

USART与UART的区别:
usart:全双工通用同步/异步串行收发器
uart:全双工通用异步串行收发器
usart支持同步通讯时比uart多一条时钟线,usart用于异步通讯时跟uart一样。

单工: 只能单向传输数据
半双工:某一个时刻只能接受或者发送的双向传输数据
全双工:可以同时进行接收和发送数据

异步:发送方发送一帧数据后不需要接受方应答继续发送下一帧
同步:发送方发送完一帧数据后需要等待接受方应答才能发下一帧

串行:每一传输按位进行,在一个导线上一位一位一次传输
并行:每次传输按两位或者两位以上进行,通过几根导线同一时刻可以进行多位传输

波特率:一秒钟内传输多少位数据,单位:Bps(byte 字节每秒),发送设备与接收设备的波特率必须一致。

USART/UART是一种通用的标准接口,根据导线的电压等不同也分为很多的同类,比如: RS485, RS422,RS232

stm32的usart编程应用的步骤:
1.串口设备的的初始化
2.串口中断优先级的定义
3.发送函数USART_SendData、接收函数USART_ReceiveData

第二步是为了当串口接收到数据的时候系统会从主程序中跳转到中断里面去,然后再进行数据的读取,当然如果少了这一步也是可以通讯的,方法就是在主函数中不断查询是否收到数据,如此会占用很多资源,所以在所用的芯片有中断的情况下最好使用中断。

void USART1_Config(unsigned int bound)
{
	GPIO_InitTypeDef GPIO_InitStructure;
	USART_InitTypeDef USART_InitStructure;
	
	/*使能端口时钟*/
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1 | RCC_APB2Periph_GPIOA, ENABLE);
	
    /*USART1-TX输出脚配置*/
	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9;
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP; //复用推挽输出
	GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
	GPIO_Init(GPIOA, &GPIO_InitStructure);
	
    /*USART1-RX接收脚配置*/
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10;
	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;//一帧中传输或者接收的数据位数(8位/9位)
	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_ITConfig(USART1, USART_IT_RXNE, ENABLE);
	/*配置好参数之后,使能USART1外设*/
	USART_Cmd(USART1, ENABLE); 
	/*USART1终端优先级的配置*/
}	
void NVIC_Configuration(void)
{
	NVIC_InitTypeDef NVIC_InitStructure; 
	/*设置优先级分组*/
	NVIC_PriorityGroupConfig(NVIC_PriorityGroup_0);

	NVIC_InitStructure.NVIC_IRQChannel = USART1_IRQn;//指定IRQ通道	 
	NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 1;//指定先占优先级
	NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0;//从优先级
	
	NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE; //定义的IRQ是被使能还是失能
	NVIC_Init(&NVIC_InitStructure);
}

/*重定向c库函数printf到USART1,调用printf会将需要打印的数据从串口发送出去*/
int fputc(int ch, FILE *f)
{
	USART_SendData(USART1, (unsigned char) ch);
	while( USART_GetFlagStatus(USART1,USART_FLAG_TC)!= SET);	 //等到USART1发送完成
	return (ch);
}

/*串口发送一个字符*/
void USART_SendChar(USART_TypeDef* pUSARTx, uint8_t c)
{
    USART_SendData(pUSARTx, c);
    while (USART_GetFlagStatus(pUSARTx, USART_FLAG_TXE) == RESET);
}
/
*串口发送一个字符串*/
void USART_SendString(USART_TypeDef* pUSARTx, char* str)
{
    uint32_t n = 0;
 
    while (*(str + n) != '\0')
    {
        USART_SendChar(pUSARTx, *(str + n));
        n++;
    }
    while (USART_GetFlagStatus(pUSARTx, USART_FLAG_TXE) == RESET);
}

int main(void)
{
	USART1_Config(9600);
	NVIC_Configuration();
	printf(“********************************”);
	printf(“hello ward!\r\n”);
	printf(“********************************”);
	While(1){} //阻止程序结束
	
}

还要在stm32f10x_it.c(中断函数一般都会放在这里)文件里面实现串口中断的处理函数:

void USART1_IRQHandler(void)
{
	unsigned char code;
	if(USART_GetITStatus(USART1, USART_IT_RXNE) != RESET)
	{ 	
	    code=USART_ReceiveData(USART1);
	  	printf("%c",code);    //将接受到的数据直接返回打印
	}
}

完成以上步骤,接好线就可以在电脑上使用串口调试助手进行调试,stm32会将所接收到的数据从串口再发送出去。
重定向c库的printf函数还需要在项目设置里勾选:
这里写图片描述

仅供参考,错误之处以及不足之处还望多多指教

  • 14
    点赞
  • 81
    收藏
    觉得还不错? 一键收藏
  • 5
    评论
下面是一个基于STM32F407芯片的USART2串口DMA乒乓缓冲实例。 首先,需要在CubeMX中配置USART2和DMA。USART2的波特率、数据位、停止位等参数可以根据实际需求进行配置。DMA的通道、模式、传输方向、数据大小等参数也需要配置。在配置DMA时,需要注意设置DMA的循环模式,以实现乒乓缓冲。 接下来,可以使用以下代码实现USART2的DMA乒乓缓冲: ```c #include "stm32f4xx_hal.h" #define BUFFER_SIZE 128 // 缓冲区大小 uint8_t usart2_rx_buffer[2][BUFFER_SIZE]; // 乒乓缓冲区 uint8_t current_buffer_index = 0; // 当前使用的缓冲区索引 void HAL_UART_RxCpltCallback(UART_HandleTypeDef *huart) { if(huart == &huart2) // 判断是USART2接收中断 { HAL_UART_Receive_DMA(&huart2, usart2_rx_buffer[current_buffer_index], BUFFER_SIZE); // 启动下一次DMA传输 current_buffer_index = (current_buffer_index + 1) % 2; // 切换缓冲区 } } int main(void) { HAL_Init(); SystemClock_Config(); MX_GPIO_Init(); MX_DMA_Init(); MX_USART2_UART_Init(); HAL_UART_Receive_DMA(&huart2, usart2_rx_buffer[current_buffer_index], BUFFER_SIZE); // 启动第一次DMA传输 while(1) { // 处理接收到的数据 if(current_buffer_index == 0) { // 处理 usart2_rx_buffer[1] 中的数据 } else { // 处理 usart2_rx_buffer[0] 中的数据 } } } ``` 在上面的代码中,当USART2接收到数据时,会触发HAL_UART_RxCpltCallback()回调函数。在该回调函数中,会启动下一次DMA传输,并切换缓冲区。在主程序中,可以根据current_buffer_index的值来判断当前使用的缓冲区,以处理接收到的数据。 需要注意的是,在处理接收到的数据时,应该尽可能快地将数据从缓冲区中取出,以避免缓冲区溢出。如果处理速度较慢,可以考虑增大缓冲区的大小。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 5
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值