STM32单片机开发-UART&USART串口收发(二)不定长数据收发DMA串口收发-STM32CubeMx项目

目录

一、概述

二、程序实现

三、不定长数据收发

四、总结 


一、概述

        本篇文章我们进入STM32串口收发功能第二部分的教学,主要讲解DMA的收发模式,源码将在最后免费提供

        开发工具:STM32CubeMx、STM32开发板(型号不限)

        基础文章:STM32单片机开发-UART&USART串口收发-STM32CubeMx项目生成


二、程序实现

        首先我们继续打开上次的STM32CubeMx工程文件。

        打开USART1的DMA设置,按照下图添加:

        随后我们重新生成并打开文件。

        代码不需要做更多改变,在原有基础上,仅是将函数名最后的IT变为DMA。

uint8_t receive[10];
void HAL_UART_RxCpltCallback(UART_HandleTypeDef *huart)
{
	HAL_UART_Transmit_DMA(&huart1,receive,10);//发送函数
	HAL_UART_Receive_DMA(&huart1,receive,10);//接收函数
}
/* USER CODE END 0 */

/**
  * @brief  The application entry point.
  * @retval int
  */
int main(void)
{
  /* USER CODE BEGIN 1 */

  /* USER CODE END 1 */

  /* MCU Configuration--------------------------------------------------------*/

  /* Reset of all peripherals, Initializes the Flash interface and the Systick. */
  HAL_Init();

  /* USER CODE BEGIN Init */

  /* USER CODE END Init */

  /* Configure the system clock */
  SystemClock_Config();

  /* USER CODE BEGIN SysInit */

  /* USER CODE END SysInit */

  /* Initialize all configured peripherals */
  MX_GPIO_Init();
  MX_DMA_Init();
  MX_USART1_UART_Init();
  /* USER CODE BEGIN 2 */
	
  /* USER CODE END 2 */

  /* Infinite loop */
  /* USER CODE BEGIN WHILE */
	HAL_UART_Receive_DMA(&huart1,receive,10);//接收函数
	while (1)
  {
    /* USER CODE END WHILE */

    /* USER CODE BEGIN 3 */
		
		
		
  }
  /* USER CODE END 3 */
}

        我们编译并调试该代码,打开串口调试助手,关于调试的方法,在上一篇文章中已经介绍。

         功能实现,但目前又一个问题,如果我们发送五位或者规定位数以下的数据,程序则不会做出反应,需要达到位数之后才能做出反馈。


        因此,我们需要换一种写法。

三、不定长数据收发

        接收不定长数据我们需要使用空闲中断,顾名思义,当串口处于空闲状态,即进入中断,便可以实现不定长接收。

        我们需要在主函数中添加初始化语句:

__HAL_UART_ENABLE_IT(&huart1,UART_IT_IDLE);//使能空闲中断

        随后,我们在USART1的中断处理函数(即:USART1_IRQHandler)中,写入我们自己的空闲中断处理函数。

        我们首先打开stm32f1xx_it.c文件,定义外部变量(即我们的数据数组):

        随后我们在中断处理函数中写入如下代码:

void USART1_IRQHandler(void)
{
  /* USER CODE BEGIN USART1_IRQn 0 */	
	if(__HAL_UART_GET_FLAG(&huart1,UART_FLAG_IDLE)==SET)//判断是不是空闲中断
	{
		__HAL_UART_CLEAR_IDLEFLAG(&huart1);//清 中断标志位
		HAL_UART_DMAStop(&huart1);//停止DMA接收
		uint8_t len=255-__HAL_DMA_GET_COUNTER(huart1.hdmarx);//求已接收数据的长度
		HAL_UART_Transmit_DMA(&huart1,receive,len);//发送接收的数据
		HAL_UART_Receive_DMA(&huart1,receive,255);//继续等待接收
	}
  /* USER CODE END USART1_IRQn 0 */
  HAL_UART_IRQHandler(&huart1);
  /* USER CODE BEGIN USART1_IRQn 1 */

  /* USER CODE END USART1_IRQn 1 */
}

        最近,我们来看一下main.c文件中的代码结构,仅是在主函数开启了一次接收而已。

        编译并烧写程序,打开串口调试助手。

        效果完美实现。


四、总结 

        本篇文章主要讲解不定长数据的接收与发送,至此,STM32的串口通信已经全部完成,最后,附上源码供大家学习,有疑问或意见欢迎在评论区留言。

         源码:STM32串口不定长数据通信源码-文章配套资源

  • 1
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
STM32 HAL库中,使用DMA实现串口数据收发也是非常简单的,只需调用相应的HAL库函数即可。以下是一个基本的示例: ```c #include "stm32f10x.h" #include "stm32f10x_hal.h" #define USART_RX_BUFFER_SIZE 256 volatile uint8_t usart_rx_buffer[USART_RX_BUFFER_SIZE]; volatile uint16_t usart_rx_write_index = 0; volatile uint16_t usart_rx_read_index = 0; UART_HandleTypeDef huart1; void usart_init(void) { GPIO_InitTypeDef GPIO_InitStruct; __HAL_RCC_GPIOA_CLK_ENABLE(); __HAL_RCC_USART1_CLK_ENABLE(); __HAL_RCC_DMA1_CLK_ENABLE(); GPIO_InitStruct.Pin = GPIO_PIN_9; GPIO_InitStruct.Mode = GPIO_MODE_AF_PP; GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_HIGH; HAL_GPIO_Init(GPIOA, &GPIO_InitStruct); GPIO_InitStruct.Pin = GPIO_PIN_10; GPIO_InitStruct.Mode = GPIO_MODE_INPUT; GPIO_InitStruct.Pull = GPIO_NOPULL; HAL_GPIO_Init(GPIOA, &GPIO_InitStruct); huart1.Instance = USART1; huart1.Init.BaudRate = 115200; huart1.Init.WordLength = UART_WORDLENGTH_8B; huart1.Init.StopBits = UART_STOPBITS_1; huart1.Init.Parity = UART_PARITY_NONE; huart1.Init.Mode = UART_MODE_TX_RX; huart1.Init.HwFlowCtl = UART_HWCONTROL_NONE; HAL_UART_Init(&huart1); HAL_NVIC_SetPriority(DMA1_Channel5_IRQn, 0, 0); HAL_NVIC_EnableIRQ(DMA1_Channel5_IRQn); HAL_UART_Receive_DMA(&huart1, (uint8_t*)usart_rx_buffer, USART_RX_BUFFER_SIZE); } void HAL_UART_RxCpltCallback(UART_HandleTypeDef *huart) { usart_rx_write_index = USART_RX_BUFFER_SIZE - huart->hdmarx->Instance->CNDTR; while (usart_rx_read_index != usart_rx_write_index) { uint8_t data = usart_rx_buffer[usart_rx_read_index]; usart_rx_read_index = (usart_rx_read_index + 1) % USART_RX_BUFFER_SIZE; // 处理接收到的数据 // ... if (usart_rx_read_index == usart_rx_write_index) { break; } } HAL_UART_Receive_DMA(&huart1, (uint8_t*)usart_rx_buffer, USART_RX_BUFFER_SIZE); } int main(void) { usart_init(); while (1) { // 主循环中不需要处理接收到的数据 } } void DMA1_Channel5_IRQHandler(void) { HAL_DMA_IRQHandler(huart1.hdmarx); } ``` 在上述代码中,我们使用了HAL库函数 `HAL_UART_Receive_DMA` 来启动USART1的DMA接收,同时使用了HAL库函数 `HAL_UART_RxCpltCallback` 来处理接收到的数据。需要注意的是,我们需要在 `HAL_UART_RxCpltCallback` 中再次调用 `HAL_UART_Receive_DMA` 来启动下一轮DMA接收。 此外,在HAL库中,我们也需要在主函数中调用 `HAL_NVIC_EnableIRQ` 函数来使能DMA中断,并且需要在中断处理函数中调用 `HAL_DMA_IRQHandler` 函数来处理DMA中断。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

智慧农行-推动传统农业数字化转型

打赏支持获得问题解答机会

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

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

打赏作者

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

抵扣说明:

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

余额充值