内存管理单元MMU,负责虚拟地址到物理地址的转换,并提供硬件进制的内存访问权限检查,操作系统通过MMU可以实现各个用户进程自己独立的地址空间,
通过内存权限的检查可以保证每个进程使用的内存不被其他进程破坏。具体操作系统怎么运用MMU的就比较复杂了,我们只看看最原始的裸机程序怎么配置寄存器操作MMU建立虚拟地址到物理地址映射,了解其原理。
ARM提供4种映射长度:段(1M),大页(64K),小页(4K),极小页(1k)
按1M的长度映射就是说一段1M的虚拟地址映射一段1M的物理地址,这1M的地址访问权限是一样的,明显如果按1k,4k映射,内存访问权限将控制的更精细,当然也更复杂。
我们看看1M的精度怎么映射,假设要映射虚拟地址范围(0x0~0x4000000)到物理地址范围(0xB0000000~0xB4000000), 每0x100000的地址长度是1M,那么范围就是64M.
这个映射是以1M对齐的,就是要已(0x0~0x100000)这样映射,而不能(0x1~0x100001),可以看出虚拟地址和物理地址低20位变化是一样的,这样我们只需要一个4byte的内存空间存放每1M的虚拟地址右移20后与物理地址右移20位建立映射,总共4*64byte。代码大致如下:
unsigned long va = 0x0;
unsigned long pa = 0xB0000000;
while(va<0x4000000){
ttb[va >> 20] = pa | 2;
va += 0x100000;
pa += 0x100000;
}
这样给定一个虚拟地址0x3000400,右移20作为ttb的index就可以找到其后12位对应的物理地址0xB3000000+0x400就是物理地址,当然这样运算是mmu自己做的,我们只需要把ttb的内存地址附给mmu的寄存器。上面的过程叫建立页表,页表建完后,启动MMU,我们操作的就是虚拟地址了。启动MMU是操作cp15协处理器,下面代码会有说明,了解下。
下面的例子我们用虚拟地址来点led.
#include "regs.h"
void (*uart_asm_putc)(int c) = 0x0202391c;
void (*uart_asm_putx)(int x) = 0x02023940;
#define GPM4CON (*(volatile unsigned int *)0xB10002E0)
#define GPM4DAT (*(volatile unsigned int *)0xB10002E4)
void init_ttb(unsigned long *ttb_base);
void mmap(unsigned long *ttb_base, unsigned long va, unsigned long pa);
void memset(char *buf, char ch, int size);
void led_blink(void);
void delay(volatile int time)
{
for(; time > 0; time-- )
;
}
void main(void)
{
unsigned long c1_flags, ttb = 0x73000000;
volatile int *p = 0x52345678;
*p = 0x8;
init_ttb(ttb);
mmap(ttb, 0x12345678, 0x52345678);
mmap(ttb, 0xB10002E0, 0x110002E0);
c1_flags = 1 | (1 << 3) | ( 1 << 11) | (1 << 28);
__asm__ __volatile__ (
"mvn r0, #0 n"
"mcr p15, 0, r0, c3, c0, 0n"
"mcr p15, 0, %1, c2, c0, 0n" //configure ttb
"mrc p15, 0, r0, c1, c0, 0n"
"orr %0, r0, %0n"
"mcr p15, 0, %0, c1, c0, 0n" //enable mmu
:
: "r" (c1_flags), "r" (ttb)
: "r0"
);
p = 0x12345678;
uart_asm_putc('r');
uart_asm_putc('n');
uart_asm_putc('c');
uart_asm_putc('y');
uart_asm_putc('j');
uart_asm_putc(':');
uart_asm_putx(*p);
uart_asm_putc('r');
uart_asm_putc('n');
led_blink();
}
void init_ttb(unsigned long *ttb_base)
{
unsigned long va, pa;
memset(ttb_base, 0x00, 16 * 1024 );
for (va = 0x00000000; va < 0x10000000; va += 0x100000) { //Others
pa = va;
ttb_base[ va >> 20] = (pa & 0xfff00000) | 2;
}
for (va = 0x10000000; va < 0x14000000; va += 0x100000) { //SFR
pa = va;
ttb_base[ va >> 20] = (pa & 0xfff00000) | 2;
}
for (va = 0x40000000; va < 0x80000000; va += 0x100000) { //DRAM
pa = va;
ttb_base[ va >> 20] = (pa & 0xfff00000) | 2;
}
}
void mmap(unsigned long *ttb_base, unsigned long va, unsigned long pa)
{
ttb_base[ va >> 20] = (pa & 0xfff00000) | 2;
}
void memset(char *buf, char ch, int size)
{
int i;
for (i = 0; i < size; i ++)
buf[i] = ch;
}
void led_blink(void)
{
unsigned long tmp = 0;
int i = 0;
/*
* GPM4_0-GPM4_3 设置为输出功能
*/
tmp = GPM4CON;
tmp &= ~0xffff;
tmp |= 0x1111;
GPM4CON = tmp;
/*
* 实现流水灯
*/
while(1)
{
GPM4DAT = i;
if (++i == 16)
i = 0;
delay(9999999);
}
}
代码位置:https://github.com/cyj1988jyc/luoji4412/
上一篇:xynos4412裸机开发 —— A/D转换器
下一篇:tiny4412 裸机程序 八、重定位到DRAM及LCD实验
推荐阅读最新更新时间:2024-11-11 11:18
设计资源 培训 开发板 精华推荐
- 典型应用 TDA1521Q 2 x 12 W 高保真音频功率放大器非对称供电
- AM2G-2405SH30Z 5V 2 瓦 DC-DC 转换器的典型应用
- 【创意PCB】月亮网络时钟
- PixArt PMW3360DM-T2QU :光学鼠标传感器分线板
- TAR5S50U 点稳压器(低压差稳压器)的典型应用
- ZXCW15SWEVAL,基于 ZXCW8100 150W/8 欧姆数字放大器的评估板
- C880093_PIC单片机方案验证板-SOP-8封装
- LT3091HT7 负输出电流监视器的典型应用
- 黄淮学院立创杯设计大赛——#1004956A#——#贾起航#
- 使用 Richtek Technology Corporation 的 RT8011APQW 的参考设计