一、背景
  最近做个项目,需要使用STM32,还是以前一样的观点,时钟就是MCU心脏,供血即时钟频率输出,想要弄明白一个MCU,时钟是一个非常好的切入点。言归正传,网上已经有太多大神详述过STM32的详细配置方法了,在此就简单介绍下STM32时钟系统,以及如何配置做个简单记录,方便以后的快速开发。

二、正文
  废话不多说,上一张STM32F10xx的时钟树图:

  

  由图可知,STM32F10XX有两级时钟
  第一级时钟
    * 高速内部时钟(HSI)
    * 锁相环时钟(PLLCLK)
    * 高速外部时钟(HSE)
  第二级时钟
    * 低速内部时钟(LSI)
    * 低速外部时钟(LSE)

  又由图可知,
    * HSE由外部晶振从"OSC_OUT","OSC_IN"两脚输入提供。
    * LSE由外部晶振从"OSC32_OUT","OSC32_IN"两脚输入提供。
    * HSI由8MHZ高速内部RC震荡电路提供。
    * LSI由40kHZ低速内部RC震荡电路提供。

  STM32F10XX还可通过MCO脚向外提供时钟输出。时钟来源有PLLCLK/HSI/HSE/SYSCLK,由MCO选择器来选择。

  研究过时钟来源,再来研究时钟的去向,MCU自身要能正常运作,即需要一个时钟,这个时钟既是系统时钟(SYSCLK),而基本上所有外设的时钟均来自于这个系统时钟(SYSCLk)。然后由系统时钟对外提供各种外设时钟。详见图。  

  当然,也有例外,USB时钟必须为48MHZ,这里的USB时钟(USBCLK)由PLLCLK直接提供,RTC时钟 (RTCCLK)也不是来源于系统时钟(SYSCLK),详见图。  

  时钟结构大体也就如此,不再深究,网上有许多更加深入的讲解,接下来说说如何去配置。用代码来说明问题:

  先贴文件"system_stm32f10x.c",此文件即库文件。里面有一个很重要的函数"SystemInit()"

  解析过STM32启动代码的朋友都应该知道,这个函数跑在进入main函数之前,里面做的事情即是配置系统时钟。代码如下:

  1. void SystemInit (void)
  2. {
  3. /* Reset the RCC clock configuration to the default reset state(for debug purpose) */
  4. /* Set HSION bit */
  5. RCC->CR |= (uint32_t)0x00000001;
  6.  
  7. /* Reset SW, HPRE, PPRE1, PPRE2, ADCPRE and MCO bits */
  8. #ifndef STM32F10X_CL
  9. RCC->CFGR &= (uint32_t)0xF8FF0000;
  10. #else
  11. RCC->CFGR &= (uint32_t)0xF0FF0000;
  12. #endif /* STM32F10X_CL */
  13.  
  14. /* Reset HSEON, CSSON and PLLON bits */
  15. RCC->CR &= (uint32_t)0xFEF6FFFF;
  16.  
  17. /* Reset HSEBYP bit */
  18. RCC->CR &= (uint32_t)0xFFFBFFFF;
  19.  
  20. /* Reset PLLSRC, PLLXTPRE, PLLMUL and USBPRE/OTGFSPRE bits */
  21. RCC->CFGR &= (uint32_t)0xFF80FFFF;
  22.  
  23. #ifdef STM32F10X_CL
  24. /* Reset PLL2ON and PLL3ON bits */
  25. RCC->CR &= (uint32_t)0xEBFFFFFF;
  26.  
  27. /* Disable all interrupts and clear pending bits */
  28. RCC->CIR = 0x00FF0000;
  29.  
  30. /* Reset CFGR2 register */
  31. RCC->CFGR2 = 0x00000000;
  32. #elif defined (STM32F10X_LD_VL) || defined (STM32F10X_MD_VL) || (defined
  33.  
  34. STM32F10X_HD_VL)
  35. /* Disable all interrupts and clear pending bits */
  36. RCC->CIR = 0x009F0000;
  37.  
  38. /* Reset CFGR2 register */
  39. RCC->CFGR2 = 0x00000000;
  40. #else
  41. /* Disable all interrupts and clear pending bits */
  42. RCC->CIR = 0x009F0000;
  43. #endif /* STM32F10X_CL */
  44.  
  45. #if defined (STM32F10X_HD) || (defined STM32F10X_XL) || (defined STM32F10X_HD_VL)
  46. #ifdef DATA_IN_ExtSRAM
  47. SystemInit_ExtMemCtl();
  48. #endif /* DATA_IN_ExtSRAM */
  49. #endif
  50.  
  51. /* Configure the System clock frequency, HCLK, PCLK2 and PCLK1 prescalers */
  52. /* Configure the Flash Latency cycles and enable prefetch buffer */
  53. SetSysClock(); //此函数在该函数末尾
  54.  
  55. #ifdef VECT_TAB_SRAM
  56. SCB->VTOR = SRAM_BASE | VECT_TAB_OFFSET; /* Vector Table Relocation in Internal
  57.  
  58. SRAM. */
  59. #else
  60. SCB->VTOR = FLASH_BASE | VECT_TAB_OFFSET; /* Vector Table Relocation in Internal
  61.  
  62. FLASH. */
  63. #endif
  64.  
  65. }
  66.  
  67. static void SetSysClock(void)
  68. {
  69. // 根据宏定义来设置时钟。
  70. #ifdef SYSCLK_FREQ_HSE
  71. SetSysClockToHSE();
  72. #elif defined SYSCLK_FREQ_24MHz
  73. SetSysClockTo24();
  74. #elif defined SYSCLK_FREQ_36MHz
  75. SetSysClockTo36();
  76. #elif defined SYSCLK_FREQ_48MHz
  77. SetSysClockTo48();
  78. #elif defined SYSCLK_FREQ_56MHz
  79. SetSysClockTo56();
  80. #elif defined SYSCLK_FREQ_72MHz
  81. SetSysClockTo72(); // 以设置成最大频率72MHZ为例
  82. #endif
  83.  
  84. /* If none of the define above is enabled, the HSI is used as System clock
  85. source (default after reset) */
  86. }
  87.  
  88. static void SetSysClockTo72(void)
  89. {
  90. __IO uint32_t StartUpCounter = , HSEStatus = ;
  91.  
  92. /* SYSCLK, HCLK, PCLK2 and PCLK1 configuration ---------------------------*/
  93. /* Enable HSE */
  94. RCC->CR |= ((uint32_t)RCC_CR_HSEON);
  95.  
  96. /* Wait till HSE is ready and if Time out is reached exit */
  97. do
  98. {
  99. HSEStatus = RCC->CR & RCC_CR_HSERDY;
  100. StartUpCounter++;
  101. } while((HSEStatus == ) && (StartUpCounter != HSE_STARTUP_TIMEOUT));
  102.  
  103. if ((RCC->CR & RCC_CR_HSERDY) != RESET)
  104. {
  105. HSEStatus = (uint32_t)0x01;
  106. }
  107. else
  108. {
  109. HSEStatus = (uint32_t)0x00;
  110. }
  111.  
  112. if (HSEStatus == (uint32_t)0x01)
  113. {
  114. /* Enable Prefetch Buffer */
  115. FLASH->ACR |= FLASH_ACR_PRFTBE;
  116.  
  117. /* Flash 2 wait state */
  118. FLASH->ACR &= (uint32_t)((uint32_t)~FLASH_ACR_LATENCY);
  119. FLASH->ACR |= (uint32_t)FLASH_ACR_LATENCY_2;
  120.  
  121. /* HCLK = SYSCLK */
  122. RCC->CFGR |= (uint32_t)RCC_CFGR_HPRE_DIV1;
  123.  
  124. /* PCLK2 = HCLK */
  125. RCC->CFGR |= (uint32_t)RCC_CFGR_PPRE2_DIV1;
  126.  
  127. /* PCLK1 = HCLK */
  128. RCC->CFGR |= (uint32_t)RCC_CFGR_PPRE1_DIV2;
  129.  
  130. #ifdef STM32F10X_CL
  131. /* Configure PLLs ------------------------------------------------------*/
  132. /* PLL2 configuration: PLL2CLK = (HSE / 5) * 8 = 40 MHz */
  133. /* PREDIV1 configuration: PREDIV1CLK = PLL2 / 5 = 8 MHz */
  134.  
  135. RCC->CFGR2 &= (uint32_t)~(RCC_CFGR2_PREDIV2 | RCC_CFGR2_PLL2MUL |
  136. RCC_CFGR2_PREDIV1 | RCC_CFGR2_PREDIV1SRC);
  137. RCC->CFGR2 |= (uint32_t)(RCC_CFGR2_PREDIV2_DIV5 | RCC_CFGR2_PLL2MUL8 |
  138. RCC_CFGR2_PREDIV1SRC_PLL2 | RCC_CFGR2_PREDIV1_DIV5);
  139.  
  140. /* Enable PLL2 */
  141. RCC->CR |= RCC_CR_PLL2ON;
  142. /* Wait till PLL2 is ready */
  143. while((RCC->CR & RCC_CR_PLL2RDY) == )
  144. {
  145. }
  146.  
  147. /* PLL configuration: PLLCLK = PREDIV1 * 9 = 72 MHz */
  148. RCC->CFGR &= (uint32_t)~(RCC_CFGR_PLLXTPRE | RCC_CFGR_PLLSRC | RCC_CFGR_PLLMULL);
  149. RCC->CFGR |= (uint32_t)(RCC_CFGR_PLLXTPRE_PREDIV1 | RCC_CFGR_PLLSRC_PREDIV1 |
  150. RCC_CFGR_PLLMULL9);
  151. #else
  152. /* PLL configuration: PLLCLK = HSE * 9 = 72 MHz */
  153. RCC->CFGR &= (uint32_t)((uint32_t)~(RCC_CFGR_PLLSRC | RCC_CFGR_PLLXTPRE |
  154. RCC_CFGR_PLLMULL));
  155. RCC->CFGR |= (uint32_t)(RCC_CFGR_PLLSRC_HSE | RCC_CFGR_PLLMULL9);
  156. #endif /* STM32F10X_CL */
  157.  
  158. /* Enable PLL */
  159. RCC->CR |= RCC_CR_PLLON;
  160.  
  161. /* Wait till PLL is ready */
  162. while((RCC->CR & RCC_CR_PLLRDY) == )
  163. {
  164. }
  165.  
  166. /* Select PLL as system clock source */
  167. RCC->CFGR &= (uint32_t)((uint32_t)~(RCC_CFGR_SW));
  168. RCC->CFGR |= (uint32_t)RCC_CFGR_SW_PLL;
  169.  
  170. /* Wait till PLL is used as system clock source */
  171. while ((RCC->CFGR & (uint32_t)RCC_CFGR_SWS) != (uint32_t)0x08)
  172. {
  173. }
  174. }
  175. else
  176. { /* If HSE fails to start-up, the application will have wrong clock
  177. configuration. User can add here some code to deal with this error */
  178. }
  179. }

  

  文件"system_stm32f10x.c"的属性默认被设置为只读文件,最好不要更改里面的东西,毕竟启动以及复位之后,最先运行的是这个文件内的函数。

  如果想要更改MCU时钟频率,建议自己写一个函数来进行更改,代码如下:

  1. /**
  2. * @brief 配置系统时钟(72MHz)
  3. * @param None
  4. * @retval None
  5. */
  6. void RCC_Configuration(void)
  7. {
  8. ErrorStatus HSEStartUpStatus;
  9.  
  10. RCC_DeInit();
  11. RCC_HSEConfig(RCC_HSE_ON);
  12. HSEStartUpStatus = RCC_WaitForHSEStartUp();
  13. if(HSEStartUpStatus == SUCCESS) {
  14. FLASH_PrefetchBufferCmd(FLASH_PrefetchBuffer_Enable);
  15. FLASH_SetLatency(FLASH_Latency_2);
  16. // HCLK(AHBCLK) = SYSCLK = 72MHZ
  17. RCC_HCLKConfig(RCC_SYSCLK_Div1);
  18. // APB2CLK = HCLK = 72MHZ
  19. RCC_PCLK2Config(RCC_HCLK_Div1);
  20. // APB1CLK = HCLK/4 = 18MHZ
  21. RCC_PCLK1Config(RCC_HCLK_Div4);
  22. // 选择PLLCLK为输入时钟,PLLMUL9倍频. 8MHZ*9 = 72MHZ
  23. RCC_PLLConfig(RCC_PLLSource_HSE_Div1, RCC_PLLMul_9);
  24. RCC_PLLCmd(ENABLE);
  25. while(RCC_GetFlagStatus(RCC_FLAG_PLLRDY) == RESET){
            ;
  26. }
  27. //PLL作为系统时钟的输入
  28. RCC_SYSCLKConfig(RCC_SYSCLKSource_PLLCLK);
  29. while(RCC_GetSYSCLKSource() != 0x08) {
            ;
  30. }
  31. }
  32. }

  至此,记录完毕。

记录地点:深圳WZ

记录时间:2016年8月16日

Cotex-M3内核STM32F10XX系列时钟及其配置方法的更多相关文章

  1. Cotex-M3内核LPC17xx系列时钟及其配置方法

    一.背景: 最近正在接手一个项目,核心芯片既是LPC17XX系列MCU,内核为ARM的Cotex-M3内核. 想要玩转一个MCU,就一定得搞定其时钟! 时钟对MCU而言,就好比人类的心脏.由其给AHB ...

  2. 国产新芯片连不上J-Link?芯海CS32L010系列芯片JLink配置方法

    疫情以来芯片供货紧张,特别是ST的MCU一芯难求.所以很多产品不得不切换成国产.不过也是经过使用后才发现,很多国产芯片的性能还是挺好的.由于芯片比较新,官方J-Link还没有支持,所以调试和烧录有些不 ...

  3. STM32F2系列时钟的配置

    前一节详细介绍了系统默认的时钟配置,及各路时钟输出是多少,这是默认配置的,但实际使用的时钟默认的时钟并不符合要求,所以就得知道如何调用库函数进行配置. 最好的资料就是查阅stm32f2xx_rcc.c ...

  4. STM32F7系列时钟相关问题:HSE模式配置(旁路模式、非旁路模式

    从时钟源的角度,分为两类外部时钟(E)和内部时钟(I).从时钟速率的角度,分为两类高速时钟(HS)和低速时钟(LS).而把它们组合起来就有四种时钟:HSE.HIS.LSE.LSI.至于为什么会有这么复 ...

  5. STM32入门系列-STM32时钟系统,时钟使能配置函数

    之前的推文中说到,当使用一个外设时,必须先使能它的时钟.怎么通过库函数使能时钟呢?如需了解寄存器配置时钟,可以参考<STM32F10x中文参考手册>"复位和时钟控制(RCC)&q ...

  6. STM32 M3内核的位带操作原理及步骤

    STM32 M3内核的位带操作原理及步骤 一.位带操作有什么用?什么是位带操作 位带操作的作用:可以实现对某一GPIO口寄存器(或SRAM内存中)的某一bit位直接写0或1,达到控制GPIO口输出(或 ...

  7. Spartan6系列之芯片配置模式详解

    1.   配置概述 Spartan6系列FPGA通过把应用程序数据导入芯片内部存储器完成芯片的配置.Spart-6 FPGA可以自己从外部非易失性存储器导入编程数据,或者通过外界的微处理器.DSP等对 ...

  8. [转载]:STM32为什么必须先配置时钟再配置GPIO

    转载来源 :http://blog.csdn.net/fushiqianxun/article/details/7926442 [原创]:我来添两句,就是很多同学(包括我)之前搞低端单片机,到了stm ...

  9. ANDROID Porting系列二、配置一个新产品

    ANDROID Porting系列二.配置一个新产品 详细说明 下面的步骤描述了如何配置新的移动设备和产品的makefile运行android. 1.         目录//vendor/创建一个公 ...

随机推荐

  1. 原生JS中常用的Window和DOM对象操作汇总

    一.常用的Window对象操作 Window对象中又包含了document.history.location.Navigator和screen几个对象,每个对象又有自己的属性方法,这里window可以 ...

  2. python使用cookielib库示例分享

    Python中cookielib库(python3中为http.cookiejar)为存储和管理cookie提供客户端支持,下面是使用示例 该模块主要功能是提供可存储cookie的对象.使用此模块捕获 ...

  3. Linux下安装py-leveldb

    1.下载源代码 svn checkout http://py-leveldb.googlecode.com/svn/trunk/ py-leveldb-read-only 2.安装辅助工具 sudo ...

  4. Bumped Map And Normal Map

    http://freespace.virgin.net/hugo.elias/graphics/x_polybm.htm 先留着,准备以后开垦

  5. IAR Usage

    ctrl+shift+f: 全局搜索

  6. HTML中<meta>标签如何正确使用

    HTML中<meta>标签如何正确使用 如果我们在浏览器中按下F12或者Ctrl+shift+J,便可以打开开发者工具,在element中即可看到<head>元素中有不少< ...

  7. 从Paxos到ZooKeeper-二、ZooKeeper和Paxos

    ZooKeeper为分布式应用提供了高效且可靠的分布式协调服务,提供了诸如tong'yi统一命名服务.配置管理和分布式锁等分布式的基础服务.在解决分布式数据一致性方面,ZooKeeper并没有直接采用 ...

  8. SQL 操作语句

    SQL Server T-SQL高级查询 高级查询在数据库中用得是最频繁的,也是应用最广泛的. Ø 基本常用查询 --select select * from student; --all 查询所有 ...

  9. Linux服务器间文件传输

    利用scp传输文件 1.从服务器下载文件 scp username@servername:/path/filename /tmp/local_destination 例如scp codinglog@1 ...

  10. Bois设置教程

    BIOS设置图解教程之Award篇 (目前主板上常见的BIOS主要为AMI与AWARD两个系列,如何辨别BIOS品牌系列请移步,本文详细讲解Award系列的BIOS设置图解教程,如果你的BIOS为AM ...