linux的initcall机制,Linux 的 initcall函数

66b52468c121889b900d4956032f1009.png

8种机械键盘轴体对比

本人程序员,要买一个写代码的键盘,请问红轴和茶轴怎么选?

kernel启动调用流程:

start_kernel -> rest_init -> kernel_thread -> init -> do_basic_setup -> do_initcalls

module_init/* initcalls are now grouped by functionality into separate

* subsections. Ordering inside the subsections is determined

* by link order.

* For backwards compatibility, initcall() puts the call in

* the device init subsection.

*

* The `id' arg to __define_initcall() is needed so that multiple initcalls

* can point at the same handler without causing duplicate-symbol build errors.

*/

#define __define_initcall(level,fn,id)

static initcall_t __initcall_##fn##id __used

__attribute__((__section__(".initcall" level ".init"))) = fn

#define core_initcall(fn) __define_initcall("1",fn,1)

#define postcore_initcall(fn) __define_initcall("2",fn,2)

#define arch_initcall(fn) __define_initcall("3",fn,3)

#define subsys_initcall(fn) __define_initcall("4",fn,4)

#define fs_initcall(fn) __define_initcall("5",fn,5)

#define rootfs_initcall(fn) __define_initcall("rootfs",fn,rootfs)

#define device_initcall(fn) __define_initcall("6",fn,6)

#define late_initcall(fn) __define_initcall("7",fn,7)

#define __initcall(fn) device_initcall(fn)

/**

* module_init() - driver initialization entry point

* @x: function to be run at kernel boot time or module insertion

*

* module_init() will either be called during do_initcalls() (if

* builtin) or at module insertion time (if a module). There can only

* be one per module.

*/

#define module_init(x) __initcall(x);

可以发现这些 *_initcall(fn) 最终都是通过 __define_initcall(level,fn) 宏定义生成的。#define __define_initcall(level,fn,id)

static initcall_t __initcall_##fn##id __used

__attribute__((__section__(".initcall" level ".init"))) = fn

定义一个 initcall_t 型的初始化函数,函数存放在 .initcall”level”.init section内。

.initcall”level”.init section定义在vmlinux.lds内。

__initcall_start 和 __initcall_end 定义在vmlinux.lds内,表示 initcall section 的起始和结束地址。__initcall_start = .;

*(.initcall1.init)

*(.initcall2.init)

*(.initcall3.init)

*(.initcall4.init)

*(.initcall5.init)

*(.initcallrootfs.init)

*(.initcall6.init)

*(.initcall7.init)

__initcall_end = .;

正好包括了上面 init.h 里定义的从 core_initcall 到 late_initcall 等7个level等级的 .initcall”level”.init section.

因此通过不同的 *_initcall 声明的函数指针最终都会存放不同level等级的 .initcall”level”.init section 内。

这些不同level的section按level等级高低依次存放。

do_initcalls

kernel启动调用流程:

start_kernel -> rest_init -> kernel_thread -> init -> do_basic_setup -> do_initcallsstatic void noinline __init_refok rest_init(void)

__releases(kernel_lock)

{

int pid;

kernel_thread(kernel_init, NULL, CLONE_FS | CLONE_SIGHAND);

numa_default_policy();

pid = kernel_thread(kthreadd, NULL, CLONE_FS | CLONE_FILES);

kthreadd_task = find_task_by_pid_ns(pid, &init_pid_ns);

unlock_kernel();

/*

* The boot idle thread must execute schedule()

* at least once to get things moving:

*/

init_idle_bootup_task(current);

preempt_enable_no_resched();

schedule();

preempt_disable();

/* Call into cpu_idle with preempt disabled */

cpu_idle();

}

通过 kernel_thread 创建一个内核线程执行init函数。

其实这里创建的即Linux的1号进程(init进程), 为linux中所有其他进程的父进程。static void __init do_initcalls(void)

{

initcall_t *call;

for (call = __early_initcall_end; call < __initcall_end; call++)

do_one_initcall(*call);

/* Make sure there is no pending stuff from the initcall sequence */

flush_scheduled_work();

}

__initcall_start 和 __initcall_end 定义在 vmlinux.lds 内,表示 initcall section 的起始和结束地址。

依次循环调用预先存储在 initcall section 内的所有各个级别的初始化函数。

tips

rest_init 函数原型:static void noinline __init_refok rest_init(void)

__releases(kernel_lock)

其中 __releases(kernel_lock) 用于kernel代码静态检测

Sparse 诞生于 2004 年, 是由linux之父开发的, 目的就是提供一个静态检查代码的工具, 从而减少linux内核的隐患.

内核代码中还有一个简略的关于 Sparse的说明文件: Documentation/sparse.txt

Sparse通过 gcc 的扩展属性 __attribute__ 以及自己定义的 __context__ 来对代码进行静态检查.__acquires(x) __attribute__((context(x, 0, 1))) 参数x 在执行前引用计数必须是0,执行后,引用计数必须为1

__releases(x) __attribute__((context(x, 1, 0))) 与 __acquires(x) 相反

__acquire(x) __context__(x, 1) 参数x 的引用计数 + 1

__release(x) __context__(x, -1) 与 __acquire(x) 相反

其中 __acquires(x) 和 __releases(x), __acquire(x) 和 __release(x) 必须配对使用, 否则 Sparse 会给出警告void __lockfunc _raw_spin_lock(raw_spinlock_t *lock) __acquires(lock);

void __lockfunc _raw_spin_unlock(raw_spinlock_t *lock) __releases(lock);

# define __acquires(x) __attribute__((context(x,0,1)))

# define __releases(x) __attribute__((context(x,1,0)))

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值