IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> 嵌入式 -> 5. ARM时钟系统 -> 正文阅读

[嵌入式]5. ARM时钟系统


时钟是同步工作系统的同步信号,SOC内部有很多控制器,例如CPU,串口等外设,这些外设需要协同工作,互相通信,就需要同步的时钟系统来指挥。这个就是Soc的时钟系统。

时钟设置编码实现

查手册和系统时钟图算出各位置应该配置的值:
系统时钟图
计算分频器每部分应该设置的值
在这里插入图片描述
之前的各时钟是由uboot程序设置好的,现在我们在跳转到C语言的main函数之前自己重新设置相应的时钟。
start.s:

.global _start
_start:
		b reset
		b undef
		b swi
		b abt_pre
		b abt_dat
		b reserved
		b irq
		b fiq
reset:
		@重新映射异常向量表到0x40008000 
		ldr r0, =0x40008000 
		mcr p15, 0, r0, c12, c0, 0
		@打开中断的总开关
		mrs r0, cpsr
		bic r0, r0, #(0x1 << 7)
		msr cpsr, r0 
		bl clock_init
		bl main
		
undef:
swi:
abt_pre:
abt_dat:
reserved:
irq:
		ldr sp, =0x420000000   @初始化栈指针
		stmfd sp!, {r0-r12,lr}
		bl irq_handler         @跳到中断函数
		ldmfd sp!, {r0-r12,lr}
		subs pc, lr, #4
fiq:

clock.s:

.global clock_init
clock_init:
			@先关闭所有PLL
			@CLK_SRC0
			ldr r0, =0xE0100200
			ldr r1, =0x0
			str r1, [r0]
			
			@设置PLL
			@APLL_LOCK
			ldr r0, =0xE0100000
			ldr r1, =30*24
			str r1, [r0]
			@MPLL_LOCK
			ldr r0, =0xE0100008
			ldr r1, =200*24
			str r1, [r0]
			@EPLL_LOCK
			ldr r0, =0xE0100010
			ldr r1, =375*24
			str r1, [r0]
			@VPLL_LOCK
			ldr r0, =0xE0100020
			ldr r1, =100*24
			str r1, [r0]
			
			@设置倍频因子
			@APLL_CON0
			ldr r0, =0xE0100100
			ldr r1, =(1<<31)|(125<<16)|(3<<8)|(1<<0)
			str r1, [r0]
			
			@MPLL_CON
			ldr r0, =0xE0100108
			ldr r1, =(1<<31)|(667<<16)|(12<<8)|(1<<0)
			str r1, [r0]
			
			@配置分频器
			@CLK_DIV0
			ldr r0, =0xE0100300
			ldr r1, =(1<<28)|(4<<20)|(1<<20)|(3<<16)|(1<<12)|(4<<8)|(4<<4)|0
			str r1, [r0]
			
			@选择使用PLL
			@CLK_SRC0
			ldr r0, =0xE0100200
			ldr r1, =0x1111
			str r1, [r0]
delay:
			mov r0, 0x100000
d:
			subs r0, r0, #1
			bne d
			
			mov pc, lr

main.c

#define WTCON   (*(volatile unsigned int *)0xE2700000))
#define WTDAT   (*(volatile unsigned int *)0xE2700004))
#define WTCNT 	(*(volatile unsigned int *)0xE2700008))
#define WTCLRINT (*(volatile unsigned int *)0xE270000C))
void key1_handler(void)
{
	int (*printf)(char *format,...)=(void *)0x3ff13e54;
	printf("hello wdt..\n");
	//清理GPIO里面的中断记录
	WTCLRINT  = 1;//清理中断

}

void wdt_init()
{
	WTCON = (65<<8)|(0x1<<5)|(0x1<<3)|(0x1<<2)|0;
	//t=(32*(65+1))/ 66M = 32us
	WTCNT = 1000*1000/32;//1s
	WTDAT = 1000*1000/32; //首次喂狗
}
int main()
{
	//初始化看门狗
	wdt_init();
	//初始化中断控制器
	request_irq(27,wdt_handler);//看门狗是27号中断
	while(1)
	{
		;
	}
	return 0;
}

irq.c:

#define VICOINTSELECT (*(volatile unsigned int *)0xF200000C)
#define VIC1INTSELECT (*(volatile unsigned int *)0xF210000C)
#define VIC2INTSELECT (*(volatile unsigned int *)0xF220000C)
#define VIC3INTSELECT (*(volatile unsigned int *)0xF230000C)

#define VIC0VECTADDR (volatile unsigned int *)0xF2000100
#define VIC1VECTADDR (volatile unsigned int *)0xF2100100
#define VIC2VECTADDR (volatile unsigned int *)0xF2200100
#define VIC3VECTADDR (volatile unsigned int *)0xF2300100

#define VIC0INTENABLE (*(volatile unsigned int *)0xF2000010)
#define VIC1INTENABLE (*(volatile unsigned int *)0xF2100010)
#define VIC2INTENABLE (*(volatile unsigned int *)0xF2200010)
#define VIC3INTENABLE (*(volatile unsigned int *)0xF2300010)

#define VIC0ADDRESS (*(volatile unsigned int *)0xF2000F00)
#define VIC1ADDRESS (*(volatile unsigned int *)0xF2100F00)
#define VIC2ADDRESS (*(volatile unsigned int *)0xF2200F00)
#define VIC3ADDRESS (*(volatile unsigned int *)0xF2300F00)

//中断控制器初始化函数
int request_irq(int irq, void(*handler)())
{
	if(irq>=0 && irq <= 31)
	{
		VICOINTSELECT = 0;//初始化中断控制器
		VIC0VECTADDR[irq] = (unsigned int)handler; //注册中断函数
		VIC0INTENABLE |= 0x1<<irq; //开启中断
	}
	else if(irq>=32 && irq<=63)
	{
		irq-=32;
		VIC1INTSELECT = 0;
		VIC1VECTADDR[irq] = (unsigned int)handler;
		VIC1INTENABLE |= 0x1<<irq; //开启中断
	}
	else if(irq>=64 && irq <=95)
	{
		irq-=64;
		VIC2INTSELECT = 0;
		VIC2VECTADDR[irq] = (unsigned int)handler;
		VIC2INTENABLE |= 0x1<<irq; //开启中断
	}
	else if(irq>=96 && irq <=128)
	{
		irq-=96;
		VIC3INTSELECT = 0;
		VIC3VECTADDR[irq] = (unsigned int)handler;
		VIC3INTENABLE |= 0x1<<irq; //开启中断
	}

}

//中断处理函数
void irq_handler()
{
	void(*handler)();
	
	//判断是那个中断发生
	if(VIC0ADDRESS !=0 )
		handler=(void *)VIC0ADDRESS;
	else if(VIC1ADDRESS !=0)
		handler=(void *)VIC1ADDRESS;
	else if(VIC2ADDRESS !=0)
		handler=(void *)VIC2ADDRESS;
	else if(VIC3ADDRESS !=0)
		handler=(void *)VIC3ADDRESS;
		
	//执行绑定的中断处理函数
	handler();
	
	//清理中断控制器
	VIC0ADDRESS=0;
	VIC1ADDRESS=0;
	VIC2ADDRESS=0;
	VIC3ADDRESS=0;
}

编译和测试

编译
因为串口用的是PCLK,如果按手册设置正确,则打印正常。
结果
修改clock.s:

ldr r1, =(0<<28)|(4<<20)|(1<<20)|(3<<16)|(1<<12)|(4<<8)|(4<<4)|0

再次编译运行:
结果
因为时钟变快了,则看门狗触发中断速度变快,而串口波特率也变了,所以是乱码。

  嵌入式 最新文章
基于高精度单片机开发红外测温仪方案
89C51单片机与DAC0832
基于51单片机宠物自动投料喂食器控制系统仿
《痞子衡嵌入式半月刊》 第 68 期
多思计组实验实验七 简单模型机实验
CSC7720
启明智显分享| ESP32学习笔记参考--PWM(脉冲
STM32初探
STM32 总结
【STM32】CubeMX例程四---定时器中断(附工
上一篇文章      下一篇文章      查看所有文章
加:2021-11-18 11:20:34  更:2021-11-18 11:22:14 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2025年1日历 -2025/1/6 19:26:07-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码