JZ2440开发板——S3C2440的时钟体系

news2025/1/12 23:09:36

参考博客

(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及以后的裸机程序)。

这里韦的课程为何不加呢?估计与我环境不一样?

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寄存器来关闭某些模块。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2133532.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

通过防火墙分段增强网络安全

什么是网络分段‌ 随着组织规模的扩大&#xff0c;管理一个不断扩大的网络成为一件棘手的事情&#xff0c;同时确保安全性、合规性、性能和不间断的运行可能是一项艰巨的任务。为了克服这一挑战&#xff0c;网络管理员部署了网络分段&#xff0c;这是一种将网络划分为更小且易…

QT::QComboBox自定义左击事件信号

因为QComboBox没有自定义的clink信号&#xff0c;所以自己新建一个MyComBox类继承QComboBox&#xff0c;并且添加自定义的左击信号&#xff0c;以及使用该信号连接一个槽函数 mycombobox.h #ifndef MYCOMBOBOX_H #define MYCOMBOBOX_H#include <QComboBox> #include &l…

使用程序集解析的方式内嵌dll到exe中

选择一个项目&#xff08;demo3&#xff09;&#xff0c;来进行内嵌。正常dll文件是可以在Bin–Debug里面看到的。 在Program里面添加内容 Program.cs里的全部代码 using System; using System.Collections.Generic; using System.Linq; using System.Reflection; using Syste…

【面试八股总结】Redis持久化

Redis 实现了数据持久化的机制&#xff0c;这个机制会把数据存储到磁盘&#xff0c;这样在 Redis 重启就能够从磁盘中恢复原有的数据。 Redis 共有三种数据持久化的⽅式&#xff1a; AOF 日志&#xff1a;每执行一条写操作命令&#xff0c;就把该命令以追加的方式写入到⼀个文…

7.5图像缩放

实验原理 在OpenCV&#xff08;Open Source Computer Vision Library&#xff09;中&#xff0c;resize函数用于调整图像的尺寸。这个函数非常有用&#xff0c;尤其是在进行图像预处理时&#xff0c;比如在图像识别或机器学习任务中需要统一输入图像的大小。 下面是基于C的re…

Qt与Udp

(1)绑定端口 (2)广播 用udp实现广播通信_udp广播-CSDN博客 数据的发送是面向整个子网的&#xff0c;任何一台在子网中的计算机都可以接收到相同的数据。 如果一台机器希望向其他N台机器发送信息&#xff0c;这时候可以使用UDP的广播。 --------------- 广播地址&#xff1…

大数据-133 - ClickHouse 基础概述 全面了解

点一下关注吧&#xff01;&#xff01;&#xff01;非常感谢&#xff01;&#xff01;持续更新&#xff01;&#xff01;&#xff01; 目前已经更新到了&#xff1a; Hadoop&#xff08;已更完&#xff09;HDFS&#xff08;已更完&#xff09;MapReduce&#xff08;已更完&am…

如何编译OpenHarmony SDK API

往期知识点记录&#xff1a; 鸿蒙&#xff08;HarmonyOS&#xff09;应用层开发&#xff08;北向&#xff09;知识点汇总 startup子系统之syspara_lite系统属性部件 &#xff08;1&#xff09; startup子系统之syspara_lite系统属性部件 &#xff08;2&#xff09; startup子系…

【数据集】城市不透水面数据集CLUD-Urban

城市不透水面数据集CLUD-Urban 数据概述数据下载参考 数据概述 1、论文-ESSD-A 30 m resolution dataset of China’s urban impervious surface area and green space, 2000–2018 空间分辨率&#xff1a;30 m 数据下载 数据下载&#xff1a;A 30-meter resolution data…

Grafana面板-linux主机详情(使用标签过滤主机监控)

1. 采集器添加labels标签区分业务项目 targets添加labels &#xff08;模板中使用的project标签&#xff09; … targets: [‘xxxx:9100’] labels: project: app2targets: [‘xxxx:9100’] labels: project: app1 … 2. grafana面板套用 21902 模板 演示

航空航司reese84逆向

reese84逆向 Reese84 是一种用于保护网站防止自动化爬虫抓取的防护机制&#xff0c;尤其是在航空公司网站等需要严格保护数据的平台上广泛使用。这种机制通过复杂的指纹识别和行为分析技术来检测和阻止非人类的互动。例如&#xff0c;Reese84 可以通过分析访问者的浏览器指纹、…

软件开发人员的真实面

我相信我们都看过视频上那些名为“软件工程师的一天”的视频。这些视频通常只展示一些日常任务&#xff0c;比如吃饭、打字和参加会议。我对这些视频未能展示软件开发工作的真实内容感到失望。这些内容往往只关注表面活动&#xff0c;却忽略了工作中的思维挑战和解决问题的部分…

新升级|优化航拍/倾斜模型好消息,支持处理多套贴图模型!

【天元轻量化软件】一直在不断地追求进步和完善&#xff0c;以满足更多用户的各种需求。 电脑登录天元官网免费体验&#xff1a;天元轻量化软件官网 本次我们对“智能PBR”功能进行了更新。更新后的“智能PBR”支持带多套贴图的模型进行使用。 本轮更新后&#xff0c;主要受益…

CISP-PTE CMS sqlgun靶场

sql靶场有个搜索框先点一下go&#xff0c;有回显说明存在漏洞 有个xss 然后在这里尝试sql注入 输入 -1 union select 1,2,3# 有回显可以查看数据库 然后查询数据库&#xff0c;用户 查询数据库的表名 查询它的数据这里admin用户的密码是md5加密 去解密看看 然后扫描ip目录发…

linux-L5.linux查看应用占用的资源top

启动 top 命令&#xff1a; 打开终端&#xff0c;输入 top 并按回车键。 查看进程信息&#xff1a; 默认情况下&#xff0c;top 会显示系统的整体资源使用情况&#xff0c;包括 CPU、内存、磁盘 I/O 和网络 I/O 等信息。然后它会列出当前运行的进程&#xff0c;以及它们分别占…

Leetcode面试经典150题-138.随机链表的复制

题目比较简单&#xff0c;重点是理解思想&#xff0c;random不管&#xff0c;copy一定要放在next 而且里面的遍历过程不能省略 解法都在代码里&#xff0c;不懂就留言或者私信 /* // Definition for a Node. class Node {int val;Node next;Node random;public Node(int val…

【脑机接口】脑机接口性能的电压波形的尖峰分类和阈值比较

Comparison of spike sorting and thresholding of voltage waveforms for intracortical brain–machine interface performance 脑机接口性能的电压波形的尖峰分类和阈值比较论文下载&#xff1a;摘要1 介绍2 方法2.1数据获取2.2spike sorting 技术2.3神经数据分析 3结果3.1神…

【机器学习】线性动态系统的基本概念以及卡尔曼滤波器的概念和应用方式

引言 线性动态系统&#xff08;Linear Dynamical System&#xff0c;LDS&#xff09;是一类特殊的动态系统&#xff0c;其中系统的状态转移和观测过程都是线性的 文章目录 引言一、线性动态系统1.1 LDS的基本组成1.2 LDS的数学表示1.2.1 状态方程1.2.2 观测方程LDS的应用 1.3 L…

计算机网络27、28——Linux命令1、2

1、虚拟机网络前方路径内容 用户名机器名&#xff1a;/$ $表示普通用户&#xff0c;#表示root用户 2、Linux不分盘&#xff0c;都是绝对路径 /表示根目录&#xff0c;表示计算机文件夹下 ~是当前用户的家&#xff0c;表示home文件夹下自己的文件夹 3、bin文件夹下的是可执…

【C++】—— list 的了解与使用

【C】—— list 的了解与使用 1 list 的函数接口2 迭代器2.1 简单使用 list 的迭代器2.2 迭代器的划分2.3 不同迭代器的使用场景2.3.1 sort2.3.2 reverse2.3.3 find 3 emplace_back4 操作函数4.1 sort4.1.1 list中sort介绍4.1.2 list 中 sort 与算法库中 sort 效率比较 4.2 mer…