內核在啟動時可以傳遞一個字符串命令行,來控制內核啟動的過程,例如:
"console=ttyS2,115200 mem=64M@0xA0000000"
這裡指定了控制台是串口2,波特率是115200,內存大小是64M,物理基地址是0xA0000000。
另外我們可以在內核中定義一些全局變量,使用這些全局變量控制內核的配置,例如usb驅動中定義了
static int nousb; /* Disable USB when built into kernel image */
這個變量為1,則整個usb驅動不初始化,如果想將其置1,可在字符串命令行中添加nousb=1。
在操作該變量之前,還要讓系統知道該變量,方法是:
__module_param_call("",nousb,param_set_bool,param_get_bool,&nousb,0444); __module_param_call這個宏定義在kernel\include\linux\moduleparam.h
原型如下:
#define __module_param_call(prefix, name, set, get, arg, perm) \ static char __param_str_##name[] = prefix #name; \ static struct kernel_param const __param_##name \ __attribute_used__ \ __attribute__ ((unused,__section__ ("__param"),aligned(sizeof(void *)))) \ = { __param_str_##name, perm, set, get, arg }
它定義了一個kernel_param類型的變量,這個變量被放到了段__param,
kernel_param結構體的定義是:
struct kernel_param { const char *name; unsigned int perm; param_set_fn set; param_get_fn get; void *arg; }; __param這個段的聲明有些平台是在arch/../../vmlinux.lds.S,而大多數平台是放到 kernel\include\asm-generic\vmlinux.lds.h中,定義如下: __param : AT(ADDR(__param) - LOAD_OFFSET) { \ VMLINUX_SYMBOL(__start___param) = .; \ *(__param) \ VMLINUX_SYMBOL(__stop___param) = .; \ }
內核啟動時就會對字符串命令進行解析,在kernel\init\main.c中,內核啟動函數start_kernel中
對外部數組進行了聲明:
extern struct kernel_param __start___param[], __stop___param[];
然後調用函數parse_args對數組進行解析:
parse_args("Booting kernel", command_line, __start___param, __stop___param - __start___param, &unknown_bootoption);
其中command_line就是要解析的字符串命令行,unknown_bootoption是函數指針,它用來獲取指定參數的=右邊的值。
parse_args就會在數組中找到和nousb名稱一樣的kernel_param變量,並調用它的set函數對其進行付值。