STM32 CAN使用记录:bxCAN基础通讯

2023-09-14 20:15:00

目的

CAN是非常常用的一种数据总线,被广泛用在各种车辆系统中。这篇文章将对STM32中CAN的使用做个示例。

CAN的一些基础介绍可以参考下面文章:
《CAN基础概念》https://blog.csdn.net/Naisu_kun/article/details/132814079

本文使用STM32F407作为主控芯片,CAN电路设计如下:
在这里插入图片描述

本文使用使用STM32CubeIDE进行开发。

关键配置与代码

轮询方式

在这里插入图片描述
在这里插入图片描述

除了默认生成的代码只需在 main.c 中手动添加一些代码即可:

int main(void)
{
  HAL_Init();
  SystemClock_Config();
  MX_GPIO_Init();
  MX_CAN1_Init();

  /**************** 以下为过滤器设置 ****************/
  CAN_FilterTypeDef  sFilterConfig;

  sFilterConfig.FilterMode = CAN_FILTERMODE_IDMASK; // 使用MASK模式
  sFilterConfig.FilterScale = CAN_FILTERSCALE_32BIT; // 过滤器使用32bit模式
  sFilterConfig.FilterIdHigh = 0x0000;
  sFilterConfig.FilterIdLow = 0x0000;
  sFilterConfig.FilterMaskIdHigh = 0x0000;
  sFilterConfig.FilterMaskIdLow = 0x0000;
  sFilterConfig.FilterFIFOAssignment = CAN_RX_FIFO0; // 经过过滤的数据放到FIFO0
  sFilterConfig.FilterBank = 0; // 使用第0组过滤器
  sFilterConfig.SlaveStartFilterBank = 14; // 从CAN过滤器其实地址
  sFilterConfig.FilterActivation = ENABLE; // 使能过滤器

  if(HAL_CAN_ConfigFilter(&hcan1, &sFilterConfig) != HAL_OK)
  {
    // 过滤器设置失败
  }

  /**************** 以下为启动CAN外设 ****************/
  HAL_CAN_Start(&hcan1); // 启动CAN

  while (1)
  {

	/**************** 以下为接收消息并回发处理 ****************/
	if(HAL_CAN_GetRxFifoFillLevel(&hcan1, CAN_RX_FIFO0) != 0) // 接收队列不为0,有数据可读
	{
		CAN_RxHeaderTypeDef   RxHeader; // 用来保存接收到的数据帧头部信息
		uint8_t               RxData[8]; // 用来保存接收数据端数据

		if(HAL_CAN_GetRxMessage(&hcan1, CAN_RX_FIFO0, &RxHeader, RxData) == HAL_OK) // 从接收队列中读取数据帧
		{
			CAN_TxHeaderTypeDef   TxHeader; // 用来保存发送数据帧头部信息
			uint8_t               TxData[8]; // 用来保存发送数据帧数据
			uint32_t              TxMailbox; // 用来保存发送数据帧所使用的邮箱号

			if(RxHeader.IDE == CAN_ID_STD) // 标准数据帧
			{
				TxHeader.StdId = RxHeader.StdId; // 标准帧ID
			}
			if(RxHeader.IDE == CAN_ID_EXT) // 扩展数据帧
			{
				TxHeader.ExtId = RxHeader.ExtId; // 扩展帧ID
			}
			TxHeader.IDE = RxHeader.IDE; // ID类型

			TxHeader.RTR = RxHeader.RTR; // CAN_RTR_DATA 数据帧;CAN_RTR_REMOTE 远程帧

			TxHeader.DLC = RxHeader.DLC; // 数据段字节数

			for(int i=0; i<TxHeader.DLC; i++)
			{
				TxData[i] = RxData[i];
			}

			while(HAL_CAN_GetTxMailboxesFreeLevel(&hcan1) == 0); // 等待有发送邮箱可用

			HAL_CAN_AddTxMessage(&hcan1, &TxHeader, TxData, &TxMailbox); // 发送数据帧
		}
	}
  }
}

上面代码中的过滤器是接收数据才需要的。过滤器可以使只有ID匹配上的报文才接收到接收FIFO中。

过滤器最常使用MASK模式,该模式下与 FilterMaskId1 的bit位对应的 FilterId bit位的值必须完全匹配,比如下面示例:

  • FilterId = 0x00000233 FilterMaskId = 0xFFFFFFFF 只接收ID为 0x00000233 的报文;
  • FilterId = 0x00000233 FilterMaskId = 0xFFFFFF00 可以接收ID为 0x000002XX 的报文(X可以为任意值);
  • FilterId = 0x00000000 FilterMaskId = 0x00000000 可以接收ID为 0xXXXXXXXX 的报文(X可以为任意值);

需要注意的是就算不想过滤任何报文也需要设置过滤器,因为需要通过过滤器来指定接收的报文存放到哪个FIFO中。

中断方式

中断方式基础配置与上面相同,唯一的变化是使能相关中断:
在这里插入图片描述
(下面代码中实际只用到RX0中断)

除了默认生成的代码只需在 main.c 中手动添加一些代码即可:

#include "main.h"

CAN_HandleTypeDef hcan1;

void SystemClock_Config(void);
static void MX_GPIO_Init(void);
static void MX_CAN1_Init(void);

/**************** 以下为重写中断回调函数 ****************/
// 注意下面的一些回调函数其实是在HAL库中已经若定义好的,我们只需要在保持函数类型不变的情况下重写实现其实际功能

// Fifo0收到消息回调
void HAL_CAN_RxFifo0MsgPendingCallback(CAN_HandleTypeDef *hcan)
{
	if(hcan->Instance == CAN1) // 如果是来自CAN1的数据
	{
		CAN_RxHeaderTypeDef   RxHeader; // 用来保存接收到的数据帧头部信息
		uint8_t               RxData[8]; // 用来保存接收数据端数据

		if(HAL_CAN_GetRxMessage(&hcan1, CAN_RX_FIFO0, &RxHeader, RxData) == HAL_OK) // 从接收队列中读取数据帧
		{
			CAN_TxHeaderTypeDef   TxHeader; // 用来保存发送数据帧头部信息
			uint8_t               TxData[8]; // 用来保存发送数据帧数据
			uint32_t              TxMailbox; // 用来保存发送数据帧所使用的邮箱号

			if(RxHeader.IDE == CAN_ID_STD) // 标准数据帧
			{
				TxHeader.StdId = RxHeader.StdId; // 标准帧ID
			}
			if(RxHeader.IDE == CAN_ID_EXT) // 扩展数据帧
			{
				TxHeader.ExtId = RxHeader.ExtId; // 扩展帧ID
			}
			TxHeader.IDE = RxHeader.IDE; // ID类型

			TxHeader.RTR = RxHeader.RTR; // CAN_RTR_DATA 数据帧;CAN_RTR_REMOTE 远程帧

			TxHeader.DLC = RxHeader.DLC; // 数据段字节数

			for(int i=0; i<TxHeader.DLC; i++)
			{
				TxData[i] = RxData[i];
			}

			while(HAL_CAN_GetTxMailboxesFreeLevel(&hcan1) == 0); // 等待有发送邮箱可用

			HAL_CAN_AddTxMessage(&hcan1, &TxHeader, TxData, &TxMailbox); // 发送数据帧
		}
	}

}

// void HAL_CAN_RxFifo0FullCallback(CAN_HandleTypeDef *hcan) { } // Fifo0队满回调

// void HAL_CAN_TxMailbox0AbortCallback(CAN_HandleTypeDef *hcan) { } // TxMailbox0取消传输回调
// void HAL_CAN_TxMailbox0CompleteCallback(CAN_HandleTypeDef *hcan) { } // TxMailbox0传输完成回调
// void HAL_CAN_TxMailbox1AbortCallback(CAN_HandleTypeDef *hcan) { } // TxMailbox1取消传输回调
// void HAL_CAN_TxMailbox1CompleteCallback(CAN_HandleTypeDef *hcan) { } // TxMailbox1传输完成回调
// void HAL_CAN_TxMailbox2AbortCallback(CAN_HandleTypeDef *hcan) { } // TxMailbox2取消传输回调
// void HAL_CAN_TxMailbox2CompleteCallback(CAN_HandleTypeDef *hcan) { } // TxMailbox2传输完成回调

int main(void)
{
  HAL_Init();
  SystemClock_Config();
  MX_GPIO_Init();
  MX_CAN1_Init();

  /**************** 以下为过滤器设置 ****************/
  CAN_FilterTypeDef  sFilterConfig;

  sFilterConfig.FilterMode = CAN_FILTERMODE_IDMASK; // 使用MASK模式
  sFilterConfig.FilterScale = CAN_FILTERSCALE_32BIT; // 过滤器使用32bit模式
  sFilterConfig.FilterIdHigh = 0x0000;
  sFilterConfig.FilterIdLow = 0x0000;
  sFilterConfig.FilterMaskIdHigh = 0x0000;
  sFilterConfig.FilterMaskIdLow = 0x0000;
  sFilterConfig.FilterFIFOAssignment = CAN_RX_FIFO0; // 经过过滤的数据放到FIFO0
  sFilterConfig.FilterBank = 0; // 使用第0组过滤器
  sFilterConfig.SlaveStartFilterBank = 14; // 从CAN过滤器其实地址
  sFilterConfig.FilterActivation = ENABLE; // 使能过滤器

  if(HAL_CAN_ConfigFilter(&hcan1, &sFilterConfig) != HAL_OK)
  {
    // 过滤器设置失败
  }

  /**************** 以下为启动中断 ****************/
  HAL_CAN_ActivateNotification(&hcan1, CAN_IT_RX_FIFO0_MSG_PENDING); // 使能FIFO0数据接收中断

  /**************** 以下为启动CAN外设 ****************/
  HAL_CAN_Start(&hcan1); // 启动CAN

  while (1)
  {
  }
}

收发测试

本示例演示结果可以通过各种CAN工具配合上位机软件进行测试:
在这里插入图片描述

示例链接

仓库地址: https://github.com/NaisuXu/STM32_MCU_Examples

本文中的示例位于仓库中 CAN_RxTxPoll_F407CAN_RxTxIT_F407

总结

STM32中使用CAN非常方便,进行配置生成代码后只需要设置过滤器,然后就可以收发数据了。

更多推荐

Python异常处理之分享

异常处理在项目开发中,异常处理是不可或缺的。异常处理帮助人们debug,通过更加丰富的信息,让人们更容易找到bug的所在。异常处理还可以提高程序的容错性。我们之前在讲循环对象的时候,曾提到一个StopIteration的异常,该异常是在循环对象穷尽所有元素时的报错。我们以它为例,来说明基本的异常处理。一个包含异常的程序

Python-requests库入门指南

介绍Python编写的HTTP库,能够发送HTTP和HTTPS请求,并且获取响应。在测试服务器响应方面经常使用。下载pipinstallrequests使用常用的格式requests.get(url,params=None,**kwargs)requests.post(url,data=None,json=None,*

ChatGPT在电子健康记录和医疗信息查询中的应用前景如何?

电子健康记录(EHRs)和医疗信息查询在现代医疗保健系统中起着至关重要的作用。它们有助于提高患者护理的质量,提高医疗保健的效率,减少错误,促进患者参与,并促进医学研究和数据驱动的决策。ChatGPT作为一种人工智能技术,在这一领域具有巨大的潜力,可以改善EHR的创建、维护和利用,以及医疗信息查询的效率和准确性。以下是C

命令模式-

定义:又叫动作模式或事务模式。指的是将一个请求封装成一个对象,使发出请求的责任和执行请求的责任分割开,然后可以使用不同的请求把客户端参数化,这样可以使得两者之间通过命令对象进行沟通,从而方便将命令对象进行储存、传递、调用、增加与管理。应用场景:1、对于很多数的请求-响应模式的功能,比较适合使用命令模式,命令模式对实现记

【ComfyUI】安装 之 window版

文章目录序言步骤下载comfyUI配置大模型和vae下载依赖组件启动生成图片解决办法序言由于stablediffusionwebui无法做到对流程进行控制,只是点击个生成按钮后,一切都交给AI来处理。但是用于生产生活是需要精细化对各个流程都要进行控制的。故也就有个今天的猪脚:Comfyui步骤下载comfyui项目配置

Python工程师Java之路(p)Maven聚合和继承

文章目录依赖管理依赖传递可选依赖和排除依赖继承与聚合依赖管理指当前项目运行所需的jar,一个项目可以设置多个依赖<!--设置当前项目所依赖的所有jar--><dependencies><!--设置具体的依赖--><dependency><!--依赖所属群组id--><groupId>org.springframewor

Java中ArrayList 和 LinkedList 的区别

ArrayList和LinkedList都是Java中常见的集合类,它们用于存储和操作数据。它们之间的主要区别在于其底层数据结构和性能特征:底层数据结构:ArrayList使用动态数组作为其底层数据结构。动态数组的容量会根据需要自动扩展。LinkedList使用双向链表作为其底层数据结构。每个元素都包含对前一个和后一个

【基础篇】四、SpringBoot整合第三方技术

文章目录1、SpringBoot整合Junit2、SpringBoot整合MyBatis3、SpringBoot整合MyBatisPlus4、SpringBoot整合Druid1、SpringBoot整合Junit步骤:导入测试对应的starter测试类使用@SpringBootTest修饰使用自动装配的形式添加要测试

Hexo在多台电脑上提交和更新

文章目录1.博客搭建2.创建一个新文件夹new,用于上传hexo博客目录3.github上创建hexo分支并设置为默认分支创建hexo分支将hexo分支设置为默认分支4.进入新建的文件夹中gitclone,再上传相关文件至hexo分支1.clone下来的文件夹内应该有个.gitignore文件,用来忽略一些不需要的文件

Docker安装ElasticSearch/ES 7.10.0

目录前言安装ElasticSearch/ES安装步骤1:准备1.安装docker2.搜索可以使用的镜像。3.也可从dockerhub上搜索镜像。4.选择合适的redis镜像。安装步骤2:拉取ElasticSearch镜像1拉取镜像2查看已拉取的镜像安装步骤3:创建容器创建容器方式1:快速创建容器安装步骤4:运行容器安装

ELT in ByteHouse 实践与展望

更多技术交流、求职机会,欢迎关注字节跳动数据平台微信公众号,回复【1】进入官方交流群谈到数据仓库,一定离不开使用Extract-Transform-Load(ETL)或Extract-Load-Transform(ELT)。将来源不同、格式各异的数据提取到数据仓库中,并进行处理加工。传统的数据转换过程一般采用Extra

热文推荐