多个CAN同时发送数据,会死机

现在有7 8个STM32同时用CAN发送数据,其中有些STM32的CAN发送不出来,应该是被仲裁掉了,这时候关闭其他的STM32,这个还是发送不出来,看起来死机了一样,请问这是一种什么现象,应该怎么解决

解决方法:

1.检查CAN控制器的配置和设置,确保它们正确并与网络中的其他节点相匹配。

2.检查CAN网络连接和其它节点的状态,确保网络通信正常。

3.如果存在优先级冲突的问题,可以尝试通过调整消息的优先级来解决。在STM32中,可以通过设置CAN控制器的滤波器或者使用优先级寄存器来调整消息的优先级。

4.如果问题仍然存在,建议对CAN控制器进行调试或重置。可以使用调试接口或重新上电来重置CAN控制器

【相关推荐】



  • 这有个类似的问题, 你可以参考下: https://ask.csdn.net/questions/5433677
  • 我还给你找了一篇非常好的博客,你可以看看是否有帮助,链接:STM32 HAL库CAN同时发送超过3条推荐方法
  • 您还可以看一下 Abel小智老师的嵌入式开发系统学习路线 从基础到项目 精品教程 工程师必备课程 物联网课程中的 STM32如何使用定时器?小节, 巩固相关知识点
  • 除此之外, 这篇博客: 【STM32】单片机之CAN通讯应用中的 3、CAN初始化配置过程(⭐⭐⭐⭐⭐) 部分也许能够解决你的问题, 你可以仔细阅读以下内容或跳转源博客中阅读:

    初始化函数:u8 CAN1_Mode_Init(u8 tsjw,u8 tbs2,u8 tbs1,u16 brp,u8 mode)

    (1)首先定义四个结构体变量(后面配置过程中会用到)

    	GPIO_InitTypeDef GPIO_InitStructure; 
        CAN_InitTypeDef        CAN_InitStructure;
      	CAN_FilterInitTypeDef  CAN_FilterInitStructure;	
    #if CAN1_RX0_INT_ENABLE 
       	NVIC_InitTypeDef  NVIC_InitStructure;  //有可能会用到,不是必须配置的
    #endif
    

    注意:结构体已经在库函数中定义过了。鼠标右键单击GPIO_InitTypeDef,点击Go To Definition of 'GPIO_InitTypeDef'即可找到结构体。在stm32f4xx_can.h文件中。

    (2)使能相关时钟,初始化GPIO,配置相关引脚的复用功能 。

    //使能相关时钟
    RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOA, ENABLE);//使能PORTA时钟	                   											 
    RCC_APB1PeriphClockCmd(RCC_APB1Periph_CAN1, ENABLE);//使能CAN1时钟	
    	
    //初始化GPIO
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_11| GPIO_Pin_12;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF;//复用功能
    GPIO_InitStructure.GPIO_OType = GPIO_OType_PP;//推挽输出
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_100MHz;//100MHz
    GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_UP;//上拉
    GPIO_Init(GPIOA, &GPIO_InitStructure);//初始化PA11,PA12
    
    //引脚复用映射配置
    GPIO_PinAFConfig(GPIOA,GPIO_PinSource11,GPIO_AF_CAN1); //GPIOA11复用为CAN1
    GPIO_PinAFConfig(GPIOA,GPIO_PinSource12,GPIO_AF_CAN1); //GPIOA12复用为CAN1
    

    (3)CAN单元设置(工作模式和波特率)

    //CAN单元设置
    //CAN工作模式设置
    CAN_InitStructure.CAN_TTCM=DISABLE;	//非时间触发通信模式   
    CAN_InitStructure.CAN_ABOM=DISABLE;	//软件自动离线管理	  
    CAN_InitStructure.CAN_AWUM=DISABLE;//睡眠模式通过软件唤醒(清除CAN->MCR的SLEEP位)
    CAN_InitStructure.CAN_NART=ENABLE;	//禁止报文自动传送 
    CAN_InitStructure.CAN_RFLM=DISABLE;	//报文不锁定,新的覆盖旧的  
    CAN_InitStructure.CAN_TXFP=DISABLE;	//优先级由报文标识符决定 
    CAN_InitStructure.CAN_Mode= mode;	 //模式设置,0为普通模式;1为回环模式
    //配置波特率 
    CAN_InitStructure.CAN_SJW=tsjw;	//重新同步跳跃宽度(Tsjw)为tsjw+1个时间单位 CAN_SJW_1tq~CAN_SJW_4tq
    CAN_InitStructure.CAN_BS1=tbs1; //Tbs1范围CAN_BS1_1tq ~CAN_BS1_16tq
    CAN_InitStructure.CAN_BS2=tbs2;//Tbs2范围CAN_BS2_1tq ~	CAN_BS2_8tq
    CAN_InitStructure.CAN_Prescaler=brp;  //分频系数(Fdiv)为brp+1
    	
    CAN_Init(CAN1, &CAN_InitStructure);   // 初始化CAN1 
    

    (4)设置过滤器

    //配置过滤器
    CAN_FilterInitStructure.CAN_FilterNumber=0;	  //过滤器0
    CAN_FilterInitStructure.CAN_FilterMode=CAN_FilterMode_IdMask; 
    CAN_FilterInitStructure.CAN_FilterScale=CAN_FilterScale_32bit; //32位 
    CAN_FilterInitStructure.CAN_FilterIdHigh=0x0000;//32位ID
    CAN_FilterInitStructure.CAN_FilterIdLow=0x0000;
    CAN_FilterInitStructure.CAN_FilterMaskIdHigh=0x0000;//32位MASK
    CAN_FilterInitStructure.CAN_FilterMaskIdLow=0x0000;
    CAN_FilterInitStructure.CAN_FilterFIFOAssignment=CAN_Filter_FIFO0;//过滤器0关联到FIFO0
    CAN_FilterInitStructure.CAN_FilterActivation=ENABLE; //激活过滤器0
    
    CAN_FilterInit(&CAN_FilterInitStructure);//滤波器初始化
    

    (5)CAN_RX0中断使能(没有用到,可以不配置)

    #if CAN1_RX0_INT_ENABLE	
        CAN_ITConfig(CAN1,CAN_IT_FMP0,ENABLE);//FIFO0消息挂号中断允许.		    
      
      	NVIC_InitStructure.NVIC_IRQChannel = CAN1_RX0_IRQn;
      	NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 1; // 主优先级为1
      	NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0;        // 次优先级为0
      	NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
      	NVIC_Init(&NVIC_InitStructure);
    #endif
    

    (6)发送消息及CAN状态获取

    //can发送一组数据(固定格式:ID为0X12,标准帧,数据帧)	
    //len:数据长度(最大为8)				     
    //msg:数据指针,最大为8个字节.
    u8 CAN1_Send_Msg(u8* msg,u8 len)
    {	
      //发送消息
      u8 mbox;
      u16 i=0;
      CanTxMsg TxMessage;
      TxMessage.StdId=0x12;	 // 标准标识符为0
      TxMessage.ExtId=0x12;	 // 设置扩展标示符(29位)
      TxMessage.IDE=0;		  // 使用扩展标识符
      TxMessage.RTR=0;		  // 消息类型为数据帧,一帧8位
      TxMessage.DLC=len;							 // 发送两帧信息
      for(i=0;i<len;i++)
      TxMessage.Data[i]=msg[i];				 // 第一帧信息          
      mbox= CAN_Transmit(CAN1, &TxMessage);  
      //CAN状态获取 
      i=0;
      while((CAN_TransmitStatus(CAN1, mbox)==CAN_TxStatus_Failed)&&(i<0XFFF))i++;	//等待发送结束
      if(i>=0XFFF)return 1;
      return 0;		
    }
    

    (7)接收消息及CAN状态获取

    //情况1:
    #if CAN1_RX0_INT_ENABLE	//使能RX0中断
    //中断服务函数			    
    void CAN1_RX0_IRQHandler(void)
    {
      	CanRxMsg RxMessage;
    	int i=0;
        CAN_Receive(CAN1, 0, &RxMessage);
    	for(i=0;i<8;i++)
    	printf("rxbuf[%d]:%d\r\n",i,RxMessage.Data[i]);
    }
    #endif
    
    //情况2:
    //can口接收数据查询
    //buf:数据缓存区;	 
    u8 CAN1_Receive_Msg(u8 *buf)
    {		   		   
     	//接收消息
     	u32 i;
    	CanRxMsg RxMessage;
    	//CAN状态获取
        if(CAN_MessagePending(CAN1,CAN_FIFO0)==0)return 0;//没有接收到数据,直接退出 
        CAN_Receive(CAN1, CAN_FIFO0, &RxMessage);//读取数据	
        for(i=0;i<RxMessage.DLC;i++)
        buf[i]=RxMessage.Data[i];  
    	return RxMessage.DLC;	
    }
    

    编程要点:
    (1) 使能 CAN 外设的时钟;
    (2) 初始化 CAN 通讯使用的目标引脚及端口时钟;
    (3) 配置 CAN 外设的工作模式、位时序以及波特率;
    (4) 配置过滤器的工作方式。


如果你已经解决了该问题, 非常希望你能够分享一下解决方案, 写成博客, 将相关链接放在评论区, 以帮助更多的人 ^-^