【正点原子STM32】STM32时钟系统(时钟树、时钟源、分频器和倍频系数、锁相环、STM32CubeMX时钟树、系统时钟配置步骤)

news2025/1/13 7:43:22

一、认识时钟树

  • 1.1、什么是时钟?
  • 1.2、认识时钟树(F1)
  • 1.3、认识时钟树(F4)
  • 1.4、认识时钟树(F7)
  • 1.5、认识时钟树(H7)

二、配置系统时钟

  • 2.1、系统时钟配置步骤
  • 2.2、外设时钟使能和失能
  • 2.3、sys_stm32_clock_init 函数(F1)
    • HAL_RCC_OscConfig()函数(F1)
    • HAL_RCC_ClockConfig函数(F1)
  • 2.4、sys_stm32_clock_init 函数(F4/F7)
    • HAL_RCC_OscConfig()函数(F4/F7)
    • HAL_RCC_ClockConfig()函数(F4/F7)
  • 2.5、sys_stm32_clock_init 函数(H7)
    • HAL_RCC_OscConfig()函数(H7)
    • HAL_RCC_ClockConfig()函数(H7)
    • HAL_RCCEx_PeriphCLKConfig()函数(H7)

三、总结

一、认识时钟树

1.1、什么是时钟?

在这里插入图片描述
时钟(Clock)在数字电路和计算机系统中是一种周期性的信号,用于同步和协调系统中各个部分的操作。时钟信号在单片机中是非常重要的,它指导着各个组件的工作,确保它们在正确的时间执行相应的任务。时钟信号通常是一个方波信号,具有固定的周期和占空比。

时钟树(Clock Tree)是指在数字系统或芯片中,将一个主时钟信号分配和扩展到整个系统中各个模块和部分的一种层次结构。时钟树的目的是确保系统中所有的时钟域都能够按照规定的时序要求同步运行。时钟树通常包括以下关键元素:

  1. 主时钟源(Clock Source): 时钟树的起点通常是一个主时钟源,这可以是外部晶振、振荡器或其他时钟源。

  2. 时钟分频器(Clock Divider): 时钟分频器用于将主时钟信号分频,以产生其他模块或部分所需的时钟频率。这样可以根据需要调整时钟频率,使得不同的部分能够以不同的速度运行。

  3. 时钟缓冲器(Clock Buffer): 时钟缓冲器用于放大和驱动时钟信号,确保时钟信号能够准确地传播到系统中的各个部分。

  4. 时钟分配网络(Clock Distribution Network): 时钟分配网络将时钟信号传送到系统中的各个模块。这通常包括时钟线路、时钟树分支等。

时钟树的设计需要考虑时序要求、功耗、噪声和电磁兼容性等因素。精心设计的时钟树能够确保系统各个部分协同工作,提高系统的稳定性和性能。在数字集成电路中,时钟树的设计是一个复杂而关键的任务。

1.2、认识时钟树(F1)

在这里插入图片描述
H、L、S、I、E代表了一些常见的缩写,通常与单片机或者数字电路中的时钟、速度和内外部资源有关。
含义:

  1. H:high

    • 通常表示高电平或高状态。在数字电路中,可能指的是高电压或逻辑高。
  2. L:low

    • 通常表示低电平或低状态。在数字电路中,可能指的是低电压或逻辑低。
  3. S:speed

    • 可能表示速度。在数字电路或通信领域中,S通常用于表示速度、传输速率或时钟频率。
  4. I:internal

    • 可能表示内部。在数字电路中,I可能用于表示内部时钟源、内部资源等。
  5. E:external

    • 可能表示外部。在数字电路中,E可能用于表示外部时钟源、外部资源等。

分频器(Divider)和倍频系数(Multiplier)

分频器和倍频系数是在时钟系统中常见的概念,用于调整时钟信号的频率。让我们更详细地了解这两个概念:

  1. 分频器(Divider):

    • 定义: 分频器用于将输入时钟信号分频,即减小其频率。
    • 作用: 分频器通常用于将高频时钟信号分频为较低的频率,以供给某些模块或外设使用。
    • 公式: 分频后的频率可以通过原始频率除以分频系数得到,即新频率 = 原始频率 / 分频系数。
  2. 倍频系数(Multiplier):

    • 定义: 倍频系数用于将输入时钟信号倍频,即增加其频率。
    • 作用: 倍频系数通常用于提高时钟信号的频率,以满足系统对更高时钟频率的需求。
    • 公式: 倍频后的频率可以通过原始频率乘以倍频系数得到,即新频率 = 原始频率 * 倍频系数。

在实际应用中,分频器和倍频系数的配置对于调整系统时钟至关重要,特别是在使用锁相环(PLL)等技术进行时钟调整时。在微控制器或数字系统中,通过配置分频器和倍频系数,可以灵活地适应不同的硬件要求,以满足外设的时钟需求,同时确保系统时钟的稳定性和准确性。

总的来说,分频器和倍频系数是时钟系统中的两个关键元素,通过它们的配置,可以实现对时钟频率的灵活调整,以满足不同应用场景的需求。

锁相环(Phase-Locked Loop,PLL)

锁相环(Phase-Locked Loop,PLL)是一种控制系统,用于调整输出信号的频率和相位,使其与输入信号保持特定的关系。在PLL中,常用的表达式是F/M*N/P,其中:

  1. F:

    • 表示输出信号的频率。
  2. M:

    • 表示输入信号的参考倍频系数(Multiplier)。
  3. N:

    • 表示反馈倍频系数。
  4. P:

    • 表示分频系数(Divider)。

这个表达式的含义是,通过调整M、N和P的值,可以控制输出频率F。下面是更详细的解释:

  • 输入信号:

    • 输入信号经过倍频(M倍)得到一个高频信号。
  • 反馈:

    • 从输出中采样一部分反馈回输入,以与输入信号同步。
  • 输出频率:

    • 输出频率是输入信号经过M倍频、N倍频、再通过P分频得到的。

具体来说,输出频率F可以通过以下公式计算:

F = F 输入 × M × N P F = \dfrac{F_{\text{输入}} \times M \times N}{P} F=PF输入×M×N

在实际应用中,通过调整M、N和P的值,可以实现对输出频率的精确控制。这种调整通常用于微控制器、射频电路、时钟产生器等应用中,以满足系统对不同频率的需求。
分频器是一种电路或模块,用于将输入的时钟信号分频,即减小其频率。这是在数字电路中常见的一种功能,它允许在系统中使用不同频率的时钟信号,以适应不同的模块和外设的工作需求。分频器通常通过使用可编程的分频系数,将输入时钟信号的频率降低到所需的水平。

以下是分频器的一些关键概念:

  1. 分频系数(Divider Ratio):

    • 分频系数是用于调整输入时钟频率的比例。它是分频器的一个参数,用于控制输出频率。
    • 分频系数 N N N 决定了输入频率与输出频率之间的关系,输出频率等于输入频率除以 N N N
    • 公式: 输出频率 = 输入频率 N \text{输出频率} = \frac{\text{输入频率}}{N} 输出频率=N输入频率
  2. 整数分频和分数分频:

    • 整数分频是指分频系数 N N N 为整数,例如 2、3、4 等。
    • 分数分频是指分频系数 N N N 可以是分数,例如 1 / 2 1/2 1/2 1 / 3 1/3 1/3 等。在某些应用中,分数分频可以提供更灵活的频率调整。
  3. 分频器的应用:

    • 分频器常用于微控制器、数字信号处理器(DSP)、时钟生成器、通信系统等领域。
    • 在微控制器中,分频器用于为外设提供适当的时钟频率。
    • 在通信系统中,分频器可用于调整射频信号的频率。
  4. 分频器类型:

    • 固定分频器:分频系数是固定的,通常在设计中确定。
    • 可编程分频器:分频系数可以通过软件或硬件配置进行调整,提供更大的灵活性。

总体而言,分频器是数字电路中的重要组件,允许系统中的不同部分以不同的时钟频率运行,以满足不同模块的工作要求。在时钟系统设计中,分频器起到了关键的作用,有助于优化电路的性能和功耗。

时钟安全系统(CSS)和自由运行时钟(FCLK)

这两个概念涉及到STM32微控制器的时钟管理和安全特性。让我们更详细地了解它们:

  1. 时钟安全系统(Clock Security System,CSS):

    • 功能: CSS是STM32微控制器上的一项安全特性,旨在监测外部高速时钟(HSE)的稳定性。
    • 作用: 如果HSE启动失败(例如,外部晶振故障),CSS会自动切换到内部高速时钟(HSI),并触发非可屏蔽中断(NMI)。
    • 应用: 这个特性可以用于提高系统的稳定性,防止由于外部时钟问题导致系统工作不正常。
  2. 自由运行时钟(Free Running Clock,FCLK):

    • 功能: FCLK是用于采样中断和调试模块计时的时钟。
    • 作用: 在特定的应用场景中,即使系统进入休眠模式,FCLK仍然保持运行,以确保一些必要的计时和中断采样。
    • 应用: 主要用于允许在休眠状态下执行一些必要的计时或调试操作,以保持对系统状态的监控。

这两个特性都是STM32系列微控制器提供的一些高级时钟管理和安全功能,有助于提高系统的可靠性和稳定性。在使用这些特性时,建议仔细查阅相关的STM32芯片手册和参考资料,以确保正确配置和使用。

STM32F103时钟树简图

在这里插入图片描述
理清这些概念的时序关系和逻辑,我们可以按照以下步骤来组织:

  1. 高速时钟源OSC(HSE、HSI):

    • HSE(High-Speed External)和 HSI(High-Speed Internal)是两种不同的时钟源。
    • HSE是外部高速晶振,HSI是芯片内部的高速时钟源。
  2. 锁相环PLL:

    • 锁相环(PLL)用于调整时钟频率,将其倍频到系统所需的频率。
    • 使用 HAL_RCC_OscConfig() 配置时钟源,选择 HSE 或 HSI 以及其他相关配置。
  3. 分频系数:

    • 配置PLL的分频系数,以调整输出时钟的频率。
    • 这可能包括主时钟(SYSCLK)、AHB时钟、APB1时钟、APB2时钟等。
    • 使用 HAL_RCC_ClockConfig() 进行系统时钟、总线和外设时钟的配置。
  4. 外设时钟使能:

    • 使用 __HAL_RCC_PPP_CLK_ENABLE() 使能外设的时钟。
    • 在这里,PPP表示具体的外设,比如UART、SPI、I2C等。
  5. 扩展外设时钟(RTC/ADC/USB):

    • 使用 HAL_RCCEx_PeriphCLKConfig() 扩展外设的时钟。
    • 这通常包括RTC、ADC、USB等外设。
  6. 内核时钟:

    • 内核时钟是供给CPU内核的时钟。
    • 它通常与系统时钟相同,但可能有差异。
    • 在配置时钟时需要确保内核时钟的要求得到满足。
  7. 具体时钟频率设置:

    • 在上述配置完成后,系统时钟 SYSCLK、AHB总线时钟 HCLK、APB1总线时钟和 APB2总线时钟会根据你的配置和分频系数产生相应频率。
    • 根据具体的需求,设置最大的 APB1 和 APB2 时钟频率(36M 和 72M)。
  8. 低速时钟源OSC(LSI、LSE):

    • LSI(Low-Speed Internal)和 LSE(Low-Speed External)是低速时钟源。
    • 通常用于RTC等外设,LSE的频率是32.768KHz,LSI的频率是40KHz。
  9. IWDG、RTC:

    • IWDG(Independent Watchdog)和 RTC(Real-Time Clock)是一些特殊的定时器或时钟相关的外设。
    • 它们可能需要特殊的时钟配置或者需要特别的时钟源。

总体而言,上述步骤涵盖了从时钟源的选择、通过PLL调整时钟频率、配置各个时钟分频系数,到最终配置系统总线和外设时钟的过程。在具体的应用中,需要根据芯片手册和硬件设计的要求进行调整。

STM32F103时钟树

在这里插入图片描述
这些概念,包括它们之间的关系:

  1. PLLXTPRE:

    • PLLXTPRE用于配置PLL输入时钟源的预分频因子。
    • 通常,可以选择不预分频(PLLXTPRE_DIV1)或者2分频(PLLXTPRE_DIV2)。
  2. PLLSRC:

    • PLLSRC用于选择PLL的输入时钟源,通常可以选择HSE(外部高速晶振)或HSI(内部高速时钟源)。
  3. PLLMUL:

    • PLLMUL用于配置PLL的倍频因子,即将输入时钟的频率乘以多少倍。
    • 具体的倍频因子可以是2、3、4、…、16。
  4. PLLCLK:

    • PLLCLK是指通过PLL倍频后得到的时钟。
  5. SYSCLK:

    • SYSCLK是系统的主时钟,通常由PLLCLK提供。
    • 通过HAL_RCC_ClockConfig()配置PLL、分频因子等,以设置SYSCLK的频率。
  6. HCLK:

    • HCLK是AHB总线的时钟,通常与SYSCLK相同,或者是SYSCLK的一部分。
    • 通过HAL_RCC_ClockConfig()设置HCLK的频率。
  7. Cortex系统时钟:

    • 这是Cortex-M处理器内部的时钟,通常等于HCLK。
  8. 滴答定时器:

    • 滴答定时器通常是一个用于生成定时中断的内部计时器。
    • 它的时钟源可以是SYSCLK或HCLK,通过HAL_InitTick()等函数进行配置。
  9. FCLK:

    • FCLK通常是指Flash存储器的时钟,其频率与HCLK相同。
  10. TIMXCLK:

    • TIMXCLK是指定时器(TIM)的时钟源,可以是内部时钟源或者外部时钟源。
    • 如果APB1预分频系数等于1,则TIMXCLK的频率不变,否则频率会×2。
  11. PCLK:

    • PCLK是指外设总线的时钟,分别对应APB1和APB2总线。
    • 通过HAL_RCC_ClockConfig()配置APB1和APB2的分频因子。
  12. I2SCLK:

    • I2SCLK是I2S接口的时钟。
  13. USBCLK:

    • USBCLK是USB接口的时钟。
  14. RTCCLK:

    • RTCCLK是实时时钟(RTC)的时钟源。
    • 通常可以选择LSE(低速外部时钟)或LSI(低速内部时钟)。
  15. RC-IWDGCLK:

    • RC-IWDGCLK是独立看门狗(IWDG)的时钟源,可以选择LSI。
  16. MCO:

    • MCO是主时钟输出,通过配置可以将SYSCLK、HSI、HSE等输出到MCO引脚,用于外部设备的时钟监测或其他应用。

这些概念的详细理解有助于在STM32CubeMX或其他开发环境中正确配置时钟和时序。请根据具体的硬件和应用需求,在配置时钟时参考芯片手册和相关文档。

STM32CubeMX时钟树(F103)

在这里插入图片描述

1.3、认识时钟树(F4)

在这里插入图片描述

STM32F407时钟树

在这里插入图片描述

STM32F429时钟树

在这里插入图片描述

STM32F4时钟树简图

在这里插入图片描述

STM32CubeMX时钟树(F407)

在这里插入图片描述

STM32CubeMX时钟树(F429)

在这里插入图片描述
为了让 FSUSB 满足 48MHz 把 Main PLL 的倍频系数配置成X384 而不是X360这样系统时钟就超频了
在这里插入图片描述
首先,为了让 FSUSB 满足 48MHz,我们需要将 Main PLL 的倍频系数配置为 X384。以下是一个示例代码,以确保系统时钟的配置满足 USB 的需求。

在这里,我将使用 HAL 库的一些函数作为例子,实际的代码可能会因为使用的具体库和芯片型号而有所不同。以下是一个基本的流程:

#include "stm32f4xx_hal.h"

void SystemClock_Config(void);
void Error_Handler(void);

int main(void)
{
    HAL_Init();

    // 配置系统时钟
    SystemClock_Config();

    // 初始化USB外设(这里使用USB FS作为示例)
    HAL_PCD_MspInit(&hpcd_USB_FS);
    HAL_PCD_Init(&hpcd_USB_FS);

    // 进行其他初始化和应用程序代码

    while (1)
    {
        // 主循环
    }
}

void SystemClock_Config(void)
{
    RCC_OscInitTypeDef RCC_OscInitStruct = {0};
    RCC_ClkInitTypeDef RCC_ClkInitStruct = {0};

    __HAL_RCC_PWR_CLK_ENABLE();
    __HAL_PWR_VOLTAGESCALING_CONFIG(PWR_REGULATOR_VOLTAGE_SCALE1);

    // 配置 HSE 为 8MHz 外部晶振
    RCC_OscInitStruct.OscillatorType = RCC_OSCILLATORTYPE_HSE;
    RCC_OscInitStruct.HSEState = RCC_HSE_ON;
    RCC_OscInitStruct.HSEPredivValue = RCC_HSE_PREDIV_DIV1;
    RCC_OscInitStruct.PLL.PLLState = RCC_PLL_ON;
    RCC_OscInitStruct.PLL.PLLSource = RCC_PLLSOURCE_HSE;
    RCC_OscInitStruct.PLL.PLLM = 8;  // 分频系数,原始频率 / 8 = 1MHz
    RCC_OscInitStruct.PLL.PLLN = 384; // 倍频系数,1MHz * 384 = 384MHz
    RCC_OscInitStruct.PLL.PLLP = RCC_PLLP_DIV2; // 384MHz / 2 = 192MHz (SYSCLK)
    RCC_OscInitStruct.PLL.PLLQ = 8;  // 分频系数,384MHz / 8 = 48MHz (USB频率)
    
    if (HAL_RCC_OscConfig(&RCC_OscInitStruct) != HAL_OK)
    {
        Error_Handler();
    }

    RCC_ClkInitStruct.ClockType = RCC_CLOCKTYPE_SYSCLK | RCC_CLOCKTYPE_HCLK | RCC_CLOCKTYPE_PCLK1 | RCC_CLOCKTYPE_PCLK2;
    RCC_ClkInitStruct.SYSCLKSource = RCC_SYSCLKSOURCE_PLLCLK;
    RCC_ClkInitStruct.AHBCLKDivider = RCC_SYSCLK_DIV1;
    RCC_ClkInitStruct.APB1CLKDivider = RCC_HCLK_DIV4;
    RCC_ClkInitStruct.APB2CLKDivider = RCC_HCLK_DIV2;
    
    if (HAL_RCC_ClockConfig(&RCC_ClkInitStruct, FLASH_LATENCY_3) != HAL_OK)
    {
        Error_Handler();
    }

    // 更新系统时钟配置
    SystemCoreClockUpdate();
}

void Error_Handler(void)
{
    while (1)
    {
        // 处理错误
    }
}

这个代码片段中,关键的部分是 RCC_OscInitStructRCC_ClkInitStruct 结构体的配置,特别是 PLL.PLLN 的值设置为 384,以满足 48MHz 的 USB 频率要求。这个例程假设使用的是 STM32Cube HAL 库,并在 main() 函数中调用了 HAL_Init()。实际上,具体的配置可能需要根据你的硬件设计和使用的芯片型号进行调整。在编写代码时,请参考相关的芯片手册和硬件设计文档。

1.4、认识时钟树(F7)

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

STM32F7时钟树简图

在这里插入图片描述

STM32CubeMX时钟树(F767/F750)

在这里插入图片描述

1.5、认识时钟树(H7)

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

STM32H7时钟树简图

在这里插入图片描述

STM32H7时钟树-低频部分简图

在这里插入图片描述### STM32CubeMX时钟树(H7)
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

二、配置系统时钟

2.1、系统时钟配置步骤

在这里插入图片描述
系统时钟配置步骤,下面是对每个步骤进行一些详细解释:

  1. 配置HSE_VALUE:

    • stm32xxxx_hal_conf.h 文件中,你需要设置 HSE_VALUE 宏定义,用于告诉HAL库外部晶振的频率。这是时钟系统的基础,确保HAL库能正确配置时钟。
  2. 调用SystemInit()函数(可选):

    • SystemInit() 函数通常包含在启动文件中,用于执行一些基本的系统初始化。在这里,它可以执行一些硬件的初始化,例如设置向量表和启用FPU等。这一步通常是由编译器自动调用的,你只需确保启动文件正确。
  3. 选择时钟源,配置PLL:

    • 使用 HAL_RCC_OscConfig() 函数配置时钟源和PLL。
    • 通过填写 RCC_OscInitTypeDef 结构体,可以指定HSE或HSI作为时钟源,并配置PLL的倍频系数和分频系数。
    • 示例如下:
      RCC_OscInitTypeDef RCC_OscInitStruct;
      RCC_OscInitStruct.OscillatorType = RCC_OSCILLATORTYPE_HSE;
      RCC_OscInitStruct.HSEState = RCC_HSE_ON;
      RCC_OscInitStruct.PLL.PLLState = RCC_PLL_ON;
      RCC_OscInitStruct.PLL.PLLSource = RCC_PLLSOURCE_HSE;
      RCC_OscInitStruct.PLL.PLLM = 8;   // 时钟源分频
      RCC_OscInitStruct.PLL.PLLN = 336; // PLL倍频
      RCC_OscInitStruct.PLL.PLLP = RCC_PLLP_DIV2; // PLL分频
      RCC_OscInitStruct.PLL.PLLQ = 7;   // USB OTG FS, SDIO and RNG分频
      if (HAL_RCC_OscConfig(&RCC_OscInitStruct) != HAL_OK)
      {
          Error_Handler();
      }
      
  4. 选择系统时钟源,配置总线分频器:

    • 使用 HAL_RCC_ClockConfig() 函数配置系统时钟和总线分频器。
    • 通过填写 RCC_ClkInitTypeDef 结构体,可以选择PLL为系统时钟源,并配置AHB、APB1、APB2等分频系数。
    • 示例如下:
      RCC_ClkInitTypeDef RCC_ClkInitStruct;
      RCC_ClkInitStruct.ClockType = RCC_CLOCKTYPE_SYSCLK | RCC_CLOCKTYPE_HCLK | RCC_CLOCKTYPE_PCLK1 | RCC_CLOCKTYPE_PCLK2;
      RCC_ClkInitStruct.SYSCLKSource = RCC_SYSCLKSOURCE_PLLCLK;
      RCC_ClkInitStruct.AHBCLKDivider = RCC_SYSCLK_DIV1;
      RCC_ClkInitStruct.APB1CLKDivider = RCC_HCLK_DIV4;
      RCC_ClkInitStruct.APB2CLKDivider = RCC_HCLK_DIV2;
      if (HAL_RCC_ClockConfig(&RCC_ClkInitStruct, FLASH_LATENCY_5) != HAL_OK)
      {
          Error_Handler();
      }
      
  5. 配置扩展外设时钟(可选):

    • 使用 HAL_RCCEx_PeriphCLKConfig() 函数配置扩展外设时钟,如RTC、ADC、USB等。
    • 示例如下:
      RCC_PeriphCLKInitTypeDef PeriphClkInitStruct;
      PeriphClkInitStruct.PeriphClockSelection = RCC_PERIPHCLK_USB | RCC_PERIPHCLK_RTC;
      PeriphClkInitStruct.UsbClockSelection = RCC_USBCLKSOURCE_PLL;
      PeriphClkInitStruct.RTCClockSelection = RCC_RTCCLKSOURCE_LSE;
      if (HAL_RCCEx_PeriphCLKConfig(&PeriphClkInitStruct) != HAL_OK)
      {
          Error_Handler();
      }
      

这些步骤涵盖了从外部晶振到PLL配置,再到系统时钟和总线分频的整个过程。确保在配置时钟时参考芯片手册和硬件设计文档,以确保满足具体硬件的要求。

2.2、外设时钟使能和失能

在这里插入图片描述

在HAL库中,使能和失能外设时钟的方法通常使用了一些预定义的宏和函数。以下是对这两个操作的分析:

  1. 使能外设时钟:

    • 通常使用形如 __HAL_RCC_XXX_CLK_ENABLE() 的宏,其中 XXX 表示具体的外设(比如GPIOA、USART1等)。
    • 这个宏实际上是在底层调用了HAL库中相应外设的时钟使能函数,这个函数会配置相应的时钟使能寄存器,以确保外设可以正常工作。
  2. 失能外设时钟:

    • 通常使用形如 __HAL_RCC_XXX_CLK_DISABLE() 的宏,其中 XXX 表示具体的外设。
    • 类似地,这个宏底层会调用HAL库中相应外设的时钟失能函数,将时钟使能寄存器中的相应位清零,以禁用外设的时钟。

这些宏实际上是对底层寄存器的访问进行了封装,方便用户使用。例如,对于GPIOA的时钟使能:

__HAL_RCC_GPIOA_CLK_ENABLE();    // 使能 GPIOA 时钟
__HAL_RCC_GPIOA_CLK_DISABLE();   // 禁止 GPIOA 时钟

上述操作会调用底层的函数,具体实现会涉及到对相应寄存器的设置或清零。这些宏和函数的实现在HAL库的源代码中,用户一般无需直接关注底层实现,只需要正确使用这些宏即可。

这种封装提高了代码的可读性和可维护性,同时确保了对寄存器的操作是正确的。

2.3、sys_stm32_clock_init 函数(F1)

HAL_RCC_OscConfig()函数(F1)

在这里插入图片描述

HAL_RCC_ClockConfig函数(F1)

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

2.4、sys_stm32_clock_init 函数(F4/F7)

HAL_RCC_OscConfig()函数(F4/F7)

在这里插入图片描述

HAL_RCC_ClockConfig()函数(F4/F7)

在这里插入图片描述

2.5、sys_stm32_clock_init 函数(H7)

HAL_RCC_OscConfig()函数(H7)

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

HAL_RCC_ClockConfig()函数(H7)

在这里插入图片描述

HAL_RCCEx_PeriphCLKConfig()函数(H7)

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

三、总结

在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1414329.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

贝叶斯增量式跨域适应:少样本 + 跨模态学习 + 知识保留和推断【fundus + OCT】,做视网膜病变

贝叶斯深度学习:增量式少样本学习跨域适应 贝叶斯多目标函数 跨模态学习 fundus OCT,做视网膜病变 核心思想设计网络:寻找分类模型、损失函数实验结果混淆矩阵与注意力图评估 总结 核心思想 论文:https://arxiv.org/pdf/2110.…

【微信小程序开发】模板与配置

文章目录 前言1. WXML 模板语法1.1 数据绑定1.2 事件绑定1.2.1 bindtap 事件1.2.2 bindinput 事件 1.3 条件渲染1.4 列表渲染 2. WXSS 模板样式2.1 什么是 rpx 尺寸单位2.2 什么是样式导入2.3 全局样式和局部样式2.3.1 全局配置2.3.1.1 window 节点常用的配置项:2.3…

反函数

一、反函数的定义 对每个y∈f(D),都有唯一的x∈D,使得f(x)y。即一个y对应一个x 举例 求得反函数的方法也比较简单: 在y f(x)中将y与x互换;重新整理为用x的表达式表示y的格式 。 二、反函数的图像性质 一个函数和反函数之间…

SpringBoot中阿里云OSS的使用

目录 1 登录/注册阿里云并进入控制台 2 进入OSS控制台 3 创建bucket 4 查看bucket 5 获取AccessKey 6 查看帮助文档 7 添加Maven依赖 8 获取示例代码并改造成工具类 9 测试 1 登录/注册阿里云并进入控制台 2 进入OSS控制台 3 创建bucket 4 查看bucket 5 获取AccessKe…

Unity MonoBehaviour 生成dll

dllllllllllllll🥓 🥙vs创建类库项目🧀添加UnityEngine、UnityEditor引用🍕添加MonoBehaviour类🦪设置dll生成路径🍿生成dll🍔使用dll中的Mono类 🥙vs创建类库项目 🧀添加…

Kotlin快速入门系列3

Kotlin条件、循环控制 IF条件控制 与Java类似,一个if语句可包含布尔表达式和一条或多条语句。 fun compare(a:Int,b:Int) : Int{//常规传统用法var max aif (b>a) max breturn max//使用elsevar mMax : Intif(a>b){mMax a}else{mMax b}return mMax//使…

【游戏客户端开发的进阶路线】

*** 游戏客户端开发的进阶路线 春招的脚步越来越近,我们注意到越来越多的同学们都在积极学习游戏开发,希望能在这个充满活力的行业中大展拳脚。 当我们思考如何成为游戏开发领域的佼佼者时,关键在于如何有效规划学习路径。 🤔 我…

【学网攻】 第(9)节 -- 路由器使用以及原理

系列文章目录 目录 系列文章目录 文章目录 前言 一、路由器是什么? 二、实验 1.引入 总结 文章目录 【学网攻】 第(1)节 -- 认识网络【学网攻】 第(2)节 -- 交换机认识及使用【学网攻】 第(3)节 -- 交换机配置聚合端口【学网攻】 第(4)节 -- 交换机划分Vlan…

代码随想录算法训练营第32天 | 122.买卖股票的最佳时机II + 55. 跳跃游戏 + 45.跳跃游戏II

今日任务 122.买卖股票的最佳时机II 55. 跳跃游戏 45.跳跃游戏II 122.买卖股票的最佳时机II - Medium 题目链接:力扣(LeetCode)官网 - 全球极客挚爱的技术成长平台 给你一个整数数组 prices ,其中 prices[i] 表示某支股票第 i…

高中数学:集合

一、基本概念与关系 1、元素 2、集合 集合中元素的特性 1、确定性。2、无序性。3、互异性。 3、空集,用∅符号表示 4、元素与集合的关系是属于关系,用∈符号表示 5、集合与集合的关系是包含关系。用⊆或者⊊符号表示 子集与真子集。 A⊊B > A⊆B 反之…

HCIP复习课(bgp实验)

1、ip配置: R1: R2: R3: R4: R5: R6: R7: R8: 2、bgp配置: R1: R2: R3: R4: R5: R6:…

Apache Flink连载(二十六):TaskSlot任务槽-(1)-TaskSlotSlotSharingGroup

🏡 个人主页:IT贫道-CSDN博客 🚩 私聊博主:私聊博主加WX好友,获取更多资料哦~ 🔔 博主个人B栈地址:豹哥教你学编程的个人空间-豹哥教你学编程个人主页-哔哩哔哩视频 提交到集群中的Flink程序最终会转换成一个个的Subtask,Subtask是Flink任务调度的基本单元,这些t…

LaTeX基础使用【系列一】

🌈个人主页:godspeed_lucip 🔥 系列专栏:LaTeX基础使用 🌏1 LaTeX基本结构🍉1.1 基本结构🍉1.2 小tip1.2.1 注释的写法1.2.2 数学公式 🌏2 LaTeX中文的处理🥑2.1 前提工作…

C++ 重点内容:友元

目录 友元函数: 友元成员函数: 友元类: 友元是否有悖于OOP? 总结: 类因为具有封装和信息隐藏的特性(类外函数无法访问类的私有、保护成员),C提出友元解决特定的编程需要;友元分…

第八篇 交叉编译华为云Iot SDK到Orangepi3B

本篇主要内容: 一、交叉编译华为云Iot SDK依赖1.宿主机安装交叉编译工具链(1)选择下载交叉编译工具链(2)解压、添加环境变量、重启2.交叉编译依赖库(0) 准备工作(1) 交叉…

Ant Design Mini - 支付宝小程序官方推出的免费开源 UI 组件库,新增支持微信小程序,实用性大大增加

支付宝小程序官方的 UI 组件库开始支持运行在微信小程序上了,如果要开发这两家小程序平台,这套组件很合适。 Ant Design Mini 也简称 antd-mini ,是一套运行在支付宝小程序的 UI 组件库,UI 设计遵循 Ant Design 规范,…

【PyTorch】使用PyTorch创建卷积神经网络并在CIFAR-10数据集上进行分类

前言 在深度学习的世界中,图像分类任务是一个经典的问题,它涉及到识别给定图像中的对象类别。CIFAR-10数据集是一个常用的基准数据集,包含了10个类别的60000张32x32彩色图像。在本博客中,我们将探讨如何使用PyTorch框架创建一个简…

TensorFlow2实战-系列教程2:神经网络分类任务

🧡💛💚TensorFlow2实战-系列教程 总目录 有任何问题欢迎在下面留言 本篇文章的代码运行界面均在Jupyter Notebook中进行 本篇文章配套的代码资源已经上传 1、Mnist数据集 下载mnist数据集: %matplotlib inline from pathlib imp…

vs2019报错MSB4019 找不到导入的项目“BuildCustomizations\CUDA 9.2.props”

在VS中执行生成,报错如下:严重性 代码 说明 项目 文件 行 禁止显示状态 错误 MSB4019 找不到导入的项目“D:\Microsoft Visual Studio\2019\Community\MSBuild\Microsoft\VC\v160\BuildCustomizations\CUDA 9.2.props”。请确认 Import 声明“D:\Microso…

在autodl训练yolov8时卡在下载字体

1.问题 在autodl训练yolov8到这一步之后会卡住很久 2. 解决办法 Ctric中断后发现是下载Arial字体卡住了,这个字体需要从外网中下载 先手动从链接中下载https://ultralytics.com/assets/Arial.ttf ,然后上传到autodl。然后将这个文件移动到/root/.config/…