linux启动参数 mem,Linux启动参数及实现 __setup与early_param

#define __setup_param(str, unique_id, fn,

early) /

static char

__setup_str_##unique_id[] __initdata =

str; /

static

struct obs_kernel_param

__setup_##unique_id /

__attribute_used__ /

__attribute__((__section__(".init.setup"))) /

__attribute__((aligned((sizeof(long))))) /

= { __setup_str_##unique_id, fn, early }

#define __setup(str,

fn) /

__setup_param(str, fn, fn, 0)

#define early_param(str,

fn) /

__setup_param(str, fn, fn, 1)

__setup与early_param不同的是,early_param

宏注册的内核选项必须要在其他内核选项之前被处理。

在函数start_kernel中,parse_early_param处理early_param定义的参数,parse_args处理__setup定义的参数。

parse_early_param();

parse_args("Booting kernel", static_command_line,

__start___param,

__stop___param - __start___param,

&unknown_bootoption);

1,所有的系统启动参数都是由形如 static int __init foo(char *str);的函数来支持的

注:#define

__init __attribute__ ((__section__ (".init.text")))

申明所有的启动参数支持函数都放入.init.text段

2.1,用__setup宏来导出参数的支持函数

__setup("foo=" , foo);

展开后就是如下的形式

static char __setup_str_foo[] __initdata =

"foo="; static struct obs_kernel_param

__setup_foo __attribute_used__ __attribute__((__section__(".init.setup"))) __attribute__((aligned((sizeof(long))))) = { __setup_str_foo, foo, 0 };//"foo=",foo,0

也就是说,启动参数(函数指针)被封装到obs_kernel_param结构中,

所有的内核启动参数形成内核映像.init.setup段中的一个

obs_kernel_param数组

2.2用early_param宏来申明需要'早期'处理的启动参数,例如在

arch/i386/kernel/setup.c就有如下的申明:

early_param("mem", parse_mem);

展开后和__setup是一样的只是early参数不一样,因此会在do_early_param

中被处理

3,内核对启动参数的解析:下面函数历遍obs_kernel_param数组,调用

支持函数

static int __init do_early_param(char *param, char *val)

{

struct

obs_kernel_param *p;

for (p =

__setup_start; p < __setup_end; p++) {

if (p->early && strcmp(param, p->str) == 0) {

if (p->setup_func(val) != 0)

printk(KERN_WARNING

"Malformed early option '%s'/n", param);

}

}

return

0;

}

这个函数在parse_early_param中被调用,而parse_early_param在start_kernel

中被调用,parse_early_param之后的parse_args会调用下面函数

static int __init obsolete_checksetup(char *line)

{

struct

obs_kernel_param *p;

int

had_early_param = 0;

p =

__setup_start;

do {

int n = strlen(p->str);

if (!strncmp(line, p->str, n)) {

if (p->early) {

if (line[n] == '/0' || line[n] == '=')

had_early_param = 1;

} else if (!p->setup_func) {

printk(KERN_WARNING "Parameter %s is obsolete,"

" ignored/n", p->str);

return 1;

} else if (p->setup_func(line + n))//调用支持函数

return 1;

}

p++;

} while (p

< __setup_end);

return

had_early_param;

}

init/main.c中启动参数申明列表:

__setup("nosmp", nosmp);

__setup("maxcpus=", maxcpus);

__setup("reset_devices", set_reset_devices);

__setup("debug", debug_kernel);

__setup("quiet", quiet_kernel);

__setup("loglevel=", loglevel);

__setup("init=", init_setup);

__setup("rdinit=", rdinit_setup);

__setup("initcall_debug", initcall_debug_setup);

原文:http://blog.csdn.net/tommy_wxie/article/details/8041487

---------------------------------------------------------------------------------------------------------------------

在Linux-2.6.30中,有如下字符被__early_param定义:

./arch/arm/kernel/setup.c:__early_param("mem=", early_mem);

./arch/arm/mm/init.c:__early_param("initrd=", early_initrd);

./arch/arm/mm/mmu.c:__early_param("cachepolicy=",

early_cachepolicy);

./arch/arm/mm/mmu.c:__early_param("nocache", early_nocache);

./arch/arm/mm/mmu.c:__early_param("nowb",

early_nowrite);

./arch/arm/mm/mmu.c:__early_param("ecc=", early_ecc);

./arch/arm/mm/mmu.c:__early_param("vmalloc=", early_vmalloc);

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值