下面给出一个更具体的 CAN 发送报文的程序流程说明,结合 HAL 库的利用及代码示例,帮助你相识每一步的具体操纵和内部原理。
一、系统与外设初始化
1.1 HAL 库初始化
在 main() 函数开头,首先调用 HAL 库初始化函数:
- 作用:重置外设、初始化系统定时器,并设置 NVIC 分组等。
- 细节:这一步包管后续调用 HAL 库函数时,各个全局变量和停止配置已停当。
1.2 系统时钟配置
调用时钟配置函数(通常由 CubeMX天生):
- 作用:设置系统时钟源、PLL 频率、各总线的分频系数。
- 细节:CAN 模块依赖于时钟,必须包管 CAN 所在总线的时钟已使能。
1.3 GPIO 初始化
调用初始化 GPIO 的函数(通常在 gpio.c 中界说,如 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 停止优先级等。
- 例如:
- void HAL_CAN_MspInit(CAN_HandleTypeDef* hcan)
- {
- if(hcan->Instance==CAN1)
- {
- __HAL_RCC_CAN1_CLK_ENABLE();
- __HAL_RCC_GPIOA_CLK_ENABLE();
-
- GPIO_InitTypeDef GPIO_InitStruct = {0};
- // 配置 CAN_TX、CAN_RX 所对应的引脚(假设为 PA11、PA12)
- GPIO_InitStruct.Pin = GPIO_PIN_11 | GPIO_PIN_12;
- GPIO_InitStruct.Mode = GPIO_MODE_AF_PP;
- GPIO_InitStruct.Pull = GPIO_NOPULL;
- GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_VERY_HIGH;
- GPIO_InitStruct.Alternate = GPIO_AF9_CAN1;
- HAL_GPIO_Init(GPIOA, &GPIO_InitStruct);
-
- // 配置 NVIC 中断(可选,根据需要启用接收/错误中断)
- HAL_NVIC_SetPriority(CAN1_RX0_IRQn, 0, 0);
- HAL_NVIC_EnableIRQ(CAN1_RX0_IRQn);
- }
- }
复制代码 - 作用:为后续 CAN 模块初始化提供硬件资源支持。
二、CAN 外设初始化及滤波器配置
2.1 配置 CAN_HandleTypeDef 并调用 HAL_CAN_Init()
在 main() 或专用初始化函数中:
- CAN_HandleTypeDef hcan1;
- hcan1.Instance = CAN1;
- hcan1.Init.Prescaler = 16; // 根据时钟计算波特率
- hcan1.Init.Mode = CAN_MODE_NORMAL; // 工作模式(正常/回环等)
- hcan1.Init.SyncJumpWidth = CAN_SJW_1TQ;
- hcan1.Init.TimeSeg1 = CAN_BS1_1TQ;
- hcan1.Init.TimeSeg2 = CAN_BS2_1TQ;
- hcan1.Init.TimeTriggeredMode = DISABLE;
- hcan1.Init.AutoBusOff = DISABLE;
- hcan1.Init.AutoWakeUp = DISABLE;
- hcan1.Init.AutoRetransmission = ENABLE;
- hcan1.Init.ReceiveFifoLocked = DISABLE;
- hcan1.Init.TransmitFifoPriority = DISABLE;
- if (HAL_CAN_Init(&hcan1) != HAL_OK)
- {
- // 初始化失败处理
- Error_Handler();
- }
复制代码
- 作用:通过调用 HAL_CAN_Init(),配置 CAN 控制器的各种参数。
- 细节:调用过程中会自动调用 HAL_CAN_MspInit() 完成低级资源初始化。
2.2 配置 CAN 滤波器
CAN 的滤波器决定了哪些报文会被 CAN 模块吸收。调用 HAL 提供的函数或用户封装的函数来配置滤波器。
- CAN_FilterTypeDef canFilterConfig;
- canFilterConfig.FilterActivation = ENABLE;
- canFilterConfig.FilterBank = 0; // 滤波器编号(根据硬件数量选择)
- canFilterConfig.FilterFIFOAssignment = CAN_FILTER_FIFO0;
- canFilterConfig.FilterIdHigh = 0x0000; // 根据需求设置过滤的 ID(高位)
- canFilterConfig.FilterIdLow = 0x0000; // (低位)
- canFilterConfig.FilterMaskIdHigh = 0x0000; // 掩码,高位(0表示不过滤,即接收所有)
- canFilterConfig.FilterMaskIdLow = 0x0000; // 掩码,低位
- canFilterConfig.FilterMode = CAN_FILTERMODE_IDMASK;
- canFilterConfig.FilterScale = CAN_FILTERSCALE_32BIT;
- if (HAL_CAN_ConfigFilter(&hcan1, &canFilterConfig) != HAL_OK)
- {
- // 滤波器配置失败处理
- Error_Handler();
- }
复制代码
- 作用:指定 CAN 模块只吸收符合条件的报文。
- 细节:滤波器必须在 CAN 启动之前配置完成。通常在 HAL_CAN_Init() 之后,HAL_CAN_Start() 之前完成滤波器配置。
2.3 启动 CAN 模块
调用启动函数:
- if (HAL_CAN_Start(&hcan1) != HAL_OK)
- {
- // 启动失败处理
- Error_Handler();
- }
复制代码
- 作用:使 CAN 模块从初始化状态进入正常工作状态,此时发送和吸收功能均可利用。
- 细节:启动后,可以使能停止,开始吸收和发送数据。
三、构造和发送 CAN 报文
3.1 构造发送数据结构
通常利用 HAL 库提供的 CAN_TxHeaderTypeDef 结构体,同时准备数据数组。例如:
- CAN_TxHeaderTypeDef TxHeader;
- uint8_t TxData[8] = {0x11, 0x22, 0x33, 0x44, 0x55, 0x66, 0x77, 0x88};
- TxHeader.StdId = 0x123; // 标准标识符
- TxHeader.ExtId = 0x01; // 如果使用扩展 ID,此项有效
- TxHeader.RTR = CAN_RTR_DATA; // 数据帧(非远程帧)
- TxHeader.IDE = CAN_ID_STD; // 标准帧
- TxHeader.DLC = 8; // 数据长度:8 字节
- TxHeader.TransmitGlobalTime = DISABLE;
复制代码
- 作用:设置报文的 ID、数据长度、数据帧类型等。
- 细节:字段设置应根据应用协议要求,确保吸收端能正确解析数据。
3.2 将报文写入发送邮箱并启动发送
调用 HAL 库的发送函数:
- uint32_t TxMailbox;
- if (HAL_CAN_AddTxMessage(&hcan1, &TxHeader, TxData, &TxMailbox) != HAL_OK)
- {
- // 发送失败处理,例如邮箱未空或总线错误
- Error_Handler();
- }
复制代码
- 作用:将构造好的报文放入 CAN 控制器的发送邮箱,由硬件完成后续发送过程。
- 细节:
- TxMailbox 用于返回所利用的发送邮箱编号(CAN 发送邮箱通常有 3 个)。
- 如果返回错误,则需要根据错误码进行重发或错误处置惩罚。
3.3 发送过程监控与确认
- 轮询检查:可以通过轮询 HAL_CAN_GetTxMailboxesFreeLevel(&hcan1) 来判断是否尚有空邮箱。
- 停止回调:也可以利用 HAL 提供的回调函数,例如 HAL_CAN_TxMailbox0CompleteCallback()(如果启用了对应停止),来确认某个邮箱完成发送。
- 错误处置惩罚:在发送过程中,如果检测到错误(例如仲裁失败、总线错误等),需要调用错误回调 HAL_CAN_ErrorCallback() 进行处置惩罚。
四、CAN 吸收(补充说明)
固然主要讨论发送流程,但在实际应用中,发送报文后 CAN 控制器也可能吸收到相应数据。一般流程如下:
- 停止处置惩罚
- 当 CAN 模块检测到有报文到达 FIFO0 时,HAL 库会调用回调函数:
- void HAL_CAN_RxFifo0MsgPendingCallback(CAN_HandleTypeDef *hcan)
- {
- CAN_RxHeaderTypeDef RxHeader;
- uint8_t RxData[8];
- if(HAL_CAN_GetRxMessage(hcan, CAN_RX_FIFO0, &RxHeader, RxData) == HAL_OK)
- {
- // 对接收到的数据进行处理
- }
- }
复制代码
- 数据处置惩罚
- 在回调函数中,读取报文内容,根据 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企服之家,中国第一个企服评测及商务社交产业平台。 |