嵌入式培训机构四个月实训课程笔记(完整版)-Linux ARM驱动编程第一天-IO和时钟(物联技术666)

链接:https://pan.baidu.com/s/1E4x2TX_9SYhxM9sWfnehMg?pwd=1688
提取码:1688

上午:ARM的I/O口
下午:ARM的时钟设定
教学内容:
1、S3C2440的I/O口
S3C2440A共有289个管脚,其中130个可配置为多功能复用输入/输出口,共分为9组,即PORTA~PORTH、PORTJ按照位数的不同,可分为:
---PORTA(GPA):23位输出口;
---PORTB(GPB):11位输入/输出口;
---PORTC(GPC):16位输入/输出口;
---PORTD(GPD):16位输入/输出口;
---PORTE(GPE):16位输入/输出口;
---PORTF(GPF):08位输入/输出口;
---PORTG(GPG):16位输入/输出口;
---PORTH(GPH):11位输入/输出口;
---PORTJ (GPJ): 13位输入/输出口;
对于I/O的控制,一般由GPGCON和GPGDAT和GPGUP三种寄存器控制,其中:
GPGCON:为I/O的设置,可以设置为00=输入,01=输出,10=允许中断,11=保留
GPGDAT:和I/O口交流的数据寄存器,读写都是对GPGDAT操作
GPGUP:设置是否上拉电阻,0:设置;1:不设置
例如:

GPGCON的地址是0x56000060,可读可写,复位后默认为0
程序设计,例如:
//***********************************************
#include "2440addr.h" //声明S3C2440的寄存器及其常用变量
#include "2440lib.h"
#include "uart.h"
#define GPGCON (*(volatile unsigned long *)0x56000060) //不可优化,为UL型
#define GPGUP (*(volatile unsigned long *)0x56000068)
#define GPGDAT (*(volatile unsigned long *)0x56000064)
GPGCON &=~(0x03<<0*2)|(0x03<<1*2)|(0x03<<2*2); //利用移位对GPGCON设置
GPGCON |=(0x01<<0*2)|(0x01<<1*2)|(0x01<<2*2);
//rGPGCON &=~(0x03<<0*2)|(0x03<<1*2)|(0x03<<2*2); //不需要前面的#define,用的是2440addr.h的头文件声明
//***********************************************************
2、时钟和电源
S3C2440的时钟可以选用晶振(XTAL),也可以使用外部时钟(EXTCLK),由系统复位时,在复位信号上升沿对引脚OM3、OM2所测的状态来确定。其对应关系如下表所示:
对时钟而言,晶振的频率一般不能高于100M,而对于现在的嵌入式芯片而言,速度越来越快,为了获得更高的频率可以采用锁相环和选频技术从而获得更高的频率,ARM9的具体时钟实现如下:

从1,2可知,通过OM[3:2]选择时钟信号来源,第3是含有两个锁相环MPLL、UPLL,它们分别产生系统所需要的时钟MPLL:FCLK---CPU,HCLK---AHB总线,PCLK---APB总线;UPLL: 产生UCLK供USB(48MHz)和Camera;
1)、对于系统上电开始,PLL是没有启动的,所以FCLK=Fin(晶振频率);
2)、设置MPLL几个寄存器后,等待一定时间,MPLL才能输出稳定,之后才能使用;
3)、(LOCKTIME寄存器)等待时间一般不要设定,使用默认时间即可;
MPLLCON:设置MPLL锁相环的输出频率
UPLLCON:设置UPLL锁相环的输出频率
CLKCON:控制所有设备的时钟关开,睡眠,空闲模式的设定
CLKSLOW:慢模式下的相关设置
CLKDIVN:分频设置UCLK,HCLK,PCLK的频率
对于正常模式,只要一般只要设置MPLLCON和CLKDIVN,CPU频率FCLK=MPLLCLK
例如:
//*********************************************
//设置MPLL锁相环的输出频率
void ChangeMPllValue(int mdiv,int pdiv,int sdiv)
{
rMPLLCON = (mdiv<<12) | (pdiv<<4) | sdiv;
}
//设置HCLK和PCLK
void ChangeClockDivider(int hdivn_val,int pdivn_val)
{
int hdivn=2, pdivn=0;
switch(hdivn_val)
{
case 11: hdivn=0; break;
case 12: hdivn=1; break;
case 13:
case 16: hdivn=3; break;
case 14:
case 18: hdivn=2; break;
}
switch(pdivn_val){
case 11: pdivn=0; break;
case 12: pdivn=1; break;
}
rCLKDIVN = (hdivn<<1) | pdivn;
switch(hdivn_val) {
case 16: break;
case 18: break;
}
}
//**********************************************

相关推荐
为美好的生活献上中指1 分钟前
java每日精进 4.29【框架之自动记录日志并插入如数据库流程分析】
java·linux·数据库
mljy.13 分钟前
Linux《进程概念(中)》
linux
JhonKI23 分钟前
【Linux网络】深入解析I/O多路转接 - Select
linux·运维·网络
伤不起bb23 分钟前
Nginx 核心功能
linux·服务器·nginx
灏瀚星空1 小时前
从基础到实战的量化交易全流程学习:1.3 数学与统计学基础——线性代数与矩阵运算 | 矩阵基础
笔记·python·学习·线性代数·数学建模·金融·矩阵
The-Dog1 小时前
Linux命令使用记录(自用)
linux
Linux运维老纪2 小时前
Ansible 铸就 Linux 安全之盾(Ansible Builds Linux Security Shield)
linux·服务器·网络·安全·云计算·ansible·运维开发
唐青枫2 小时前
Linux apropos 命令使用详解
linux
刘大猫.2 小时前
Centos Ubuntu RedOS系统类型下查看系统信息
linux·ubuntu·centos·ip·ifconfig·redos·查询系统信息
YuSun_WK3 小时前
程序&进程&多任务&线程
linux·运维·服务器