(1) 这是TI平台的OMAP4430的机器实现核心文件/arch/arm/mach-omap2/board-omap3pandora.c 中的一段代码
MACHINE_START(OMAP3_PANDORA, "Pandora Handheld Console")
.phys_io = 0x48000000,
.io_pg_offst = ((0xd8000000) >> 18) & 0xfffc,
.boot_params = 0x80000100,
.map_io = omap3pandora_map_io,
.init_irq = omap3pandora_init_irq,
.init_machine = omap3pandora_init,
.timer = &omap_timer,
MACHINE_END
这里赋值了影射io、初始化irq、初始化机器等函数指针,其中的omap3pandora_map_io()、(影射io空间)
omap3pandora_init_irq()、(初始化板级的中断系统)
omap3pandora_init() 、(OMAPZOOM平台的板级初始化函数,如flash、usb、串口、camera、wifi等初始化)
都是在此文件中实现的初始化函数
omap_timer是为当前机器实现的定时器
Io地址影射
如果想叫处理器知道外围设备上的存储空间,首先要建立虚拟地址到总线地址的影射,总线地址到物里地址转换是硬件完成的。外围设备的存储范围是4G的顶端,而应设的虚拟地址时在896M以上,就是high_mm开始的地方,页表示在init_mm,这是专门为内核使用的页表,不与一般进程的页表在一起,所以在唤出时候,就找不到它的存在,所以内核页表示不会唤出的。同时虚拟地址也是在4G最后的128M里面,由一个虚拟存储区管理。
Offset 要应设的物理地址
Size 大小范围
返回影射成功的虚拟地址
Void* iormap(unsigned long offset ,unsigned longsize)
1 是对offset的一些测试,不能是1M下的应为这些已经建立影射,不能重复影射。
2 申请一个vm_struct结构,当然这个东西是在slab管理的,
3
申请完就获得了虚拟地址,就要在init_mm的页表里建立影射了。加一句,正是这里,所以有时候会出现页面异常,所以要同步
(2)(这是转载的s3c2410的平台机器信息)
MACHINE_START(VR1000, "Thorcom-VR1000")
/* Maintainer: Ben Dooks <
[email protected]
> */
.phys_ram = S3C2410_SDRAM_PA,
.phys_io = S3C2410_PA_UART,
.io_pg_offst = (((u32)S3C24XX_VA_UART) >> 18) & 0xfffc,
.boot_params = S3C2410_SDRAM_PA + 0x100,
.map_io = vr1000_map_io,
.init_irq = s3c24xx_init_irq,
.timer = &s3c24xx_timer,
MACHINE_END
这两个是宏定义 原型为
#define MACHINE_START(_type,_name) /
const struct machine_desc __mach_desc_##_type /
__attribute__((__section__(".arch.info.init"))) = { /
.nr = MACH_TYPE_##_type, /
.name = _name,
#define MACHINE_END /
};
struct machine_desc 的原型为:
struct machine_desc {
/*
* Note! The first five elements are used
* by assembler code in head-armv.S
*/
unsigned int nr; /* architecture number */
unsigned int phys_ram; /* start of physical ram */
unsigned int phys_io; /* start of physical io */
unsigned int io_pg_offst; /* byte offset for io
* page tabe entry */
const char *name; /* architecture name */
unsigned long boot_params; /* tagged list */
unsigned int video_start; /* start of video RAM */
unsigned int video_end; /* end of video RAM */
unsigned int reserve_lp0 :1; /* never has lp0 */
unsigned int reserve_lp1 :1; /* never has lp1 */
unsigned int reserve_lp2 :1; /* never has lp2 */
unsigned int soft_reboot :1; /* soft reboot */
void (*fixup)(struct machine_desc *,
struct tag *, char **,
struct meminfo *);
void (*map_io)(void);/* IO mapping function */
void (*init_irq)(void);
struct sys_timer *timer; /* system tick timer */
void (*init_machine)(void);
};
MACH_TYPE_SMDK2410定义在arch/include/asm-arm/mach-types.h内,值为193.
/* arch/include/asm-arm/mach-types.h */
#define MACH_TYPE_SMDK2410 193
这个值是机器的类型值,编译时由arch/arm/tool/mach-types里面定义的数据生成的。
/* arch/arm/tool/mach-types */
smdk2410 ARCH_SMDK2410 SMDK2410 193
由上发现,MACHINE_START主要是定义了"struct machine_desc"的类型,放在 section(".arch.info.init"),是初始化数据,Kernel 起来之后将被丢弃。
各个成员函数在不同时期被调用:
1. .init_machine 在 arch/arm/kernel/setup.c 中被 customize_machine 调用,放在 arch_initcall() 段里面,会自动按顺序被调用。
2. init_irq在start_kernel() --> init_IRQ() --> init_arch_irq() 被调用
3. map_io 在 setup_arch() --> paging_init() --> devicemaps_init()被调用
其他主要都在 setup_arch() 中用到