S3C2440时钟体系

文章目录

    • 1 s3c2440时钟框图
      • 1.1 时钟来源
      • 1.2 时钟分配
      • 1.3 上电时时钟的切换
    • 2 编程提高运行时钟
      • 2.1 CLKDIVN寄存器
      • 2.2 PLL COntrol寄存器
      • 2.3 设置CPU工作于异步模式
      • 2.4 完整代码

1 s3c2440时钟框图

s3c2440时钟框图如下:
S3C2440时钟体系_第1张图片

1.1 时钟来源

时钟来源由OM[3:2]的引脚电平决定,会在上电后复位信号出现时进行锁存。
S3C2440时钟体系_第2张图片

1.2 时钟分配

S3C2440时钟体系_第3张图片

  1. jz2440的外部输入晶振为12MHz,然后经过MPLL(Main PLL)和UPLL(USB PLL),经过MPLL后的信号直接提供给FCLK,通过HDIVN分频给HCLK,通过PDIVN分频给PCLK,再传给下面的各个设备。
  2. S3C2440的工作频率,FCLK最高400MHz,HCLK最高136MHz,PCLK最高68MHz。
  3. 与CPU密切相关的,工作于FCLK;一些对性能要求较高的设备,像LCD显示、相机等,在AHB BUS,H即为High,高速之意,工作于HCLK;另外一些对性能要求不那么高的低速设备,在APB BUS,P即为Peripheral之意,工作在PCLK。

1.3 上电时时钟的切换

S3C2440时钟体系_第4张图片

  1. 复位芯片,等待电源稳定后nREST才输出高电平。
  2. 根据OM[3:2]的值,确定FCLK=外部晶振的值。
  3. PLL锁存OM[3:2]的值。
  4. 开始设置PLL。
  5. PLL设置中,CPU停止工作。
  6. PLL设置完毕,FCLK等于PLL输出的时钟,CPU开始工作。

2 编程提高运行时钟

都是死知识,这里直接复制百问网中的内容:原文链接。

怎么编程控制MPLL、HDIV、PDIV,使FCLK=400MHz,HCLK=100MHz,PLCK=50MHz?
需要设置MPLLCON的FCLK=400MHz,设置CLKDIVN的HCLK=FCLK/4,PCLK=FCLK/8。

2.1 CLKDIVN寄存器

S3C2440时钟体系_第5张图片
想设置HCLK=FCLK/4需要将HDIVN[2:1]设置为10,同时将CAMDIVN[9]设置为0。查看CAMDIVN[9]的初始值默认就是0,因此只需要设置HDIVN[2:1]为10。

S3C2440时钟体系_第6张图片

想设置PCLK=FCLK/8需要将PDIVN[0]设置为1,因此整个CLKDIVN寄存器设置如下:

/* CLKDIVN(0x4C000014) = 0X5, tFCLK:tHCLK:tPCLK = 1:4:8  */
	ldr r0, =0x4C000014
	ldr r1, =0x5
	str r1, [r0]

2.2 PLL COntrol寄存器

在手册的PLL VALUE SELECTION TABLE里列出了常见情况PLL的设置,我们输入的是晶振的12MHz,输出需要400MHz,因此根据表格需要设置 MDIV=92(0x5C),PDIV=1,SDIV=1;
S3C2440时钟体系_第7张图片

在手册介绍了MPLL的m、p、s与MDIV、PDIV、SDIV之间的关系:
S3C2440时钟体系_第8张图片
PLL控制寄存器如下:
S3C2440时钟体系_第9张图片
因此需要配置(92<<12)|(1<<4)|(1<<0):

/* 设置MPLLCON(0x4C000004) = (92<<12)|(1<<4)|(1<<0) 
	 *  m = MDIV+8 = 92+8=100
	 *  p = PDIV+2 = 1+2 = 3
	 *  s = SDIV = 1
	 *  FCLK = 2*m*Fin/(p*2^s) = 2*100*12/(3*2^1)=400M
	 */
	ldr r0, =0x4C000004
	ldr r1, =(92<<12)|(1<<4)|(1<<0)
	str r1, [r0]

2.3 设置CPU工作于异步模式

手册还提到,需要SetAsyncBusMode。

// MMU_SetAsyncBusMode
mrc p15,0,r0,c1,c0,0
orr r0,r0,#R1_nF:OR:R1_iA
mcr p15,0,r0,c1,c0,0

2.4 完整代码

.text
.global _start

_start:

	/* 关闭看门狗 */
	ldr r0, =0x53000000
	ldr r1, =0
	str r1, [r0]

	/* 设置MPLL, FCLK : HCLK : PCLK = 400m : 100m : 50m */
	/* LOCKTIME(0x4C000000) = 0xFFFFFFFF */
	ldr r0, =0x4C000000
	ldr r1, =0xFFFFFFFF
	str r1, [r0]

	/* CLKDIVN(0x4C000014) = 0X5, tFCLK:tHCLK:tPCLK = 1:4:8  */
	ldr r0, =0x4C000014
	ldr r1, =0x5
	str r1, [r0]

	/* 设置CPU工作于异步模式 */
	mrc p15,0,r0,c1,c0,0
	orr r0,r0,#0xc0000000   //R1_nF:OR:R1_iA
	mcr p15,0,r0,c1,c0,0

	/* 设置MPLLCON(0x4C000004) = (92<<12)|(1<<4)|(1<<0) 
	 *  m = MDIV+8 = 92+8=100
	 *  p = PDIV+2 = 1+2 = 3
	 *  s = SDIV = 1
	 *  FCLK = 2*m*Fin/(p*2^s) = 2*100*12/(3*2^1)=400M
	 */
	ldr r0, =0x4C000004
	ldr r1, =(92<<12)|(1<<4)|(1<<0)
	str r1, [r0]

	/* 一旦设置PLL, 就会锁定lock time直到PLL输出稳定
	 * 然后CPU工作于新的频率FCLK
	 */
	
	

	/* 设置内存: sp 栈 */
	/* 分辨是nor/nand启动
	 * 写0到0地址, 再读出来
	 * 如果得到0, 表示0地址上的内容被修改了, 它对应ram, 这就是nand启动
	 * 否则就是nor启动
	 */
	mov r1, #0
	ldr r0, [r1] /* 读出原来的值备份 */
	str r1, [r1] /* 0->[0] */ 
	ldr r2, [r1] /* r2=[0] */
	cmp r1, r2   /* r1==r2? 如果相等表示是NAND启动 */
	ldr sp, =0x40000000+4096 /* 先假设是nor启动 */
	moveq sp, #4096  /* nand启动 */
	streq r0, [r1]   /* 恢复原来的值 */
	

	bl main

halt:
	b halt

你可能感兴趣的:(Linux)