总线上有两个设备,主机:100ms周期发送数据。从机:以不同波特率发送数据,再把从机波特率调节至主机波特率一致无法通信。
环境:VSCODE + IDF-v5.0
我们先看下ESP32技术参考手册中TWAI中对错误状态计数器的描述。当TEC>255时,说明总线发生严重错误,此时ESP32的机制会关闭总线进入离线状态。这也是导致我们无法通信的原因。

在ESP32的技术参考手册上也描述了怎么恢复离线状态。

首先我们需要知道CAN(TWAI)的总线状态。这个我们可以通过IDF的API函数来获取中线的各种状态。
代码如下:
- /**
- * @brief Structure to store status information of TWAI driver
- */
- typedef struct {
- twai_state_t state; /**< Current state of TWAI controller (Stopped/Running/Bus-Off/Recovery) */
- uint32_t msgs_to_tx; /**< Number of messages queued for transmission or awaiting transmission completion */
- uint32_t msgs_to_rx; /**< Number of messages in RX queue waiting to be read */
- uint32_t tx_error_counter; /**< Current value of Transmit Error Counter */
- uint32_t rx_error_counter; /**< Current value of Receive Error Counter */
- uint32_t tx_failed_count; /**< Number of messages that failed transmissions */
- uint32_t rx_missed_count; /**< Number of messages that were lost due to a full RX queue (or errata workaround if enabled) */
- uint32_t rx_overrun_count; /**< Number of messages that were lost due to a RX FIFO overrun */
- uint32_t arb_lost_count; /**< Number of instances arbitration was lost */
- uint32_t bus_error_count; /**< Number of instances a bus error has occurred */
- } twai_status_info_t;
-
- /**
- * @brief Get current status information of the TWAI driver
- *
- * @param[out] status_info Status information
- *
- * @return
- * - ESP_OK: Status information retrieved
- * - ESP_ERR_INVALID_ARG: Arguments are invalid
- * - ESP_ERR_INVALID_STATE: TWAI driver is not installed
- */
- esp_err_t twai_get_status_info(twai_status_info_t *status_info)
同时CAN(TWAI)总线的状态有以下几种状态。
- /**
- * @brief TWAI driver states
- */
- typedef enum {
- TWAI_STATE_STOPPED, /**< Stopped state. The TWAI controller will not participate in any TWAI bus activities */
- TWAI_STATE_RUNNING, /**< Running state. The TWAI controller can transmit and receive messages */
- TWAI_STATE_BUS_OFF, /**< Bus-off state. The TWAI controller cannot participate in bus activities until it has recovered */
- TWAI_STATE_RECOVERING, /**< Recovering state. The TWAI controller is undergoing bus recovery */
- } twai_state_t;
当我们获取到现在的总线状态为OFF时,我们就可以知道我们的CAN(TWAI)中线已经离线,我们要开始进入恢复流程,我们直接调用IDF的API进入恢复流程,我们可以通过以下代码来实现。
- /**
- * @brief Start the bus recovery process
- *
- * This function initiates the bus recovery process when the TWAI driver is in
- * the bus-off state. Once initiated, the TWAI driver will enter the recovering
- * state and wait for 128 occurrences of the bus-free signal on the TWAI bus
- * before returning to the stopped state. This function will reset the TX queue,
- * clearing any messages pending transmission.
- *
- * @note The BUS_RECOVERED alert can be enabled to alert the application when
- * the bus recovery process completes.
- *
- * @return
- * - ESP_OK: Bus recovery started
- * - ESP_ERR_INVALID_STATE: TWAI driver is not in the bus-off state, or is not installed
- */
- esp_err_t twai_initiate_recovery(void);
-
- twai_status_info_t twai_status_info;
- twai_get_status_info(&twai_status_info);
-
- // 总线关闭
- if(twai_status_info.state == TWAI_STATE_BUS_OFF)
- {
- // 总线复位
- twai_initiate_recovery();
- }
当我们执行完twai_initiate_recovery();函数之后CAN(TWAI)总线会清空所有错误计数,但是此时总线还不能直接收发数据,因为此时我们的总线状态为TWAI_STATE_STOPPED状态,我们此时需要切换到TWAI_STATE_RUNNING状态我们才能正常收发数据,这个状态的切换我们也是通过API函数来切换。代码如下:
twai_start();
问题总结:解决这个问题只需要在收发任务中加入以下代码就可以对总线状态进行监控。进入恢复流程。
- // 读取总线状态
- twai_status_info_t twai_status_info;
-
- twai_get_status_info(&twai_status_info);
- // 总线关闭
- if(twai_status_info.state == TWAI_STATE_BUS_OFF)
- {
- twai_initiate_recovery();
- }
- // 总线停止
- else if(twai_status_info.state == TWAI_STATE_STOPPED)
- {
- twai_start();
- }
如果帮到了你,请给我一个关注😍。