CAN总线学习笔记|CAN盒测试STM32的CAN中断接收

发布者:Xiangsi最新更新时间:2023-08-01 来源: elecfans关键字:CAN总线  STM32  CAN  中断接收 手机看文章 扫描二维码
随时随地手机看文章

CAN基础知识介绍文中介绍了CAN协议相关的基础知识,以及STM32F4芯片的CAN控制器相关知识,下面将通过实例,利用STM32CubeMX图形化配置工具,并配合CAN盒,来实现CAN通讯的中断收发测试


1. STM32CubeMX配置

CAN是挂载在APB1总线上,设置PCLK1时钟频率到最大45MHz

3ef368ea-957d-11ed-bfe3-dac502259ad0.png

激活CAN1,配置位时序参数,其他基本参数以及工作模式(此处设置为Normal普通模式)3f18da80-957d-11ed-bfe3-dac502259ad0.png

CAN波特率的计算公式:只需要知道BS1和BS2的设置,以及APB1的时钟频率,就可以方便的计算出波特率。比如设置TS1=8、TS2=6和BRP=6,在APB1频率为45Mhz的条件下,即可得到CAN通信的波特率=45000/6/(8+6+1)=500Kbps
3f48a260-957d-11ed-bfe3-dac502259ad0.png

激活USART1作为调试串口,配置相关LED对应的GPIO引脚作为指示灯

2.MDK-ARM编程

2.1 几个重要的结构体

CAN 初始化结构体:此结构体内容,可由STM32CubeMX工具进行配置

typedefstruct

{

uint32_tPrescaler;/*配置CAN外设的时钟分频,可设置为1-1024*/

uint32_tMode;/*配置CAN的工作模式,回环或正常模式*/

uint32_tSyncJumpWidth;/*配置SJW极限值*/

uint32_tTimeSeg1;/*配置BS1段长度*/

uint32_tTimeSeg2;/*配置BS2段长度*/

FunctionalStateTimeTriggeredMode;/*是否使能TTCM时间触发功能*/

FunctionalStateAutoBusOff;/*是否使能ABOM自动离线管理功能*/

FunctionalStateAutoWakeUp;  /*是否使能AWUM自动唤醒功能*/

FunctionalStateAutoRetransmission;/*是否使能NART自动重传功能*/

FunctionalStateReceiveFifoLocked;/*是否使能RFLM锁定FIFO功能*/

FunctionalStateTransmitFifoPriority;/*配置TXFP报文优先级的判定方法*/

}CAN_InitTypeDef;

发送及接收头结构体:主要用于构造发送报文,以及接收报文。收发发文时,需要自定义头结构体变量


typedefstruct

{

uint32_tStdId;/*存储报文的标准标识符11位,0-0x7FF*/

uint32_tExtId;/*存储报文的扩展标识符29位,0-0x1FFFFFFF*/

uint32_tIDE;/*存储IDE扩展标志*/

uint32_tRTR;/*存储RTR远程帧标志*/

uint32_tDLC;/*存储报文数据段的长度,0-8*/

FunctionalStateTransmitGlobalTime;

}CAN_TxHeaderTypeDef;


typedefstruct

{

uint32_tStdId;/*存储报文的标准标识符11位,0-0x7FF.*/

uint32_tExtId;/*存储报文的扩展标识符29位,0-0x1FFFFFFF*/

uint32_tIDE;/*存储IDE扩展标志*/

uint32_tRTR;/*存储RTR远程帧标志*/

uint32_tDLC;/*存储报文数据段的长度,0-8*/

uint32_tTimestamp;

uint32_tFilterMatchIndex;

}CAN_RxHeaderTypeDef;

过滤器结构体:STM32CubeMX不会初始化过滤器的相关内容,需要自己添加

typedefstruct

{

uint32_tFilterIdHigh;/*CAN_FxR1寄存器的高16位*/

uint32_tFilterIdLow;/*CAN_FxR1寄存器的低16位*/

uint32_tFilterMaskIdHigh;/*CAN_FxR2寄存器的高16位*/

uint32_tFilterMaskIdLow;/*CAN_FxR2寄存器的低16位*/

uint32_tFilterFIFOAssignment;/*设置经过筛选后数据存储到哪个接收FIFO*/

uint32_tFilterBank;/*筛选器编号,范围0-27,CAN1是0-13,CAN2是14-27*/

uint32_tFilterMode;/*筛选器模式*/

uint32_tFilterScale;/*设置筛选器的尺度*/

uint32_tFilterActivation;/*是否使能本筛选器*/

uint32_tSlaveStartFilterBank;/*CAN2起始过滤器组*/

}CAN_FilterTypeDef;

2.2 程序编写


生成工程后,打开can.c文件,可见STM32CubeMX已经对位时序参数、其他基本参数以及工作模式进行了初始化。但是并没有初始化过滤器的相关内容,因此需要我们自己添加,并在CAN初始化时调用


//下面的设置只使能了FIFO0,并不过滤任何消息

voidCAN_Filter_Config(){

 CAN_FilterTypeDefsFilterConfig;


 sFilterConfig.FilterBank=0;//筛选器编号,CAN1是0-13,CAN2是14-27

 sFilterConfig.FilterMode=CAN_FILTERMODE_IDMASK;//采用掩码模式

 sFilterConfig.FilterScale=CAN_FILTERSCALE_32BIT;//设置筛选器的尺度,采用32位

 sFilterConfig.FilterIdHigh=0X0000;//过滤器ID高16位,即CAN_FxR1寄存器的高16位

 sFilterConfig.FilterIdLow=0X0000;//过滤器ID低16位,即CAN_FxR1寄存器的低16位

 sFilterConfig.FilterMaskIdHigh=0X0000;//过滤器掩码高16位,即CAN_FxR2寄存器的高16位

 sFilterConfig.FilterMaskIdLow=0X0000;//过滤器掩码低16位,即CAN_FxR2寄存器的低16位

 sFilterConfig.FilterFIFOAssignment=CAN_RX_FIFO0;//设置经过筛选后数据存储到哪个接收FIFO

 sFilterConfig.FilterActivation=ENABLE;//是否使能本筛选器

 sFilterConfig.SlaveStartFilterBank=14;//指定为CAN1分配多少个滤波器组


 if(HAL_CAN_ConfigFilter(&hcan1,&sFilterConfig)!=HAL_OK)

 {

  Error_Handler();

 }

}

编写发送和接收数据函数:此处将格式固定为标准数据帧,ID为12

uint8_tCAN1_Send_Msg(uint8_t*msg,uint8_tlen){

 uint16_ti=0;

 uint32_ttxMailBox;

 uint8_tsend_buf[8];


 txHeader.StdId=12;

 txHeader.ExtId=12;

 txHeader.IDE=CAN_ID_STD;

 txHeader.RTR=CAN_RTR_DATA;

 txHeader.DLC=len;


 for(i=0;i< len; i++)

    send_buf[i] = msg[i];

 

  if(HAL_CAN_AddTxMessage(&hcan1, &txHeader, send_buf, &txMailBox) != HAL_OK)

    return 1; 

  return 0;

}


void HAL_CAN_RxFifo0MsgPendingCallback(CAN_HandleTypeDef *hcan){

  uint16_t i = 0;

  uint8_t buf[8] = {0};

 

  if(hcan->Instance==CAN1){

  printf("*******************************

");

  printf("RecvviaSTM32F429Interrupt

");


  HAL_CAN_GetRxMessage(&hcan1,CAN_RX_FIFO0,&rxHeader,buf);


  if(rxHeader.IDE==CAN_ID_STD)

   printf("StdIdID:%d

",rxHeader.StdId);

  else

   printf("ExtIdID:%d

",rxHeader.ExtId);

  printf("

");

  printf("CANIDE:%d

",rxHeader.IDE);printf("

");

  printf("CANRTR:%d

",rxHeader.RTR);printf("

");

  printf("CANDLC:%d

",rxHeader.DLC);printf("

");

  printf("RecvData:");


  for(i=0;i< rxHeader.DLC; i++)

      printf("%c ",buf[i]);

 

    printf("

");

    printf("*******************************

");

  }

}

默认Cubemx生成的代码并没有can start,没有调用HAL_CAN_Start(&hcan1) 来使能CAN,因此需要在CAN初始化代码中添加

voidMX_CAN1_Init(void){

 ......

 /*USERCODEBEGINCAN1_Init2*/

 CAN_Filter_Config();

 HAL_CAN_Start(&hcan1);

 HAL_CAN_ActivateNotification(&hcan1,CAN_IT_RX_FIFO0_MSG_PENDING);

/*USERCODEENDCAN1_Init2*/

}

主函数main.c中,代码如下

intmain(void){

HAL_Init();

SystemClock_Config();

MX_GPIO_Init();

MX_CAN1_Init();

MX_USART1_UART_Init();

/*USERCODEBEGIN2*/

 uint8_tret;

 printf("CANTesting....!

");

 uint8_ttxdata[8]={78,79,82,77,65,76,33,32};

/*USERCODEEND2*/

/*Infiniteloop*/

/*USERCODEBEGINWHILE*/

while(1)

{

  HAL_GPIO_TogglePin(GPIOB,GPIO_PIN_1);

  printf("StartSenddata...

");


  ret=CAN1_Send_Msg(txdata,8);

  if(ret==0)

   printf("STM32F429CANSendsuccess!

");

  else

   printf("STM32F429CANSendfailed!

");


  HAL_Delay(3000);

/*USERCODEENDWHILE*/

/*USERCODEBEGIN3*/

}

/*USERCODEEND3*/

}


3. 下载测试

将CAN盒与STM32的CAN接口连接起来,CAN盒连接电脑,使用CAN调试软件接收和发送数据,如下图示

3f5877f8-957d-11ed-bfe3-dac502259ad0.png

STM32工程编译无误后,下载到开发板,可以看到系统运行时D1指示灯不断闪烁,串口不断的打印STM32发送CAN消息成功的信息。使用CAN调试软件,可以看到CAN盒接收到了STM32发出的数据。使用调试软件,发送CAN数据给STM32

3f959af2-957d-11ed-bfe3-dac502259ad0.png

串口调试助手中可以看到,STM32通过中断接收到了CAN盒发来的数据

3fb3c7ac-957d-11ed-bfe3-dac502259ad0.png


关键字:CAN总线  STM32  CAN  中断接收 引用地址:CAN总线学习笔记|CAN盒测试STM32的CAN中断接收

上一篇:STM32单片机学习笔记(5):ADC模数转换器
下一篇:关于STM32的除以0运算问题

推荐阅读最新更新时间:2024-11-07 02:53

Microchip开始提供业界第一款外部CAN灵活数据速率控制器
电子网消息,Microchip Technology Inc.(美国微芯科技公司)日前宣布,开始提供业界第一款外部CAN灵活数据速率(CAN FD)控制器。采用MCP2517FD,设计人员能够很快从CAN 2.0升级到CAN FD,受益于CAN FD增强协议。 CAN FD相对于传统的CAN 2.0有很多优势,包括更快的数据速率和数据字节消息扩展等。前沿的MCP2517FD CAN FD控制器可用于任何单片机(MCU),使开发人员能够轻松实现这一技术,而且完全不需要从新设计系统。由于CAN FD的应用和转换还处于起步阶段,因此,目前可用的CAN FD MCU数量还有限。此外,换一个系统MCU会明显增加成本,开发时间和风险
[半导体设计/制造]
STM32 编译指令 #pragma pack 的配对使用
#pragma pack 可以用来指定数据结构的成员变量的内存对齐数值。 可选值为: 1、2、4、8、16。 使用 pack 指令要配对使用,以避免意外影响项目中其他源文件的结构成员的内存对齐。 如果影响了其他源文件的结构成员内存对齐,那么在你按照默认对齐来计算那些结构成员占用内存大小或者使用指针移动计算结构成员偏移位置的时候,就可能会出现意料之外的异常。 主要可能的异常是内存定位错误或非法内存访问,结果可能导致错误的定位或数值,极端的情况下可能导致程序崩溃。 下面的例子用来展示基本的配对使用方式。 1)#pragma pack(n)的配对使用 #pragma pack(1) //内存对齐设置为1个字节
[单片机]
STM32 PWM输出
a) 目的:基础PWM输出,以及中断配合应用。输出选用PB1,配置为TIM3_CH4,是目标板的LED6控制脚。 b) 对于简单的PWM输出应用,暂时无需考虑TIM1的高级功能之区别。 c) 初始化函数定义: void TIM_Configuration(void); //定义TIM初始化函数 d) 初始化函数调用: TIM_Configuration(); //TIM初始化函数调用 e) 初始化函数,不同于前面模块,TIM的初始化分为两部分 基本初始化和通道初始化: void TIM_Configuration(void)//TIM初始化函数
[单片机]
STM32入门笔记(3)——GPIO
STM32的每个I/O口都由7个寄存器控制:2个32位端口配置寄存器CRL和CRH;2个32位的数据寄存器IDR和ODR;1个32位的置位复位寄存器BSRR;1个16位的复位寄存器BRR;1个32位的锁存寄存器LCKR;最常用的有4个:CRL/CRH/IDR/ODR;因此通常情况下只需要配置好以上四个寄存器就可以了。而CRH和CRL作用是完全一样的,只不过CRH控制的是高8位的IO而CRL控制的是低8位的,因此IO的控制就显的比较简单了。 (1)使能IO时钟 老规矩,用到哪个IO使能哪组IO时钟。例:RCC- APB2ENR|=1 2; // 使能PORTA时钟 (2)配置IO输入/输出模式 STM32的输入输
[单片机]
基于PAJ7620和STM32手势识别应用案例
1.简介  手势识别是计算机科学和语言技术中的一个主题,目的是通过数学算法来识别人类手势。 手势可以源自任何身体运动或状态,但通常源自面部或手。 本领域中的当前焦点包括来自面部和手势识别的情感识别。 用户可以使用简单的手势来控制或与设备交互,而无需接触他们。姿势,步态和人类行为的识别也是手势识别技术的主题。手势识别可以被视为计算机理解人体语言的方式,从而在机器和人之间搭建比原始文本用户界面或甚至GUI(图形用户界面)更丰富的桥梁。 手势识别使人们能够与机器进行通信,并且无需任何机械设备即可自然交互。 使用手势识别的概念,可以将手指指向计算机屏幕,使得光标将相应地移动。 这可能使常规输入设备(如鼠标,键盘甚至触摸屏)变得冗余。越
[单片机]
基于PAJ7620和<font color='red'>STM32</font>手势识别应用案例
STM32与FPGA通信写数据出错问题解决方法
项目中需要使用STM32和FPGA通信,使用的是地址线和数据线,在FPGA中根据STM32的读写模式A的时序完成写入和读取。之前的PCB设计中只使用了8跟数据线和8根地址线,调试过程中没有发现什么问题,在现在的PCB中使用了8根地址线和16根数据线,数据宽度也改成了16位,刚开始是读取数据不正确,后来发现了问题,STM32在16位数据宽度下有个内外地址映射的问题,只需要把FPGA中的设定的地址乘以2在STM32中访问就可以了,但是在写操作的时候会出现写当前地址的时候把后面的地址写成0的情况,比如说我给FPGA中定义的偏移地址0x01写一个16位数据,按照地址映射,在STM32中我把地址写入0x02,。实际测试发现这个地址上的数据是
[单片机]
STM32模拟IIC读写24CXX
文件(iic.h): #define SDA_IN() {GPIOB- CRL&=0X0FFFFFFF;GPIOB- CRL|=0X80000000;} #define SDA_OUT() {GPIOB- CRL&=0X0FFFFFFF;GPIOB- CRL|=0X30000000;} #define IIC_SCL PBout(6) #define IIC_SDA PBout(7) #define READ_SDA PBin(7) void bsp_iic_init(void); void bsp_iic_start(void); void bsp_iic_st
[单片机]
基于ARM的橡塑机械双层网络监控器设计
0 引言 橡塑机械是现代制造技术中一类重要工具,是一种由数控装置、伺服驱动装置、机床主体和其他辅助装置构成的机电一体化产品,它被广泛应用在橡塑产品加工制造领域。随着科技的进步,橡塑机械正朝着高速度、智能化、多功能化、高可靠性的方向发展。在监控系统方面,尤其是近年来,也发展得更为集成化、网络化。作为提升橡塑机械性能的一个重要组成部分,其监控系统也得到了长足的发展。本文设计了一款基于ARM9、嵌入式Linux以及CAN总线技术的应用于橡塑机械的双层网络监控器,并介绍了成功应用于橡塑机械的实例。 1 双层网络系统整体设计 系统基于工业现场监控集成化、自动化的理念设计了下层CAN总线结构。在现场监控部分,监控器所需要完成
[单片机]
小广播
设计资源 培训 开发板 精华推荐

最新单片机文章
何立民专栏 单片机及嵌入式宝典

北京航空航天大学教授,20余年来致力于单片机与嵌入式系统推广工作。

换一换 更多 相关热搜器件

 
EEWorld订阅号

 
EEWorld服务号

 
汽车开发圈

电子工程世界版权所有 京B2-20211791 京ICP备10001474号-1 电信业务审批[2006]字第258号函 京公网安备 11010802033920号 Copyright © 2005-2024 EEWORLD.com.cn, Inc. All rights reserved