优质服务商推荐更多服务商>

你了解linux驱动的入口?

4103

 以module_init(Demo_init);为例

    你了解linux驱动的入口?_设计制作_可编程逻辑  

定义文件:

include\     linux   \init.h

#define module_init(x) __initcall(x);

#define __initcall(fn) device_initcall(fn)

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

#define __define_initcall(level,fn,id) \
sta     ti   c initcall_t __initcall_##fn##id __attribu     te   _used__ \
__attribute__((__sec  TI on__(".initcall" level ".init"))) = fn

展开为

sta  TI c initcall_t __initcall_Demo_init6 __attribute_used__ __attribute__((__sec  TI on__(".initcall6.init"))) = Demo_init;

typedef int (*initcall_t)(void);

这里

typedef int (init_fnc_t)  (void); 定义一种函数类型
typedef int (*init_fnc_t) (void); 定义一种类型的函数指针

所以展开的宏定义就是定义名为__initcall_Demo_init6的函数指针

属性有两个:

1.

在gcc 3.4之前的编译器被展开成__attribute__((unused))来禁止编译器弹出有关函数没有被用到的的警告信息

在gcc 3.4之后被展开成__attribute__((used))功能一样

2.加载到段.initcall6.init,其地址为Demo_init的地址

段的分布顺序在链接脚本中有

编译内核后,会有vmlinux.lds的打印信息,里面有各段位置

__initcall_start = .;

*(.initcallearly.init) __early_initcall_end = .;

*(.initcall0.init)

*(.initcall0s.init)

*(.initcall1.init)

*(.initcall1s.init)

*(.initcall2.init)

*(.initcall2s.init)

*(.initcall3.init)

*(.initcall3s.init)

*(.initcall4.init)

*(.initcall4s.init)

*(.initcall5.init)

*(.initcall5s.init)

*(.initcallrootfs.init)

*(.initcall6.init)

*(.initcall6s.init)

*(.initcall7.init)

*(.initcall7s.init)

__initcall_end = .;

当insmod的时候,内核从initcall6.init段中读取到驱动入口地址,然后跳转到该地址去执行入口函数,

一般入口函数会进行注册驱动,例如

register_chrdev(unsigned int major, const char * name, const struct file_opera  TI ons * fops)

    usb   _register(struct usb_driver * driver)

s     pi   _register_driver(struct spi_driver * sdrv)

等等注册函数,再依次调用相应设备结构体中的ioctl或者直接调用file_operations结构体



特别声明:本文仅供交流学习 , 版权归属原作者,并不代表蚂蚜网赞同其观点和对其真实性负责。若文章无意侵犯到您的知识产权,损害了您的利益,烦请与我们联系vmaya_gz@126.com,我们将在24小时内进行修改或删除。

相关推荐: