【内核数据结构】 内核链表分析

一、简介:



Linux中的链表使用两个指针,可以方便的构成双向链表,实际上,通常它都组织成双向循环链表,不同于数据结构书上的链表,这里的节点只有链表指针,没有链表的数据,下边我将对内核中使用的 include/linux/list.h 进行函数说明和生动的图形解释。

二、函数:


我们先来看看

1. 链表数据结构 list_head 的定义:


[cpp] view plain copy

print?

  1. struct list_head {
  2. struct list_head *next, *prev;
  3. };

【1】只有前后节点指针,没有数据

2. 声明和初始化:有两种方法



①声明的时候初始化一个链表 LIST_HEAD 宏

[cpp] view plain copy

print?

  1. #define LIST_HEAD_INIT(name) { &(name), &(name) } // 链表的pre和next指针都指向了节点自己的首地址
  2. #define LIST_HEAD(name) \
  3. struct list_head name = LIST_HEAD_INIT(name)

②运行时初始化链表 INIT_LIST_HEAD 函数:

[cpp] view plain copy

print?

  1. static inline void INIT_LIST_HEAD(struct list_head *list)
  2. {
  3. list->next = list;
  4. list->prev = list;
  5. }

注意:

此处说的声明的时候简单的理解为不在函数内部,而运行时指的就是在函数内部了

图形:

3. 插入/删除/合并


a) 插入


对链表的插入操作有两种:

在表头插入 list_add函数 和

在表尾插入 list_add_tail函数

[cpp] view plain copy

print?

  1. static inline void list_add(struct list_head *new, struct list_head *head) // new:要添加的新的链表的首地址,head:链表的中的位置
  2. {
  3. __list_add(new, head, head->next);
  4. }
  5. static inline void list_add_tail(struct list_head *new, struct list_head *head)
  6. {
  7. __list_add(new, head->prev, head);
  8. }

可以看到他们调用了相同的 __list_add 函数:

[cpp] view plain copy

print?

  1. static inline void __list_add(struct list_head *new,
  2. struct list_head *prev,
  3. struct list_head *next)
  4. {
  5. next->prev = new;
  6. new->next = next;
  7. new->prev = prev;
  8. prev->next = new;
  9. }

【1】对于这个函数,他是将list_add和list_add_tail的共性的部分抽离了出来,给我们分析很大的障碍,我们只分析 list_add 和 list_add_tail 函数

图形:

  • list_add 部分:

网络上的一张图更全面的展示了在使用中的链表的结构:

  • list_add_tail 部分:

画图总结:

【1】上边图形的画法中,要前两步划在外边沿

【2】对list链表的头和尾的快速记忆的方法,我们可以看待内核中的链表为 向右行驶的贪吃蛇

b) 删除



对链表的删除操作函数有两种:

list_del函数

list_del_init函数

[cpp] view plain copy

print?

  1. static inline void list_del(struct list_head *entry) // entry:要删除的链表的首地址
  2. {
  3. __list_del(entry->prev, entry->next); // 这不就是 __list_del_entry(entry) 吗!!
  4. entry->next = LIST_POISON1;
  5. entry->prev = LIST_POISON2;
  6. }
  7. static inline void list_del_init(struct list_head *entry)
  8. {
  9. __list_del_entry(entry);
  10. INIT_LIST_HEAD(entry); // 运行中初始化链表节点
  11. }
  12. static inline void __list_del_entry(struct list_head *entry)
  13. {
  14. __list_del(entry->prev, entry->next);
  15. }

【1】list_del函数中entry的next和prev指针指向了LIST_POISON1和LIST_POISON2位置,对他们进行访问都将引起页故障,保护不在链表中的节点项不可访问

他们调用了相同的 __list_del 函数:

[cpp] view plain copy

print?

  1. static inline void __list_del(struct list_head * prev, struct list_head * next)
  2. {
  3. next->prev = prev;
  4. prev->next = next;
  5. }

图形:

我们来删除有3个元素的链表的中间的一个:list_del(&new)

list_del_init 函数不再画图,唯一的不同是把删除下来的图的next和prev指针指向了自己的首地址

c) 替换



对链表的替换操作有两个:

list_replace函数

list_replace_init函数

[cpp] view plain copy

print?

  1. static inline void list_replace(struct list_head *old,
  2. struct list_head *new)
  3. {
  4. new->next = old->next;
  5. new->next->prev = new;
  6. new->prev = old->prev;
  7. new->prev->next = new;
  8. }
  9. static inline void list_replace_init(struct list_head *old,
  10. struct list_head *new)
  11. {
  12. list_replace(old, new);
  13. INIT_LIST_HEAD(old);
  14. }

图形:

list_replace_init函数的图形此处也不再画

d) 搬移



搬移的含义是将原本属于一个链表的节点移动到另一个链表的操作,有两个函数:

list_move函数

list_move_tail函数

[cpp] view plain copy

print?

  1. /**
  2. * list_move - 把从一个链表上删除的节点添加到另外的一个链表的头部
  3. * @list: 我们要移动的节点
  4. * @head: 要移动到的另外的一个链表头
  5. */
  6. static inline void list_move(struct list_head *list, struct list_head *head)
  7. {
  8. __list_del_entry(list);
  9. list_add(list, head);
  10. }
  11. /**
  12. * list_move_tail - 添加到另外的一个链表的尾部
  13. * @list: the entry to move
  14. * @head: the head that will follow our entry
  15. */
  16. static inline void list_move_tail(struct list_head *list,
  17. struct list_head *head)
  18. {
  19. __list_del_entry(list);
  20. list_add_tail(list, head);
  21. }

图形:

e) 合并



合并在这里的意思就是合并了,是将两个独立的链表合并成为一个链表,合并的时候根据合并的位置的不同可以分为:

合并到头部的 list_splice函数

合并到尾部的 list_splice_tail函数:(这两个函数有推荐使用的函数)

[cpp] view plain copy

print?

  1. /**
  2. * list_splice - join two lists, this is designed for stacks
  3. * @list: the new list to add.
  4. * @head: the place to add it in the first list.
  5. */
  6. static inline void list_splice(const struct list_head *list,
  7. struct list_head *head)
  8. {
  9. if (!list_empty(list))
  10. __list_splice(list, head, head->next);
  11. }
  12. /**
  13. * list_splice_tail - join two lists, each list being a queue
  14. * @list: the new list to add.
  15. * @head: the place to add it in the first list.
  16. */
  17. static inline void list_splice_tail(struct list_head *list,
  18. struct list_head *head)
  19. {
  20. if (!list_empty(list))
  21. __list_splice(list, head->prev, head);
  22. }
  23. static inline void list_splice_init(struct list_head *list, // 推荐使用,防止混乱
  24. struct list_head *head)
  25. {
  26. if (!list_empty(list)) {
  27. __list_splice(list, head, head->next);
  28. INIT_LIST_HEAD(list);                   <--- 跟list_splice唯一的不同
  29. }
  30. }
  31. static inline void list_splice_tail_init(struct list_head *list, // 推荐使用,防止混乱
  32. struct list_head *head)
  33. {
  34. if (!list_empty(list)) {
  35. __list_splice(list, head->prev, head);
  36. INIT_LIST_HEAD(list);                   <--- 跟list_splice_tail_init唯一的不同
  37. }
  38. }
  39. static inline void __list_splice(const struct list_head *list,
  40. struct list_head *prev,
  41. struct list_head *next)
  42. {
  43. struct list_head *first = list->next;
  44. struct list_head *last = list->prev;
  45. first->prev = prev;
  46. prev->next = first;
  47. last->next = next;
  48. next->prev = last;
  49. }

图形:

这张图虽然画出来了,比起看程序,虽然好点,但是理解起来还是有很大的问题,此处就借鉴别人的一张图来说明这个list_splice函数实现了什么:

链表合并list_splice(&list1,&list2) (此图片引自:http://www.ibm.com/developerworks/cn/linux/kernel/l-chain/

对于这张图的说明如下:

假设当前有两个链表,表头分别是list1和list2(都是struct list_head变量),当调用list_splice(&list1,&list2)时,只要list1非空,list1链表的内容将被挂接在list2链表上,位于list2和list2.next(原list2表的第一个节点)之间。新list2链表将以原list1表的第一个节点为首节点,而尾节点不变。

4. 找到链表中的数据



前边提到的函数都是操作的链表节点的入口,但是对于我们真正有意义的是节点上的数据,链表的头上没有数据,其他的节点上都是带有数据的。如何从一个链表节点的入口得到节点的数据呢?要用到以下的函数:

list_entry函数

[cpp] view plain copy

print?

  1. /**
  2. * list_entry - 获得含链表入口的结构体首地址
  3. * @ptr:    member的首地址
  4. * @type:   容器的类型
  5. * @member: 要得到他的容器的某个成员
  6. */
  7. #define list_entry(ptr, type, member) \
  8. container_of(ptr, type, member)
  9. #define container_of(ptr, type, member) ({          \
  10. const typeof( ((type *)0)->member ) *__mptr = (ptr); \
  11. (type *)( (char *)__mptr - offsetof(type,member) );})
  12. #define offsetof(TYPE, MEMBER) ((size_t) &((TYPE *)0)->MEMBER) // 将数据结构体放到0地址处,天然的结构体中成员的首地址就是成员在结构体中的偏移量

一个简单的例子:

main.c

[cpp] view plain copy

print?

  1. #include <stdio.h>
  2. #include "list.h"
  3. LIST_HEAD(device_list);
  4. typedef struct device_struct
  5. {
  6. unsigned char *devname;
  7. struct list_head entry;
  8. } device_struct_s;
  9. int main(int argc, const char *argv[])
  10. {
  11. device_struct_s led;
  12. device_struct_s *led2;
  13. led.devname = "led";
  14. /* 添加到链表的前边 */
  15. list_add(&led.entry, &device_list);
  16. /* 得到含有链表节点的数据结构体的首地址 */
  17. led2 = list_entry(device_list.next, device_struct_s, entry);
  18. printf("led2.devname = %s\n", led2->devname);
  19. return 0;
  20. }

【1】list.h 你需要复制linux内核中的list.h头文件,并且把list_head的定义和其他需要包含进来的结构体或者宏包含进来,编译后执行的结果如下:

led2.devname = led

5. 遍历链表



对linux内核的遍历可以分为遍历链表和遍历链表中的结构体:

从头开始遍历链表,list_for_each宏,

从头开始遍历链表中的结构体,list_for_each_entry宏:

[cpp] view plain copy

print?

  1. /**
  2. * list_for_each - 迭代/遍历 链表
  3. * @pos:    the &struct list_head to use as a loop cursor.
  4. * @head:   要遍历的链表头
  5. */
  6. #define list_for_each(pos, head) \
  7. for (pos = (head)->next; pos != (head); pos = pos->next)
  8. /**
  9. * list_for_each_entry  - 遍历含链表节点入口的结构体
  10. * @pos:    the type * to use as a loop cursor.
  11. * @head:   要遍历的链表头
  12. * @member: 结构体中链表入口的名字
  13. */
  14. #define list_for_each_entry(pos, head, member)              \
  15. for (pos = list_entry((head)->next, typeof(*pos), member);   \
  16. &pos->member != (head);     \
  17. pos = list_entry(pos->member.next, typeof(*pos), member))

一个简单的例子:

[cpp] view plain copy

print?

  1. #include <stdio.h>
  2. #include "list.h"
  3. LIST_HEAD(device_list);
  4. typedef struct device_struct
  5. {
  6. unsigned char *devname;
  7. struct list_head entry;
  8. } device_struct_s;
  9. int main(int argc, const char *argv[])
  10. {
  11. device_struct_s led, gpio, beep, *tmp;
  12. led.devname = "led";
  13. gpio.devname = "gpio";
  14. beep.devname = "beep";
  15. /* 一个一个往链表的前边添加 */
  16. list_add(&led.entry, &device_list);
  17. list_add(&gpio.entry, &device_list);
  18. list_add(&beep.entry, &device_list);
  19. /* 1. 遍历链表的入口的首地值 */
  20. struct list_head *i;
  21. list_for_each(i, &device_list)
  22. {
  23. tmp = list_entry(i, device_struct_s, entry);
  24. printf("tmp.devname = %s\n", tmp->devname);
  25. }
  26. /* 2. 遍历含链表的入口的结构体的首地值 */
  27. device_struct_s *j;
  28. list_for_each_entry(j, &device_list, entry)
  29. {
  30. printf("j.devname = %s\n", j->devname);
  31. }
  32. return 0;
  33. }

【1】list.h 你需要复制linux内核中的list.h头文件,并且把list_head的定义和其他需要包含进来的结构体或者宏包含进来,编译后执行的结果如下:

tmp.devname = beep
tmp.devname = gpio
tmp.devname = led
j.devname = beep
j.devname = gpio
j.devname = led

另外:

  1. linux内核的链表中提供了反向遍历链表的宏list_for_each_prev和list_for_each_entry_reverse,他们分别是list_for_each和list_for_each_entry的反方向的实现,使用方法完全一样。
  2. 如果遍历不是从链表头开始,而是从已知的某个节点pos开始,则要使用list_for_each_entry_continue宏(使用方法同list_for_each_entry宏)。
  3. 如果想实现如果pos有值则从pos开始遍历,如果没有则从链表的头开始遍历,为此,Linux专门提供了一个list_prepare_entry(pos,head,member)宏,将它的返回值作为list_for_each_entry_continue()的pos参数,就可以满足这一要求。

我们将list_for_each_prev和list_for_each_entry_reverse的代码和执行结果也写下来:

[cpp] view plain copy

print?

  1. printf("list_for_each_prev()\n");
  2. /* 3. 反向遍历链表的入口的首地值 */
  3. struct list_head *k;
  4. list_for_each_prev(k, &device_list)
  5. {
  6. tmp = list_entry(k, device_struct_s, entry);
  7. printf("tmp.devname = %s\n", tmp->devname);
  8. }
  9. printf("list_for_each_reverse()\n");
  10. /* 4. 反向遍历含链表的入口的结构体的首地值 */
  11. device_struct_s *g;
  12. list_for_each_entry_reverse(g, &device_list, entry)
  13. {
  14. printf("g.devname = %s\n", g->devname);
  15. }

【1】此部分是在上边的main.c中实现的

【2】结合上边代码整个的执行结果如下:

list_for_each()
tmp.devname =beep
tmp.devname =gpio
tmp.devname =led
list_for_each_entry()
j.devname = beep
j.devname = gpio
j.devname = led
list_for_each_prev()   <--- 可以看到遍历结果是从尾部遍历到头部
tmp.devname = led
tmp.devname = gpio
tmp.devname = beep
list_for_each_reverse()   <--- 可以看到遍历结果是从尾部遍历到头部
g.devname = led
g.devname = gpio
g.devname = beep

6. 安全性


只讲一点判断链表是不是为空:

list_empty宏

[cpp] view plain copy

print?

  1. static inline int list_empty(const struct list_head *head)
  2. {
  3. return head->next == head;
  4. }

时间: 2024-10-08 21:34:04

【内核数据结构】 内核链表分析的相关文章

AT&amp;T汇编语言与GCC内嵌汇编,Linux内核数据结构之链表

最近在看<Linux内核源代码情景分析>,作者毛德操.书中刚开始介绍了AT&T汇编语言与GCC内嵌汇编,以及Linux内核数据结构之链表.可惜书中介绍的不够全面.因为推荐大家阅读下面两篇文章.很不错. AT&T汇编语言与GCC内嵌汇编:http://grid.hust.edu.cn/zyshao/Teaching_Material/OSEngineering/Chapter2.pdf. Linux内核数据结构之链表:http://www.cnblogs.com/Anker/p/

Linux内核数据结构之链表

与经典双向链表比较 ??经典双向链表如图.其中有一个pre指针和一个next指针,数据是在链表的节点内. ??内核链表如图.每一个链表节点内只有一个pre指针和一个next指针,整个链表节点嵌入到了一个需要使用链表的结构体内. 内核链表介绍 ??内核链表节点结构体定义如图.其中next指针指向下一个链表节点,prev指针指向前一个链表节点. ??前面已经说过,内核链表节点是嵌入到数据节点内的,那么就产生了一个问题,如何访问到链表所在结构体的指针呢? ??内核链表中通过list_entry宏来访问

linux内核系列(一)内核数据结构之链表

双向链表 传统链表与linu内核链表的区别图: 图一 图二 从上图中看出在传统链表中各种不同链表间没有通用性,因为各个数据域不同,而在linux内核中巧妙将链表结构内嵌到数据域结构中使得不同结构之间能连接起来: 链表的常用操作 内核中链表实现文件路径:include/linux/list.h 链表结构定义 struct list_head {     struct list_head *next, *prev; }; 获取结构入口地址(list_entry) #define list_entry

Linux内核数据结构——链表

目录 目录 简介 单向链表 双向链表 环形链表 Linux内核中的链表实现 offsetof container_of container_of 第一部分 container_of 第二部分 链表初始化 向链表中增加一个节点 删除节点 移动节点 判断链表是否为空 遍历链表 Demo测试 mlisth mlistc 执行结果 简介 最近在学习Android Binder驱动程序实现的时候,发现里面的数据结构用到了struct list_head.而我google发现struct list_head

Linux内核klist链表分析

1.前言 在Linux内核源码中,除了简洁的list链表还有klist链表,它是list链表的线程安全版本,在结构中提供了整个链表的自旋锁,对链表节点查找.插入和删除等操作,都需要先获得这个自旋锁,klist链表节点数据结构klist_node引入了引用计数,只有当节点的的引用计数为0时,才允许该节点从klist链表中移除. 2.klist链表相关结构 内核源码中,klist相关的头文件在include/linux/klist.h,实现的文件在lib/klist.c中,接下来分析一下klist链

[原理分析]linux内核中的链表原理实践

摘要: 本文根据linux内核的链表定义,尝试自己来完成相关的接口设计,并且构建测试用例,来体会下链表接口的用法. 正文: 首先来看下linux内核提供的链表接口大致如下: struct head_node{ struct head_node* pre; struct head_node* next; }; 即节点中只有两个指针:一个指向前一个元素,一个指向后一个元素:假设我们现在要使用该接口,跟一个int值联合在一起形成数据节点,那么存在两种可能的方式: case1: typedef stru

喜羊羊系列之数据结构内核链表

博客地址:http://blog.csdn.net/muyang_ren 内核链表示意图 关于内核链表与简单的双向连表,是否有过疑惑 内核链表定义: struct list_head{ struct list_head *next,*prev; }; struct doublelist{ datatype data; struct list_head list; }double_list; 简单的双向链表定义: struct doublelist{ datatype data; struct d

(转)Linux内核基数树应用分析

Linux内核基数树应用分析 ——lvyilong316 基数树(Radix tree)可看做是以二进制位串为关键字的trie树,是一种多叉树结构,同时又类似多层索引表,每个中间节点包含指向多个节点的指针数组,叶子节点包含指向实际对象的指针(由于对象不具备树节点结构,因此将其父节点看做叶子节点). 图1是一个基数树样例,该基数树的分叉为4(2^2),树高为4,树的每个叶子结点用来快速定位8位文件内偏移,可以定位4x4x4x4=256(叶子节点的个数)页,如:图中虚线对应的两个叶子结点的路径组成值

《Linux内核设计与实现》读书笔记(六)- 内核数据结构

内核数据结构贯穿于整个内核代码中,这里介绍4个基本的内核数据结构. 利用这4个基本的数据结构,可以在编写内核代码时节约大量时间. 主要内容: 链表 队列 映射 红黑树 1. 链表 链表是linux内核中最简单,同时也是应用最广泛的数据结构. 内核中定义的是双向链表. 1.1 头文件简介 内核中关于链表定义的代码位于: include/linux/list.h list.h文件中对每个函数都有注释,这里就不详细说了. 其实刚开始只要先了解一个常用的链表操作(追加,删除,遍历)的实现方法, 其他方法