RCC: reset clock control 复位和时钟控制器。
注意: 任何一个外设在使用之前,必须首先使能其相应的时钟
来源:YSYCLK,HSI,HSE,PLLCLK二分频
配置函数在stm32f10x_rcc.h头文件中
寄存器描述在stm32f10x.h头文件中
typedef struct
{
//重要
__IO uint32_t CR; //HSI,HSE,CSS,PLL等的使能和就绪标志位
__IO uint32_t CFGR; //PLL等选择器的时钟源选择,分频系数设定
//
__IO uint32_t CIR; // 清除/使能 时钟就绪中断
__IO uint32_t APB2RSTR; //APB2线上外设复位寄存器
__IO uint32_t APB1RSTR; //APB1线上外设复位寄存器
//重要
__IO uint32_t AHBENR; //DMA,SDIO等时钟使能
__IO uint32_t APB2ENR; //APB2线上外设时钟使能
__IO uint32_t APB1ENR; //APB1线上外设时钟使能
//
__IO uint32_t BDCR; //备份域控制寄存器
__IO uint32_t CSR; //控制状态寄存器
} RCC_TypeDef;
RCC_LSEConfig();
RCC_HSEConfig();
RCC_HSICmd();
RCC_LSICmd();
RCC_PLLCmd(); // ……
RCC_PLLConfig ();
RCC_SYSCLKConfig();
RCC_RTCCLKConfig(); //…
RCC_HCLKConfig();
RCC_PCLK1Config();
RCC_PCLK2Config(); //…
RCC_APB1PeriphClockCmd(): //APB1线上外设时钟使能
RCC_APB2PeriphClockCmd(); //APB2线上外设时钟使能
RCC_AHBPeriphClockCmd(); //AHB线上外设时钟使能
RCC_ADCCLKConfig ();
RCC_RTCCLKConfig();
RCC_GetClocksFreq();
RCC_GetSYSCLKSource();
RCC_GetFlagStatus();
RCC_ITConfig();
RCC_GetITStatus();
RCC_ClearITPendingBit(); //…
函数位置:system_stm32f10x.c中 system_stm32f10x.h
//Reset 即为设置为0
void SystemInit (void)
{
RCC->CR |= (uint32_t)0x00000001; //Set HSION bit 打开HSION(第一位)
/* Reset SW, HPRE, PPRE1, PPRE2, ADCPRE and MCO bits */
#ifndef STM32F10X_CL //大容量芯片不执行
RCC->CFGR &= (uint32_t)0xF8FF0000;
#else
RCC->CFGR &= (uint32_t)0xF0FF0000;
#endif /* STM32F10X_CL */
/* Reset HSEON, CSSON and PLLON bits */
RCC->CR &= (uint32_t)0xFEF6FFFF;
/* Reset HSEBYP bit */
RCC->CR &= (uint32_t)0xFFFBFFFF;
/* Reset PLLSRC, PLLXTPRE, PLLMUL and USBPRE/OTGFSPRE bits */
RCC->CFGR &= (uint32_t)0xFF80FFFF;
#ifdef STM32F10X_CL //大容量芯片不会执行
/**************************************不执行*************************/
/* Reset PLL2ON and PLL3ON bits */
RCC->CR &= (uint32_t)0xEBFFFFFF;
/* Disable all interrupts and clear pending bits */
RCC->CIR = 0x00FF0000;
/* Reset CFGR2 register */
RCC->CFGR2 = 0x00000000;
#elif defined (STM32F10X_LD_VL) || defined (STM32F10X_MD_VL) || (defined STM32F10X_HD_VL)
/* Disable all interrupts and clear pending bits */
RCC->CIR = 0x009F0000;
/* Reset CFGR2 register */
RCC->CFGR2 = 0x00000000;
/**************************************不执行*************************/
#else
/* Disable all interrupts and clear pending bits */
RCC->CIR = 0x009F0000;
#endif /* STM32F10X_CL */
#if defined (STM32F10X_HD) || (defined STM32F10X_XL) || (defined STM32F10X_HD_VL)
#ifdef DATA_IN_ExtSRAM
SystemInit_ExtMemCtl();
#endif /* DATA_IN_ExtSRAM */
#endif
SetSysClock();
#ifdef VECT_TAB_SRAM
SCB->VTOR = SRAM_BASE | VECT_TAB_OFFSET; /* Vector Table Relocation in Internal SRAM. */
#else
SCB->VTOR = FLASH_BASE | VECT_TAB_OFFSET; /* Vector Table Relocation in Internal FLASH. */
#endif
}
TICKINT:是否产生中断
COUNTFLAG:避免误读
对于STM32,外部时钟源是 HCLK(AHB总线时钟)的1/8;内核时钟是 HCLK时钟
void SysTick_CLKSourceConfig(uint32_t SysTick_CLKSource)
//Systick时钟源选择 misc.c文件中
//配置CTRL寄存器
//入口参数
//SysTick_CLKSource_HCLK_Div8(外部时钟源) SysTick的定时器为72/8=9MHZ
//或SysTick_CLKSource_HCLK(内部时钟) SysTick的定时器为72MHZ
static __INLINE uint32_t SysTick_Config(uint32_t ticks)
//初始化systick,时钟为HCLK,并开启中断
//core_cm3.h/core_cm4.h文件中
//作用:开启SysTick中断,并配置ticks(两个中断中间有多少个周期)
void SysTick_Handler(void); //Systick中断服务函数:
static __IO uint32_t TimingDelay;
void Delay(__IO uint32_t nTime)
{
TimingDelay = nTime;
while(TimingDelay != 0); //TimingDelay自动减,减到0退出while
}
void SysTick_Handler(void) //每到1ms中断时运行此函数
{
if (TimingDelay != 0x00) //不等于0减一
{
TimingDelay--;
}
}
int main(void)
{ //…
if (SysTick_Config(SystemCoreClock / 1000))
//systick时钟为HCLK,中断时间间隔1ms 72000000Hz/1000=72000
//即每两次中断中有72000次间隔,运行72000次1/72000000秒=1ms
{
while (1); //SysTick_Config()有返回值,配置成功返回0跳过死循环
}
while(1)
{ Delay(200);//2ms
// …
}
}
void delay_init()
{
#if SYSTEM_SUPPORT_OS //如果需要支持OS.
u32 reload;
#endif
SysTick_CLKSourceConfig(SysTick_CLKSource_HCLK_Div8);
//选择外部时钟 HCLK/8 9MHz
fac_us=SystemCoreClock/8000000; //微秒因子
//为系统时钟的1/8,一微秒为9个时钟周期
#if SYSTEM_SUPPORT_OS //如果需要支持OS.
reload=SystemCoreClock/8000000; //每秒钟的计数次数 单位为K
reload*=1000000/delay_ostickspersec; //根据delay_ostickspersec设定溢出时间
//reload为24位寄存器,最大值:16777216,在72M下,约合1.86s左右
fac_ms=1000/delay_ostickspersec; //代表OS可以延时的最少单位
SysTick->CTRL|=SysTick_CTRL_TICKINT_Msk; //开启SYSTICK中断
SysTick->LOAD=reload; //每1/delay_ostickspersec秒中断一次
SysTick->CTRL|=SysTick_CTRL_ENABLE_Msk; //开启SYSTICK
#else
fac_ms=(u16)fac_us*1000; //非OS下,代表每个ms需要的systick时钟数
#endif
}
#else //不用OS时
//延时nus
//nus为要延时的us数.
void delay_us(u32 nus)
{
u32 temp;
SysTick->LOAD=nus*fac_us; //时间加载,将两次中断中的周期数给LOAD重装载值
SysTick->VAL=0x00; //清空计数器,清零后使能定时器会重新加载
SysTick->CTRL|=SysTick_CTRL_ENABLE_Msk ; //开始倒数,即使能
do
{
temp=SysTick->CTRL;
}while((temp&0x01)&&!(temp&(1<<16))); //等待时间到达
SysTick->CTRL&=~SysTick_CTRL_ENABLE_Msk; //关闭计数器
SysTick->VAL =0X00; //清空计数器
}
//延时nms
//注意nms的范围
//SysTick->LOAD为24位寄存器,所以,最大延时为:
//nms<=0xffffff*8*1000/SYSCLK
//SYSCLK单位为Hz,nms单位为ms
//对72M条件下,nms<=1864