设备初始化函数宏

  • 名称描述
    __devinit用于标记初始化设备的函数,例如,对于PCI驱动程序,用于初始化的函数pci_driver->probe就是用此宏标识的。被其它由_devinit标记的函数调用的函数通常也由_devinit标记。
    __devexit用于标记设备卸载时被调用的函数。
    __devexit_p用于初始化由__devexit 标记的函数的指针。如果内核既支持模块也支持热拔插,则__devexit_p(fn)返回fn,否则返回NULL。可以参考“其它优化”一节
    __devinitdata用于标记函数使用的已初始化的数据,而这些函数兼顾设备初始化(如被_devinit标记),因此共享其属性。
    __devexitdata与__devinitdata类似但与__devexit关联匹配.
  • 内核初始化优化宏

    内核使用了大量不同的宏来标记具有不同作用的函数和数据结构。如宏__init、__devinit等。这些宏在include/linux/init.h头文件中定义。编译器通过这些宏可以把代码优化放到合适的内存位置,以减少内存占用和提高内核效率。

    下面是一些常用的宏:

    • __init,标记内核启动时使用的初始化代码,内核启动完成后不再需要。以此标记的代码位于.init.text内存区域。它的宏定义是这样的:

      #define _ _init    _ _attribute_ _ ((_ _section_ _ (".text.init")))
    • __exit,标记退出代码,对于非模块无效。

    • __initdata,标记内核启动时使用的初始化数据结构,内核启动完成后不再需要。以此标记的代码位于.init.data内存区域。

    • __devinit,标记设备初始化使用的代码。

    • __devinitdata,标记初始化设备数据结构的函数。

    • __devexit,标记移除设备时使用的代码。

    • xxx_initcall,一系列的初始化代码,按降序优先级排列。

                             初始化代码的内存结构


    _init_begin -------------------
    | .init.text | ---- __init
    |-------------------|
    | .init.data | ---- __initdata
    _setup_start |-------------------|
    | .init.setup | ---- __setup_param
    __initcall_start |-------------------|
    | .initcall1.init | ---- core_initcall
    |-------------------|
    | .initcall2.init | ---- postcore_initcall
    |-------------------|
    | .initcall3.init | ---- arch_initcall
    |-------------------|
    | .initcall4.init | ---- subsys_initcall
    |-------------------|
    | .initcall5.init | ---- fs_initcall
    |-------------------|
    | .initcall6.init | ---- device_initcall
    |-------------------|
    | .initcall7.init | ---- late_initcall
    __initcall_end |-------------------|
    | |
    | ... ... ... |
    | |
    __init_end -------------------

    初始化代码的特点是:在系统启动运行,且一旦运行后马上退出内存,不再占用内存。

    对于驱动程序模块来说,这些优化标记使用的情况如下:

    • 通过module_init()和module_exit()函数调用的函数就需要使用__init和__exit宏来标记。

    • pci_driver数据结构不需标记。

    • probe()和remove()函数应该使用__devinit和__devexit标记,且只能标记probe()和remove()

    • 如果remove()使用__devexit标记,则在pci_driver结构中要用__devexit_p(remove)来引用remove()函数。

    • 如果你不确定需不需要添加优化宏则不要添加。

  • 你可能感兴趣的:(数据结构,优化,Module,null,编译器)