CAN 五: CAN编程实践

发布时间:2023年12月20日

1、CAN基本驱动步骤

(1)CAN参数初始化

  • 工作模式、波特率等
  • 函数:HAL_CAN_Init

(2)使能CAN时钟和初始化相关引脚

  • GPIO模式设为复用功能模式
  • 函数:HAL_CAN_MspInit(CAN的初始化回调函数)

(3)设置过滤器

  • 过滤器的配置
  • 函数:HAL_CAN_ConfigFilter

(4)CAN数据的接收和发送

  • HAL_CAN_AddTxMessage? 发送消息
  • HAL_CAN_GetRxMessage? 接收数据

(5)使能CAN相关中断/设置NVIC/编写中断服务函数

  • NVIC:中断控制器
  • __HAL_CAN_ENABLE_IT (可选)

2、开发环境

(1)KeilMDK:V5.38.0.0

(2)STM32CubeMX:V6.8.1

(3)MCU:STM32F407ZGT6

3、实验目的

(1)使用回环模式实现自发自收。

(2)CAN发送数据,然后接收数据,将接收到的数据通过串口发送出去。

4、原理图

(1)CAN芯片选择TJA1040。

(2)CAN_TX接PA12,CAN_RX接PA11。

5、STM32CubeMX创建工程及配置

5.1、补充内容

(1)查阅数据手册,CAN外设接在总线APB1上,时钟频率此处配置为36MHz。

(2)CubeMX创建工程、配置时钟、串口不做详细介绍。

5.2、CubeMX中CAN的配置

(1)使能CAN外设。

(2)配置CAN的参数

  • Bit Timings Parameters:位时序参数
    • Prescaler:分频系数
    • Time Quanta in Bit Segment 1:时间段1(配置为9标识9个时间单元)
    • Time Quanta in Bit Segment 2:时间段2
    • ReSynchronization Jump Width:重新同步跳跃宽度
    • TS1=8、TS2=7、BRP=3,波特率 = 36000 / [( 9 + 8 + 1 ) * 4] = 500Kbps
  • Basic Parameters:基本参数
    • Time Triggered Communication Mode:时间触发通信方式
    • Automatic Bus-Off Management:总线自动离线管理
    • Automatic Wake-Up Mode:自动唤醒模式
    • Automatic Retransmission:自动重发
    • Receive Fifo Locked Mode:接收Fifo锁定模式
    • Transmit Fifo Priority:发送Fifo优先级
  • Advanced Parameters:先进的参数
    • Operating Mode:操作模式

6、KeilMDK软件编写

6.1、CAN相关函数

CAN_TxHeaderTypeDef g_can1_txheader;    /* CAN发送结构体 */
CAN_RxHeaderTypeDef g_can1_rxheader;    /* CAN接收结构体 */
CAN_HandleTypeDef hcan1;     // CAN控制句柄

/* CAN1 init function */
void MX_CAN1_Init(void)
{
  hcan1.Instance = CAN1;
  hcan1.Init.Prescaler = 4;                  /* 分频系数 */            
  hcan1.Init.Mode = CAN_MODE_LOOPBACK;       /* 工作模式设置 环回模式:自发自收 */
  hcan1.Init.SyncJumpWidth = CAN_SJW_1TQ;    /* 重新同步跳跃宽度 */
  hcan1.Init.TimeSeg1 = CAN_BS1_9TQ;         /* 时间段1 */
  hcan1.Init.TimeSeg2 = CAN_BS2_8TQ;         /* 时间段2 */
  hcan1.Init.TimeTriggeredMode = DISABLE;    /* 禁止时间触发通信模式 */
  hcan1.Init.AutoBusOff = DISABLE;           /* 禁止自动离线管理 */
  hcan1.Init.AutoWakeUp = DISABLE;           /* 禁止自动唤醒 */
  hcan1.Init.AutoRetransmission = DISABLE;   /* 禁止自动重发 */
  hcan1.Init.ReceiveFifoLocked = DISABLE;    /* 禁止接收FIFO锁定 */
  hcan1.Init.TransmitFifoPriority = DISABLE; /* 禁止发送FIFO优先级 */
  if (HAL_CAN_Init(&hcan1) != HAL_OK)
  {
    Error_Handler();
  }

}

void HAL_CAN_MspInit(CAN_HandleTypeDef* canHandle)
{

  GPIO_InitTypeDef GPIO_InitStruct = {0};
  if(canHandle->Instance==CAN1)
  {
  /* USER CODE BEGIN CAN1_MspInit 0 */

  /* USER CODE END CAN1_MspInit 0 */
    /* CAN1 clock enable */
    __HAL_RCC_CAN1_CLK_ENABLE();

    __HAL_RCC_GPIOA_CLK_ENABLE();
    /**CAN1 GPIO Configuration
    PA11     ------> CAN1_RX
    PA12     ------> CAN1_TX
    */
    GPIO_InitStruct.Pin = GPIO_PIN_11|GPIO_PIN_12;
    GPIO_InitStruct.Mode = GPIO_MODE_AF_PP;
    GPIO_InitStruct.Pull = GPIO_NOPULL;
    GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_VERY_HIGH;
    GPIO_InitStruct.Alternate = GPIO_AF9_CAN1;
    HAL_GPIO_Init(GPIOA, &GPIO_InitStruct);

  /* USER CODE BEGIN CAN1_MspInit 1 */

  /* USER CODE END CAN1_MspInit 1 */
  }
}


/*
**功能:CAN过滤器配置
**参数:无
**返回值:无
 */
void can_filter_config(void)
{
	CAN_FilterTypeDef can_filterconfig;
    /* 过滤器是接收所有报文,不筛选 */
    can_filterconfig.FilterMode = CAN_FILTERMODE_IDMASK;   /* 过滤器模式:标识符掩码模式(屏蔽位模式)*/
    can_filterconfig.FilterScale = CAN_FILTERSCALE_32BIT;  /* 过滤器位宽:32位位宽 */
   
	//STID[10:3] STID[2:0] EXID[17:13]         EXID[12:5] EXID[4:0] IDE RTR 0
	can_filterconfig.FilterIdHigh = 0;        /* ID高字节 */               
    can_filterconfig.FilterIdLow  = 0;        /* ID低字节 */
    can_filterconfig.FilterMaskIdHigh = 0;    /* 掩码高字节 */
    can_filterconfig.FilterMaskIdLow  = 0;    /* 掩码低字节 */
	
    can_filterconfig.FilterBank = 0;                         /* 选择过滤器组 */
    can_filterconfig.FilterFIFOAssignment = CAN_FilterFIFO0; /* 过滤器关联FIFO */
    can_filterconfig.FilterActivation = CAN_FILTER_ENABLE;   /* 过滤器使能 */
    can_filterconfig.SlaveStartFilterBank = 14;
    HAL_CAN_ConfigFilter(&hcan1, &can_filterconfig);
    
}

/* 发送消息数据函数 */
void can_send_message(uint32_t id, uint8_t *buf, uint8_t len)
{

    uint32_t tx_mail = CAN_TX_MAILBOX0;  /* 发送邮箱 */
    
    g_can1_txheader.ExtId = id;          /* 扩展标识符 */
    g_can1_txheader.DLC = len;           /* 数据长度 */
    g_can1_txheader.IDE = CAN_ID_EXT;    /* 帧格式(标准帧或扩展帧) */
    g_can1_txheader.RTR = CAN_RTR_DATA;  /* 帧类型(数据帧或远程帧) */
    
    HAL_CAN_AddTxMessage(&hcan1, &g_can1_txheader, buf, &tx_mail);
    // 等待发送完成
    while(HAL_CAN_GetTxMailboxesFreeLevel(&hcan1) != 3);
}

/* 接收数据函数 */
uint8_t can_receive_message(uint8_t *buf)
{
    if (HAL_CAN_GetRxFifoFillLevel(&hcan1, CAN_RX_FIFO0) == 0)
    {
        return 0;
    }
    
    HAL_CAN_GetRxMessage(&hcan1, CAN_RX_FIFO0, &g_can1_rxheader, buf);
    
		// 返回接收数据长度
    return g_can1_rxheader.DLC;
}

6.2、main.c部分代码

#include "can.h"


int main(void)
{
    /* USER CODE BEGIN 1 */
    uint8_t can_sen_buf[8] = {0, 1, 2, 3, 4, 5, 6, 7};  // can发送数据
    uint8_t can_rec_len = 0;                            // can接收数据长度
    uint8_t can_rec_buf[8] = {0};                       // can接收数据缓冲区
   

    MX_CAN1_Init();
    /* USER CODE BEGIN 2 */
	// 1、已经使能CAN时钟和初始化CAN
	// 2、配置CAN接收过滤器
	can_filter_config();
	// 3、启动CAN设备
	HAL_CAN_Start(&hcan1);
    while (1)
    {
	    printf("hello world\r\n");
	    can_send_message(0xF0000000, can_sen_buf, 8);
	    can_rec_len = can_receive_message(can_rec_buf);
        
        if (can_rec_len)
	    {
		    for (uint8_t i = 0; i < can_rec_len; i++)
		    {
			    printf("%x ",can_rec_buf[i]);
		    }
		    printf("\r\n");
	    }
    }
}

6.3、完整工程下载地址

(1)完整工程存储再码云。

(2)STM32_CSDN: CSDN中STM32专栏的所有示例代码

文章来源:https://blog.csdn.net/weixin_42727214/article/details/135030325
本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。