Linux内核设备驱动之内核中链表的使用笔记整理
作者:Engineer-Bruce_Yang 发布时间:2021-09-15 10:24:32
/********************
* 内核中链表的应用
********************/
(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.静态
如果在编译时静态创建链表,并且直接引用它,如下:
struct my_struct mine={
.lost = LIST_HEAD_INIT(mine.list);
.dog = 0,
.cat = NULL
};
//或
static LIST_HEAD(fox);
/*等于struct list_head fox = LIST_HEAD_INIT(fox); */
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);
ptr: list_head指针
type_of_struct: 包含ptr的结构体类型
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);
来源:https://blog.csdn.net/morixinguan/article/details/79706265


猜你喜欢
- 怎么删除docker里建立的容器1、首先使用docker -s -a命令列出所有容器# docker ps -aCONTAINER ID &
- 从刚开始做站开始自己就有一个目标,那就是绝对不能做垃圾站,但是回首一个月以来做站的历程,发现自己的网站确实已经变得垃圾起来,没有任何的新意,
- 今天和妈妈、大福一起去逛街,5小时。大福买了粉色的T恤,我买了粉色的凉鞋,妈妈买了粉色的套装。我们还买了很多吃的,每个人都知道另外的人喜欢吃
- 一、什么是Shiro Apache Shiro是一个强大易用的Java安全框架,提供了认证、授权、加密和会话管理等功能: 
- 如何选择CMS?这里打个问号,说明这篇文章不是来解决问题的,是来发问的。建站难不?难,细数一下,首先要学会这些东西。1、域名选择;2、空间选
- Windows操作系统的IIS是大家最常用的Web服务器之一。IIS功能强大、简单易用,但也容易受到恶意攻击,它的安全性一直是大家谈论的焦点
- 概述使用docker已经有一年多了,最近意识到,我在快速编排服务的时候,shell脚本里用到的git还是原生的于是打算也将git容器化,在d
- docker attach命令docker attach [options] 容器会连接到正在运行的容器,然后将容器的标准输入、输出和错误流
- 你的计算机是否适合安装Windows 2000在开始安装Windows 2000之前,为保证安装的顺利和成功,必须保证硬件符合下列最低的需求
- 距离ubuntu最新版发布已经差不多半年了,博主近来对linux系统有了兴趣,奈何资金不足无法购置一台新机来安装ubuntu。所以想到了虚拟
- 刚刚无意中在统计中发现有一个来自法国的访问者从http://editors.dmoz.org/editors/editunrev2.cgi这
- 1 BlueHostsanp用的第一个国外主机,Cpanel后台,支持SSH,性能还是挺稳定的,毕竟是一家老牌的主机商了。但是会有大概500
- 常常在些站长论坛看到这样的求助帖子“为什么我的站是原创文章搜索收录才几页啊?这是什么原因呀?哪位高手帮我分析分析啊!”其实造成网站收录少的原
- phpcms实现站内搜索自动对应频道的代码:<form method="get"
- SEO的未来发展趋势互联网是一个庞大的信息和数据来源,大部分的互联网用户依靠搜索引擎找资料,组织网上信息是一个庞大的任务,搜索引擎优化目的就
- 软件下载下载软件链接链接: https://pan.baidu.com/s/1sfR8qGNEzuzMLq8nr9O-HQ 提取码: 5mk
- 大家先来看看问题描述:新建虚拟机的列表中没有64位系统选项,如何解决?操作系统:Windows 7 64位;虚拟机:Oracle VM Vi
- 什么是baiduspider?baiduspider是百度搜索引擎的一个自动程序。它的作用是访问互联网上的html网页,建立索引数据库,使用
- 如何解决VMWare中Ubuntu重启后无法上网,具体如下原因:没有设置静态的DNS,导致系统每次重启都会把先前设置的DNS删除。重启后无法
- VMware Server提供了一个免费的并且比较容易进入的服务器虚拟化方法,不过部署它的几个方面——尤其是配置,可能很棘手。即使