参考博客

(1)S3C2440-裸机篇-05 | S3C2440时钟体系详解(FCLK、PCLK、HCLK) 

一、三种时钟(FCLK、HCLK、PCLK)

如下图所示,S3C2440的时钟控制逻辑,给整个芯片提供三种时钟:

(1)FCLK:用于CPU核;

(2)HCLK:用于接在AHB总线上的设备,比如LCD控制器、存储器控制器、中断控制器、USB主机模块等;

(3)PCLK:用于接在APB总线上的设备,比如看门狗、IIS、I2C、ADC、UART等。

另外由数据手册可知(如下图所示),CPU最大的工作频率可达400MHz,高速设备最大的工作频率可达136MHz,低速设备最大的工作频率是68MHz。 

二、如何产生三种时钟 

由底板原理图可知(如下图所示),时钟源是12MHz的晶振。

如何由12MHz提高到400MHz?这需要使用到 PLL(锁相环)。

我们来看一下时钟产生框图:

由图可知有两个时钟源:一个是晶振提供时钟源,另一个是通过外部引脚提供时钟源。具体选择哪个时钟源,由选择器的OM[3:2]来决定,如下图所示:

由于 OM[3:2] 引脚接地,所以其值为00,则选择晶振作为时钟源,如下所示:

由图可知,S3C2440有两个PLL,分别叫做MPLL(main PLL)、UPLL(usb PLL)。UPLL专用于USB设备,MPLL用于设置FCLK、HCLK、PCLK。它们的设置方法类似,这里以MPLL为例。

上电时,MPLL没被启动,FCLK等于外部输入的时钟(一般是晶振产生的12Mhz时钟),我们称之为Fin。如果要提高系统的时钟,需要使用软件来启用MPLL。其上电时序图如下所示:

(1)LOCKTIME 寄存器:用于设置锁相时间

Fin进入MPLL后,需要经过一定的时长(时长可以通过 LOCKTIME 寄存器进行设置,我们一般使用默认值0xFFFF-FFFF就好),MPLL才能输出倍频后的 FCLK。

(2)MPLLCON 寄存器:用于设置FCLK与Fin的倍数关系

已知给 MPLL 输入的 Fin=12MHz,如果想让MPLL输出的 FCLK = 400MHz(因为CPU最大的工作频率可达400MHz),该如何设置呢?可以通过 MPLLCON 寄存器进行设置。

有如下公式:

从MPLL输出的FCLK = (2*m*Fin)/(p*2^s)
    
    m = MDIV(即 MPLLCON[19:12] 的值 )+ 8
    p = PDIV(即 MPLLCON[9:4] 的值 )+ 2
    s = SDIV(即 MPLLCON[1:0] 的值 )

数据手册会给出FCLK典型值的设置推荐值,如下图所示,我们编程时使用这些推荐值即可(虽然也可以由公式自己推算)。

(3)CLKDIVN寄存器:用于设置FCLK、HCLK、PCLK的比例

上面已经得到FCLK,那如何由它进一步得到HCLK、PCLK呢?可以将FCLK进行分频,得到HCLK、PCLK,这意味着FCLK、HCLK、PCLK三者存在比例关系。具体的分频系数,可以通过CLKDIVN寄存器进行设置,比如通过 CLKDIVN[2:1] 设置将FCLK分频多少以得到HCLK。

过程总结如下图所示:

三、编程实践

3.1 编程前的分析 

在编程之前,注意数据手册有下面的一段描述:

它表明,如果HDIVN的值不设为0(为0则表示HCLK=FCLK/1,而HCLK一般不等于FCLK,所以一般不会设置为0的),则需要添加上图红框内的代码(注意将“R1_nF…”这个宏转换为实际值)。

假设我们需要设置FCLK=400MHz,HCLK=100MHz,PCLK=50MHz。

则根据第二节的描述,我们需要设置MPLLCON寄存器、CLKDIVN寄存器:

(1)关于MPLLCON寄存器的设置。由于400MHz是典型值, 我们使用数据手册给出的设置:

MDIV(即 MPLLCON[19:12] 的值 ):设置为92(0x5c)

PDIV(即 MPLLCON[9:4] 的值 ):设置为1

SDIV(即 MPLLCON[1:0] 的值 ):设置为1

那么MPLLCON寄存器的值应该设置为:(92<<12)|(1<<4)|(1<<0)

(2)关于CLKDIVN寄存器的设置。

由于 HCLK(100MHz) = FCLK(400MHz) / 4,所以CLKDIVN[2:1] = 0b10;而且CAMDIVN[9]要设置为0(初始值默认也为0,那么设不设置好像都行)。

由于 PCLK(50MHz) = HCLK(100MHz) / 2,所以CLKDIVN[0] = 1;

综合起来,CLKDIVN寄存器要设置为0b101=0x5。

3.2 编程实践

完整的代码见链接(课程提供的代码):

(1)其中start.S文件内容如下(我仿写的):

.text
.global _start

_start:
	
	//关看门狗
	ldr r0,=0x53000000
	ldr r1,=0
	str r1,[r0]

/*******************************************/	
	//设置HDIV、PDIV的分频系数,使得FCLK : HCLK : PCLK=400:100:50
	//通过寄存器CLKDIVN来设置分频系数
	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的值为0xc0000000
	mcr p15,0,r0,c1,c0,0
	
	//设置MPLL的锁相时间
	/* LOCKTIME(0x4C000000) = 0xFFFFFFFF */
	ldr r0, =0x4C000000
	ldr r1, =0xFFFFFFFF
	str r1, [r0]
	
	//设置MPLL,使它输出400MHz
	ldr r0,=0x4c000004
	ldr r1,=(92<<12)|(1<<4)|(1<<0)
	str r1,[r0]
/******************************************************/

	//设置栈
	/*判断nor/nand启动方式,并设置相应的栈
	 *如何判断启动方式:写0到0地址,然后再读出来,
	 *如果得到0,则表示地址0的内容被修改了,它对应着sram,意味着nand启动
	 *否则为nor启动(因为nor不能直接写)
	 */
	//ldr r0,[0]  //读出原来的值进行备份
	mov r1,#0
	ldr r0,[r1]
	str r1,[r1] //将0写到0地址
	ldr r2,[r1] //将0地址的内容读出来
	cmp r1,r2   // r1==r2? 如果相等则表示是nand启动
	ldr sp,=0x4000000+0x1000 //先假设是nor启动(nor启动时,内部的SRAM映射到0x40000000,4096=0x1000)
	moveq sp,#4096 //如果相等则表示nand启动,将sp指向内部SRAM的最高地址处
	streq r0,[r1]  //如果相等则表示nand启动,恢复原来的值
	
	bl main

halt:
	b halt

(2)led.c文件的内容如下(我仿写的):

#include "s3c2440_soc.h"

void delay(volatile int d)
{
	while (d--);
}

int main(void)
{
	//设置GPFCON让GPF4/5/6配置为输出引脚
	GPFCON &= ~((3<<8)|(3<<10)|(3<<12));//先清零
	GPFCON |= ((1<<8)|(1<<10)|(1<<12));//置位,设置为输出引脚
	
	GPFDAT=0xff;//全部熄灭
	//循环点亮
	while(1)
	{
		GPFDAT=0xff;//全部熄灭
		GPFDAT=0xef;//让LED1亮
		delay(100000);
		GPFDAT=0xff;//让LED1灭
	
		GPFDAT=0xdf;//让LED2亮
		delay(100000);
		GPFDAT=0xff;//让LED2灭
	
		GPFDAT=0xbf;//让LED4亮
		delay(100000);
	}
	return 0;
}

//课程的版本
#if 0

int main(void)
{
	int val = 0;  /* val: 0b000, 0b111 */
	int tmp;

	/* 设置GPFCON让GPF4/5/6配置为输出引脚 */
	GPFCON &= ~((3<<8) | (3<<10) | (3<<12));
	GPFCON |=  ((1<<8) | (1<<10) | (1<<12));

	/* 循环点亮 */
	while (1)
	{
		tmp = ~val;
		tmp &= 7;
		GPFDAT &= ~(7<<4);
		GPFDAT |= (tmp<<4);
		delay(100000);
		val++;
		if (val == 8)
			val =0;
	}
	return 0;
}

#endif

 代码依据是数据手册与原理图中的相关内容,如下所示:

由下面的原理图可知,GPF4~GPF6引脚输出低电平时,对应的LED1、LED2、LED4会亮。

3.3 现象与分析

1、在课程提供的代码(见上面提到的链接)目录下执行make时,在进行链接时报错:

xjh@ubuntu:~/iot/embedded_basic/jz2440/armBareMachine/clk$ make
arm-linux-gcc -c -o led.o led.c
arm-linux-gcc -c -o start.o start.S
arm-linux-ld -Ttext 0 led.o start.o -o led.elf
led.o:(.ARM.exidx+0x0): undefined reference to `__aeabi_unwind_cpp_pr0'
led.o:(.ARM.exidx+0x8): undefined reference to `__aeabi_unwind_cpp_pr1'
make: *** [all] Error 1
xjh@ubuntu:~/iot/embedded_basic/jz2440/armBareMachine/clk$

在网上查询解决方法,是在 arm-linux-gcc 命令加上 -nostdlib 这个选项。它表示不链接系统标准启动文件和标准库文件,只把指定的文件传递给连接器。这个选项常用于编译内核、bootloader等程序,它们不需要启动文件、标准库文件(书P35)。经过测试,这样修改之后的确可以编译通过。

回顾一下朱老师的裸机课程,发现如果 arm-linux-ld 时只有一个待连接的.o文件,则Makefile中的arm-linux-gcc命令不需要加上 -nostdlib 这个选项(比如chapter4->8.leds.s),如果有两个待链接的.o文件,则需要加上该选项(比如chapter5->3.set_sp_s及以后的裸机程序)。这是为何?

另外韦老师的课程为何不需要加nostdlib 这个选项?估计与我环境不一样?后来回过头来看,的确是因为交叉编译工具链版本不一样,见链接:解决编译问题:undefined reference to `__aeabi_uidivmod‘-CSDN博客

2、一些编程注意事项

(1) &=,这两个符号不能有空格,即不能写成“& =”

GPFCON & = ~((3<<8)|(3<<10)|(3<<12)); //会报错

 (2)Makefile文件中arm-linux-ld时,.o文件第一个必须是start.o文件!否则可以连接成功,但烧写到开发板后没有现象。

(3)直接写成 ldr r0,[0] 貌似会报错,要写成:

mov r1,#0
ldr r0,[r1]

(4)课程的led.c文件用的是位操作,我没有仔细分析其代码,而是直接赋值修改。有时间分析一下其代码。

3、烧写现象

以NorFlash启动,在uboot的shell界面下按“n”,使用“usb下载线+dnw”方式将生成的led.bin烧写到NandFlash中。然后改为NandFlash启动,可以看见三颗LED灯在快速地循环点亮。

 如果将start.S文件中两条“/**************/”之间的内容(也就是时钟初始化部分)删掉,重新编译烧写运行,可以看见三颗LED灯依然在循环点亮,但速度明显慢许多!(此时FCLK应该是12MHz,而HCLK与PCLK又是多少呢?)

四、总结

1、深入讲解了S3C2440芯片的结构

掌握了S3C2440的时钟体系架构和上电复位时序,其时钟源有两个:外部晶振或者外部时钟,通过OM[3:2]硬件选择;其内部主要调整频率的PLL有两个:MPLL(产生FCLK)和UPLL(产生UCLK);其主要的时钟频率有三个(FCLK->CPU使用,HCLK->AHB总线高速外设使用,PCLK->APB总线低速外设使用),其中HCLK和PCLK由FCLK分频而来。

2、学习了如何进行芯片操作

掌握了如何编程设置寄存器控制S3C2440的时钟频率,比如本节设置FCLK=400Mhz,HCLK=100Mhz,PCLK=50Mhz。

3、其他一些启发

可以关闭某些模块的时钟,以达到省电的目的。 比如设置CLKCON寄存器来关闭某些模块。

Logo

助力广东及东莞地区开发者,代码托管、在线学习与竞赛、技术交流与分享、资源共享、职业发展,成为松山湖开发者首选的工作与学习平台

更多推荐