Can通信流程

打印 上一主题 下一主题

主题 937|帖子 937|积分 2811

下面给出一个更具体的 CAN 发送报文的程序流程说明,结合 HAL 库的利用及代码示例,帮助你相识每一步的具体操纵和内部原理。

一、系统与外设初始化

1.1 HAL 库初始化

在 main() 函数开头,首先调用 HAL 库初始化函数:
  1. HAL_Init();
复制代码


  • 作用:重置外设、初始化系统定时器,并设置 NVIC 分组等。
  • 细节:这一步包管后续调用 HAL 库函数时,各个全局变量和停止配置已停当。
1.2 系统时钟配置

调用时钟配置函数(通常由 CubeMX天生):
  1. SystemClock_Config();
复制代码


  • 作用:设置系统时钟源、PLL 频率、各总线的分频系数。
  • 细节:CAN 模块依赖于时钟,必须包管 CAN 所在总线的时钟已使能。
1.3 GPIO 初始化

调用初始化 GPIO 的函数(通常在 gpio.c 中界说,如 MX_GPIO_Init()):
  1. MX_GPIO_Init();
复制代码


  • 作用:初始化全部用到的 GPIO,包罗 CAN_TX 和 CAN_RX 所对应的引脚。
  • 细节:这些引脚需要配置为“复勤奋能”(Alternate Function),并设置对应的 AF 映射(如 GPIO_AF9_CAN1),以便与 CAN 外设关联。
1.4 CAN 外设低级硬件初始化

通过 HAL 库的 MSP 回调函数进行:


  • 在 HAL_CAN_MspInit() 中,使能 CAN 所在外设的时钟、配置相关 GPIO、设置 NVIC 停止优先级等。
  • 例如:
    1. void HAL_CAN_MspInit(CAN_HandleTypeDef* hcan)
    2. {
    3.     if(hcan->Instance==CAN1)
    4.     {
    5.         __HAL_RCC_CAN1_CLK_ENABLE();
    6.         __HAL_RCC_GPIOA_CLK_ENABLE();
    7.         
    8.         GPIO_InitTypeDef GPIO_InitStruct = {0};
    9.         // 配置 CAN_TX、CAN_RX 所对应的引脚(假设为 PA11、PA12)
    10.         GPIO_InitStruct.Pin = GPIO_PIN_11 | GPIO_PIN_12;
    11.         GPIO_InitStruct.Mode = GPIO_MODE_AF_PP;
    12.         GPIO_InitStruct.Pull = GPIO_NOPULL;
    13.         GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_VERY_HIGH;
    14.         GPIO_InitStruct.Alternate = GPIO_AF9_CAN1;
    15.         HAL_GPIO_Init(GPIOA, &GPIO_InitStruct);
    16.         
    17.         // 配置 NVIC 中断(可选,根据需要启用接收/错误中断)
    18.         HAL_NVIC_SetPriority(CAN1_RX0_IRQn, 0, 0);
    19.         HAL_NVIC_EnableIRQ(CAN1_RX0_IRQn);
    20.     }
    21. }
    复制代码
  • 作用:为后续 CAN 模块初始化提供硬件资源支持。

二、CAN 外设初始化及滤波器配置

2.1 配置 CAN_HandleTypeDef 并调用 HAL_CAN_Init()

在 main() 或专用初始化函数中:
  1. CAN_HandleTypeDef hcan1;
  2. hcan1.Instance = CAN1;
  3. hcan1.Init.Prescaler = 16;             // 根据时钟计算波特率
  4. hcan1.Init.Mode = CAN_MODE_NORMAL;      // 工作模式(正常/回环等)
  5. hcan1.Init.SyncJumpWidth = CAN_SJW_1TQ;
  6. hcan1.Init.TimeSeg1 = CAN_BS1_1TQ;
  7. hcan1.Init.TimeSeg2 = CAN_BS2_1TQ;
  8. hcan1.Init.TimeTriggeredMode = DISABLE;
  9. hcan1.Init.AutoBusOff = DISABLE;
  10. hcan1.Init.AutoWakeUp = DISABLE;
  11. hcan1.Init.AutoRetransmission = ENABLE;
  12. hcan1.Init.ReceiveFifoLocked = DISABLE;
  13. hcan1.Init.TransmitFifoPriority = DISABLE;
  14. if (HAL_CAN_Init(&hcan1) != HAL_OK)
  15. {
  16.     // 初始化失败处理
  17.     Error_Handler();
  18. }
复制代码


  • 作用:通过调用 HAL_CAN_Init(),配置 CAN 控制器的各种参数。
  • 细节:调用过程中会自动调用 HAL_CAN_MspInit() 完成低级资源初始化。
2.2 配置 CAN 滤波器

CAN 的滤波器决定了哪些报文会被 CAN 模块吸收。调用 HAL 提供的函数或用户封装的函数来配置滤波器。
  1. CAN_FilterTypeDef canFilterConfig;
  2. canFilterConfig.FilterActivation = ENABLE;
  3. canFilterConfig.FilterBank = 0;                      // 滤波器编号(根据硬件数量选择)
  4. canFilterConfig.FilterFIFOAssignment = CAN_FILTER_FIFO0;
  5. canFilterConfig.FilterIdHigh = 0x0000;                 // 根据需求设置过滤的 ID(高位)
  6. canFilterConfig.FilterIdLow = 0x0000;                  // (低位)
  7. canFilterConfig.FilterMaskIdHigh = 0x0000;             // 掩码,高位(0表示不过滤,即接收所有)
  8. canFilterConfig.FilterMaskIdLow = 0x0000;              // 掩码,低位
  9. canFilterConfig.FilterMode = CAN_FILTERMODE_IDMASK;
  10. canFilterConfig.FilterScale = CAN_FILTERSCALE_32BIT;
  11. if (HAL_CAN_ConfigFilter(&hcan1, &canFilterConfig) != HAL_OK)
  12. {
  13.     // 滤波器配置失败处理
  14.     Error_Handler();
  15. }
复制代码


  • 作用:指定 CAN 模块只吸收符合条件的报文。
  • 细节:滤波器必须在 CAN 启动之前配置完成。通常在 HAL_CAN_Init() 之后,HAL_CAN_Start() 之前完成滤波器配置。
2.3 启动 CAN 模块

调用启动函数:
  1. if (HAL_CAN_Start(&hcan1) != HAL_OK)
  2. {
  3.     // 启动失败处理
  4.     Error_Handler();
  5. }
复制代码


  • 作用:使 CAN 模块从初始化状态进入正常工作状态,此时发送和吸收功能均可利用。
  • 细节:启动后,可以使能停止,开始吸收和发送数据。

三、构造和发送 CAN 报文

3.1 构造发送数据结构

通常利用 HAL 库提供的 CAN_TxHeaderTypeDef 结构体,同时准备数据数组。例如:
  1. CAN_TxHeaderTypeDef TxHeader;
  2. uint8_t TxData[8] = {0x11, 0x22, 0x33, 0x44, 0x55, 0x66, 0x77, 0x88};
  3. TxHeader.StdId = 0x123;           // 标准标识符
  4. TxHeader.ExtId = 0x01;            // 如果使用扩展 ID,此项有效
  5. TxHeader.RTR = CAN_RTR_DATA;      // 数据帧(非远程帧)
  6. TxHeader.IDE = CAN_ID_STD;        // 标准帧
  7. TxHeader.DLC = 8;                 // 数据长度:8 字节
  8. TxHeader.TransmitGlobalTime = DISABLE;
复制代码


  • 作用:设置报文的 ID、数据长度、数据帧类型等。
  • 细节:字段设置应根据应用协议要求,确保吸收端能正确解析数据。
3.2 将报文写入发送邮箱并启动发送

调用 HAL 库的发送函数:
  1. uint32_t TxMailbox;
  2. if (HAL_CAN_AddTxMessage(&hcan1, &TxHeader, TxData, &TxMailbox) != HAL_OK)
  3. {
  4.     // 发送失败处理,例如邮箱未空或总线错误
  5.     Error_Handler();
  6. }
复制代码


  • 作用:将构造好的报文放入 CAN 控制器的发送邮箱,由硬件完成后续发送过程。
  • 细节

    • TxMailbox 用于返回所利用的发送邮箱编号(CAN 发送邮箱通常有 3 个)。
    • 如果返回错误,则需要根据错误码进行重发或错误处置惩罚。

3.3 发送过程监控与确认



  • 轮询检查:可以通过轮询 HAL_CAN_GetTxMailboxesFreeLevel(&hcan1) 来判断是否尚有空邮箱。
  • 停止回调:也可以利用 HAL 提供的回调函数,例如 HAL_CAN_TxMailbox0CompleteCallback()(如果启用了对应停止),来确认某个邮箱完成发送。
  • 错误处置惩罚:在发送过程中,如果检测到错误(例如仲裁失败、总线错误等),需要调用错误回调 HAL_CAN_ErrorCallback() 进行处置惩罚。

四、CAN 吸收(补充说明)

固然主要讨论发送流程,但在实际应用中,发送报文后 CAN 控制器也可能吸收到相应数据。一般流程如下:

  • 停止处置惩罚

    • 当 CAN 模块检测到有报文到达 FIFO0 时,HAL 库会调用回调函数:
      1. void HAL_CAN_RxFifo0MsgPendingCallback(CAN_HandleTypeDef *hcan)
      2. {
      3.     CAN_RxHeaderTypeDef RxHeader;
      4.     uint8_t RxData[8];
      5.     if(HAL_CAN_GetRxMessage(hcan, CAN_RX_FIFO0, &RxHeader, RxData) == HAL_OK)
      6.     {
      7.         // 对接收到的数据进行处理
      8.     }
      9. }
      复制代码

  • 数据处置惩罚

    • 在回调函数中,读取报文内容,根据 ID 和数据内容进行相应的业务处置惩罚。


五、总结整个流程


  • 初始化阶段

    • HAL 初始化、系统时钟、GPIO 配置、CAN MSP 初始化(使能时钟、GPIO、NVIC 等)。

  • CAN 模块初始化

    • 配置 CAN 参数(波特率、模式等),调用 HAL_CAN_Init()。
    • 配置滤波器(决定哪些报文被吸收),调用 HAL_CAN_ConfigFilter()。
    • 启动 CAN 模块,调用 HAL_CAN_Start()。

  • 发送阶段

    • 构造发送报文(填写 CAN_TxHeaderTypeDef、数据数组)。
    • 将报文写入发送邮箱,调用 HAL_CAN_AddTxMessage()。
    • 监控发送过程,处剃头送成功或错误。

这种具体流程确保在发送报文前,全部硬件和软件配置都已完成,并且在发送过程中对可能出现的错误提供了检查和处置惩罚机制。通过这种模块化设计,整个 CAN 通信过程清晰而可靠,便于后续的调试和维护。

免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!更多信息从访问主页:qidao123.com:ToB企服之家,中国第一个企服评测及商务社交产业平台。
回复

使用道具 举报

0 个回复

倒序浏览

快速回复

您需要登录后才可以回帖 登录 or 立即注册

本版积分规则

冬雨财经

金牌会员
这个人很懒什么都没写!
快速回复 返回顶部 返回列表