网站备案去哪备案今天头条新闻
在裸机环境下实现CAN数据的接收和发送,需要通过 硬件寄存器操作 或 HAL库函数 结合 手动实现的队列 来完成。以下是完整的接收和发送流程实现:
1. 硬件初始化
首先初始化CAN控制器和GPIO:
void CAN_Init(void) {// 1. 使能CAN时钟__HAL_RCC_CAN1_CLK_ENABLE();// 2. 配置CAN GPIO(以PB8/PB9为例)GPIO_InitTypeDef GPIO_InitStruct = {0};GPIO_InitStruct.Pin = GPIO_PIN_8 | GPIO_PIN_9; // CAN_RX/CAN_TXGPIO_InitStruct.Mode = GPIO_MODE_AF_PP;GPIO_InitStruct.Pull = GPIO_NOPULL;GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_HIGH;GPIO_InitStruct.Alternate = GPIO_AF9_CAN1;HAL_GPIO_Init(GPIOB, &GPIO_InitStruct);// 3. 配置CAN控制器hcan.Instance = CAN1;hcan.Init.Prescaler = 6; // 波特率 = APB1时钟/(Prescaler*(BS1+BS2+SJW))hcan.Init.Mode = CAN_MODE_NORMAL; // 正常模式hcan.Init.SyncJumpWidth = CAN_SJW_1TQ;hcan.Init.TimeSeg1 = CAN_BS1_8TQ; hcan.Init.TimeSeg2 = CAN_BS2_3TQ;hcan.Init.TimeTriggeredMode = DISABLE;hcan.Init.AutoBusOff = DISABLE;hcan.Init.AutoWakeUp = DISABLE;hcan.Init.AutoRetransmission = ENABLE; // 自动重传hcan.Init.ReceiveFifoLocked = DISABLE;hcan.Init.TransmitFifoPriority = DISABLE;HAL_CAN_Init(&hcan);// 4. 配置CAN过滤器(接收所有消息)CAN_FilterTypeDef filter;filter.FilterIdHigh = 0x0000;filter.FilterIdLow = 0x0000;filter.FilterMaskIdHigh = 0x0000;filter.FilterMaskIdLow = 0x0000;filter.FilterFIFOAssignment = CAN_RX_FIFO0; // 使用FIFO0filter.FilterBank = 0;filter.FilterMode = CAN_FILTERMODE_IDMASK;filter.FilterScale = CAN_FILTERSCALE_32BIT;filter.FilterActivation = ENABLE;HAL_CAN_ConfigFilter(&hcan, &filter);// 5. 启动CAN和中断HAL_CAN_Start(&hcan);HAL_CAN_ActivateNotification(&hcan, CAN_IT_RX_FIFO0_MSG_PENDING); // 启用FIFO0中断
}
2. 数据接收实现
(1)中断中接收数据并存入队列
void CAN_RX0_IRQHandler(void) {CanRxMsg_t rxMsg;// 从CAN硬件读取数据if (HAL_CAN_GetRxMessage(&hcan, CAN_RX_FIFO0, &rxMsg.id, rxMsg.data, &rxMsg.length) == HAL_OK) {// 存入接收队列(中断安全)if (!Queue_SendFromISR(&canRxQueue, &rxMsg, sizeof(CanRxMsg_t))) {// 队列满时的处理(如点亮LED警告)HAL_GPIO_WritePin(LED_GPIO_Port, LED_Pin, GPIO_PIN_SET);}}HAL_CAN_IRQHandler(&hcan); // 处理其他CAN中断
}
(2)主循环处理接收队列
while (1) {CanRxMsg_t rxMsg;// 从队列取出数据if (Queue_Receive(&canRxQueue, &rxMsg, sizeof(CanRxMsg_t))) {// 处理接收到的数据(示例:通过串口打印)printf("ID:0x%lX, Data:", rxMsg.id);for (int i = 0; i < rxMsg.length; i++) {printf("%02X ", rxMsg.data[i]);}printf("\n");}// 其他任务...
}
3. 数据发送实现
(1)发送数据到队列
void SendCANMessage(uint32_t id, uint8_t *data, uint8_t length) {CanTxMsg_t txMsg;txMsg.id = id;txMsg.length = length;memcpy(txMsg.data, data, length);// 非阻塞发送到队列if (!Queue_Send(&canTxQueue, &txMsg, sizeof(CanTxMsg_t))) {// 队列满处理printf("TX Queue Full!\n");}
}
(2)主循环处理发送队列
while (1) {CanTxMsg_t txMsg;// 检查发送队列if (Queue_Receive(&canTxQueue, &txMsg, sizeof(CanTxMsg_t))) {// 通过CAN硬件发送CAN_TxHeaderTypeDef txHeader;txHeader.StdId = txMsg.id;txHeader.IDE = CAN_ID_STD;txHeader.RTR = CAN_RTR_DATA;txHeader.DLC = txMsg.length;uint32_t mailbox;if (HAL_CAN_AddTxMessage(&hcan, &txHeader, txMsg.data, &mailbox) != HAL_OK) {printf("CAN Send Failed!\n");}}// 其他任务...HAL_Delay(1); // 防止CPU占用率100%
}
4. 关键机制说明
机制 实现方式
中断安全 在中断中使用 __disable_irq()/__enable_irq() 保护队列操作
非阻塞设计 所有队列操作立即返回,不等待
硬件发送 主循环轮询发送队列,通过 HAL_CAN_AddTxMessage 发送
错误处理 队列满时触发标志位或LED警告
数据一致性 使用 memcpy 保证数据完整拷贝
5. 完整示例流程
初始化
int main(void) {HAL_Init();SystemClock_Config();Queues_Init();CAN_Init();while (1) {ProcessRxQueue(); // 处理接收队列ProcessTxQueue(); // 处理发送队列}
}
外部调用发送:
// 发送示例数据
uint8_t data[] = {0x01, 0x02, 0x03};
SendCANMessage(0x123, data, 3);
中断自动接收:
数据到达 → 触发 CAN_RX0_IRQHandler → 存入 canRxQueue → 主循环处理。
6. 性能优化建议
DMA发送:对于高频发送,配置CAN TX DMA。
双缓冲接收:使用两个队列交替处理,避免数据覆盖。
优先级控制:为CAN中断设置合适优先级(通常高于普通任务)。
通过这种方式,裸机系统可以实现与RTOS类似的可靠通信,同时保持更高的实时性和更低的内存开销。