时间:2021-05-19
/******************** * 内核中链表的应用 ********************/
(1)介绍
在Linux内核中使用了大量的链表结构来组织数据,包括设备列表以及各种功能模块中的数据组织。这些链表大多采用在include/linux/list.h实现的一个相当精彩的链表数据结构。
链表数据结构的定义很简单:
struct list_head { struct list_head *next, *prev;};list_head结构包含两个指向list_head结构的指针prev和next,内核的数据结构通常组织成双循环链表。
和以前介绍的双链表结构模型不同,这里的list_head没有数据域。在Linux内核链表中,不是在链表结构中包含数据,而是在数据结构中包含链表节点。如:
struct my_struct{ struct list_head list; unsigned long dog; void *cat;};linux中的链表没有固定的表头,从任何元素开始访问都可以。遍历链表仅仅需要从某个节点开始,沿指针访问下一个节点,直到又重新回到最初这个节点就可以了。每个独立的节点都可以被称作是链表头。
(2)链表的初始化
a.静态
如果在编译时静态创建链表,并且直接引用它,如下:
b.动态
struct my_struct *p;p = kmalloc(GFP_KERNEL, sizeof(my_struct));p->dog = 0;p->cat = NULL;INIT_LIST_HEAD(&p->list);(3)操作链表
内核提供了一组函数来操作链表。
注意!这些函数都使用一个或多个list_head结构体指针作参数。定义在<linux/list.h>
a.增加节点
list_add(struct list_head *new, struct list_head *head);//向指定链表的head节点后面插入new节点b.把节点增加到链表尾
list_add_tail(struct list_head *new, struct list_head *head);//向指定链表的head节点前面插入new节点c.从链表删除一个节点
list_del(struct list_head *entry);//将entry从链表中移走d.把节点从一个链表移到另一个链表
list_move(struct list_head *list, struct list_head *head);从一个链表中摘除list项,然后将其插入head的后面
e.list_empty(struct list_head *head);链表为空返回非0值,否则返回0
f.合并链表
list_splice(struct list_head *list, struct list_head *head);//注意!新的链表不包括list节点(4)遍历链表
链表本身不重要,访问到那个包含链表的结构体才重要
a.从链表指针获得包含该链表的结构体的指针
list_entry(struct list_head *ptr, type_of_struct, field_name);如:
my_struct *p = (list_head *ptr, my_struct, list);b.遍历链表
list_for_each(struct list_head *cursor, struct list_head *list);//常常和list_entry配套使用//注意!用list_for_each遍历时,不包括头节点c.遍历的同时获得大结构体指针
list_for_each_entry(type *cursor, struct list_head *list, member);d.遍历链表的同时释放每个被遍历到的节点
list_for_each_entry_safe(type *cursor, type *tmp; struct list_head *list, member);总结
以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,谢谢大家对的支持。如果你想了解更多相关内容请查看下面相关链接
声明:本页内容来源网络,仅供用户参考;我单位不保证亦不表示资料全面及准确无误,也不保证亦不表示这些资料为最新信息,如因任何原因,本网内容或者用户因倚赖本网内容造成任何损失或损害,我单位将不会负任何法律责任。如涉及版权问题,请提交至online#300.cn邮箱联系删除。
提问:我想要知道Linux系统中内核内置的模块,以及每个模块有哪些参数。有什么方法可以得到内置模块和设备驱动的列表,以及它们的详细信息呢?现代Linux内核正在
1.Linux内核驱动模块机制静态加载,把驱动模块编进内核,在内核启动时加载动态加载,把驱动模块编为ko,在内核启动后,需要用时加载2.编写内核驱动#inclu
在前一篇文章中,我们介绍了如何在Ubuntu上为Android系统编写Linux内核驱动程序。在这个名为hello的Linux内核驱动程序中,创建三个不同的文件
Android是基于Linux内核的操作系统,但是,运行在Android设备上的内核其实与Google选择的LTS版本Linux内核有很大不同。 在到达每
当你在编译一个设备驱动模块时,你需要在系统中安装内核头文件。内核头文件同样在你编译与内核直接链接的用户空间程序时需要。当你在这些情况下安装内核头文件时,你必须确