0
  • 聊天消息
  • 系統(tǒng)消息
  • 評論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

STM32H743的FDCAN發(fā)送線程卡死的處理方法

冬至配餃子 ? 來源:小白小白小白白 ? 作者:小白小白小白白 ? 2023-10-12 11:37 ? 次閱讀

芯片型號STM32H743IIT6,測試時發(fā)現(xiàn)如果外面沒有連接CAN設(shè)備,程序調(diào)用CAN發(fā)送時會一直等待發(fā)送反饋,導(dǎo)致相關(guān)線程掛起。

分析發(fā)現(xiàn)是卡在can.c文件的168行_can_int_tx函數(shù):rt_completion_wait(&(tx_tosnd->completion), RT_WAITING_FOREVER);

rt_inline int _can_int_tx(struct rt_can_device *can, const struct rt_can_msg *data, int msgs)
{
int size;
struct rt_can_tx_fifo *tx_fifo;
RT_ASSERT(can != RT_NULL);
size = msgs;
tx_fifo = (struct rt_can_tx_fifo *) can->can_tx;
RT_ASSERT(tx_fifo != RT_NULL);
while (msgs)
{
rt_base_t level;
rt_uint32_t no;
rt_uint32_t result;
struct rt_can_sndbxinx_list tx_tosnd = RT_NULL;
rt_sem_take(&(tx_fifo->sem), RT_WAITING_FOREVER);
level = rt_hw_interrupt_disable();
tx_tosnd = rt_list_entry(tx_fifo->freelist.next, struct rt_can_sndbxinx_list, list);
RT_ASSERT(tx_tosnd != RT_NULL);
rt_list_remove(&tx_tosnd->list);
rt_hw_interrupt_enable(level);
no = ((rt_uint32_t)tx_tosnd - (rt_uint32_t)tx_fifo->buffer) / sizeof(struct rt_can_sndbxinx_list);
tx_tosnd->result = RT_CAN_SND_RESULT_WAIT;
if (can->ops->sendmsg(can, data, no) != RT_EOK)
{
/
send failed. */
level = rt_hw_interrupt_disable();
rt_list_insert_after(&tx_fifo->freelist, &tx_tosnd->list);
rt_hw_interrupt_enable(level);
rt_sem_release(&(tx_fifo->sem));
continue;
}
can->status.sndchange = 1;
rt_completion_wait(&(tx_tosnd->completion), RT_WAITING_FOREVER);

說明一直在等待完成信號,而發(fā)送完成信號的地方在can.c文件的900行rt_hw_can_isr函數(shù):

rt_completion_done(&(tx_fifo->buffer[no].completion));

case RT_CAN_EVENT_TX_DONE:
case RT_CAN_EVENT_TX_FAIL:
{
    struct rt_can_tx_fifo *tx_fifo;
    rt_uint32_t no;
    no = event > > 8;
    tx_fifo = (struct rt_can_tx_fifo *) can- >can_tx;
    RT_ASSERT(tx_fifo != RT_NULL);
    if ((event & 0xff) == RT_CAN_EVENT_TX_DONE)
    {
        tx_fifo- >buffer[no].result = RT_CAN_SND_RESULT_OK;
    }
    else
    {
        tx_fifo- >buffer[no].result = RT_CAN_SND_RESULT_ERR;
    }
    rt_completion_done(&(tx_fifo- >buffer[no].completion));
    break;
}

然后想到如果can總線沒有其他設(shè)備,CAN發(fā)送報文應(yīng)該屬于出錯的情況,查看drv_fdcan.c文件中關(guān)于幾種中斷的處理,發(fā)現(xiàn)故障后的回調(diào)函數(shù)里沒有調(diào)用rt_hw_can_isr。

于是參考HAL_FDCAN_TxBufferCompleteCallback函數(shù)的處理方式,對HAL_FDCAN_ErrorCallback進(jìn)行了如下處理。

void HAL_FDCAN_ErrorCallback(FDCAN_HandleTypeDef *hfdcan)
{
rt_uint32_t tmp_u32Errcount;
rt_uint32_t tmp_u32status;
uint32_t ret = HAL_FDCAN_GetError(hfdcan);
if(hfdcan->Instance == FDCAN1)
{
#ifdef BSP_USING_FDCAN1
//can1
if( (ret & FDCAN_IT_ARB_PROTOCOL_ERROR) &&
(hfdcan->Instance->CCCR & FDCAN_CCCR_INIT_Msk))
{
//hfdcan->Instance->CCCR |= FDCAN_CCCR_CCE_Msk;
hfdcan->Instance->CCCR &= ~FDCAN_CCCR_INIT_Msk;
st_DrvCan1.device.status.errcode = 0xff;
}
else
{
tmp_u32Errcount = st_DrvCan1.fdcanHandle.Instance->ECR;
tmp_u32status = st_DrvCan1.fdcanHandle.Instance->PSR;
st_DrvCan1.device.status.rcverrcnt = (tmp_u32Errcount>>8)&0x000000ff;
st_DrvCan1.device.status.snderrcnt = (tmp_u32Errcount)&0x000000ff;
st_DrvCan1.device.status.lasterrtype = tmp_u32status&0x000000007;
rt_hw_can_isr(&st_DrvCan1.device, RT_CAN_EVENT_TX_FAIL);
}
#endif / BSP_USING_FDCAN1 /
}
else
{
#ifdef BSP_USING_FDCAN2
if( (ret & FDCAN_IT_ARB_PROTOCOL_ERROR) &&
(hfdcan->Instance->CCCR & FDCAN_CCCR_INIT_Msk))
{
//hfdcan->Instance->CCCR |= FDCAN_CCCR_CCE_Msk;
hfdcan->Instance->CCCR &= ~FDCAN_CCCR_INIT_Msk;
st_DrvCan2.device.status.errcode = 0xff;
}
else
{
//can2
tmp_u32Errcount = st_DrvCan2.fdcanHandle.Instance->ECR;
tmp_u32status = st_DrvCan2.fdcanHandle.Instance->PSR;
st_DrvCan2.device.status.rcverrcnt = (tmp_u32Errcount>>8)&0x000000ff;
st_DrvCan2.device.status.snderrcnt = (tmp_u32Errcount)&0x000000ff;
st_DrvCan2.device.status.lasterrtype = tmp_u32status&0x000000007;
rt_hw_can_isr(&st_DrvCan2.device, RT_CAN_EVENT_TX_FAIL);
}
#endif / BSP_USING_FDCAN2 /
}
}

經(jīng)過測試發(fā)現(xiàn)即使CAN總線上沒有別的設(shè)備,調(diào)用發(fā)送函數(shù)也不會一直等待,而是返回發(fā)送失敗。

聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點(diǎn)僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問題,請聯(lián)系本站處理。 舉報投訴
  • CAN總線
    +關(guān)注

    關(guān)注

    145

    文章

    1908

    瀏覽量

    130502
  • 中斷處理
    +關(guān)注

    關(guān)注

    0

    文章

    94

    瀏覽量

    10940
  • FIFO存儲
    +關(guān)注

    關(guān)注

    0

    文章

    103

    瀏覽量

    5953
  • STM32H743
    +關(guān)注

    關(guān)注

    0

    文章

    24

    瀏覽量

    1633
收藏 人收藏

    評論

    相關(guān)推薦

    請問STM32H743 FDCAN自舉程序使用的CANFD的GPIO是哪倆個?

    FDCAN1_Tx引腳說的是PH14和PH13,但是在實際應(yīng)用中使用的STM32H743ZI的144pin芯片中沒有PH14和PH13引腳,那么該芯片的FDCAN自舉程序使用的CANFD的GPIO是哪倆個?
    發(fā)表于 03-18 06:47

    STM32H743以太網(wǎng)與高速USB同時工作遇到的疑問求解

    報文的時候,高速USB也進(jìn)行收發(fā)報文,則USB通信會偶發(fā)性異常,PC端使用的是libusb,PC端顯示USB報文已經(jīng)成功發(fā)送STM32H743了,但是STM32H743卻沒有收到這個USB報文,是偶發(fā)性沒有
    發(fā)表于 04-09 07:53

    stm32H743can配置 精選資料分享

    STM32H743兩路can的配置FDCAN_HandleTypeDef FDCAN1_Handler;FDCAN_RxHeaderTypeDef
    發(fā)表于 08-17 08:50

    STM32H743 Flash用來存什么

    STM32H743 Flash用來存什么?STM32H743 Flash的硬件特性和基本操作是什么?
    發(fā)表于 09-24 09:41

    NUCLEO STM32H743怎么使用?

    NUCLEO STM32H743怎么使用?
    發(fā)表于 11-08 06:25

    如何調(diào)試STM32H743的兩串口?

    如何調(diào)試STM32H743的兩串口?
    發(fā)表于 02-28 08:51

    STM32H743FDCAN導(dǎo)致CAN發(fā)送線程卡死處理方法

    芯片型號STM32H743IIT6,測試時發(fā)現(xiàn)如果外面沒有連接CAN設(shè)備,程序調(diào)用CAN發(fā)送時會一直等待發(fā)送反饋,導(dǎo)致相關(guān)線程掛起。分析發(fā)現(xiàn)是卡在can.c文件的168行_can_in
    發(fā)表于 02-28 15:02

    STM32H743 FDCAN自舉程序使用的CANFD的GPIO是哪倆個?

    FDCAN1_Tx引腳說的是PH14和PH13,但是在實際應(yīng)用中使用的STM32H743ZI的144pin芯片中沒有PH14和PH13引腳,那么該芯片的FDCAN自舉程序使用的CANFD的GPIO是哪倆個?
    發(fā)表于 08-07 06:57

    STM32H743芯片上實現(xiàn)CAN通信的步驟

    本篇筆記主要介紹,在STM32H743芯片上實現(xiàn)CAN通信,封裝為BSP驅(qū)動,為之后實現(xiàn)CAN的高層通信打下基礎(chǔ)。
    的頭像 發(fā)表于 09-14 14:33 ?1.6w次閱讀
    在<b class='flag-5'>STM32H743</b>芯片上實現(xiàn)CAN通信的步驟

    stm32h750/stm32h743原理圖和pcb源文件

    stm32在目前使用非常廣泛,但是目前很多人都還停留在stmf1/f4僅僅只有72/128m主頻階段,stm32h743采用arm m7架構(gòu),高達(dá)400m主頻的處理器,為我們的控制提供強(qiáng)有力的支持
    發(fā)表于 12-08 09:21 ?152次下載
    <b class='flag-5'>stm32h</b>750/<b class='flag-5'>stm32h743</b>原理圖和pcb源文件

    stm32h743外部RAM非字節(jié)對齊訪問,引起的hard fault

    stm32h743外部RAM非字節(jié)對齊訪問,引起的hard fault
    發(fā)表于 12-09 09:21 ?5次下載
    <b class='flag-5'>stm32h743</b>外部RAM非字節(jié)對齊訪問,引起的hard fault

    STM32H743+CubeMX-解決FDCAN控制器無法接收遠(yuǎn)程幀

    文章目錄一. 前言二. STM32H743編程參考手冊三. HAL_FDCAN_ConfigGlobalFilter( )一. 前言第一次接觸STM32H7的FDCAN控制器時,覺得要
    發(fā)表于 12-20 19:03 ?16次下載
    <b class='flag-5'>STM32H743</b>+CubeMX-解決<b class='flag-5'>FDCAN</b>控制器無法接收遠(yuǎn)程幀

    STM32H743 FDCAN FIFO接收管理分析(HAL庫)

    MCU:H743野火挑戰(zhàn)者前言H743具備兩個接收FIFO,分別是FIFO 0和FIFO 1,功能相同;H743提供的是FDCAN,FDCAN
    發(fā)表于 12-20 19:39 ?15次下載
    <b class='flag-5'>STM32H743</b> <b class='flag-5'>FDCAN</b> FIFO接收管理分析(HAL庫)

    STM32cube實現(xiàn)STM32H743的USB驅(qū)動

    STM32cube實現(xiàn)STM32H743的USB驅(qū)動基于STM32cube實現(xiàn)STM32H743的USB驅(qū)動說明,CUBE配置未講述。USB驅(qū)動說明1、
    發(fā)表于 12-28 19:53 ?37次下載
    用<b class='flag-5'>STM32</b>cube實現(xiàn)<b class='flag-5'>STM32H743</b>的USB驅(qū)動

    stm32h743軟件模擬i2c驅(qū)動

    stm32h743 實現(xiàn)gpio模擬i2c資料分享
    發(fā)表于 12-01 11:12 ?7次下載