本章目标:
了解S3C2410/S3C2440地址空间的布局
掌握如何通过总线形式访问扩展的外设,比如内存、NOR Flash、网卡等
····································································································
总线的使用方法是嵌入式低层开发的基础,了解它之后,再根据外设的具体特性,就可以驱动外设了。
6.1 使用存储控制器访问外设的原理
6.1.1 S3C2410/S3C2440的地址空间
S3C2410/S3C2440的“存储控制器”提供了访问外部设备所需要的信息,它有如下特性:
① 支持小字节序、大字节序(通过软件选择);
② 每个BANK的地址空间为128MB,总共1GB(8 BANKs);
③ 可编程控制的总线位宽(8/16/32-bit),不过BANK0只能选择两种位宽(16/32-bit);
④ 总共8个BANK,BANK0~BANK5可以支持外接ROM、SRAM等,BANK6~BANK7除了可以支持
ROM、SRAM外,还支持SDRAM等;
⑤ BANK0~BANK6共7个BANK的起始地址是固定的;
⑥ BANK7的起始地址可编程选择;
⑦ BANK6、BANK7的地址空间大小是可编程控制的;
⑧ 每个的访问周期均可编程控制;
⑨ 可以通过外部的“wait”信号延长总线的访问周期;
⑩ 在外接SDRAM时,支持自刷新(self-refresh)和省电模式(power down mode)。
S3C2410/S3C2440对外引出的27根地址线ADDR0~ADDR26的访问范围只有128MB,CPU对外
还引出了8根片选信号nGCS0~nGCS7,对应于BANK0~BANK7,当访问BANKx的地址空间时,
nGCSx引脚输出低电平用来选中外接的设备。这8个BANK的地址空间如下图6.1所示:
S3C2410/S3C2440作为32位的CPU,可以使用的地址范围理论上达到4GB。除去上述用于连接外设
的1GB地址空间外,还有一部分是CPU内部寄存器的地址,剩下的地址空间没有使用。
S3C2410/S3C2440的寄存器地址范围都处于0x480 0000~0x5ff ffff,各功能部件的寄存器大体相同,
如下表所示:
6.1.2 存储控制器与外设的关系
本书所用开发板使用了存储控制器的BANK0~BANK6,分别外接了如下设备:
NOR Flash、IDE接口、10M网卡CS8900A、100M网卡DM9000、扩展串口芯片16C2550、
SDRAM,连线方式如下图6.2所示。
外设的访问地址 = 地址线确定的地址 + BANK的起始地址。
比如:扩展串口。
(1)它使用nGCS5,起始地址为0x2800 0000。
(2)nCSA=ADDR24||nGCS5,nCSB=!ADDR24||nGCS5。当ADDR24和nGCS5均为低电平
时选中扩展串口A;当ADDR24为高电平、nGCS5为低电平时,选中扩展串口B。
(3)CPU的ADDR0~ADDR2连接到扩展串口的A0~A2,所以访问空间为8字节。
综上所述,扩展串口A的访问空间为:0x2800 0000~0x2800 0007;扩展串口B的
访问空间为:0x2900 0000~0x2900 0007(bit24为1)。
图4.2:
BANK0~BANK5的连接方式都是类似的,BANK6连接SDRAM时复杂一点,CPU提供了一
组用于SDRAM的信号。
① SDRAM时钟有效信号SCKE;
② SDRAM时钟信号SCLK0/SCLK1;
③ 数据掩码信号DQM0/DQM1/DQM2/DQM3;
④ SDRAM片选信号nSCS0(它与nGCS6是同一引脚的两个功能);
⑤ SDRAM行地址选通脉冲信号nSRAS;
⑥ SDRAM列地址选通脉冲信号nSCAS;
⑦ 写允许信号nWE(它不是专用于SDRAM的)。
SDRAM的内部是一个存储阵列,如同表格一样,将数据填进去。
SDRAM的寻址基本原理:先指定一个行,在指定一个列,就可以准确地找到所需要的单元格。
这个单元格被称为存储单元。这个表格(存储阵列)就是逻辑Bank(简称L-Bank),SDRAM一般含有
4个L-BANK。
SDRAM的逻辑结构如图6.3所示。
可以想象,对SDRAM的访问可以分为如下4个步骤:
(1)CPU发出的片选信号nSCS0有效,它选中SDRAM芯片;
(2)SDRAM中有4个L-Bank,需要两根地址信号来选中其中一个,从图6.2可知使用ADDR24、
ADDR25作为L-Bank的选择信号;
(3)对被选中的芯片进行统一行、列(存储单元)寻址;
根据SDRAM芯片的列地址线数目设置CPU的相关寄存器后,CPU就会从32位地址中自动分出
L-Bank选择信号、行地址信号、列地址信号,然后先后发出行地址信号、列地址信号。L-Bank选
择信号在发出行地址信号的同时发出,并维持到列地址信号结束。
在图6.2中,行地址、列地址公用地址线ADDR2~ADDR14(BANK6位宽为32,ADDR0/1没有使
用),使用nSRAS、nSCAS两个信号来区分它们。比如本开发板中,使用两根地址线ADDR24、
ADDR25作为L-Bank的选择信号;SDRAM芯片K4S561632的行地址数为13,列地址数为9,所以
当nSRAS信号有效时,ADDR2~ADDR14上发出的是行地址信号,它对应32位地址空间的bit[23:11];
当nSCAS信号有效时,ADDR2~ADDR10上发出的是列地址信号,它对应32位地址空间的bit[10:2]。
由于图6.2中BANK6以32位的宽度连接SDRAM,ADDR0、ADDR1恒为0,不参与译码。
(4)找到存储单元后,被选中的芯片就要进行统一的数据传输了。
开发板中使用两片16位的SDRAM芯片并联组成32位的位宽,与CPU的32根数据线(DATA0~DATA31)相连。
BANK6的起始地址为0x3000 0000,所以SDRAM的访问地址为0x3000 0000~0x33ff ffff,共64MB。
对图6.2中连接的外设,它们的访问地址(物理地址)如表6.2所示。
6.1.3 存储控制器的寄存器使用方法
存储控制器共有13个寄存器,BANK0~BANK5只需要设置BWSCON和BANKCONx(x为0~5)两个
寄存器:BANK6和BANK7外接SDRAM时,除BWSCON和BANKCONx(x为6、7)外,还要设置REFRESH、
BANKSIZE、MRSRB6、MRSRB7等4个寄存器。下面分类说明:
1.位宽和等待控制寄存器BWSCON
每4位控制一个BANK,最高4位对应BANK7、接下来4位对应BANK6,以此类推。
(1)STx:启动/禁止SDRAM的数据掩码引脚,对于SDRAM, 此位为0;对于SRAM,此位为1;
(2)WSx:是否使用存储器的WAIT信号,通常设为0;
(3)DWx:使用两位来设置相应BANK的位宽:
0b00:8位,0b01:16位,0b10:32位,0b11:保留;
BANK0比较特殊,只支持16和32位两种位宽,它没有ST0和WS0,DW0([2:1])只读——由硬件跳线决定:
0b01:16位,0b10:32位;
对于本开发板,没有使用BANK7,根据表6.1可以确定BWSCON的值为:0x2201 1110。
2.BANK控制寄存器BANKCONx(x为0~5)
这几个寄存器用来控制BANK0~BANK5外接设备的访问时序,使用默认的0x0700即可满足
本开发板所接各外设的要求。
3.BANK控制寄存器BANKCONx(x为6~7)
在8个BANK中,只有BANK6和BANK7可以外接SRAM或SDRAM,所以BANKCON6~
BANKCON7与BANKCON0~BANKCON5有点不同。
(1)MT([16:15]):用于设置本BANK外接的是ROM/SRAM还是SDRAM。
当MT = 0b00:时,接SRAM,此寄存器与BANKCON0~BANKOCN5类似;
当MT = 0b11:时,接SDRAM,此寄存器其他值如下设置。
(2)Trcd([3:2]):RAS to CAS delay,设为推荐值0b01。
(3)SCAN([1:0]):SDRAM的列地址位数,对于本开发板使用的SDRAM K4S561632,
列地址位数为9,所以SCAN = 0b01。如果使用其他型号的SDRAM,需要查看数据手册来
决定SCAN的取值。0b00:8位;0b01:9位;0b10:10位。
综上所述,本开发板中BANKCON6、7均设为0x0001 8005。
4.刷新控制寄存器REFRESH:设为0x008c 0000 + R_CNT
(1)REFEN([23]):0 = 禁止SDRAM的刷新功能,1 = 开启SDRAM的刷新功能。
(2)TREFMD([22]):SDRAM的刷新模式,0 = CBR/Auto Refresh,1 = Self Refresh(一般在系统休眠时使用)。
(3)Trp([21:20]):设为0即可。
(4)Tsrc([19:18]):设为默认值0b11即可。
(5)Refresh Counter([10:0]):即上述的R_CNT,
R_CNT = 2^11 + 1 - SDRAM时钟频率(MHz) * SDRAM刷新周期(us);
其中SDRAM时钟频率就是HCLK,SDRAM的刷新周期在SDRAM数据手册上有标明,
在本开发板所使用的SDRAM 数据手册上,可以看见“ 64ms refresh period(8K Cycle)”。所以,
刷新周期 = 64ms/8192 = 7.8125us。
在未使用PLl时,SDRAM时钟频率等于晶振频率12MHz。
现在可以计算:R_CNT = 2^11 + 1 - 12*7.8125 = 1955。
所以,在未使用PLL时,REFRESH = 0x008c 0000 + 195 = 0x008C 07A3。
5.BANKSIZE寄存器BANKSIZE
(1)BURST_EN([7])。
0 = ARM核禁止突发传输,1 = ARM核支持突发传输。
(2)SCKE_EN([5])。
0 = 不使用SCKE信号令SDRAM进入省电模式,1 = 使用SCKEN信号令SDRAM进入省电模式。
(3)SCLK_EN([4])。
0 = 时刻发出SCLK信号,1 = 仅在访问SDRAM期间发出SCLK信号(推荐)。
(4)BK76MAP([2:0]):设置BANK6/7的大小。
BANK6/7对应的地址空间与BANK0~5不同:
BANK0~5的地址空间大小都是固定的128MB,地址范围是(x * 128M)到(x+1)*128M -1,x表示0~5。
BANK6/7的大小是可变的,以保持这两个空间的地址连续,即BANK7的起始地址会随着
它们的大小变化。
BK76MAP的取值意义如下:
0b010 = 128MB/128MB,0b001 = 64MB/64MB,0b000 = 32M/M,
0b111 = 16M/16M,0b110 = 8M/8M,0b101 = 4M/4M,0b100 = 2M/2M
本开发板BANK6外接64MB的SDRAM,令[2:0] = 0b001,表示BANK6/7的容量都是64MB,
虽然BANK7没使用。
综上所述,开发板的BANKSIZE寄存器的值可算得0xB1。
6.SDRAM模式设置寄存器MRSRBx(x为6~7)
能修改的只有位CL([6:4]),这是SDRAM时序的一个时间参数:
[work] 0b000 = 1 clock,0b010 = 2 clocks,0b011 = 3 clocks
SDRAM K4S561632不支持CL = 1的情况,所以此位取值0b010(CL = 2)或0b011(CL = 3)。
本开发板取最保守值0b011,所以MRSRB6/7的值为0x30。
6.2 存储控制器操作实例:使用SDRAM
6.2.1 代码详解及程序的复制、跳转过程
从NAND Flash启动CPU时,CPU会通过内部的硬件将NAND Flash开始的4KB数据复
制到称为“Steppingstone”的4KB的内部RAM中(起始地址为0),然后跳到地址0开始执行。
本实例先使用汇编语言设置好存储控制器,使外接的SDRAM可用;然后把程序从
Steppingstone复制到SDRAM处;最后跳到SDRAM中执行。
源代码在/work/hardware/sdram目录中,包含两个文件head.S和leds.c。其中leds.c和第5
章中的leds的代码完全一样,也是让3个led从0~7轮流计数。
重点在head.S,它的作用是设置SDRAM,将程序复制到SDRAM,然后跳到SDRAM
继续执行。head.S的代码如下:
@****************************************
@ File:head.S
@ 功能:设置SDRAM,将程序复制到SDRAM,然后跳到SDRAM继续执行
@****************************************
.equ MEM_CTL_BASE, 0x48000000
.equ SDRAM_BASE, 0x30000000
.text
.global _start
_start:
bl disable_watch_dog @关闭WATCHDOG
bl memsetup @设置存储控制器
bl copy_steppingston_to_sdram @复制代码到SDRAM中
ldr pc, =on_sdram @跳到SDRAM中继续执行
on_sdram:
ldr sp, =0x34000000 @设置栈
bl main
halt_loop:
b halt_loop
disable_watch_dog:
@往WATCHDOG寄存器写0即可
mov r1, #0x53000000
mov r2, #0x0
str r2, [r1]
mov pc, lr @返回
copy_steppingstone_to_sdram:
@将Steppingstone的4KB数据全部复制到SDRAM中去
@Steppingstone起始地址为0x0000 ,SDRAM中起始地址为0x3000
mov r1, #
ldr r2, =SDRAM_BASE
mov r3, #*
:
ldr r4, [r1], # @从Steppingstone(0x0000 )读取4字节的数据,并让源地址加4
str r4, [r2], # @将此4字节数据复制到SDRAM中,并让目的地址加4
cmp r1, r3 @判断是否完成:源地址是否等于Steppingstone的末地址?
bne 1b @若没有复制完,继续
mov pc, lr @返回
memsetup:
@设置存储控制器以便使用SDRAM等外设
mov r1, #MEM_CTL_BASE @存储控制器的13个寄存器的开始地址
adr1 r2, mem_cfg_val @这13个值的起始存储地址
add r3, r1, # @13* =
:
ldr r4, [r2], # @读取“待设置值”,并让r2加4
str r4, [r1], # @将此值写入寄存器,并让r1加4
cmp r1, r3 @判断是否设置完所有13个寄存器
bne 1b @若没有写完,继续
mov pc, lr @返回
.align
mem_cfg_val:
@存储控制器13个寄存器的设置值
.long 0x22011110 @BWSCON
.long 0x00000700 @BANKCON0
.long 0x00000700 @BANKCON1
.long 0x00000700 @BANKCON2
.long 0x00000700 @BANKCON3
.long 0x00000700 @BANKCON4
.long 0x00000700 @BANKCON5
.long 0x00018005 @BANKCON6
.long 0x00018005 @BANKCON7
.long 0x008c07a3 @REFRESH
.long 0x000000b1 @BANKSIZE
.long 0x00000030 @MRSRB6
.long 0x00000030 @MRSRB7
head.S
程序是如何通过第15行的“ ldr pc, =on_sdram”指令来完成的?
程序标号“on_sdram”这个地址值在连接程序时被确定为0x3000 0010(这个是SDRAM的
地址),执行“ldr pc, =on_sdram”后,程序一下子就跳到SDRAM中去了。
“on_sdram”这个地址值为什么等于0x3000 0010?
Makefile中连接程序的命令为“arm-linux-ld -Ttext 0x30000000 head.o sdram.o -o
sdram_elf”,意思就是代码段的起始地址为0x3000 0000,即程序的第一条指令(第12行)的
连接地址为0x3000 0000,第二条指令(第13行)的连接地址为0x3000 0004,...,第5条指
令(第17行)的连接地址为0x3000 0010,其程序标号“on_sdram”的值就是0x3000 0010。
虽然第12~14行指令的连接地址都在SDRAM中,但是由于他们都是位置无关的相对
跳转指令,所以可以在Steppingstone里执行。
Makefile如下(注意第4行,“-Ttext 0x30000000”指定了代码段的起始地址):
sdram.bin : head.S leds.c
arm-linux-gcc -c -o head.o head.S
arm-linux-gcc -c -o leds.o leds.c
arm-linux-ld -Ttext 0x30000000 head.o leds.o -o sdram_elf
arm-linux-objcopy -O binary -S sdram_elf sdram.bin
arm-linux-objdump -D -m arm sdram_elf > sdram.dis
clean:
rm -f sdram.dis sdram.bin sdram_elf *.o
Makefile
为了更形象地了解本程序,下面用图6.4来演示程序的复制、跳转过程。
6.2.2 实例测试
在sdram目录中执行make指令生成可执行文件sdram.bin后,下载到板子上运行。可以
发现与leds程序相比,LED闪烁得更慢,原因是外部SDRAM的性能比内部SRAM差一些。
把程序从性能更好的内部SRAM移到外部SDRAM中去,是否多此一举呢?内部SRAM
只有4KB大小,如果程序大于4KB,那么就不能指望完全利用内部SRAM来运行了,得想
办法把存储在NAND Flash中的代码复制到SDRAM中去。对于NAND Flash中的前4KB,
芯片自动把它复制到内部SRAM中,可以很轻松地再把它复制到SDRAM中(实验代码中的
函数copy_steppingston_to_sdram就有此功能),要复制4KB后面的代码需要使用NAND Flash
控制器来读取NAND Flash,这就是第8章的内容。
附:代码:
链接: https://pan.baidu.com/s/1kV24a9L 密码: tfab
- JZ2440 裸机驱动 第13章 LCD控制器(1)
本章目标 了解LCD显示器的接口及时序: 掌握S3C2410/S3C2440 LCD控制器的使用方法: 了解帧缓冲区的概念,掌握如何设置帧缓冲区来显示图像: 13.1 LCD和LCD控制器 13.1 ...
- JZ2440 裸机驱动 第13章 LCD控制器(2)
13.2 TFT LCD显示实例 13.2.1 程序设计 本实例的目的是从串口输出一个菜单,从中选择各种方法进行测试,比如画线. 画圆.显示单色.使用调色板等. 13.2.2代码详解 ...
- JZ2440 裸机驱动 第10章 系统时钟和定时器
本章目标 了解S3C2410/S3C2440的时钟体系结构 掌握通过设置MPLL改变系统时钟的方法 掌握在不同的频率下设置存储控制器的方法 掌握PWM定时器的用法 ...
- JZ2440 裸机驱动 第7章 内存管理单元MMU
本章目标: 了解虚拟地址和物理地址的关系: 掌握如何通过设置MMU来控制虚拟地址到物理地址的转化: 了解MMU的内存访问权限机制: 了解TLB.Cache.Write ...
- JZ2440 裸机驱动 第8章 NAND Flash控制器
本章目标 了解NAND Flash 芯片的接口 掌握通过NAND Flash控制器访问NAND Flash的方法 8.1 NAND Flash介绍和NAND Flash控制器使用 NAND ...
- JZ2440 裸机驱动 第14章 ADC和触摸屏接口
本章目标: 了解S3C2410/S3C2440和触摸屏的结构: 了解电阻触摸屏的工作原理和等效电路图: 了解S3C2410/S3C2440触摸屏控制器的多种工作模式: ...
- JZ2440 裸机驱动 第12章 I2C接口
本章目标: 了解I2C总线协议: 掌握S3C2410/S3C2440中I2C接口的使用方法: 12.1 I2C总线协议及硬件介绍 12.1.1 I2C总线协议 1 I2C总线的概念 2 I2C总线的信 ...
- JZ2440 裸机驱动 第5章 GPIO接口
本章目标: 掌握嵌入式开发的步骤:编程.编译.烧写程序.运行 通过GPIO的操作了解软件如何控制硬件 5.1 GPIO硬件介绍 S3C2440A有130个多功能输入/输出口引脚 ...
- JZ2440 裸机驱动 第9章 中断体系结构
本章目标: 了解ARM体系CPU的7种工作模式 了解S3C2410/S3C2440中断体系结构 掌握S3C2410/S3C2440的中断服务程序的编写方法 9.1 S3C241 ...
随机推荐
- json批量设置DIV属性
<!doctype html> <html> <head> <meta charset="utf-8"> <title> ...
- SGU 124. Broken line 射线法 eps的精准运用,计算几何 难度:3
124. Broken line time limit per test: 0.25 sec. memory limit per test: 4096 KB There is a closed bro ...
- 申请和使用github共享代码
1.申请github帐号 https://github.com/join?source=header-home 2.创建项目 2.1 或者: 2.2 输入信息 2.3创建成功,地址及基本命令提示 3. ...
- C# 自动发送邮件被系统当做垃圾邮件退回的处理方法
前两天做了一个发送邮件的定时任务.其功能为在每天的某一时刻,系统自动从银行查询前一天的交易明细到本地服务器,解析后生成为各自不同Excel文件,并打包成zip压缩文件,然后抄送附件给财务对账.其中,邮 ...
- tensorflow 模型保存后的加载路径问题
import tensorflow as tf #保存模型 saver = tf.train.Saver() saver.save(sess, "e://code//python//test ...
- 关于edge detecte
1.注意render Texture 双击render Texture ,要勾选 use viewport dimensions,否则出现边框和模型不合的情况 2.注意edge detecte 的状态 ...
- SpringContextUtil spring上下文获取工具类
package com.midea.biz; import org.springframework.beans.BeansException; import org.springframework.c ...
- 去除字符串中的emoji字符
对于使用utf8编码的mysql数据库来说,如果字符串中存在emoji小图像,是不能存进数据库中的,查了一下,原因大概是因为utf8编码可以存1-3个字节的字符,但是emoji是4个字节:解决方法可以 ...
- L213
The world lost seven astronauts of Space Shuttle Columbia(哥伦比亚号航天飞机) this month. It broughthome the ...
- chrome flash插件地址
C:\Users\Administrator\AppData\Local\Google\Chrome\User Data\PepperFlash 火狐