# 九：SystemInit()实现的功能

花了一天的时间，总算是了解了SystemInit()函数实现了哪些功能，初学STM32，，现记录如下（有理解错误的地方还请大侠指出）：

使用的是3.5的库，用的是STM32F107VC，开发环境RVMDK4.23

我已经定义了STM32F10X\_CL，SYSCLK\_FREQ\_72MHz

**函数调用顺序：**

**startup\_stm32f10x\_cl.s（启动文件） → SystemInit() →  SetSysClock () → SetSysClockTo72()**

初始化时钟用到的RCC寄存器**复位值：**

RCC\_CR = 0x0000 xx83; RCC\_CFGR = 0x0000 0000；RCC\_CIR = 0x0000 0000; RCC\_CFGR2 = 0x0000 0000;

**SystemInit()**

在调用 SetSysClock()之前RCC寄存器的值如下（都是一些与运算，或运算，在此就不赘述了）：

RCC->CR = 0x0000 0083;  RCC->CIR = 0x00FF0000; RCC->CFGR2 = 0x00000000;至于这些寄存器都代表着什么意思，详见芯片资料RCC寄存器，该文重点不在此处；

**SetSysClock()函数如下：**

static void SetSysClock(void)

{

#ifdef SYSCLK\_FREQ\_HSE

  SetSysClockToHSE();

#elif defined SYSCLK\_FREQ\_24MHz

  SetSysClockTo24();

#elif defined SYSCLK\_FREQ\_36MHz

  SetSysClockTo36();

#elif defined SYSCLK\_FREQ\_48MHz

  SetSysClockTo48();

#elif defined SYSCLK\_FREQ\_56MHz

  SetSysClockTo56();

#elif defined SYSCLK\_FREQ\_72MHz //我的定义的是SYSCLK\_FREQ\_72MHz，所以调用SetSysClockTo72()

  SetSysClockTo72();

#endif

}

**SetSysClockTo72()函数如下：**

static void SetSysClockTo72(void)

{

  \_\_IO uint32\_t StartUpCounter = 0, HSEStatus = 0;

   /\* SYSCLK, HCLK, PCLK2 and PCLK1 configuration ---------------------------\*/

  /\* Enable HSE \*/

  RCC->CR |= ((uint32\_t)RCC\_CR\_HSEON);

  /\* Wait till HSE is ready and if Time out is reached exit \*/

  do

  {

    HSEStatus = RCC->CR & RCC\_CR\_HSERDY;

    StartUpCounter++;

  } while((HSEStatus == 0) && (StartUpCounter != HSE\_STARTUP\_TIMEOUT));

  if ((RCC->CR & RCC\_CR\_HSERDY) != RESET)

  {

    HSEStatus = (uint32\_t)0x01;

  }

  else

  {

    HSEStatus = (uint32\_t)0x00;

  }

  if (HSEStatus == (uint32\_t)0x01)

  {

    /\* Enable Prefetch Buffer \*/

    FLASH->ACR |= FLASH\_ACR\_PRFTBE;

    /\* Flash 2 wait state \*/

    FLASH->ACR &= (uint32\_t)((uint32\_t)~FLASH\_ACR\_LATENCY);

    FLASH->ACR |= (uint32\_t)FLASH\_ACR\_LATENCY\_2;

**/\* HCLK = SYSCLK \*/**

**RCC->CFGR |= (uint32\_t)RCC\_CFGR\_HPRE\_DIV1;**

**/\* PCLK2 = HCLK \*/**

**RCC->CFGR |= (uint32\_t)RCC\_CFGR\_PPRE2\_DIV1;**

**/\* PCLK1 = HCLK \*/**

**RCC->CFGR |= (uint32\_t)RCC\_CFGR\_PPRE1\_DIV2;**

#ifdef STM32F10X\_CL

    /\* Configure PLLs ------------------------------------------------------\*/

    /\* PLL2 configuration: PLL2CLK = (HSE / 5) \* 8 = 40 MHz \*/

    /\* PREDIV1 configuration: PREDIV1CLK = PLL2 / 5 = 8 MHz \*/

**RCC->CFGR2 &= (uint32\_t)~(RCC\_CFGR2\_PREDIV2 | RCC\_CFGR2\_PLL2MUL |**

**RCC\_CFGR2\_PREDIV1 | RCC\_CFGR2\_PREDIV1SRC);**

**RCC->CFGR2 |= (uint32\_t)(RCC\_CFGR2\_PREDIV2\_DIV5 | RCC\_CFGR2\_PLL2MUL8 |**

**RCC\_CFGR2\_PREDIV1SRC\_PLL2 | RCC\_CFGR2\_PREDIV1\_DIV5);**

     /\* Enable PLL2 \*/

    RCC->CR |= RCC\_CR\_PLL2ON;

    /\* Wait till PLL2 is ready \*/

    while((RCC->CR & RCC\_CR\_PLL2RDY) == 0)

    {

    }

      /\* PLL configuration: PLLCLK = PREDIV1 \* 9 = 72 MHz \*/

**RCC->CFGR &= (uint32\_t)~(RCC\_CFGR\_PLLXTPRE | RCC\_CFGR\_PLLSRC | RCC\_CFGR\_PLLMULL);**

**RCC->CFGR |= (uint32\_t)(RCC\_CFGR\_PLLXTPRE\_PREDIV1 | RCC\_CFGR\_PLLSRC\_PREDIV1 |**

**RCC\_CFGR\_PLLMULL9);**

#else

    /\*  PLL configuration: PLLCLK = HSE \* 9 = 72 MHz \*/

    RCC->CFGR &= (uint32\_t)((uint32\_t)~(RCC\_CFGR\_PLLSRC | RCC\_CFGR\_PLLXTPRE |

                                        RCC\_CFGR\_PLLMULL));

    RCC->CFGR |= (uint32\_t)(RCC\_CFGR\_PLLSRC\_HSE | RCC\_CFGR\_PLLMULL9);

#endif /\* STM32F10X\_CL \*/

    /\* Enable PLL \*/

    RCC->CR |= RCC\_CR\_PLLON;

    /\* Wait till PLL is ready \*/

    while((RCC->CR & RCC\_CR\_PLLRDY) == 0)

    {

    }

    /\* Select PLL as system clock source \*/

**RCC->CFGR &= (uint32\_t)((uint32\_t)~(RCC\_CFGR\_SW));**

**RCC->CFGR |= (uint32\_t)RCC\_CFGR\_SW\_PLL;**

    /\* Wait till PLL is used as system clock source \*/

    while ((RCC->CFGR & (uint32\_t)RCC\_CFGR\_SWS) != (uint32\_t)0x08)

    {

    }

  }

  else

  { /\* If HSE fails to start-up, the application will have wrong clock

         configuration. User can add here some code to deal with this error \*/

  }

}

**1：AHB, APB1，APB2时钟确定**

//HCLK = SYSCLK ,从下面的分析可以得出SYSCLK是使用PLLCLK时钟的，也就是72MHZ（**至于72MHZ如何得来，请看下面分析**）

//那么就是HCLK(AHB总线时钟)=PLLCLK = 72MHZ

//AHB总线时钟等于系统时钟SYSCLK，

//**也就是 AHB时钟 = HCLK = SYSCLK = 72MHZ**

/\* HCLK = SYSCLK \*/

**RCC->CFGR |= (uint32\_t)RCC\_CFGR\_HPRE\_DIV1;**

//PLCK2等于HCLK一分频， 所以PCLK2 = HCLK，HCLK = 72MHZ,

//那么PLCK2(APB2总线时钟) = 72MHZ

//APB2总线时钟等于HCLK的一分频，也就是不分频**；**

**//APB2 时钟 = HCLK = SYSCLK = 72MHZ**

/\* PCLK2 = HCLK \*/

**RCC->CFGR |= (uint32\_t)RCC\_CFGR\_PPRE2\_DIV1;**

//PCLK1 = HCLK / 2；PCLK1 等于HCLK时钟的二分频，

//那么PCLK1(APB1) = 72MHZ / 2 = 36MHZ

//APB1总线时钟等于HCLK的二分频，也就是 **APB1时钟= HCLK / 2 = 36MHZ**

/\* PCLK1 = HCLK \*/

**RCC->CFGR |= (uint32\_t)RCC\_CFGR\_PPRE1\_DIV2;**

**2：如何得出SYSCLK（系统时钟）为72MHZ(外部晶振25MHZ)**

//记得参考英文芯片资料的时钟树P115页和RCC时钟寄存器进行理解

**RCC->CFGR2 |= (uint32\_t)(RCC\_CFGR2\_PREDIV2\_DIV5 | RCC\_CFGR2\_PLL2MUL8|RCC\_CFGR2\_PREDIV1SRC\_PLL2 | RCC\_CFGR2\_PREDIV1\_DIV5);**

**RCC\_CFGR2\_PREDIV2\_DIV5:**PREDIV2 = 5； 5分频

 也就是PREDIV2对输入的外部时钟 5分频,那么PLL2和PLL3没有倍频前

是25 /5 = 5MHZ

**RCC\_CFGR2\_PLL2MUL8  :** PLL2MUL = 8； 8倍频

 8倍频后,PLL2时钟 = 5 \* 8 = 40MHZ; 因此 PLL2CLK = 40MHZ

**RCC\_CFGR2\_PREDIV1SRC\_PLL2 :** RCC\_CFGR2的第16位为1， 选择**PLL2CLK 作为PREDIV1的时钟源**

**RCC\_CFGR2\_PREDIV1\_DIV5：PREDIV1 = 5；**PREDIV1对输入时钟5分频 **PREDIV1CLK**= PLL2CLK / 5 = 8MHZ

![http://home.eeworld.com.cn/my/image/face/6.gif](data:image/gif;base64,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)![http://home.eeworld.com.cn/my/image/face/1.gif](data:image/gif;base64,R0lGODlhFAAUAOYAAP//////lPf39//33v/3zv/3vf/3nP/3lP/3hP/3jP/v1u/v7//vpf/vhP/ve//mvf/mre/m3vfmtf/mlPfmrf/mhP/mjPfmpf/mc//me//ma//etffetf/epffepf/elP/ehPfenP/ee//ec//eY//ea/fWpf/WjP/WhP/We/fWlP/Wc/fWhP/Wa//WY//WUv/WWvfWc//Oc//Oa//OWvfOc//OUvfOa+bOjP/OSvfOWubOhPfOSvfOQu/OWvfOOv/FY//FWvfFc/fFa//FUv/FSvfFWv/FQv/FOvfFQvfFOvfFMe/FOve9a+/FMf+9Qve9Wv+9Sve9Sv+9Ov+9Mfe9Ke+9Ou+9Me+9Ke+9Gfe1Sv+1Ov+1Mf+1If+1Kfe1Kfe1Ife1Ge+1Ke+1Ge+1EP+tIf+tEP+tGfetKfetIfetGfetEO+tGe+tEP+lGf+lEPelKf+lAPelGf+lCPelEPelCO+lGf+cEP+cCPecEPecCO+cCJwxAP///wAAAAAAACH/C05FVFNDQVBFMi4wAwEAAAAh+QQFZAB9ACwAAAAAFAAUAAAH/4B9goMXPlhkZGJGHoONgxJYTCwdEB0oRmJiEo6CMU4UBAwVGCQtLTJEbDWON04DBQkIDhglLi85REFsQ49XCgYBB7K0tjlHT1F2m31YIRM9WVk9GqQwOVWIYF9ofRdMHT04ERE4PyQkNkvi5FVrHjoxPDsLAgILO0ouS/P192FQYj5kiSCADx8BEbLAGFjwYEI0bQQSNIgwiw2GFBPuaXNCHj17O3jkSMIPpJI9YmS06LFj3I4eMI7QIOlSyRE0RqTQoAFNGo0iU54QSRJNyRM1UDykAXLEhQgRLo5QoeKFC5EWM56c0cNIzJMnU6dy8eKlTJkzb/DMmcOtjwQ7UUXGku1i9oybtGr1bBg0RM6Usmfo7NmT547aOHqaOKpBp4sXOioAAFChZw5iIZz6bEBDB7Jkynrg7M0syIMWO4PtaDHBKRAAIfkEBQoAfQAsAgAFABEACQAAB36AfQwICQkNDn2JfTlHU1NcfQ8ZDg2HGIkwNkVHVF5TfRYYoqIaJH0vjEidXmkjJSUaryQufZqNXF1lXTB8vX18MJk5vr9nXTYvE8rKNolIICIpKytnNjk22Dk5m6pcXmVub2ZER0eb5Uh9VN5lZ29vdX2OSEiOin3tb3hmfYEAIfkEBRQAfQAsAgAFABEACQAAB3aAfRM9WVk9Gn2JfVVkZGBffR09OBEROD+JNkuUllV9PDsLAgILO0p9S6GjpWFZEQJ8fAIRWX2usLK0t4mztbt9vaCipDs8iarESi09O5U7PTB9NEnNEaZHNDSEhjRFfU9ESYVKT2p9Ry4iIi5Hin1ELTNPZ32BACH5BAUKAH0ALAIABQARAAkAAAd+gH0MCAkJDQ59iX05R1NTXH0PGQ4NhxiJMDZFR1ReU30WGKKiGiR9L4xInV5pIyUlGq8kLn2ajVxdZV0wfL29MDCLvr1nXTYvE8nJNolIICIpKytnNjk21zk5RX2qXF5lbm9mREdHm+VIfVTeZWdvb3V9jkhIjop97W94Zn2BACH5BAkUAH0ALAAAAAAUABQAAAeOgH2Cg4SFhoeIiYqLjI2MEz1ZWT0ahFVkZGBfhB09OBEROD+CNkufoVWEPDsLAgILO0p9S6yusGGDWRECfHwCEVl9ury+wLm7gr/Bw8nGgqutrzs8grXSsoItPTugOz0wfTRJ3BGxR4M0NJGTNEV9T0RJkkpPaoRHLiIiLueDRC0znpxxRLCgwYMIEw4KBAA7)**以上是对RCC\_CFGR2进行的配置**

**--------------------------------------------------------------------------------------**

**RCC->CFGR |= (uint32\_t)(RCC\_CFGR\_PLLXTPRE\_PREDIV1 | RCC\_CFGR\_PLLSRC\_PREDIV1 |**

**RCC\_CFGR\_PLLMULL9);**

**RCC\_CFGR\_PLLXTPRE\_PREDIV1 ：**操作的是RCC\_CFGR的第17位PLLXTPRE，操作这一位和操作RCC\_CFGR2寄存器的 位[3:0]中的最低位是相同的效果

**RCC\_CFGR\_PLLSRC\_PREDIV1 ：**选择PREDIV1输出作为PLL输入时钟;PREDIV1CLK = 8MHZ,所以输入给PLL倍频的 时钟源是8MHZ

**RCC\_CFGR\_PLLMULL9 ：PLLMUL = 9；PLL倍频系数为9，也就是对 PLLCLK = PREDIV1CLK \* 8 = 72MHZ**

**以上是对RCC\_CFGR进行的配置**

---------------------------------------------------------------------------------------

//选择PLLCLK作为系统时钟源

**RCC->CFGR |= (uint32\_t)RCC\_CFGR\_SW\_PLL;**

---------------------------------------------------------------------------------------

至此基本配置已经完成，配置的时钟如下所述：

SYSCLK(系统时钟) = 72MHZ

AHB总线时钟   = 72MHZ

APB1总线时钟  = 36MHZ

APB2总线时钟  = 72MHZ

PLL时钟   = 72MHZ

PLL2时钟  = 40MHZ