STM32F4 CAN驱动配置,以及SEG1,SEG2的最佳配置

一、开发环境

平台:MDK5
单片机型号:STM32F407VG
HAL库版本:V1.8.1

二、配置程序

该程序在APB1时钟为42M速率的情况下,将CAN配置为500KBps.

void can_init(void)
{
		GPIO_InitTypeDef GPIO_InitStruct;
    
		// 开启GPIOA和CAN1时钟
		__HAL_RCC_GPIOA_CLK_ENABLE();
		__HAL_RCC_CAN1_CLK_ENABLE();
		
		// 配置PA11和PA12为CAN1模式
    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);
    
		// 配置CAN1
    hcan1.Instance = CAN1;
    hcan1.Init.Prescaler = 4;
    hcan1.Init.Mode = CAN_MODE_NORMAL;
    hcan1.Init.SyncJumpWidth = CAN_SJW_1TQ;
    hcan1.Init.TimeSeg1 = CAN_BS1_12TQ;
    hcan1.Init.TimeSeg2 = CAN_BS2_8TQ;
    hcan1.Init.TimeTriggeredMode = DISABLE;
    hcan1.Init.AutoBusOff = DISABLE;
    hcan1.Init.AutoWakeUp = DISABLE;
    hcan1.Init.AutoRetransmission = ENABLE;
    hcan1.Init.ReceiveFifoLocked = DISABLE;
    hcan1.Init.TransmitFifoPriority = DISABLE;
    if (HAL_CAN_Init(&hcan1) != HAL_OK) {
        Error_Handler();
    }

		// 配置CAN1过滤器
    CAN_FilterTypeDef sFilterConfig;
    sFilterConfig.FilterIdLow = 0x0000;
    sFilterConfig.FilterIdHigh = 0x0000;
    sFilterConfig.FilterMaskIdLow = 0x0000;
    sFilterConfig.FilterMaskIdHigh = 0x0000;
    sFilterConfig.FilterFIFOAssignment = CAN_FILTER_FIFO0;
    sFilterConfig.FilterBank = 0;
    sFilterConfig.FilterMode = CAN_FILTERMODE_IDMASK;
    sFilterConfig.FilterScale = CAN_FILTERSCALE_32BIT;
    sFilterConfig.FilterActivation = ENABLE;
    if (HAL_CAN_ConfigFilter(&hcan1, &sFilterConfig) != HAL_OK) {
        Error_Handler(); // 可以自定义错误处理函数
    }

		// 使能CAN1
    if (HAL_CAN_Start(&hcan1) != HAL_OK) {
        Error_Handler(); // 可以自定义错误处理函数
    }

		// 使能CAN中断
		HAL_CAN_ActivateNotification(&hcan1, CAN_IT_RX_FIFO0_MSG_PENDING);
}

三、关于波特率的计算

通过SystemClock_Config函数,将主频配置为168M(最大),APB1为42M(最大)。CAN的波特率=42M/(1 + TimeSeg1 + TimeSeg2) = 42M/(1+12+8)=500Kbps。

四、关于CAN传输参数的最佳配置

先附上参考链接
https://blog.csdn.net/qfmzhu/article/details/126892968
https://cloud.tencent.com/developer/article/1371907

CAN的Seg1和Seg2是有讲究的,特别是用于远距离传输的时候(我的参数配置并不是很好,不过我传的不远,影响不大)。使用最佳的采样点得到的配置,同速率下可以最大化传输距离。采样点=(1+TSEG1)/(1+TSEG1+TSEG2) ,最好在但不超过7/8位时间点上(87.5%)。如果对传输距离有要求,尽量往这个参数靠吧~

  • 0
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
对于STM32f4数码管倒计时显示,你可以使用定时器和GPIO来实现。下面是一个简单的示例代码: ```c #include "stm32f4xx.h" // 定义数码管的引脚 #define SEG_A_PIN GPIO_Pin_0 #define SEG_B_PIN GPIO_Pin_1 #define SEG_C_PIN GPIO_Pin_2 #define SEG_D_PIN GPIO_Pin_3 #define SEG_E_PIN GPIO_Pin_4 #define SEG_F_PIN GPIO_Pin_5 #define SEG_G_PIN GPIO_Pin_6 #define SEG_DP_PIN GPIO_Pin_7 // 定义数码管的共阳极连接方式 #define SEG_A_PORT GPIOA #define SEG_B_PORT GPIOA #define SEG_C_PORT GPIOA #define SEG_D_PORT GPIOA #define SEG_E_PORT GPIOA #define SEG_F_PORT GPIOA #define SEG_G_PORT GPIOA #define SEG_DP_PORT GPIOA // 定义数码管的位选引脚 #define DIGIT1_PIN GPIO_Pin_8 #define DIGIT2_PIN GPIO_Pin_9 #define DIGIT3_PIN GPIO_Pin_10 #define DIGIT4_PIN GPIO_Pin_11 // 定义数码管的位选端口 #define DIGIT1_PORT GPIOB #define DIGIT2_PORT GPIOB #define DIGIT3_PORT GPIOB #define DIGIT4_PORT GPIOB // 定义倒计时的时间 #define COUNTDOWN_TIME 60 // 定义全局变量 volatile uint32_t countdown = COUNTDOWN_TIME; // 初始化定时器 void TIM_Init(void) { TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure; NVIC_InitTypeDef NVIC_InitStructure; // 使能定时器时钟 RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM2, ENABLE); // 配置定时器参数 TIM_TimeBaseStructure.TIM_Period = 1000 - 1; // 定时器周期为1ms TIM_TimeBaseStructure.TIM_Prescaler = 8400 - 1; // 定时器预分频为8400,即定时器时钟为10kHz TIM_TimeBaseStructure.TIM_ClockDivision = TIM_CKD_DIV1; TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up; TIM_TimeBaseInit(TIM2, &TIM_TimeBaseStructure); // 使能定时器更新中断 TIM_ITConfig(TIM2, TIM_IT_Update, ENABLE); // 配置中断优先级 NVIC_InitStructure.NVIC_IRQChannel = TIM2_IRQn; NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0; NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0; NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE; NVIC_Init(&NVIC_InitStructure); // 启动定时器 TIM_Cmd(TIM2, ENABLE); } // 初始化GPIO void GPIO_Init(void) { GPIO_InitTypeDef GPIO_InitStructure; // 使能GPIO时钟 RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOA | RCC_AHB1Periph_GPIOB, ENABLE); // 配置数码管引脚为推挽输出 GPIO_InitStructure.GPIO_Mode = GPIO_Mode_OUT; GPIO_InitStructure.GPIO_OType = GPIO_OType_PP; GPIO_InitStructure.GPIO_Speed = GPIO_Speed_100MHz; GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_NOPULL; GPIO_InitStructure.GPIO_Pin = SEG_A_PIN | SEG_B_PIN | SEG_C_PIN | SEG_D_PIN | SEG_E_PIN | SEG_F_PIN | SEG_G_PIN | SEG_DP_PIN; GPIO_Init(SEG_A_PORT, &GPIO_InitStructure); // 配置位选引脚为推挽输出 GPIO_InitStructure.GPIO_Pin = DIGIT1_PIN | DIGIT2_PIN | DIGIT3_PIN | DIGIT4_PIN; GPIO_Init(DIGIT1_PORT, &GPIO_InitStructure); } // 更新数码管显示 void UpdateDisplay(uint32_t value) { // 数码管显示的数字和对应的引脚状态 uint8_t digit[4] = {value / 1000, (value / 100) % 10, (value / 10) % 10, value % 10}; uint16_t segment[8] = {SEG_A_PIN, SEG_B_PIN, SEG_C_PIN, SEG_D_PIN, SEG_E_PIN, SEG_F_PIN, SEG_G_PIN, SEG_DP_PIN}; // 关闭所有位选 GPIO_SetBits(DIGIT1_PORT, DIGIT1_PIN); GPIO_SetBits(DIGIT2_PORT, DIGIT2_PIN); GPIO_SetBits(DIGIT3_PORT, DIGIT3_PIN); GPIO_SetBits(DIGIT4_PORT, DIGIT4_PIN); // 设置数码管引脚状态 for (int i = 0; i < 4; i++) { // 设置位选引脚 switch (i) { case 0: GPIO_ResetBits(DIGIT1_PORT, DIGIT1_PIN); break; case 1: GPIO_ResetBits(DIGIT2_PORT, DIGIT2_PIN); break; case 2: GPIO_ResetBits(DIGIT3_PORT, DIGIT3_PIN); break; case 3: GPIO_ResetBits(DIGIT4_PORT, DIGIT4_PIN); break; } // 设置数码管引脚 for (int j = 0; j < 8; j++) { if ((digit[i] >> j) & 0x01) { GPIO_SetBits(SEG_A_PORT, segment[j]); } else { GPIO_ResetBits(SEG_A_PORT, segment[j]); } } // 延时一段时间,控制数码管刷新频率 for (volatile int k = 0; k < 1000; k++) ; } } // 定时器中断处理函数 void TIM2_IRQHandler(void) { if (TIM_GetITStatus(TIM2, TIM_IT_Update) != RESET) { TIM_ClearITPendingBit(TIM2, TIM_IT_Update); if (countdown > 0) { countdown--; } } } int main(void) { // 初始化定时器和GPIO TIM_Init(); GPIO_Init(); while (1) { // 更新数码管显示 UpdateDisplay(countdown); } } ``` 这段代码使用定时器TIM2来实现倒计时功能,每隔1ms触发一次定时器中断,在中断处理函数中更新倒计时的值。然后通过GPIO控制数码管的引脚状态来显示倒计时的时间。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值