Tiny210(S5PV210) U-BOOT(三)----配置时钟频率源码分析

1.设置APLL/MPLL/EPLL/EPLL锁相环时间
翻看手册,P371页,找到PLL CONTROL REGISTERS。
• (APLL_LOCK, R/W, Address = 0xE010_0000)
• (MPLL_LOCK, R/W, Address = 0xE010_0008)
• (EPLL_LOCK, R/W, Address = 0xE010_0010)
• (VPLL_LOCK, R/W, Address = 0xE010_0020) 
Tiny210(S5PV210) U-BOOT(三)----配置时钟频率源码分析_第1张图片
这里出现了一个问题,APLL/MPLL/EPLL/VPLL的锁相环的时间是不一样的,而像S5PC100,频率为667MHz,他的A/M/E/HPLL的锁相环时间均是300us,而在Exynos 4xxx中,时间又变成了250,这个时间一定要查芯片手册。那么,所以,这里我们要用4个寄存器来存不同的LOCK_OFFSET的值。查询ATPCS文档(Procedure Call Standard for the ARM v2.08),在第P15页找到,r4-r8是用来存储变量的寄存器,OK,我们选取4个寄存器来保存值。
APLL的锁时间为:24*30=720    --0x2D0
MPLL的锁时间为:24*200=4800  --0x12C0
EPLL的锁时间为:24*375=9000  --0x2328
VPLL的锁时间为:24*100=2400  --0x960
将上面四个值存入寄存器r4--r8,这里使用寄存器的别名v1--v4,然后再将寄存值的分别存入APLL_LOCK/MPLL_LOCK/EPLL_LOCK/VPLL_LOCK对应的寄存器地址中
ldr    r0, =0xE0100000                /*ELFIN_CLOCK_POWER_BASE*/


ldr    v1, =0x2D0
ldr    v2, =0x12C0
ldr    v3, =0x2328
ldr    v4, =0x960

str    v1, [r0, #0x00]                /*APLL_LOCK_OFFSET*/
str    v2, [r0, #0x04]                /*MPLL_LOCK_OFFSET*/
str    v3, [r0, #0x08]                /*EPLL_LOCK_OFFSET*/
str    v4, [r0, #0x0c]                /*VPLL_LOCK_OFFSET*/

2.开始倍频
2.1倍频APLL
我们只要把 P=3 M=125,S=1即可让输出频率为1000MHz,即MSOUT_MSYS=1000MHz,那么配置哪个寄存器可以倍频呢?继续向下看手册,在P372页找到APLL_CON0( 0xE010_0100)。
Tiny210(S5PV210) U-BOOT(三)----配置时钟频率源码分析_第2张图片
上面这个表显示出如何配置P/M/S位,那么如下代码,我们可以配置输出为2000MHz。(参考代码board/samsung/smdkc100/lowlevel_init.S -- system_clock_init)
ldr    r1, =(1<<31 |125<<16 |3<<8 |1<<0)  @APLL_VAL=(1<<31 | 445<<16 | 0x4<<8 | 1<<0)
/*APLL_CON0_OFFSET 0x100*/
str    r1, [r0, #APLL_CON_OFFSET]      @24  1000  APLL  3  125  1  2000.0  FOUTAPLL = 1000.0MHz 

2.2倍频MPLL
MPLL的P/M/S位配置为P=12 M=667,S=1,按上面的配置为MPLL_CON寄存器,Address=0xE010_0108
Tiny210(S5PV210) U-BOOT(三)----配置时钟频率源码分析_第3张图片
ldr    r1, =(1<<31 |667<<16 |12<<8 |1<<0)  @MPLL_VAL=(1<<31 |667<<16 |12<<8 |1<<0)
/*MPLL_CON0_OFFSET 0x108*/
str    r1, [r0, #MPLL_CON_OFFSET]      @24  667  MPLL  12  667  1  667.0  FOUTAPLL = 667.0MHz 

2.3倍频EPLL
EPLL的P/M/S位配置为P=3 M=48,S=2,按上面的配置配置EPLL_CON寄存器,Address=0xE010_0110
Tiny210(S5PV210) U-BOOT(三)----配置时钟频率源码分析_第4张图片
ldr    r1, =(1<<31 |108<<16 |6<<8 |3<<0)  @VPLL_VAL=(1<<31 |108<<16 |6<<8 |3<<0)
/*VPLL_CON0_OFFSET 0x120*/
str    r1, [r0, #VPLL_CON_OFFSET]      @24  54  VPLL  6  108  3  54.0  FOUTAPLL = 54.0MHz
OK,倍频完成,代码整理一下:
ldr    r1, =(1<<31 |125<<16 |3<<8 |1<<0)  @APLL_VAL=(1<<31 | 445<<16 | 0x4<<8 | 1<<0)
/*APLL_CON0_OFFSET 0x100*/
str    r1, [r0, #APLL_CON_OFFSET]      @24  1000  APLL  3  125  1  2000.0  FOUTAPLL = 1000.0MHz 
ldr    r1, =(1<<31 |667<<16 |12<<8 |1<<0)  @MPLL_VAL=(1<<31 |667<<16 |12<<8 |1<<0)
/*MPLL_CON0_OFFSET 0x108*/
str    r1, [r0, #MPLL_CON_OFFSET]      @24  667  MPLL  12  667  1  667.0  FOUTAPLL = 667.0MHz 
ldr    r1, =(1<<31 |48<<16 |3<<8 |2<<0)  @EPLL_VAL=(1<<31 |48<<16 |3<<8 |2<<0)
/*EPLL_CON0_OFFSET 0x110*/
str    r1, [r0, #EPLL_CON_OFFSET]      @24  96  EPLL  3  48  2  96.0  FOUTAPLL = 96.0MHz  
ldr    r1, =(1<<31 |108<<16 |6<<8 |3<<0)  @VPLL_VAL=(1<<31 |108<<16 |6<<8 |3<<0)
/*VPLL_CON0_OFFSET 0x120*/
str    r1, [r0, #VPLL_CON_OFFSET]      @24  54  VPLL  6  108  3  54.0  FOUTAPLL = 54.0MHz  
参考芯片手册P361页时钟生成电路图,系统的MSYS/DSYS/PSYS的CLK全部来自于APLL和MPLL,现在开始对APLL/MPLL进行分频,分给不同的设备,如Cotrex A8内核或其它设备。在分频之前,必须要先选择时钟源,否则怎么分频呢?
 
3.选择时钟源
Tiny210(S5PV210) U-BOOT(三)----配置时钟频率源码分析_第5张图片
参考芯片手册P361页时钟生成电路图,以APLL以例,查找FOUTAPLL和FINPLL
Tiny210(S5PV210) U-BOOT(三)----配置时钟频率源码分析_第6张图片

由图所知,FINPLL是没有倍频的频率,我们要选的是倍频后的,所以应该选择FOUTAPLL,依此类推,后面几位选上FOUTMPLL/FOUTEPLL/FOUTVPLL。那第[28]位,ONENADN选择什么呢?往上查一下ONENAND用的是什么CLK,在P363页查找到用的是PSYS,所以这里应该选择0。
Tiny210(S5PV210) U-BOOT(三)----配置时钟频率源码分析_第7张图片
//CLK_SRC0_OFFSET            0x200
ldr    r1, [r0, #CLK_SRC0_OFFSET]
ldr    r2, =(0<<28|1<<12 |1<<8 |1<<4 |1<<0)
orr    r1, r1, r2
str    r1, [r0, #CLK_SRC0_OFFSET]
OK,现在开始对APLL/MPLL进行分频了。
 
4.分频
继续向下查找芯片手册,找到P387页,系统时钟分频操作的寄存器是CLK_DIV0,Address = 0xE010_0300
Tiny210(S5PV210) U-BOOT(三)----配置时钟频率源码分析_第8张图片
回过头去,再去看时钟频率的产生图,最终MSYS有五个CLK, ARMCLK/HCLK_MSYS/PCLK_MSYS/HCLK_IMEM/SCLK_DMC0。P356查看常用CLK
Values for the high-performance operation:
•  freq(ARMCLK)   = 1000 MHz
•  freq(HCLK_MSYS)   = 200 MHz
•  freq(HCLK_IMEM)   = 100 MHz
•  freq(PCLK_MSYS)   = 100 MHz
•  freq(HCLK_DSYS)   = 166 MHz
•  freq(PCLK_DSYS)   = 83 MHz
•  freq(HCLK_PSYS)   = 133 MHz
•  freq(PCLK_PSYS)   = 66 MHz
•  freq(SCLK_ONENAND)    = 133 MHz, 166
在寄存器CLK_DIV0的表格上还有这样一段话:There are operating frequency limitations. The maximum operating frequency of SCLKAPLL, SCLKMPLL, SCLKA2M, HCLK_MSYS, and PCLK_MSYS are 1GHz, 667 MHz, 400 MHz, 200 MHz, and 100 MHz, respectively. These operating clock conditions must be met through CLK_DIVX configuration.
查找SCLKAPLL,找到SCLKAPLL的来源
Tiny210(S5PV210) U-BOOT(三)----配置时钟频率源码分析_第9张图片

OK,将上面列表中CLK也整理一下,列表出来:
•  freq(SCLKAPLL)   = 1000 MHz
•  freq(SCLKMPLL)   = 667 MHz
•  freq(SCLKA2M)   = 400 MHz 
接着分析图,在[19:16]要用到的MSOUT_DSYS,他的频率来自MPLL,所以MSOUT_DSYS =MPLL = 667
Tiny210(S5PV210) U-BOOT(三)----配置时钟频率源码分析_第10张图片
下面一位一位的开始分析:
[2:0]ARMCLK = 1000MHz,MOUT_MSYS = 2000MHz,ARMCLK = MOUT_MSYS/(1+1),APLL_RATIO为1,1<<0
[6:4]SCLKA2M = 400MHz,SCLKAPLL = 1000MHz,由于无法整除,而SCLKA2M最大为400,我们只要保证不超过就行,可以低一些,SCLKA2M = SCLKAPLL / (2 + 1)  = 333,A2M_RATIO为2,2<<4
[10:8]HCLK_MSYS = 200MHz,ARMCLK = 1000MHz,HCLK_MSYS = ARMCLK /(4 + 1),HCLK_MSYS_RATIO为4,4<<8
[14:12]PCLK_MSYS = 100MHz,HCLK_MSYS = 200MHz,PCLK_MSYS = HCLK_MSYS / (1 + 1) ,PCLK_MSYS_RATIO为1,1<<12  
[19:16]HCLK_DSYS = 166MHz,MOUT_DSYS = 667MHz,HCLK_DSYS = MOUT_DSYS / (3+ 1) ,HCLK_DSYS_RATIO为1,3<<16
[22:20]PCLK_DSYS = 83MHz,HCLK_DSYS = 166MHz,PCLK_DSYS = HCLK_DSYS / (1 + 1) ,PCLK_DSYS_RATIO为1,1<<20 
[27:24]HCLK_PSYS = 133MHz,MOUT_PSYS = 667MHz,HCLK_PSYS = MOUT_PSYS / (4 + 1) ,HCLK_PSYS_RATIO为1,4<<24  
[30:28]PCLK_PSYS = 66MHz,HCLK_PSYS = 133MHz,PCLK_PSYS = HCLK_PSYS / (1 + 1)  ,PCLK_PSYS_RATIO为1,1<<28 
OK,至此,分频完成,整理代码如下: 
/*将r0的值偏移300,此时地址为CLK_DIV0 = 0xE010_0300),保存到r1中*/
ldr r1, [r0, #0x300]            /*CLK_DIV0_OFFSET*/
/*将CLK_DIV0中有关分频的位全部清零0~30*/
ldr    r2, =0x7fff                /*CLK_DIV0_MASK*/
bic    r1, r1, r2

/*分频给系统时钟*/
ldr    r2, =(0<<0)|(2<<4)|(4<<8)|(1<<12)|(3<<16)|(1<<20)|(4<<24)|(1<<28)       
orr    r1, r1, r2
str    r1, [r0, #0x300]  //CLK_DIV0_OFFSET

5.延时一段时间
    /*delay*/
    mov    r1, #0x10000
1:  subs   r1, r1, #1
    bne    1b

6.返回指针
mov    pc, lr
OK,到此,系统时钟频率的全部设置工作完成。最后将代码重新归整如下:
system_clock_init:
 
    ldr    r0, =ELFIN_CLOCK_POWER_BASE    @0xe0100000
        ldr    v1, =0x2D0
        ldr    v2, =0x12C0
        ldr    v3, =0x2328
        ldr    v4, =0x960
 
        str    v1, [r0, #0x00]                /*APLL_LOCK_OFFSET*/
        str    v2, [r0, #0x04]                /*MPLL_LOCK_OFFSET*/
        str    v3, [r0, #0x08]                /*EPLL_LOCK_OFFSET*/
        str    v4, [r0, #0x0c]                /*VPLL_LOCK_OFFSET*/
 
/**************************************************************************
  APLL_CON0_OFFSET 0x100   @24  1000  APLL  3  125  1  2000.0  FOUTAPLL = 1000.0MHz
  MPLL_CON0_OFFSET 0x108   @24  667  MPLL  12  667  1  667.0  FOUTAPLL = 667.0MHz
  EPLL_CON0_OFFSET 0x110   @24   54  VPLL  6  108  3  54.0  FOUTAPLL = 54.0MHz 
***************************************************************************/
    ldr    r1, =(1<<31 |125<<16 |3<<8 |1<<0)  @APLL_VAL=(1<<31 | 445<<16 | 0x4<<8 | 1<<0)
    str    r1, [r0, #APLL_CON0_OFFSET]      
 
    ldr    r1, =(1<<31 |667<<16 |12<<8 |1<<0)  @MPLL_VAL=(1<<31 |667<<16 |12<<8 |1<<0)
    str    r1, [r0, #MPLL_CON_OFFSET]     
 
    ldr    r1, =(1<<31 |48<<16 |3<<8 |2<<0)  @EPLL_VAL=(1<<31 |48<<16 |3<<8 |2<<0)
    str    r1, [r0, #EPLL_CON_OFFSET]     
 
    ldr    r1, =(1<<31 |108<<16 |6<<8 |3<<0)  @VPLL_VAL=(1<<31 |108<<16 |6<<8 |3<<0)
    str    r1, [r0, #VPLL_CON_OFFSET]     
 
 
    //CLK_SRC0_OFFSET            0x200
    ldr    r1, [r0, #CLK_SRC0_OFFSET]
    ldr    r2, =(0<<28|1<<12 |1<<8 |1<<4 |1<<0)
    orr    r1, r1, r2
    str    r1, [r0, #CLK_SRC0_OFFSET]
 
    /*将r0的值偏移300,此时地址为CLK_DIV0 = 0xE010_0300,保存到r1
     * 将CLK_DIV0中有关分频的位全部清零0~30
     */
    ldr r1, [r0, #0x300]            /*CLK_DIV0_OFFSET*/
    ldr    r2, =0x7fff                /*CLK_DIV0_MASK*/
    bic    r1, r1, r2
 
    /*分频给系统时钟*/
    ldr    r2, =(0<<0)|(2<<4)|(4<<8)|(1<<12)|(3<<16)|(1<<20)|(4<<24)|(1<<28)      
    orr    r1, r1, r2
    str    r1, [r0, #0x300]  //CLK_DIV0_OFFSET
 

    mov    r1, #0x10000
1:    subs    r1, r1, #1
    bne    1b
    mov    pc, lr



你可能感兴趣的:(u-boot,Tiny210,s5pv210)