打印
[STM32C0]

【STM32C092RC 测评】FDCAN Loopback测试

[复制链接]
76|0
手机看帖
扫描二维码
随时随地手机跟帖
跳转到指定楼层
楼主
      CAN同样可以进行Loopback测试,我们首先连接一下CAN的基本概念。
      控制器局域网总线(CAN)作为专为实时应用量身打造的串行通讯协议,在工业及自动化领域扮演着至关重要的角色。仅需简单的双绞线,它便能实现信号的高效、稳定传输。凭借卓越的性能与广泛的适用性,CAN总线已成为全球范围内应用最为广泛的现场总线之一。我们将深入探索STM32在CAN总线方面的具体应用之法。
      本次测试中的C092型号支持FDCAN,即CAN FD(CAN with Flexible Data Rate),作为传统CAN协议的升级版,其核心升级聚焦于协议层面,物理层维持原状。CAN与CAN FD的关键差异体现在:
      传输速率:传统CAN的传输速率较为固定,而CAN FD引入可变速率机制,能根据实际需求灵活调整,尤其在大数据量或高实时性要求的场景下,显著提升数据传输速率。
      数据域长度:CAN FD扩展了数据域长度,相较于传统CAN有限的数据域,CAN FD单次通信可传输更多数据,减少通信次数,提高通信效率。
      帧格式:为适应可变速率和数据域长度的变化,CAN FD优化了帧格式,不仅支持更高效的变速率与大数据量传输,同时保持与传统CAN协议的兼容性。
      ID长度:CAN FD不仅扩展了数据域长度,还增加了标识符(ID)的长度,提供更多寻址空间,支持更多节点,增强系统扩展性和灵活性。
      我们看一下C092开发板中对应CAN接口的原理图:
      对应的单片机引脚如下:
      实物接口如下:
      Loopback测试连接如下:
      就是这么简单,紧邻常用的USB接口,C0系列设备在USB与FDCAN之间做出了巧妙的设计取舍,二者仅能择其一配备。
      接下来进行FDCAN的配置:
      先看一下时钟配置,因为C092支持最大的是48MHz,到FDCAN最大也是48MHz:
      启动FDCAN:
      下面就是具体参数的设置,我们本次主要进行一下回环测试,所以要进行如下配置:

基本参数
Clock Divider
时钟分频
  
Frame Format
CANFD模式
  
Mode
正常工作模式
  
Auto Retransmission
自动重传
  
Transmit Pause
传输暂停
  
Protocol Exception
协议异常处理
  
Nominal Sync Jump Width
裁决段同步跳转段宽度
  
Data Prescaler
数据段分频系数
  
Data Sync Jump Width
数据段同步跳转段宽度
  
Data Time Seg1
数据段时间段1
  
Data Time Seg2
数据段时间段2
  
Std Filters Nbr
标准滤波器数量
  
Ext Filters Nbr
拓展滤波器数量
  
Tx Fifo Queue Mode
发送模式
  
      
      我们按照例程中的配置参数进行一下配置:
      接下来进行比特率的配置,如下图:
       上图所示选项中,最小数值设定为1,比特率的计算公式为:比特率 = CAN时钟 / (时钟分频 × 预分频 × (Seg1 + Seg2 + 1))。即便采用此最小配置参数,理论上能达到的最大比特率可高达16MHz。
       软件方面主要是配置过滤器,这里我们主要配置过滤ID,前面各开启了一个标准滤波器和一个扩展滤波器:
<p>/* Configure standard ID reception filter to Rx FIFO 0. Only accept ID = FilterID1 */</p><p>  FDCAN_FilterTypeDef        sFilterConfig;</p><p>  sFilterConfig.IdType       = FDCAN_STANDARD_ID;</p><p>  sFilterConfig.FilterIndex  = 0U;</p><p>  sFilterConfig.FilterType   = FDCAN_FILTER_DUAL;</p><p>  sFilterConfig.FilterConfig = FDCAN_FILTER_TO_RXFIFO0;</p><p>  sFilterConfig.FilterID1    = 0x444;</p><p>  sFilterConfig.FilterID2    = 0x444; /* For acceptance, MessageID and FilterID1 must match exactly */</p><p>  if (HAL_FDCAN_ConfigFilter(&hfdcan1, &sFilterConfig) != HAL_OK)</p><p>  {</p><p>    Error_Handler();</p><p>  }</p><p>  /* Configure extended ID reception filter to Rx FIFO 1. Only accept ID between FilterID1 and FilterID2. */</p><p>  sFilterConfig.IdType       = FDCAN_EXTENDED_ID;</p><p>  sFilterConfig.FilterIndex  = 0U;</p><p>  sFilterConfig.FilterType   = FDCAN_FILTER_RANGE_NO_EIDM;</p><p>  sFilterConfig.FilterConfig = FDCAN_FILTER_TO_RXFIFO1;</p><p>  sFilterConfig.FilterID1    = 0x1111111;</p><p>  sFilterConfig.FilterID2    = 0x2222222;</p><p>  if (HAL_FDCAN_ConfigFilter(&hfdcan1, &sFilterConfig) != HAL_OK)</p><p>  {</p><p>    Error_Handler();</p><p>  }</p>

       然后是进行数据的发送和接收,这里都是通过FIFO的模式进行缓存的,并在接收节点进行接收数据打印,这里面发送配置中主要就是ID要满足上面过滤器的要求:
<p>FDCAN_TxHeaderTypeDef        txHeader;</p><p>  /* Add message to Tx FIFO */</p><p>  txHeader.Identifier          = 0x444;</p><p>  txHeader.IdType              = FDCAN_STANDARD_ID;</p><p>  txHeader.TxFrameType         = FDCAN_DATA_FRAME;</p><p>  txHeader.DataLength          = FDCAN_DLC_BYTES_12;</p><p>  txHeader.ErrorStateIndicator = FDCAN_ESI_ACTIVE;</p><p>  txHeader.BitRateSwitch       = FDCAN_BRS_ON;</p><p>  txHeader.FDFormat            = FDCAN_FD_CAN;</p><p>  txHeader.TxEventFifoControl  = FDCAN_STORE_TX_EVENTS;</p><p>  txHeader.MessageMarker       = 0x52U;</p><p>  if (HAL_FDCAN_AddMessageToTxFifoQ(&hfdcan1, &txHeader, txData0) != HAL_OK)</p><p>  {</p><p>    Error_Handler();</p><p>  }</p><p>  /* Add second message to Tx FIFO */</p><p>  txHeader.Identifier          = 0x1111112;</p><p>  txHeader.IdType              = FDCAN_EXTENDED_ID;</p><p>  txHeader.TxFrameType         = FDCAN_DATA_FRAME;</p><p>  txHeader.DataLength          = FDCAN_DLC_BYTES_12;</p><p>  txHeader.ErrorStateIndicator = FDCAN_ESI_PASSIVE;</p><p>  txHeader.BitRateSwitch       = FDCAN_BRS_ON;</p><p>  txHeader.FDFormat            = FDCAN_FD_CAN;</p><p>  txHeader.TxEventFifoControl  = FDCAN_STORE_TX_EVENTS;</p><p>  txHeader.MessageMarker       = 0xCCU;</p><p>  if (HAL_FDCAN_AddMessageToTxFifoQ(&hfdcan1, &txHeader, txData1) != HAL_OK)</p><p>  {</p><p>    Error_Handler();</p><p>  }</p><p>  /* Add third message to Tx FIFO */</p><p>  txHeader.Identifier          = 0x1111113;</p><p>  txHeader.IdType              = FDCAN_EXTENDED_ID;</p><p>  txHeader.TxFrameType         = FDCAN_DATA_FRAME;</p><p>  txHeader.DataLength          = FDCAN_DLC_BYTES_12;</p><p>  txHeader.ErrorStateIndicator = FDCAN_ESI_PASSIVE;</p><p>  txHeader.BitRateSwitch       = FDCAN_BRS_OFF;</p><p>  txHeader.FDFormat            = FDCAN_FD_CAN;</p><p>  txHeader.TxEventFifoControl  = FDCAN_STORE_TX_EVENTS;</p><p>  txHeader.MessageMarker       = 0xDDU;</p><p>  if (HAL_FDCAN_AddMessageToTxFifoQ(&hfdcan1, &txHeader, txData2) != HAL_OK)</p><p>  {</p><p>    Error_Handler();</p><p>  }</p><p>/* Get tick */</p><p>  uint32_t tickstart = HAL_GetTick();</p><p>  /* Wait transmission complete */</p><p>  while (HAL_FDCAN_GetTxFifoFreeLevel(&hfdcan1) != NB_RX_FIFO)</p><p>  {</p><p>    /* Timeout handling */</p><p>    if ((HAL_GetTick() - tickstart) > TX_TIMEOUT)</p><p>    {</p><p>      Error_Handler();</p><p>    }</p><p>  }</p><p>/*##-4 Receive messages ###################################################*/</p><p>  /* Check one message is received in Rx FIFO 0 */</p><p>  if (HAL_FDCAN_GetRxFifoFillLevel(&hfdcan1, FDCAN_RX_FIFO0) != 1U)</p><p>  {</p><p>    Error_Handler();</p><p>  }</p><p>  /* Retrieve message from Rx FIFO 0 */</p><p>  if (HAL_FDCAN_GetRxMessage(&hfdcan1, FDCAN_RX_FIFO0, &rxHeader, rxData) != HAL_OK)</p><p>  {</p><p>    Error_Handler();</p><p>  }</p><p>  /* Compare received RX message to expected data */</p><p>  if ((rxHeader.Identifier != 0x444) ||</p><p>      (rxHeader.IdType     != FDCAN_STANDARD_ID) ||</p><p>      (rxHeader.DataLength != FDCAN_DLC_BYTES_12) ||</p><p>      (BufferCmp8b(txData0, rxData, COUNTOF(rxData)) != 0U))</p><p>  {</p><p>    Error_Handler();</p><p>  }</p><p>HAL_UART_Transmit(&huart2, (uint8_t *)&rxData, 12, 0xFFFF);</p><p>  /* Check two messages are received in Rx FIFO 1 */</p><p>  if (HAL_FDCAN_GetRxFifoFillLevel(&hfdcan1, FDCAN_RX_FIFO1) != 2U)</p><p>  {</p><p>    Error_Handler();</p><p>  }</p><p>  /* Retrieve message from Rx FIFO 1 */</p><p>  if (HAL_FDCAN_GetRxMessage(&hfdcan1, FDCAN_RX_FIFO1, &rxHeader, rxData) != HAL_OK)</p><p>  {</p><p>    Error_Handler();</p><p>  }</p><p>  /* Compare received RX message to expected data */</p><p>  if ((rxHeader.Identifier != 0x1111112) ||</p><p>      (rxHeader.IdType     != FDCAN_EXTENDED_ID) ||</p><p>      (rxHeader.DataLength != FDCAN_DLC_BYTES_12) ||</p><p>      (BufferCmp8b(txData1, rxData, COUNTOF(rxData)) != 0U))</p><p>  {</p><p>    Error_Handler();</p><p>  }</p><p>HAL_UART_Transmit(&huart2, (uint8_t *)&rxData, 12, 0xFFFF);</p><p>  /* Retrieve next message from Rx FIFO 1 */</p><p>  if (HAL_FDCAN_GetRxMessage(&hfdcan1, FDCAN_RX_FIFO1, &rxHeader, rxData) != HAL_OK)</p><p>  {</p><p>    Error_Handler();</p><p>  }</p><p>  /* Compare received RX message to expected data */</p><p>  if ((rxHeader.Identifier != 0x1111113) ||</p><p>      (rxHeader.IdType     != FDCAN_EXTENDED_ID) ||</p><p>      (rxHeader.DataLength != FDCAN_DLC_BYTES_12) ||</p><p>      (BufferCmp8b(txData2, rxData, COUNTOF(rxData)) != 0U))</p><p>  {</p><p>    Error_Handler();</p><p>  }</p><p>HAL_UART_Transmit(&huart2, (uint8_t *)&rxData, 12, 0xFFFF);</p>

       最后在通过在FIFO中查询的数据进行打印,与发送的数据一致:

使用特权

评论回复
发新帖 我要提问
您需要登录后才可以回帖 登录 | 注册

本版积分规则

认证:硬件工程师
简介:有着多年硬件开发经验的专业人员,专注于医疗电子领域,热衷于对新鲜事物的探索,喜欢DIY!

455

主题

2310

帖子

11

粉丝