配置STM32F411CEU6的系统时钟-避免芯片内核锁死

一.了解

我在尝试建立HAL库的时候使用的正点原子的sys.c的工程,结果出现芯片内核锁死的问题。究其原因是因为正点原子使用的是8M的晶振,就导致程序下进去就会锁死内核

这是正点原子的晶振,这是我的板载晶振。

这是我最小系统的晶振。所以我们只能重写sys.c和sys.h文件解决问题。

二.程序

1.sys.c

cs 复制代码
#include "sys.h"
#include "stm32f4xx.h"

/**
  * @brief  STM32F411 对齐CubeMX配置:25M晶振→96MHz 系统时钟
  * @param  None
  * @note   1、完全匹配CubeMX参数:PLLM=25, PLLN=192, PLLP=2 → 96MHz
  *         2、保留超时兜底+时钟源顺序,永不卡死/GPIO乱跳
  *         3、校准延时精准到500ms,兼容HAL_Delay
  */
void SystemClock_Config(void)
{
    uint32_t timeout = 0xFFFFFFF;

    // 1. FLASH配置(96MHz必备)
    FLASH->ACR = FLASH_ACR_ICEN | FLASH_ACR_DCEN | FLASH_ACR_PRFTEN | FLASH_ACR_LATENCY_3WS;

    // 2. 复位RCC+超时兜底
    RCC->CR |= RCC_CR_HSION;                
    while((RCC->CR & RCC_CR_HSIRDY) == 0 && --timeout);
    if(timeout == 0) { Error_Handler(); }
    
    RCC->CFGR = 0x00000000;                 
    RCC->CR &= ~(RCC_CR_PLLON | RCC_CR_HSEON); 
    RCC->CIR = 0x00000000;                  

    // 3. 使能25M晶振+超时兜底
    timeout = 0xFFFFFFF;
    RCC->CR |= RCC_CR_HSEON;
    while((RCC->CR & RCC_CR_HSERDY) == 0 && --timeout);
    if(timeout == 0) { Error_Handler(); }

    if(timeout > 0) 
    {
        // 4. 完全对齐CubeMX的PLL参数(关键!)
        RCC->PLLCFGR = 0x00000000;
        RCC->PLLCFGR |= RCC_PLLCFGR_PLLSRC_HSE; // 先开时钟源(防失效)
        RCC->PLLCFGR |= 25;                 // ? CubeMX的PLLM=25
        RCC->PLLCFGR |= (192 << 6);         // ? CubeMX的PLLN=192
        RCC->PLLCFGR |= (0 << 16);          // ? CubeMX的PLLP=2(对应位00)
        RCC->PLLCFGR |= (4 << 24);          // PLLQ=4(CubeMX默认)

        // 5. 使能PLL+超时兜底
        timeout = 0xFFFFFFF;
        RCC->CR |= RCC_CR_PLLON;
        while((RCC->CR & RCC_CR_PLLRDY) == 0 && --timeout);
        if(timeout == 0) { Error_Handler(); }

        if(timeout > 0) 
        {
            // 6. 总线分频(CubeMX同款:AHB=1, APB1=2, APB2=1)
            RCC->CFGR |= RCC_CFGR_HPRE_DIV1;    
            RCC->CFGR |= RCC_CFGR_PPRE1_DIV2;   
            RCC->CFGR |= RCC_CFGR_PPRE2_DIV1;   

            // 7. 切PLL时钟源
            RCC->CFGR |= RCC_CFGR_SW_PLL;
            while((RCC->CFGR & RCC_CFGR_SWS) != RCC_CFGR_SWS_PLL);
        }
    }

    // 8. 校准延时:从400ms→500ms精准(只改这行!)
    SystemCoreClock = 96000000UL;   // ? 校准值:25M晶振+CubeMX参数,完美匹配500ms
    SysTick->CTRL &= ~(1 << 2);          // 清脏位
    SysTick->CTRL |=  (1 << 2);          // 关8分频
    HAL_SYSTICK_Config(SystemCoreClock / 1000);
    HAL_NVIC_SetPriority(SysTick_IRQn, 0, 0);
}

void Error_Handler(void)
{
    __disable_irq();
    while(1) {}
}

2.sys.h

cs 复制代码
#ifndef __SYS__H
#define __SYS__H

#include "stm32f4xx.h"  // STM32F4核心头文件
#include "stm32f4xx_hal.h" // HAL库基础定义(必须,因为用到__HAL_XXX宏)

// 声明【CubeMX原生无修改】的系统时钟配置函数
void SystemClock_Config(void);

// 声明报错函数(原代码里的调用,必须声明)
void Error_Handler(void);

#endif

三.晶振不起震

我最开始把正点原子的一个低功耗删去,程序也可以运行,但是使用的是内部低速时钟16Mhz,我发现我delay(500)ms实测只有130ms,我又开始继续修复,最终搞定了完整的版本,把时钟跑起来。我还进行了测试。

生成了四路PWM波形,发现恨我吗预计的是可以的,我们最终给他修复完成。

相关推荐
驭渊的小故事12 小时前
简单模板笔记
数据结构·笔记·算法
野犬寒鸦13 小时前
从零起步学习并发编程 || 第七章:ThreadLocal深层解析及常见问题解决方案
java·服务器·开发语言·jvm·后端·学习
陈桴浮海13 小时前
【Linux&Ansible】学习笔记合集二
linux·学习·ansible
深圳市九鼎创展科技13 小时前
瑞芯微 RK3399 开发板 X3399 评测:高性能 ARM 平台的多面手
linux·arm开发·人工智能·单片机·嵌入式硬件·边缘计算
辰哥单片机设计13 小时前
STM32项目分享:车辆防盗报警系统
stm32·单片机·嵌入式硬件
xhbaitxl13 小时前
算法学习day39-动态规划
学习·算法·动态规划
智者知已应修善业13 小时前
【洛谷P9975奶牛被病毒传染最少数量推导,导出多样例】2025-2-26
c语言·c++·经验分享·笔记·算法·推荐算法
ZH154558913113 小时前
Flutter for OpenHarmony Python学习助手实战:数据库操作与管理的实现
python·学习·flutter
Junlan2713 小时前
Cursor使用入门及连接服务器方法(更新中)
服务器·人工智能·笔记
risc12345614 小时前
如何认识结构?结构 = 要素 + 关系 + 动态
笔记