STM32
直播中

贾伟刚

7年用户 1551经验值
私信 关注
[问答]

如何使用DMA方式实现串口数据转发?

如何使用DMA方式实现串口数据转发?

回帖(1)

卜臻敏

2021-12-6 14:39:47
前言

其实之前做RM比赛的时候就要有做转发器的想法,但是当时是因为云台和底盘上下分开,通过滑环相连,为了减少通讯线路,才萌生做转发器的想法,虽然最后方案讨论不够完善,所以就搁置了。现在是因为某模块以及焊死在板子上,没办法直接使用串口进行通讯,所以不得不使用串口转发的方式,来进行模块的连接。
实现串口转发的方式有很多,各有优劣。本文主要利用DMA方式实现串口转发功能。
环境




  • 芯片:STM32F103RCT6(芯片仅做示例,更换32其他型号实现原理相同)
  • HAL库版本:1.8.0
  • STM32CubeMX版本:5.6.1

总体思路

方案一:使能两个串口的DMA接收和发送通道,使能串口空闲中断判断数据帧结束,初始化缓存数组变量。
       Created with Raphaël 2.2.0                                                                                               
优点:低CPU占用率和中断资源占用率,几乎不出现死机情况
缺点:高延时,内存占用率高,且一旦数据量超过缓存大小就会被覆盖。由于启用4条DMA通道,所以总线占用率也变高,进一步增加延时。
方案二:仅使能两个串口的DMA接收通道,使能串口接收中断判断一字节数据到达,初始化缓存数组变量。
       Created with Raphaël 2.2.0                                                                                             
优点:超低延时,内存占用率低
缺点:占用大量中断资源,有概率遇到DMA忙导致数据丢失(大约每1160字节丢1字节),多次频繁请求DMA会出现死机情况(连续发送约16k字节数据后,出现死机情况)
实现

以方案二为例,使用CubeMX新建工程
1.编辑串口参数




2.启用串口发送中断





3.勾选启用串口全局中断





4.取消选择生成HAL中断处理函数,这部分需要我们自己写






使用CubeMX生成我们的工程文件,并打开main.c。在main函数中添加如下代码


  /* USER CODE BEGIN 2 */
  __HAL_UART_ENABLE_IT(&huart1, UART_IT_RXNE); //使能串口1接收中断
  __HAL_UART_ENABLE_IT(&huart2, UART_IT_RXNE); //使能串口2接收中断
  /* USER CODE END 2 */


打开stm32f1xx_it.c,找到USARTx_IRQHandler(x为你使用的串口号,x=1,2…),编写中断处理函数。


/* 初始化缓存数组 */
uint8_t USART1RxBuff[1];
uint8_t USART2RxBuff[1];


void USART1_IRQHandler(void)
{
  /* 判断中断类型为串口接收中断 */
  if ((__HAL_UART_GET_FLAG(&huart1, UART_FLAG_RXNE) != RESET))
  {
    /* 清除中断标志位 */
    __HAL_UART_CLEAR_FLAG(&huart1, UART_FLAG_RXNE);
   
    /* 将串口接收到的数组移入缓存 */
    USART1RxBuff[0] = (uint8_t)(huart1.Instance->DR & (uint8_t)0x00FF);


    /* 启用DMA发送一字节数据 */
    HAL_UART_Transmit_DMA(&huart2, USART1RxBuff, USART1RxFIFO, 1);
  }
  
  /* 判断中断类型为串口发送中断 */
  if ((__HAL_UART_GET_FLAG(&huart1, UART_FLAG_TC) != RESET))
  {
    __HAL_UART_CLEAR_FLAG(&huart1, UART_FLAG_TC);
  }
  /* USER CODE END USART1_IRQn 0 */
}


编写DMA中断处理函数


/**
  * @brief This function handles DMA1 channel4 global interrupt.
  */
void DMA1_Channel4_IRQHandler(void)
{
  /* USER CODE BEGIN DMA1_Channel4_IRQn 0 */
  huart1.gState = HAL_UART_STATE_READY;                //重置串口状态为就绪态
  hdma_usart1_tx.State = HAL_DMA_STATE_READY;          //重置DMA状态为就绪态
  __HAL_DMA_CLEAR_FLAG(&hdma_usart1_tx, DMA_FLAG_TC4); //清除DMA传输完成标志位
  __HAL_DMA_CLEAR_FLAG(&hdma_usart1_tx, DMA_FLAG_HT4); //清除DMA半传输完成标志位
  __HAL_DMA_CLEAR_FLAG(&hdma_usart1_tx, DMA_FLAG_TE4); //清除DMA传输出错标志位
  __HAL_UNLOCK(&hdma_usart1_tx);                       //解锁DMA
  /* USER CODE END DMA1_Channel4_IRQn 0 */
}


同样的方法编写另一个串口的中断


然后编译+下载即可实现串口转发功能


后记
实际上这一版程序如文中所说并不完善,有很多正在尝试但是还是未解决的问题


丢字节的情况:尝试使用FIFO进行发送和接收,结果还是丢字节。DMA一次性发送512字节,结果发送2048,依然丢一字节,遂怀疑是硬件问题。
长时间多次请求DMA会导致死机,也就是方案二中收发到13-16k数据时会出现死机
其实这也算不上一个合格的转发,对于数据完整性没有任何保证,但是实际上要做到这些,需要上位机和下位机进行配合,如使用分包传输的方式。但是这种方式解决一些燃眉之急还是不错的。
举报

更多回帖

发帖
×
20
完善资料,
赚取积分