waiman-156411 发表于 2017-1-5 13:39:51

STM32F107,CAN发送错误,导致CAN离线状态,如何恢复CAN总线?

CAN规范里头提到,如果一个节点由于发送错误计数器的值不小于255时,它会进入bus off离线状态。
当它监听到总线上有128次连续11个“隐性”位后,允许处于bus off状态的节点转到“error active”状态。
但问题是,我的STM32F107是主机,负责发送CAN信号的,当由于信号线的问题(CANH和CANL短路),导致主机发送错误,离线了(CAN_ESR上的TEC>255),CAN上也就没有信号,那么从何而来《128次连续11个“隐性”位》进入恢复过程。

而且 CAN_MCR里的RESET置位,也不能复位CAN寄存器,CAN_ESR上错误次数还是存在。难道只能强行复位芯片才能解决?

下面是我的错误中断里的处理代码

void HAL_CAN_ErrorCallback(CAN_HandleTypeDef *hcan)
{
        uint32_t err = hcan->ErrorCode;
        uint8_t i=0;
        uint8_t status=0;
        Process_Status = 0;
       
        uint32_t canTSR = hcan->Instance->TSR;
        uint32_t canABRQ = CAN_TSR_ABRQ0;        // 终止发送
        uint32_t canTERR = CAN_TSR_TERR0;        // 发送失败
        uint32_t canALST = CAN_TSR_ALST0;        // 仲裁失败
       
        switch(err){
          case HAL_CAN_ERROR_EWG:        // EWG error   
                printf("CAN ERRROR = EWG\r\n");
                break;
          case HAL_CAN_ERROR_EPV:        // EPV error
                printf("CAN ERRROR = EPV\r\n");
                break;
          case HAL_CAN_ERROR_BOF:        // BOF error
                printf("CAN ERRROR = BOF\r\n");
                break;
          case HAL_CAN_ERROR_STF:        // Stuff error
                printf("CAN ERRROR = Stuff\r\n");
                break;
          case HAL_CAN_ERROR_FOR:        // Form error
                printf("CAN ERRROR = Form\r\n");
                break;
          case HAL_CAN_ERROR_ACK:        // Acknowledgment error
                printf("CAN ERRROR = ACK\r\n");
                break;
          case HAL_CAN_ERROR_BR:        // Bit recessive
                printf("CAN ERRROR = Bit recessive\r\n");
                break;
          case HAL_CAN_ERROR_BD:        // LEC dominant
                printf("CAN ERRROR = LEC dominant\r\n");
                break;
          case HAL_CAN_ERROR_CRC:        // LEC transfer error
                printf("CAN ERRROR = LEC transfer\r\n");
                break;
          case HAL_CAN_ERROR_NONE:        // No error
                printf("CAN ERRROR = NO\r\n");
                break;
               
          default:
                break;
        }
       
       
       
    hcan->ErrorCode = HAL_CAN_ERROR_NONE;
    hcan->State = HAL_CAN_STATE_READY;
       
        hcan->Instance->IER = 0;        // 清除全部中断
        hcan->Instance->ESR = 0;        // 清除错误计数
       
        // 清空发送邮箱       
        for(i=0;i<3;i++){
                if((canTSR & canTERR) !=0){
                        hcan->Instance->TSR |=canABRQ;
                }
                if((canTSR & canALST) !=0){
                        hcan->Instance->TSR |=canABRQ;
                }
                canABRQ <<=8;
                canTERR <<=8;
                canALST <<=8;
               
        }
        for(i=0;i<3;i++){
                hcan->Instance->sTxMailBox.TIR=0;
                hcan->Instance->sTxMailBox.TDTR=0;
                hcan->Instance->sTxMailBox.TDLR=0;
                hcan->Instance->sTxMailBox.TDHR=0;
        }
       

        BSP_CAN_Reset(hcan);        // 复位CAN寄存器 CAN_MCR RESET置位
        HAL_CAN_MspDeInit(hcan);// 注销CAN实例
        status = BSP_CAN_Init(CAN_Baudrate);        //重新初始化CAN
}




任风吹吹 发表于 2017-12-7 10:30:48

楼主你是想知道BUSOFF后如何恢复吗? 将ABOM设1后,一旦检测到条件会自动恢复的,不需要人工干预。

但楼主的问题显然不是这个,楼主主要是对这句话“当它监听到总线上有128次连续11个“隐性”位后,允许处于bus off状态的节点转到“error active”状态。”不怎么理解吧?

当CAN节点进入到BUS OFF状态后,只要使能了ABOM位,那么它还是持续检测总线是否存在128次连续11个隐性位,这个是CAN接口恢复到主动错误状态的条件,那么,关键的是,如何理解这个128次连续11个隐性位。

那么我要问的是,什么是隐性位?当CANH=CANL=2.3V时! 这里是约等于,我没找到这个符号,用等号代替。

当CANH和CANL短接时,CAN节点本身并不知道(如果没有其他外围辅助检测电路的话),从CAN节点来看,只是知道CANH=CANL=2.3V,对于CAN节点来说,这个就是IDLE状态。所以当持续一段时间后,BUS OFF状态会解除,自动恢复到主动错误状态,但是回到主动错误状态后,由于CANH与CANL还是短接(CAN BUS IDLE状态),此时若发送报文,则会导致发送错误,进而导致TEC累加,最终再次导致进入到BUSOFF状态。如此反复循环。

在此过程中,楼主如查看BUSOFF寄存器的状态,楼主会发现有时为0,有时为1,这就是返回循环导致的。

进入到BUSS OFF状态需要发送来驱动,但从BUS OFF状态恢复到主动错误状态并不需要任何操作来驱动,它就是自动检测128次连续11位隐性位这个条件来决定的。

这里关键的是,当CANH与CANL短接时,对于CAN节点来说,就是一直为IDLE,当CAN总线上没有任何消息时,也是IDLE,CAN节点本身在没有辅助电路的情况下是无法区分这两种情况的。

发表于 2017-1-5 14:42:25

没有遇到类似问题,从手册看,以下是解决方法:
如果ABOM位为’1’,bxCAN进入离线状态后,就自动开启恢复过程。
如果ABOM位为’0’,软件必须先请求bxCAN进入然后再退出初始化模式,随后恢复过程才被开启。

xmshao 发表于 2017-1-5 14:54:31

你配置ABOM为1,可以自动恢复的。你不妨验证下。

andypanfan 发表于 2017-1-6 09:12:01

研究的比较的深,还没有考虑过!!!!

waiman-156411 发表于 2017-1-6 11:07:54

安 发表于 2017-1-5 14:42
没有遇到类似问题,从手册看,以下是解决方法:
如果ABOM位为’1’,bxCAN进入离线状态后,就自动开启恢复 ...

在初始化的时候已经置位了,但就是恢复不了,重新发can也发不出去。

手册里说的恢复过程是:需要CAN总线上有128次连续11个隐形位,相当于,用CAN总线上的数据把CAN_ESR里的TEC域递减清零。才能恢复完成。但我这个是主机,停止CAN发送的话,CAN总线上也没有数据存在。所以就清除不了错误。

难道是互联产品的从CAN就是用在恢复主CAN的?
当主CAN出现故障之后,从CAN用来发数据,清除主CAN的CAN_ESR计数?然后恢复现场?

发表于 2017-1-7 11:12:00

楼主把重发机制关闭了试试。

waiman-156411 发表于 2017-1-19 09:15:58

安 发表于 2017-1-7 11:12
楼主把重发机制关闭了试试。

没有开重发机制。

CAN总线默认的状态是隐性电平“1” (没有差分信号),那么CAN总线上如果没有数据在传输,那么128次连续11个隐性位,相当于总线闲置过一段时间之后,CAN会自己恢复?

jcx0324 发表于 2017-1-19 12:24:49

可以设置自动重启

yzez 发表于 2017-12-6 17:32:36

楼主问题解决了吗。。。

hjl2832 发表于 2017-12-7 09:13:54

关注,标记,CAN问题
页: [1] 2
查看完整版本: STM32F107,CAN发送错误,导致CAN离线状态,如何恢复CAN总线?