嵌入式培训机构四个月实训课程笔记(完整版)-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;
}
}
//**********************************************

相关推荐
小安运维日记14 分钟前
Linux云计算 |【第三阶段】PROJECT1-DAY1
linux·运维·云计算·apache
pyliumy32 分钟前
rsync 全网备份
linux·运维·服务器
世俗ˊ41 分钟前
CSS入门笔记
前端·css·笔记
万河归海4281 小时前
C语言——二分法搜索数组中特定元素并返回下标
c语言·开发语言·数据结构·经验分享·笔记·算法·visualstudio
sorel_ferris1 小时前
Ubuntu-24.04中Docker-Desktop无法启动
linux·ubuntu·docker
ggb19991 小时前
【python的坑】vpn下,python request报错 check_hostname requires server_hostname
linux·运维·服务器
小O_好好学1 小时前
vi | vim基本使用
linux·编辑器·vim
-SGlow-1 小时前
Linux相关概念和重要知识点(4)(自举、vim)
linux·运维·vim
多多*2 小时前
OJ在线评测系统 登录页面开发 前端后端联调实现全栈开发
linux·服务器·前端·ubuntu·docker·前端框架
李小星同志2 小时前
高级算法设计与分析 学习笔记6 B树
笔记·学习