OEMAddressTable只存在于X86和ARM架构的平台下,用来定义从4GB的虚拟地址到物理512MB存储空间的静态映射关系,它实质是一个结构变量。
typedef struct
{
ULONG ulVirtualAddress;
ULONG ulPhysicalAddress;
ULONG ulSizeInMegs;
} AddressTableStruct;
#define MEG(A) (((A - 1)>>20) + 1)
const AddressTableStruct OEMAddressTable[] =
{
{ SDRAM_VIRTUAL_MEMORY, //虚拟地址
PHYSICAL_ADDR_SDRAM_MAIN, //物理地址
MEG(SDRAM_MAIN_BLOCK_SIZE) //这段空间的大小,以M计
},
………………………
{
0,
0,
0
}
};
如例子所示,OEMAddressTable为一个结构数组,每项的第一个成员为虚拟地址,第二个成员为对应的物理地址,最后一个成员为该段空间的大小。这个数组的最后一项必须全部为0,以示整个数组的结束。内核启动时会读取这个数组的内容以初始化MMU页表。启用MMU以后OS内核段程序(如ISR)可以用这里的虚拟地址来访问设备。当然,OEMAddressTable中所用到的每个物理地址及虚拟地址都需要在头文件中定义。
比如X86的OEMAddressTable (platform/common/src/x86/common/startup/startup.asm):
_OEMAddressTable:
dd 80000000h, 0, 04000000h //格式为:虚拟地址,物理地址,大小
注意(1)这里的大小,X86下为4MB的倍数,ARM下为1MB的倍数。
(2)这里的映射是静态的cached,虚拟地址范围是0x8000000-0x9FFFFFFF,系统会自动添加uncached段的映射,从0xA000000-0xBFFFFFFF.
CONFIG.BIB文件分两个部分,我们且称之为段,MEMORY段和CONFIG段。MEMORY段定义虚拟内存的分片方法,CONFIG段定义它的一些属性。以下是一个CONFIG.BIB文件MEMORY段的例子:
来自platform/CEPC/files/config.bib:
MEMORY
; Name Start Size Type
; ------- -------- -------- ----
; 64 MB of RAM (note: AUTOSIZE will adjust boundary)
NK 80220000 009E0000 RAMIMAGE //系统镜像
RAM 80C00000 03400000 RAM // 应用程序与文件系统区域
DMA 80100000 00030000 RESERVED ; Native DMA reserved.
BOOTARGS 801FFF00 00000100 RESERVED ; Boot arguments
EDBG_DMA 80200000 00020000 RESERVED ; EDBG DMA buffer