STM32F407ZGT6 串口1(USART1)接收DMA接收数据只有一次

因为这个bug,搞了一整天。串口1接收,使用DMA,如果配置为 NORMAL 模式,只能接收一次,后面数据不变。如果使用 CIRCLAR 模式,能够连续接收,但是不正确。

结合多方博客,终于解决问题,稳如老狗。

使用 CIRCLAR 模式,中断配置里解锁等操作,附上文件供参考,实现串口1 DMA收发传输。

#include "debug.h"

/* 
 * 外部发来的期望位置
 */
#define EX_DATA_LEN 32
 
struct XYZ_t des_pos_ = {0,0,0};
uint8_t get_cmd = 0;												// 0:处理完数据等待获取数据  1:已获取数据待处理
uint8_t pos_vel_buf[EX_DATA_LEN] = {'\0'};

/**************************************************************************************************************/
/*****************************************  PART1:串口接收底层配置	  *****************************************/

UART_HandleTypeDef usart1; 														//UART句柄
DMA_HandleTypeDef  usart1_tx_dma;      								//DMA句柄
DMA_HandleTypeDef usart1_rx_dma;

/* 
 * 初始化IO 使用串口1 		
 * bound:波特率				
 */
void DebugInit(uint32_t bound)
{
  //GPIO端口设置
	GPIO_InitTypeDef GPIO_Initure;
		
	DEBUG_ENCLK();																			// 使能时钟
	
	GPIO_Initure.Pin				= DEBUG_PIN;								// Pin
	GPIO_Initure.Mode				= GPIO_MODE_AF_PP;					// 复用推挽输出
	GPIO_Initure.Pull				= GPIO_PULLUP;							// 上拉
	GPIO_Initure.Speed				= GPIO_SPEED_FAST;					// 高速
	GPIO_Initure.Alternate			= GPIO_AF7_USART1;					// 复用为USART1
	HAL_GPIO_Init(DEBUG_GPIO,&GPIO_Initure);	   				// 初始化
			
	// UART1 TX DMA配置--DMA2 Stream7 Channel4
	__HAL_RCC_DMA2_CLK_ENABLE();																						//DMA2时钟使能	
	__HAL_LINKDMA(&usart1,hdmatx,usart1_tx_dma);    												//将DMA与USART1联系起来(发送DMA)
	
	usart1_tx_dma.Instance				=	DMA2_Stream7;                         //数据流选择
	usart1_tx_dma.Init.Channel			=	DMA_CHANNEL_4;                        //通道选择
	usart1_tx_dma.Init.Direction		=	DMA_MEMORY_TO_PERIPH;             		//存储器到外设
	usart1_tx_dma.Init.PeriphInc		=	DMA_PINC_DISABLE;                 		//外设非增量模式
	usart1_tx_dma.Init.MemInc				=	DMA_MINC_ENABLE;                     	//存储器增量模式
	usart1_tx_dma.Init.PeriphDataAlignment	=	DMA_PDATAALIGN_BYTE;    			//外设数据长度:8位
	usart1_tx_dma.Init.MemDataAlignment		=	DMA_MDATAALIGN_BYTE;      		//存储器数据长度:8位
	usart1_tx_dma.Init.Mode					=	DMA_NORMAL;                           //外设普通模式
	usart1_tx_dma.Init.Priority			=	DMA_PRIORITY_MEDIUM;               		//中等优先级
	usart1_tx_dma.Init.FIFOMode			=	DMA_FIFOMODE_DISABLE;              
	usart1_tx_dma.Init.FIFOThreshold    =	DMA_FIFO_THRESHOLD_FULL;      
	usart1_tx_dma.Init.MemBurst			=	DMA_MBURST_SINGLE;                 		//存储器突发单次传输
	usart1_tx_dma.Init.PeriphBurst	    =	DMA_PBURST_SINGLE;              			//外设突发单次传输
	
	HAL_DMA_DeInit(&usart1_tx_dma);   
	HAL_DMA_Init(&usart1_tx_dma);	
	
	// UAR1 RX DMA 
	usart1_rx_dma.Instance 			= DMA2_Stream2;
	usart1_rx_dma.Init.Channel 		= DMA_CHANNEL_4;
	usart1_rx_dma.Init.Direction 	= DMA_PERIPH_TO_MEMORY;
	usart1_rx_dma.Init.PeriphInc 	= DMA_PINC_DISABLE;
	usart1_rx_dma.Init.MemInc 		= DMA_MINC_ENABLE;
	usart1_rx_dma.Init.PeriphDataAlignment 	= DMA_PDATAALIGN_BYTE;
	usart1_rx_dma.Init.MemDataAlignment 	= DMA_MDATAALIGN_BYTE;
	usart1_rx_dma.Init.Mode 		= DMA_CIRCULAR;
	usart1_rx_dma.Init.Priority 	= DMA_PRIORITY_LOW;
	usart1_rx_dma.Init.FIFOMode 	= DMA_FIFOMODE_DISABLE;
	
	HAL_DMA_DeInit(&usart1_rx_dma);
	HAL_DMA_Init(&usart1_rx_dma);
	__HAL_LINKDMA(&usart1,hdmarx,usart1_rx_dma);
	
	//UART 初始化设置	
	usart1.Instance					= USART1;												// USART1
	usart1.Init.BaudRate			= bound;												// 波特率
	usart1.Init.WordLength			= UART_WORDLENGTH_8B;   				// 字长为8位数据格式
	usart1.Init.StopBits			= UART_STOPBITS_1;	    				// 一个停止位
	usart1.Init.Parity				= UART_PARITY_NONE;							// 无奇偶校验位
	usart1.Init.HwFlowCtl			= UART_HWCONTROL_NONE;  				// 无硬件流控
	usart1.Init.Mode				= UART_MODE_TX_RX;							// 收发模式
	usart1.Init.OverSampling 		= UART_OVERSAMPLING_16;
	HAL_UART_Init(&usart1);					    											// HAL_UART_Init()会使能UART1
	
	HAL_NVIC_EnableIRQ(USART1_IRQn);													// 使能USART1中断通道
	HAL_NVIC_SetPriority(USART1_IRQn, 5, 0);									// 优先级	
	
	printf("Usart1 init success\r\n");
}

/* 
 * USART1 DMA 中断函数		
 */
void DMA2_Stream2_IRQHandler(void)
{
	HAL_DMA_IRQHandler(&usart1_rx_dma);
}

/* 
 * 串口1中断服务函数		
 */
void USART1_IRQHandler(void)
{
	HAL_UART_IRQHandler(&usart1);
	
	// 不加这几句 USART1 DMA 有问题。如果 NORMAL模式只能接收一次,CIRCLAR 模式数据不正确
	usart1.State = HAL_UART_STATE_READY;
	usart1_rx_dma.State = HAL_DMA_STATE_READY;
	__HAL_UNLOCK(&usart1_rx_dma);

	if((__HAL_UART_GET_FLAG(&usart1,UART_FLAG_IDLE) != RESET))	// 如果线路空闲(接收完一帧)
	{ 
		__HAL_UART_CLEAR_IDLEFLAG(&usart1);												// 清除线路空闲标志
		get_cmd = 1;																							// 获取数据成功
	}
}


/* 
 * 连续打印数据任务						
 * 调试使用,很多数据定义为使用,不必纠结	
 */
void PrintDataTask(void *arg)
{
	printf("All the bugs will be killed\r\n");					
    DelayMs(20);
}


void GetDesirePosVelTask(void *arg)
{
	uint16_t len; 
	
	memset(pos_vel_buf, '\0', EX_DATA_LEN);
	
	__HAL_UART_ENABLE_IT(&usart1, UART_IT_IDLE);		// 开启串口空闲中断
    HAL_UART_Receive_DMA(&usart1, (uint8_t*)pos_vel_buf, EX_DATA_LEN);
	
	
	while (1)
	{
		if (get_cmd == 1)
		{
			get_cmd = 0;
			
			des_pos_.x = (float)Str2Double(&pos_vel_buf[2], 3) * 0.1f;
			des_pos_.y = (float)Str2Double(&pos_vel_buf[5], 3) * 0.1f;
			des_pos_.z = (float)Str2Double(&pos_vel_buf[8], 3) * 0.1f;
			
			printf("recv: %s, %.1f, %.1f, %.1f", pos_vel_buf, des_pos_.x, des_pos_.y, des_pos_.z);	// 启动接收

			memset(pos_vel_buf, '\0', EX_DATA_LEN);
			DelayMs(10);
			HAL_UART_Receive_DMA(&usart1, (uint8_t*)pos_vel_buf, EX_DATA_LEN);
		}
	
		else
		{
			DelayMs(10);
		}
	}
}

struct XYZ_t GetExDesirePosApi(void)
{
	return des_pos_;
}

/********************************************************************/
/* 							一些暂时不用关心的函数					*/
#if 1
#pragma import(__use_no_semihosting)   
//标准库需要的支持函数                 
struct __FILE 
{ 
	int handle; 
}; 

FILE __stdout;       
//定义_sys_exit()以避免使用半主机模式    
void _sys_exit(int x) 
{ 
	x = x; 
} 
//重定义fputc函数 
//注意,读取USARTx->SR能避免莫名其妙的错误   	
int fputc(int ch, FILE *f)
{ 	
	while((USART1->SR&0X40)==0);//循环发送,直到发送完毕   
	USART1->DR = (uint8_t) ch;      
	return ch;
}
#endif 
好的,以下是基于HAL库实现的代码示例: ```c #include "main.h" #include "stdio.h" #include "string.h" /* OLED 显示屏的 I2C 地址 */ #define OLED_ADDR 0x78 /* OLED 显示屏的宽度和高度 */ #define OLED_WIDTH 128 #define OLED_HEIGHT 64 /* OLED 显示屏的初始化命令 */ uint8_t OLED_Init_Cmd[] = { 0xAE, // 关闭显示 0xD5, 0x80, // 设置时钟分频因子,震荡器频率 0xA8, 0x3F, // 设置驱动路数 0xD3, 0x00, // 设置显示偏移 0x40, // 设置显示开始行 [5:0] 0x8D, 0x14, // 使能电荷泵 0x20, 0x00, // 水平寻址模式 0xA0 | 0x1, // 设置段重定义 0xC8, // 设置扫描方向 0xDA, 0x12, // 设置 COM 硬件引脚配置 0x81, 0xCF, // 设置对比度 0xD9, 0xF1, // 设置预充电周期 0xDB, 0x40, // 设置 VcomH 电压倍率 0xA4, // 全局显示开启 0xA6, // 设置显示方式: 正常显示 0xAF // 打开显示 }; /* 串口数据缓冲区 */ uint8_t uart_buf[128]; /* OLED 显示缓冲区 */ uint8_t oled_buf[1024]; /* 串口 DMA 发送完成标志 */ volatile uint8_t uart_dma_tx_complete = 0; /* 串口 DMA 发送回调函数 */ void HAL_UART_TxCpltCallback(UART_HandleTypeDef *huart) { uart_dma_tx_complete = 1; } /* OLED I2C 发送命令函数 */ void OLED_Send_Cmd(uint8_t cmd) { uint8_t data[2]; data[0] = 0x00; // Co = 0, D/C# = 0 data[1] = cmd; while(HAL_I2C_Master_Transmit(&hi2c1, OLED_ADDR, data, sizeof(data), 1000) != HAL_OK); } /* OLED I2C 发送数据函数 */ void OLED_Send_Data(uint8_t *data, uint32_t size) { uint8_t buf[17]; buf[0] = 0x40; // Co = 0, D/C# = 1 while(size > 0) { uint32_t len = size > 16 ? 16 : size; memcpy(&buf[1], data, len); while(HAL_I2C_Master_Transmit(&hi2c1, OLED_ADDR, buf, len + 1, 1000) != HAL_OK); data += len; size -= len; } } /* OLED 显示初始化函数 */ void OLED_Init() { HAL_Delay(100); for(uint32_t i = 0; i < sizeof(OLED_Init_Cmd); i++) { OLED_Send_Cmd(OLED_Init_Cmd[i]); } } /* OLED 显示清屏函数 */ void OLED_Clear() { memset(oled_buf, 0x00, sizeof(oled_buf)); } /* OLED 显示字符函数 */ void OLED_Show_Char(uint8_t x, uint8_t y, char ch) { uint8_t *p = &oled_buf[x + y / 8 * OLED_WIDTH]; for(uint32_t i = 0; i < 8; i++) { if((font_8x16[(ch - ' ')*16 + i] & 0x80) != 0) { p[0] |= 0x01 << (y % 8); } else { p[0] &= ~(0x01 << (y % 8)); } if((font_8x16[(ch - ' ')*16 + i] & 0x40) != 0) { p[1] |= 0x01 << (y % 8); } else { p[1] &= ~(0x01 << (y % 8)); } p += OLED_WIDTH; y++; } } /* OLED 显示字符串函数 */ void OLED_Show_String(uint8_t x, uint8_t y, char *str) { while(*str != '\0') { OLED_Show_Char(x, y, *str); x += 8; str++; } } /* 主函数 */ int main(void) { /* 初始化 HAL 库 */ HAL_Init(); /* 配置系统时钟 */ SystemClock_Config(); /* 配置 GPIO 串口1 */ MX_GPIO_Init(); MX_USART1_UART_Init(); /* 配置 GPIO I2C1 */ MX_GPIO_Init(); MX_I2C1_Init(); /* OLED 显示初始化 */ OLED_Init(); /* OLED 显示清屏 */ OLED_Clear(); /* 串口 DMA 发送 */ while(1) { sprintf((char*)uart_buf, "Hello World!\r\n"); uart_dma_tx_complete = 0; HAL_UART_Transmit_DMA(&huart1, uart_buf, strlen((char*)uart_buf)); while(uart_dma_tx_complete == 0); OLED_Show_String(0, 0, (char*)uart_buf); OLED_Send_Data(oled_buf, sizeof(oled_buf)); HAL_Delay(1000); } } ``` 这段代码实现了串口1在 DMA 模式下输出信息给 OLED,具体功能包括: 1. OLED 的初始化、清屏、字符和字符串显示等; 2. 串口1 的 DMA 发送; 3. 串口1 DMA 发送完成后将数据显示在 OLED 上。 需要注意的是,这段代码使用了字库文件 `font_8x16.c`,需要用户自己添加字库文件和相关函数。此外,涉及到的 HAL 库函数需要用户根据实际情况进行修改。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

大强强小强强

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值