STM32F0系列串口超时中断DMA不定长数据接收

在STM32F1中接收不定长数据都是使用空闲中断,STM32F0也支持这个中断,同样也可以用来接收不定长数据,不过F0系列的串口功能更加丰富,本次的话就主要介绍一下串口超时中断,实现和空闲中断同样的功能。
因为ST目前主推的是CubeMX + HAL + LL的结构,用习惯了SPL很难习惯HAL那种裹粽子的结构。LL(Low Layer)库的话,和SPL是差不多的,都是直观的操作寄存器,这个比较符合单片机的开发理念,所以就立马换上了LL库,但是LL目前Bug还是比较多,其次就是没有提供例程,使得所有的所有的功能都要自己码出来,不过你可以参考SPL的方式,实现起来都是大同小异的,只不过是换了套API而已。

  • 硬件环境:STM32F051K6
  • 软件环境:MDK5.25, CubeMX 5.0.1, Cube MCU Package for STM32F0 Series 1.9.0

CubeMX 配置

串口配置.png

串口相关的部分只要打开串口的全局中断即可,着重于DMA配置,首先Add一个DMA Request为USART1_RX, 通道设置为DMA1 Channel 3,方向为外设到内存(本次是串口接收,所以是串口接收寄存器到RAM)优先级为中。

在生成代码之前还需要再设置一下,将所有的初始化放在独立的.c .h中,将生成的工程修改为MDK5,然后将所有的库都从HAL修改为LL,这边不在上图自行在软件中设置。

生成代码后有以下几处需要修改

   LL_USART_EnableRxTimeout(USART1);
   LL_USART_EnableIT_RTO(USART1);
   LL_USART_EnableDMAReq_RX(USART1);
   LL_DMA_EnableChannel(DMA1, LL_DMA_CHANNEL_3);
  • LL_USART_SetRxTimeout(USART1, 10); 设置超时时长,主要是设置USARTx_RTOR寄存器,其数值是以bit为单位,举例波特率是115200的话,那一个bit即为1/115200秒,如果设置为10就为(1/115200) * 10秒,时间的长度是跟随着波特率的变化而变化的,根据自己的需要来设置

  • LL_USART_EnableRxTimeout(USART1); 使能超时接收功能

  • LL_USART_EnableIT_RTO(USART1); 使能超时中断

  • LL_USART_EnableDMAReq_RX(USART1); 使能DMA接收

  • LL_DMA_EnableChannel(DMA1, LL_DMA_CHANNEL_3); 使能DMA通道

整个串口初始化代码

注意DMA在生成代码后还需要手动配置DMA的外设地址和内存地址。

void MX_USART1_UART_Init(void)
{
  LL_USART_InitTypeDef USART_InitStruct = {0};

  LL_GPIO_InitTypeDef GPIO_InitStruct = {0};
  /* Peripheral clock enable */
  LL_APB1_GRP2_EnableClock(LL_APB1_GRP2_PERIPH_USART1);
  
  LL_AHB1_GRP1_EnableClock(LL_AHB1_GRP1_PERIPH_GPIOA);
  /**USART1 GPIO Configuration  
  PA9   ------> USART1_TX
  PA10   ------> USART1_RX 
  */
  GPIO_InitStruct.Pin = LL_GPIO_PIN_9;
  GPIO_InitStruct.Mode = LL_GPIO_MODE_ALTERNATE;
  GPIO_InitStruct.Speed = LL_GPIO_SPEED_FREQ_HIGH;
  GPIO_InitStruct.OutputType = LL_GPIO_OUTPUT_PUSHPULL;
  GPIO_InitStruct.Pull = LL_GPIO_PULL_NO;
  GPIO_InitStruct.Alternate = LL_GPIO_AF_1;
  LL_GPIO_Init(GPIOA, &GPIO_InitStruct);

  GPIO_InitStruct.Pin = LL_GPIO_PIN_10;
  GPIO_InitStruct.Mode = LL_GPIO_MODE_ALTERNATE;
  GPIO_InitStruct.Speed = LL_GPIO_SPEED_FREQ_HIGH;
  GPIO_InitStruct.OutputType = LL_GPIO_OUTPUT_PUSHPULL;
  GPIO_InitStruct.Pull = LL_GPIO_PULL_NO;
  GPIO_InitStruct.Alternate = LL_GPIO_AF_1;
  LL_GPIO_Init(GPIOA, &GPIO_InitStruct);

  /* USART1 DMA Init */
   LL_DMA_DeInit(DMA1, LL_DMA_CHANNEL_3);
  /* USART1_RX Init */
    //设置DMA外设基地址
  LL_DMA_SetPeriphAddress(DMA1, LL_DMA_CHANNEL_3, (uint32_t)(&USART1->RDR));
    //设置DMA内存基地址
  LL_DMA_SetMemoryAddress(DMA1, LL_DMA_CHANNEL_3, (uint32_t)testBuf);
    //设置DMA传输方向
  LL_DMA_SetDataTransferDirection(DMA1, LL_DMA_CHANNEL_3, LL_DMA_DIRECTION_PERIPH_TO_MEMORY);
    //设置DMA的优先级
  LL_DMA_SetChannelPriorityLevel(DMA1, LL_DMA_CHANNEL_3, LL_DMA_PRIORITY_MEDIUM);
    //设置DMA的模式
  LL_DMA_SetMode(DMA1, LL_DMA_CHANNEL_3, LL_DMA_MODE_NORMAL);
    //设置DMA缓冲区长度
  LL_DMA_SetDataLength(DMA1, LL_DMA_CHANNEL_3, 100);
    //设置DMA外设递增
  LL_DMA_SetPeriphIncMode(DMA1, LL_DMA_CHANNEL_3, LL_DMA_PERIPH_NOINCREMENT);
    //设置内存递增
  LL_DMA_SetMemoryIncMode(DMA1, LL_DMA_CHANNEL_3, LL_DMA_MEMORY_INCREMENT);
  //外设大小设置
  LL_DMA_SetPeriphSize(DMA1, LL_DMA_CHANNEL_3, LL_DMA_PDATAALIGN_BYTE);
    //内存大小设置
  LL_DMA_SetMemorySize(DMA1, LL_DMA_CHANNEL_3, LL_DMA_MDATAALIGN_BYTE);
    
    
  /* USART1 interrupt Init */
  NVIC_SetPriority(USART1_IRQn, 0);
  NVIC_EnableIRQ(USART1_IRQn);

  USART_InitStruct.BaudRate = 115200;
  USART_InitStruct.DataWidth = LL_USART_DATAWIDTH_8B;
  USART_InitStruct.StopBits = LL_USART_STOPBITS_1;
  USART_InitStruct.Parity = LL_USART_PARITY_NONE;
  USART_InitStruct.TransferDirection = LL_USART_DIRECTION_TX_RX;
  USART_InitStruct.HardwareFlowControl = LL_USART_HWCONTROL_NONE;
  USART_InitStruct.OverSampling = LL_USART_OVERSAMPLING_16;
  LL_USART_Init(USART1, &USART_InitStruct);
  LL_USART_DisableIT_CTS(USART1);
  LL_USART_DisableOverrunDetect(USART1);
  LL_USART_ConfigAsyncMode(USART1);
  LL_USART_Enable(USART1);
  LL_USART_SetRxTimeout(USART1, 10);
  LL_USART_EnableRxTimeout(USART1);
  LL_USART_EnableIT_RTO(USART1);
  //使能DMA接收
  LL_USART_EnableDMAReq_RX(USART1);
  //使能DMA通道
  LL_DMA_EnableChannel(DMA1, LL_DMA_CHANNEL_3);
}


//DMA接收函数
void USART_DMA_Rx(DMA_TypeDef* DMAx,uint32_t DMA_CHx, uint8_t *pData, uint16_t Rcv_Len)
{
    //失能DMA(设置DMA之前必须要关闭DMA)
    LL_DMA_DisableChannel(DMAx, DMA_CHx);
    //设置DMA接收缓冲区
    LL_DMA_SetMemoryAddress(DMAx, DMA_CHx, (uint32_t)pData);
    //设置DMA,通道,缓存区长度
    LL_DMA_SetDataLength(DMAx, DMA_CHx, Rcv_Len);
    //使能DMA
    LL_DMA_EnableChannel(DMAx, DMA_CHx);
}

//串口中断函数
void USART1_IRQHandler(void)
{
    if(LL_USART_IsActiveFlag_RTO(USART1))
    {
        LL_USART_ClearFlag_RTO(USART1);
        Data_Cnt = LL_DMA_GetDataLength(DMA1, LL_DMA_CHANNEL_3);
//      temp = LL_USART_ReceiveData8(USART1);
        LL_USART_TransmitData8(USART1, Data_Cnt);
        USART_DMA_Rx(DMA1, LL_DMA_CHANNEL_3, testBuf, 100);
    }
}

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
【社区内容提示】社区部分内容疑似由AI辅助生成,浏览时请结合常识与多方信息审慎甄别。
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

相关阅读更多精彩内容

  • 姓名:周崇杰 学号:16140120059 专业:机械设计制造及其自动化 转载自:http://blog.csd...
    CJbaby阅读 8,876评论 0 3
  • 姓名:周崇杰 学号:16140120059 专业:机械设计制造及其自动化 转载自:http://blog.csd...
    CJbaby阅读 10,216评论 0 5
  • # STM32之串口DMA接收不定长数据 ## 引言 在使用stm32或者其他单片机的时候,会经常使用到串口通讯,...
    杰杰T_T阅读 3,607评论 0 0
  • 关于读书,很多朋友都有这些疑惑: “我知道读书有用,我也经常读书,但为什么读过什么都记不住?” “经常想起一句读过...
    Fashion极客阅读 3,715评论 13 29
  • 建议阅读本文前先了解HashMap,鄙人文章 HashMap解析 很简单,LinkedHashMap的Entry只...
    风风风筝阅读 4,223评论 0 5

友情链接更多精彩内容