owner THIS MODULE为什么加 点 的原因

               

.owner = THIS_MODULE为什么加“点”的原因

多次看书、编程时又看到了对结构体这种定义的方法,如:
struct file_operations scull_fops = {
.owner = THIS_MODULE,
.llseek = scull_llseek,
.read = scull_read,
.write = scull_write,
.ioctl = scull_ioctl,
.open = scull_open,
.release = scull_release, };
前边一直没去真正查找倒底是怎么回事,当时只是想或许是GNU C的扩展语法,因为才谭浩强的书上也没有提到过,看过了一些C语言书也没有提到过。今天一查,原来这个是C99标准,这个目前也是最新的标准,之前我也清楚这件事,但是没意识到这是一个差别。网上找到一篇文章,写得很详细,copy过来了。原文地址:http://www.diybl.com/course/3_pr ... 2008713/133042.html
在阅读GNU/Linux内核代码时,我 们会遇到一种特殊的结构初始化方式。该方式是某些C教材(如谭二版、K&R二版)中没有介绍过的。这种方式称为指定初始化(designated initializer)。下面我们看一个例子,Linux-2.6.x/drivers/usb/storage/usb.c中有这样一个结构体初始化 项目:
static struct usb_driver usb_storage_driver = {
.owner = THIS_MODULE,
.name = \"usb-storage\",
.probe = storage_probe,
.disconnect = storage_disconnect,
.id_table = storage_usb_ids, };
乍一看,这与我们之前学过的结构体初始化差距甚远。其实这就是前面所说的指定初始化在Linux设备驱动程序中的一个应用,它源自ISO C99标准。以下我摘录了C Primer Plus第五版中相关章节的内容,从而就可以很好的理解2.6版内核采用这种方式的优势就在于由此初始化不必严格按照定义时的顺序。这带来了极大的灵活 性,其更大的益处还有待大家在开发中结合自身的应用慢慢体会。 已知一个结构,定义如下
struct book {
char title[MAXTITL];
char author[MAXAUTL];
float value; };
C99支持结构的指定初始化项目,其语法与数组的指定初始化项目近似。只是,结构的指定初始化项目使用点运算符和成员名(而不是方括号和索引值)来标识具体的元素。例如,只初始化book结构的成员value,可以这样做:
struct book surprise = { .value = 10.99 };
可以按照任意的顺序使用指定初始化项目:
struct book gift = {
.value = 25.99,
.author = \"James Broadfool\",
.title = \"Rue for the Toad\"};
正像数组一样,跟在一个指定初始化项目之后的常规初始化项目为跟在指定成员后的成员提供了初始值。

           
在Linux内核驱动开发中,您提供的代码片段是驱动程序结构体(通常是一个`platform_driver`结构体)的初始化部分。这个结构体用于注册一个平台驱动程序,并在没有设备树(Device Tree)的情况下,您仍然可以实现它。但是,需要注意的是,在现代的Linux内核版本中,推荐使用设备树来描述硬件信息。 在没有设备树的情况下,您需要在驱动代码中直接包含和初始化设备资源。通常,这意味着您需要静态定义一些数据结构来描述您的硬件设备,并在驱动初始化时直接注册这些设备。下面是一个基本的实现方法: 1. 定义设备和驱动程序的数据结构。 2. 在驱动程序的初始化函数中注册设备和驱动程序。 3. 在驱动程序的退出函数中注销设备和驱动程序。 这里是一个简化的例子,展示如何在没有设备树的情况下注册一个平台驱动程序: ```c #include <linux/module.h> #include <linux/platform_device.h> // 定义驱动程序结构体 static struct platform_driver my_driver = { .driver = { .name = "my_driver", .owner = THIS_MODULE, }, .probe = my_probe, // 驱动的探测函数 .remove = my_remove, // 驱动的移除函数 }; // 驱动的探测函数 static int my_probe(struct platform_device *pdev) { // 设备的初始化代码 return 0; } // 驱动的移除函数 static int my_remove(struct platform_device *pdev) { // 清理资源代码 return 0; } static int __init my_driver_init(void) { // 注册平台驱动程序 return platform_driver_register(&my_driver); } static void __exit my_driver_exit(void) { // 注销平台驱动程序 platform_driver_unregister(&my_driver); } module_init(my_driver_init); module_exit(my_driver_exit); MODULE_LICENSE("GPL"); MODULE_AUTHOR("Your Name"); MODULE_DESCRIPTION("A simple example Linux module."); ``` 在这个例子中,`probe`函数是当驱动程序与设备匹配时调用的函数,用于初始化设备;`remove`函数则是当设备被移除时调用的函数,用于清理资源。`platform_driver_register`函数用于注册驱动程序,而`platform_driver_unregister`用于注销驱动程序。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值