STM32 F767 ADC时钟问题

2019-07-20 16:08发布

//时钟设置函数
//Fvco=Fs*(plln/pllm);
//Fsys=Fvco/pllp=Fs*(plln/(pllm*pllp));
//Fusb=Fvco/pllq=Fs*(plln/(pllm*pllq));

//Fvco:VCO频率
//Fsys:系统时钟频率
//Fusb:USB,SDMMC,RNG等的时钟频率
//FsLL输入时钟频率,可以是HSI,HSE等.
//plln:主PLL倍频系数(PLL倍频),取值范围:50~432.
//pllm:主PLL和音频PLL分频系数(PLL之前的分频),取值范围:2~63.
//pllp:系统时钟的主PLL分频系数(PLL之后的分频),取值范围:2,4,6,8.(仅限这4个值!)
//pllq:USB/SDMMC/随机数产生器等的主PLL分频系数(PLL之后的分频),取值范围:2~15.

//外部晶振为25M的时候,推荐值:plln=432,pllm=25,pllp=2,pllq=9.
//得到:Fvco=25*(432/25)=432Mhz
//     Fsys=432/2=216Mhz
//     Fusb=432/9=48Mhz
//返回值:0,成功;1,失败。
u8 Sys_Clock_Set(u32 plln,u32 pllm,u32 pllp,u32 pllq)
{
        u16 retry=0;
        u8 status=0;
        RCC->CR|=1<<16;                                //HSE 开启
        while(((RCC->CR&(1<<17))==0)&&(retry<0X1FFF))retry++;//等待HSE RDY
        if(retry==0X1FFF)status=1;        //HSE无法就绪
        else   
        {
                RCC->APB1ENR|=1<<28;        //电源接口时钟使能
                PWR->CR1|=3<<14;                 //高性能模式,时钟可到180Mhz
                PWR->CR1|=1<<16;                 //使能过驱动,频率可到216Mhz
                PWR->CR1|=1<<17;                 //使能过驱动切换
                RCC->CFGR|=(0<<4)|(5<<10)|(4<<13);//HCLK 不分频;APB1 4分频;APB2 2分频.
                RCC->CR&=~(1<<24);                //关闭主PLL
                RCC->LLCFGR=pllm|(plln<<6)|(((pllp>>1)-1)<<16)|(pllq<<24)|(1<<22);//配置主PLL,PLL时钟源来自HSE
                RCC->CR|=1<<24;                        //打开主PLL
                while((RCC->CR&(1<<25))==0);//等待PLL准备好
                FLASH->ACR|=1<<8;                //指令预取使能.
                FLASH->ACR|=1<<9;                //使能ART Accelerator
                FLASH->ACR|=7<<0;                //8个CPU等待周期.
                RCC->CFGR&=~(3<<0);                //清零
                RCC->CFGR|=2<<0;                //选择主PLL作为系统时钟         
                while((RCC->CFGR&(3<<2))!=(2<<2));//等待主PLL作为系统时钟成功.
        }
        return status;
}  

//初始化ADC
//这里我们仅以规则通道为例
//我们默认仅开启ADC1_CH5                                                                                                                                          
void  Adc_Init(void)
{   
        //先初始化IO口
        RCC->APB2ENR|=1<<8;            //使能ADC1时钟
        RCC->AHB1ENR|=1<<0;            //使能PORTA时钟          
        GPIO_Set(GPIOA,PIN5,GPIO_MODE_AIN,0,0,GPIO_PUPD_PU);        //PA5,模拟输入,下拉   

        RCC->APB2RSTR|=1<<8;           //ADCs复位
        RCC->APB2RSTR&=~(1<<8);        //复位结束         
        ADC->CCR=1<<16;                        //ADCCLK=PCLK2/4=90/4=22.5Mhz,ADC时钟最好不要超过36Mhz
       
        ADC1->CR1=0;                           //CR1设置清零
        ADC1->CR2=0;                           //CR2设置清零
        ADC1->CR1|=0<<24;              //12位模式
        ADC1->CR1|=0<<8;            //非扫描模式       
       
        ADC1->CR2&=~(1<<1);            //单次转换模式
        ADC1->CR2&=~(1<<11);           //右对齐       
        ADC1->CR2|=0<<28;            //软件触发
       
        ADC1->SQR1&=~(0XF<<20);
        ADC1->SQR1|=0<<20;             //1个转换在规则序列中 也就是只转换规则序列1                           
        //设置通道5的采样时间
        ADC1->SMPR2&=~(7<<(3*5));//通道5采样时间清空          
        ADC1->SMPR2|=7<<(3*5);         //通道5  480个周期,提高采样时间可以提高精确度         
        ADC1->CR2|=1<<0;                   //开启AD转换器          
}       

问题:ADC时钟是PCLK2的四分频,但是PCLK2不应该是216M/2=108M吗,ADC初始化中的90M是怎么得来的?是不是官方搞错了?          

友情提示: 此问题已得到解决,问题已经关闭,关闭后问题禁止继续编辑,回答。