1. KEIL编译器预定义
cs
STM32F10X_MD,USE_STDPERIPH_DRIVER


或者:预定义外部晶振,因为标准库内部默认外部晶振是8MHz,如何是12MHz 可以在这里预定义,方便系统时钟的计算。下面的8000000改成12000000即可
cs
STM32F10X_MD,USE_STDPERIPH_DRIVER,HSE_VALUE=8000000

2. 系统时钟未配置也会默认是72MHz的原因
因为startup_stm32f10x_md.s 的文件里面有:
cpp
LDR R0, =SystemInit
BLX R0
LDR R0, =__main
BX R0
SystemInit() 是标准外设库(system_stm32f10x.c)里的函数,它会根据你定义的芯片宏(如 STM32F10X_MD)自动完成初始时钟配置:
-
如果定义了
STM32F10X_MD,SystemInit()会尝试使用 HSE(外部 8 MHz 晶振:外部晶振的频率完全由HSE_VALUE这个宏决定) 作为 PLL 输入,倍频 9,得到 72 MHz 系统时钟。 -
同时会正确设置 AHB、APB1、APB2 预分频器,以及 Flash 等待周期。
-
前提:你的板子上有外部晶振且电路正常。
所以,即使你的 main() 里没有写任何时钟配置代码,只要工程里链接了标准库的 system_stm32f10x.c,且外部晶振正常工作,系统就会跑在 72 MHz。
3. 系统时钟配置72MHz
cs
#include "stm32f10x.h"
/**
* @brief 系统时钟配置为 72 MHz(完善版)
* 主时钟源:HSE (8 MHz) -> PLL (×9) -> SYSCLK = 72 MHz
* 容错机制:若 HSE 失效,回退至 HSI 产生 48 MHz 并启用 CSS 保护
* AHB = 72 MHz
* APB1 = 36 MHz (最大限制)
* APB2 = 72 MHz
* @retval 1: 72 MHz 配置成功 0: HSE 失败,已降级为 48 MHz
*/
uint8_t SystemClock_Config(void)
{
ErrorStatus HSEStartUpStatus;
uint8_t clock72MHz_success = 0;
/* 1. 复位 RCC 配置 */
RCC_DeInit();
/* 2. 使能外部高速晶振 (HSE) */
RCC_HSEConfig(RCC_HSE_ON);
/* 3. 等待 HSE 就绪 */
HSEStartUpStatus = RCC_WaitForHSEStartUp();
if (HSEStartUpStatus == SUCCESS)
{
/* --- HSE 正常,配置 72 MHz --- */
/* 4. 设置 Flash 等待周期(72 MHz 需要 2 个等待周期) */
FLASH_SetLatency(FLASH_Latency_2);
FLASH_PrefetchBufferCmd(FLASH_PrefetchBuffer_Enable);
/* 5. 配置 AHB/APB 总线预分频器 */
RCC_HCLKConfig(RCC_SYSCLK_Div1); /* AHB = SYSCLK = 72 MHz */
RCC_PCLK2Config(RCC_HCLK_Div1); /* APB2 = HCLK = 72 MHz */
RCC_PCLK1Config(RCC_HCLK_Div2); /* APB1 = HCLK/2 = 36 MHz */
/* 6. 配置 PLL:时钟源 = HSE (不分频),倍频系数 9 -> 72 MHz */
RCC_PLLConfig(RCC_PLLSource_HSE_Div1, RCC_PLLMul_9);
/* 7. 使能 PLL */
RCC_PLLCmd(ENABLE);
/* 8. 等待 PLL 就绪 */
while (RCC_GetFlagStatus(RCC_FLAG_PLLRDY) == RESET);
/* 9. 切换系统时钟到 PLL */
RCC_SYSCLKConfig(RCC_SYSCLKSource_PLLCLK);
/* 10. 确认 PLL 已成为系统时钟(SWS 位 = 0x08) */
while (RCC_GetSYSCLKSource() != 0x08);
/* 11. 使能时钟安全系统 (CSS),监测 HSE 失效 */
RCC_ClockSecuritySystemCmd(ENABLE);
clock72MHz_success = 1;
}
else
{
/* --- HSE 启动失败,启用备份方案:HSI -> PLL -> 48 MHz --- */
/* HSI 内部 8 MHz,PLL 时钟源 HSI/2 = 4 MHz,倍频 12 -> 48 MHz */
/* 1. Flash 等待周期:48 MHz 需要 1 个等待周期 */
FLASH_SetLatency(FLASH_Latency_1);
FLASH_PrefetchBufferCmd(FLASH_PrefetchBuffer_Enable);
/* 2. 总线分频(48 MHz 下 APB1 仍为 24 MHz,符合上限) */
RCC_HCLKConfig(RCC_SYSCLK_Div1);
RCC_PCLK2Config(RCC_HCLK_Div1);
RCC_PCLK1Config(RCC_HCLK_Div2);
/* 3. 使能 HSI 并等待就绪 */
RCC_HSICmd(ENABLE);
while (RCC_GetFlagStatus(RCC_FLAG_HSIRDY) == RESET);
/* 4. 配置 PLL 使用 HSI/2 作为时钟源,倍频 12 */
RCC_PLLConfig(RCC_PLLSource_HSI_Div2, RCC_PLLMul_12);
/* 5. 使能 PLL 并等待就绪 */
RCC_PLLCmd(ENABLE);
while (RCC_GetFlagStatus(RCC_FLAG_PLLRDY) == RESET);
/* 6. 切换到 PLL 输出 */
RCC_SYSCLKConfig(RCC_SYSCLKSource_PLLCLK);
while (RCC_GetSYSCLKSource() != 0x08);
/* 7. 关闭 HSE 以节省功耗(已确认失效) */
RCC_HSEConfig(RCC_HSE_OFF);
/* 此时系统降级运行在 48 MHz,应用程序可检测该状态并报警 */
clock72MHz_success = 0;
}
return clock72MHz_success; /* 返回值让调用者判断是否达到 72 MHz */
}