STM32F103 UART DMA接收任意长度数据

使用STM32CubeMX 配置初始化模板,勾选UART 发送接收 DMA ,以及中断,TX设置单次模式,RX设置循环接收(属于大于缓冲区大小,将会覆盖前面的)

生成的工程uart初始化增加 打开空闲中断 以及 设置DMA接收缓冲 代码

打开空闲中断 __HAL_UART_ENABLE_IT(&huart1, UART_IT_IDLE);

设置DMA接收缓冲 HAL_UART_Receive_DMA(&huart1, (uint8_t *)aRxBuffer, RXBUFFERSIZE);

然后在stm32f1xx_it.c里面的 USART1_IRQHandler 上加入调用DMA空闲中断,

空闲中断里面 获取长度(HAL库是获取剩余大小) ,停止传输重新启动DMA(否则下次接收将会在第一次接收后面)

void USART1_DMAHandler(void)
{
    if (RESET != __HAL_UART_GET_FLAG(&huart1, UART_FLAG_IDLE))
    {
        __HAL_UART_CLEAR_IDLEFLAG(&huart1);
        uint32_t len = RXBUFFERSIZE - __HAL_DMA_GET_COUNTER(&hdma_usart1_rx);
        
        //停止当前DMA传输
        HAL_UART_DMAStop(&huart1);
        //重新配置DMA传输,从头传输
        HAL_UART_Receive_DMA(&huart1, (uint8_t *)aRxBuffer, RXBUFFERSIZE);
        
        printf("uartCnt = %d\n", len);
        HAL_UART_Transmit_DMA(&huart1, (uint8_t *)aRxBuffer, len);
    }
}

stm32f1xx_it.c 

/**
  * @brief This function handles USART1 global interrupt.
  */
void USART1_IRQHandler(void)
{
  /* USER CODE BEGIN USART1_IRQn 0 */
  extern void USART1_DMAHandler(void);
  USART1_DMAHandler();
  /* USER CODE END USART1_IRQn 0 */
  HAL_UART_IRQHandler(&huart1);
  /* USER CODE BEGIN USART1_IRQn 1 */

  /* USER CODE END USART1_IRQn 1 */
}

usart.c

/* USER CODE BEGIN Header */
/**
  ******************************************************************************
  * @file    usart.c
  * @brief   This file provides code for the configuration
  *          of the USART instances.
  ******************************************************************************
  * @attention
  *
  * Copyright (c) 2024 STMicroelectronics.
  * All rights reserved.
  *
  * This software is licensed under terms that can be found in the LICENSE file
  * in the root directory of this software component.
  * If no LICENSE file comes with this software, it is provided AS-IS.
  *
  ******************************************************************************
  */
/* USER CODE END Header */
/* Includes ------------------------------------------------------------------*/
#include "usart.h"

/* USER CODE BEGIN 0 */
#include <stdio.h>

#define RXBUFFERSIZE        (128)
uint32_t aRxBuffer[RXBUFFERSIZE];

/* USER CODE END 0 */

UART_HandleTypeDef huart1;
DMA_HandleTypeDef hdma_usart1_rx;
DMA_HandleTypeDef hdma_usart1_tx;

/* USART1 init function */

void MX_USART1_UART_Init(void)
{

  /* USER CODE BEGIN USART1_Init 0 */

  /* USER CODE END USART1_Init 0 */

  /* USER CODE BEGIN USART1_Init 1 */

  /* USER CODE END USART1_Init 1 */
  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;
  huart1.Init.OverSampling = UART_OVERSAMPLING_16;
  if (HAL_UART_Init(&huart1) != HAL_OK)
  {
    Error_Handler();
  }
  /* USER CODE BEGIN USART1_Init 2 */
  
    __HAL_UART_ENABLE_IT(&huart1, UART_IT_IDLE);
    if(HAL_UART_Receive_DMA(&huart1, (uint8_t *)aRxBuffer, RXBUFFERSIZE) != HAL_OK)
    {
      Error_Handler();
    }
//    if(HAL_UART_Transmit_DMA(&huart1, (uint8_t*)aRxBuffer, RXBUFFERSIZE)!= HAL_OK)
//    {
//      Error_Handler();
//    }
    
  /* USER CODE END USART1_Init 2 */

}

void HAL_UART_MspInit(UART_HandleTypeDef* uartHandle)
{

  GPIO_InitTypeDef GPIO_InitStruct = {0};
  if(uartHandle->Instance==USART1)
  {
  /* USER CODE BEGIN USART1_MspInit 0 */

  /* USER CODE END USART1_MspInit 0 */
    /* USART1 clock enable */
    __HAL_RCC_USART1_CLK_ENABLE();

    __HAL_RCC_GPIOA_CLK_ENABLE();
    /**USART1 GPIO Configuration
    PA9     ------> USART1_TX
    PA10     ------> USART1_RX
    */
    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);

    /* USART1 DMA Init */
    /* USART1_RX Init */
    hdma_usart1_rx.Instance = DMA1_Channel5;
    hdma_usart1_rx.Init.Direction = DMA_PERIPH_TO_MEMORY;
    hdma_usart1_rx.Init.PeriphInc = DMA_PINC_DISABLE;
    hdma_usart1_rx.Init.MemInc = DMA_MINC_ENABLE;
    hdma_usart1_rx.Init.PeriphDataAlignment = DMA_PDATAALIGN_BYTE;
    hdma_usart1_rx.Init.MemDataAlignment = DMA_MDATAALIGN_BYTE;
    hdma_usart1_rx.Init.Mode = DMA_CIRCULAR;
    hdma_usart1_rx.Init.Priority = DMA_PRIORITY_HIGH;
    if (HAL_DMA_Init(&hdma_usart1_rx) != HAL_OK)
    {
      Error_Handler();
    }

    __HAL_LINKDMA(uartHandle,hdmarx,hdma_usart1_rx);

    /* USART1_TX Init */
    hdma_usart1_tx.Instance = DMA1_Channel4;
    hdma_usart1_tx.Init.Direction = DMA_MEMORY_TO_PERIPH;
    hdma_usart1_tx.Init.PeriphInc = DMA_PINC_DISABLE;
    hdma_usart1_tx.Init.MemInc = DMA_MINC_ENABLE;
    hdma_usart1_tx.Init.PeriphDataAlignment = DMA_PDATAALIGN_BYTE;
    hdma_usart1_tx.Init.MemDataAlignment = DMA_MDATAALIGN_BYTE;
    hdma_usart1_tx.Init.Mode = DMA_NORMAL;
    hdma_usart1_tx.Init.Priority = DMA_PRIORITY_HIGH;
    if (HAL_DMA_Init(&hdma_usart1_tx) != HAL_OK)
    {
      Error_Handler();
    }

    __HAL_LINKDMA(uartHandle,hdmatx,hdma_usart1_tx);

    /* USART1 interrupt Init */
    HAL_NVIC_SetPriority(USART1_IRQn, 0, 0);
    HAL_NVIC_EnableIRQ(USART1_IRQn);
  /* USER CODE BEGIN USART1_MspInit 1 */

  /* USER CODE END USART1_MspInit 1 */
  }
}

void HAL_UART_MspDeInit(UART_HandleTypeDef* uartHandle)
{

  if(uartHandle->Instance==USART1)
  {
  /* USER CODE BEGIN USART1_MspDeInit 0 */

  /* USER CODE END USART1_MspDeInit 0 */
    /* Peripheral clock disable */
    __HAL_RCC_USART1_CLK_DISABLE();

    /**USART1 GPIO Configuration
    PA9     ------> USART1_TX
    PA10     ------> USART1_RX
    */
    HAL_GPIO_DeInit(GPIOA, GPIO_PIN_9|GPIO_PIN_10);

    /* USART1 DMA DeInit */
    HAL_DMA_DeInit(uartHandle->hdmarx);
    HAL_DMA_DeInit(uartHandle->hdmatx);

    /* USART1 interrupt Deinit */
    HAL_NVIC_DisableIRQ(USART1_IRQn);
  /* USER CODE BEGIN USART1_MspDeInit 1 */

  /* USER CODE END USART1_MspDeInit 1 */
  }
}

/* USER CODE BEGIN 1 */

int fputc(int ch, FILE *f)
{
    while ((USART1->SR & 0X40) == 0);

    USART1->DR = (uint8_t)ch;
    return ch;
}

/* uart IDLE DMAHandler interrupt */
void USART1_DMAHandler(void)
{
    if (RESET != __HAL_UART_GET_FLAG(&huart1, UART_FLAG_IDLE))
    {
        __HAL_UART_CLEAR_IDLEFLAG(&huart1);
        uint32_t len = RXBUFFERSIZE - __HAL_DMA_GET_COUNTER(&hdma_usart1_rx);
        
        //停止当前DMA传输
        HAL_UART_DMAStop(&huart1);
        //重新配置DMA传输,从头传输
        HAL_UART_Receive_DMA(&huart1, (uint8_t *)aRxBuffer, RXBUFFERSIZE);
        
        printf("uartCnt = %d\n", len);
        HAL_UART_Transmit_DMA(&huart1, (uint8_t *)aRxBuffer, len);
    }
}

/* USER CODE END 1 */

  • 2
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 在STM32F103系列微控制器中,可以使用DMA(直接内存访问)来实现串口接收不定长数据。下面是实现的大致步骤: 1. 首先,需要配置串口进行接收: - 初始化串口,并设置波特率、数据位、停止位等参数。 - 打开串口接收中断,以便在接收数据时触发中断。 - 开启DMAUART接收通道。 2. 配置DMA: - 设置DMA通道的源地址为串口数据寄存器地址。 - 设置DMA通道的目的地址为接收数据的存储位置,可以是单个变量或数组。 - 设置DMA数据传输的长度为最大接收数据长度。 - 配置DMA通道为循环模式,以便在接收数据时自动重新启动DMA传输。 3. 在串口接收中断的处理函数中,可以在每次接收数据时检查DMA是否已经接收到足够的数据。可以根据接收到的数据情况进行进一步处理,例如打印数据或进行其他操作。 需要注意的是,在使用DMA接收不定长数据时,需要确保DMA传输的长度足够长,以便接收到的数据不会超出DMA缓冲区的范围。此外,为了避免数据丢失或覆盖,建议在处理接收到的数据之前判断DMA是否已经完成传输。 总的来说,通过配置USART接收中断和DMA通道,可以实现STM32F103系列微控制器的串口DMA接收不定长数据的功能。这种方法可以在保证低功耗的同时,提高系统的效率和响应速度。 ### 回答2: 在STM32F103系列微控制器中,使用DMA(直接存储器访问)来接收不定长数据是一种高效和可靠的方法。 首先,我们需要配置串口通信的DMA接收功能。通过设置串口的DMA接收使能位(RXDMAEN),可以使用DMA接收数据。然后,通过配置DMA控制器的通道、缓冲区和传输长度等参数,将数据从串口接收DMA缓冲区中。 在不定长数据接收的情况下,可以通过设置DMA传输完成中断(TC)来判断数据是否接收完整。每当DMA接收到指定长度数据时,将触发一个DMA传输完成中断,在中断服务程序中可以处理接收到的数据以及进行后续操作。 另外,为了区分每一帧数据的开始和结束,可以通过给数据添加开始标志和结束标志的方式进行帧同步。当接收到一个完整的帧后,可以通过软件逻辑进行数据处理和分析。 需要注意的是,不定长数据接收可能存在干扰和错误。为了提高接收数据的可靠性,可以通过一些策略来进行数据完整性检查和错误处理,如校验和检验、超时机制等。 综上所述,使用DMA接收不定长数据需要配置串口的DMA接收使能位,并设置DMA控制器的相关参数。通过中断服务程序和逻辑判断,可以实现对不定长数据接收和处理。 ### 回答3: STM32F103是一种具有DMA(直接内存访问)功能的微控制器,它可以用于实现串口接收不定长数据。在进行串口DMA接收不定长数据之前,需要先配置串口和DMA的相关寄存器。 首先,需要使能串口的DMA接收模式。可以通过设置串口控制寄存器CR3的位DMA-RX使能。然后,将DMA的通道选择为串口的接收通道。 接下来,需要配置DMA的相关寄存器,包括源和目的地址、传输长度数据流向等。对于串口DMA接收,源地址通常是串口的数据寄存器,目的地址是存储接收数据的缓冲区。传输长度可设置为一个较大的值,以确保能够接收到不定长的数据数据流向应设置为从外设到内存。 然后,需要配置DMA的传输模式,包括循环模式和自动请求使能。循环模式可以确保DMA接收完指定长度数据后,自动重新开始传输。自动请求使能则用于自动触发DMA传输。 最后,可以通过使能DMA接收完成中断来对接收到的数据进行处理。当DMA接收完成后,会触发DMA的中断,可以在中断函数中处理接收到的数据,比如打印输出或进行其他操作。 综上所述,通过配置串口和DMA的相关寄存器,并处理DMA接收完成中断,就可以实现STM32F103串口DMA接收不定长数据

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值