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

相关推荐
Hello_Embed35 分钟前
libmodbus 移植 STM32(USB 串口后端篇)
笔记·stm32·单片机·嵌入式·freertos·libmodbus
张祥6422889041 小时前
RTKLIB源码和理论结合分析笔记三
笔记
日更嵌入式的打工仔1 小时前
0欧电阻作用
笔记
生活很暖很治愈1 小时前
Linux——孤儿进程&进程调度&大O(1)调度
linux·服务器·ubuntu
HalvmånEver1 小时前
Linux:线程同步
linux·运维·服务器·线程·同步
Zach_yuan1 小时前
自定义协议:实现网络计算器
linux·服务器·开发语言·网络
wdfk_prog1 小时前
[Linux]学习笔记系列 -- [drivers][I2C]I2C
linux·笔记·学习
VekiSon2 小时前
Linux内核驱动——杂项设备驱动与内核模块编译
linux·c语言·arm开发·嵌入式硬件
Y1rong2 小时前
linux之网络
linux
寄存器漫游者2 小时前
Linux 软件编程 - IO 编程
linux·运维·spring