Linux内核中的链表通常都组织成双向循环链表,不同于一般意义上的链表,这里的链表节点只含有链表指针(next和prev),没有链表的数据。Linux内核中使用的链表源代码位于include/linux/list.h 中,下面详细叙述。
1. 链表数据结构 list_head 的定义:
struct list_head {
struct list_head *next, *prev;
};
【注意】只有前后节点指针,没有数据 !
2. 链表的声明和初始化
2.1静态方法——编译时
#define LIST_HEAD_INIT(name) { &(name), &(name) } // 链表的pre和next指针都指向了节点自己的首地址
#define LIST_HEAD(name) \
struct list_head name = LIST_HEAD_INIT(name)
2.2动态方法——运行时
static inline void INIT_LIST_HEAD(struct list_head *list)
{
list->next = list;
list->prev = list;
}
图示:
3. 插入/删除/合并
3.1插入
3.1.1 头插——list_add
static inline void list_add(struct list_head *new, struct list_head *head)
{
__list_add(new, head, head->next);
}
3.1.2 尾插—— list_add_tail
static inline void list_add_tail(struct list_head *new, struct list_head *head)
{
__list_add(new, head->prev, head);
}
- 可以看到他们调用了相同的
__list_add 函数:
static inline void __list_add(struct list_head *new, struct list_head *prev, struct list_head *next)
{
next->prev = new;
new->next = next;
new->prev = prev;
prev->next = new;
}
3.2 删除
- 对链表的删除操作函数有两种:list_del函数和list_del_init函数:
3.2.1 list_del
static inline void list_del(struct list_head *entry)
{
__list_del(entry->prev, entry->next);
entry->next = LIST_POISON1;
entry->prev = LIST_POISON2;
}
#ifdef CONFIG_ILLEGAL_POINTER_VALUE
# define POISON_POINTER_DELTA _AC(CONFIG_ILLEGAL_POINTER_VALUE, UL)
#else
# define POISON_POINTER_DELTA 0
#endif
#define LIST_POISON1 ((void *) 0x100 + POISON_POINTER_DELTA)
#define LIST_POISON2 ((void *) 0x200 + POISON_POINTER_DELTA)
3.2.2 list_del_init
static inline void list_del_init(struct list_head *entry)
{
__list_del_entry(entry);
INIT_LIST_HEAD(entry);
}
static inline void __list_del_entry(struct list_head *entry)
{
__list_del(entry->prev, entry->next);
}
- 将链表节点entry从链表中删除后,list_del_init函数中又将entry重新初始化了(next和prev指针指向了自己)。在此不再画图,唯一的不同是把删除下来的图的next和prev指针指向了自己的首地址
- 他们调用了相同的
__list_del 函数
static inline void __list_del(struct list_head * prev, struct list_head * next)
{
next->prev = prev;
prev->next = next;
}
3.3 替换
对链表的替换操作有两个:list_replace函数和list_replace_init函数:
3.3.1 list_replace
static inline void list_replace(struct list_head *old, struct list_head *new)
{
new->next = old->next;
new->next->prev = new;
new->prev = old->prev;
new->prev->next = new;
}
3.3.2 list_replace_init
static inline void list_replace_init(struct list_head *old, struct list_head *new)
{
list_replace(old, new);
INIT_LIST_HEAD(old);
}
list_replace_init函数的图形在此处也不再画,区别就是多了初始化换下来的old节点
3.4 搬移
搬移的含义是将原本属于一个链表的节点移动到另一个链表的操作,有两个函数:list_move函数和list_move_tail函数:
3.4.1 list_move
static inline void list_move(struct list_head *list, struct list_head *head)
{
__list_del_entry(list);
list_add(list, head);
}
3.4.2 list_move_tail
static inline void list_move_tail(struct list_head *list, struct list_head *head)
{
__list_del_entry(list);
list_add_tail(list, head);
}
3.5 合并
合并在这里的意思就是合并了,是将两个独立的链表合并成为一个链表,合并的时候根据合并的位置的不同可以分为:
- 合并到头部的 list_splice函数
- 合并到尾部的 list_splice_tail函数
3.5.1 list_splice
static inline void list_splice(const struct list_head *list, struct list_head *head) {
if (!list_empty(list))
__list_splice(list, head, head->next);
}
- 假设当前有两个链表,表头分别是list1和list2(都是struct list_head变量),当调用list_splice(&list1,&list2)时,只要list1非空,list1链表的内容将被挂接在list2链表上,位于list2和list2.next(原list2表的第一个节点)之间。新list2链表将以原list1表的第一个节点为首节点,而尾节点不变。
3.5.2 list_splice_tail
static inline void list_splice_tail(struct list_head *list, struct list_head *head)
{
if (!list_empty(list))
__list_splice(list, head->prev, head);
}
- 假设当前有两个链表,表头分别是list1和list2(都是struct list_head变量),当调用list_splice_tail(&list1,&list2)时,只要list1非空,list1链表的内容将被挂接在list2链表上,位于list2尾节点和list2(原list2头)之间。新list2链表将以原list1表的最后一个节点为尾节点,而头节点不变。
3.5.3 list_splice_init
static inline void list_splice_init(struct list_head *list, struct list_head *head)
{
if (!list_empty(list)) {
__list_splice(list, head, head->next);
INIT_LIST_HEAD(list); <--- 跟list_splice唯一的不同
}
}
3.5.4 list_splice_tail_init
static inline void list_splice_tail_init(struct list_head *list, struct list_head *head)
{
if (!list_empty(list)) {
__list_splice(list, head->prev, head);
INIT_LIST_HEAD(list); <--- 跟list_splice_tail唯一的不同
}
}
- 共同的函数
__list_splice 和list_empty
static inline void __list_splice(const struct list_head *list, struct list_head *prev, struct list_head *next)
{
struct list_head *first = list->next;
struct list_head *last = list->prev;
first->prev = prev;
prev->next = first;
last->next = next;
next->prev = last;
}
static inline int list_empty(const struct list_head *head)
{
return head->next == head;
}
4. 找到链表中的数据
前边提到的函数都是操作的链表节点的入口,但是对于我们真正有意义的是节点上的数据,链表的头上没有数据,其他的节点上都是带有数据的。如何从一个链表节点的入口得到节点的数据呢?要用到以下的函数
list_entry函数:
#define list_entry(ptr, type, member) \
container_of(ptr, type, member)
#define container_of(ptr, type, member) ({ \
const typeof( ((type *)0)->member ) *__mptr = (ptr); \
(type *)( (char *)__mptr - offsetof(type,member) );})
#define offsetof(TYPE, MEMBER) ((size_t) &((TYPE *)0)->MEMBER)
举例:
main.c
#include <stdio.h>
#include <linux/list.h>
LIST_HEAD(device_list);
typedef struct device_struct
{
unsigned char *devname;
struct list_head entry;
} device_struct_s;
int main(int argc, const char *argv[])
{
device_struct_s led;
device_struct_s *ledp;
led.devname = "led";
list_add(&led.entry, &device_list);
ledp = list_entry(device_list.next, device_struct_s, entry);
printf("led.devname = %s\n", ledp->devname);
return 0;
}
5. 遍历链表
对linux内核的遍历可以分为遍历链表和遍历链表中的结构体:
从头开始遍历链表,list_for_each 宏;
从头开始遍历链表中的结构体,list_for_each_entry 宏;
5.1 遍历list_head链表
#define list_for_each(cursor, head) \
for (cursor = (head)->next; cursor != (head); cursor = cursor->next)
#define list_for_each_safe(cursor, next, head) \
for (cursor = (head)->next,next = cursor->next;\
cursor != (head);\
cursor = next,next = cursor->next)
- 注意:当我们采用
list__for_each() 函数遍历list时,如果我们删除元素(即执行list_del(cursor); ),就会导致cursor指向的元素的prev=LIST_POISON1,next=LIST_POISON2,当执行到cursor=cursor->next时,就会出现错误。
- 但是当我们使用
list_for_each_safe() 函数遍历list时,如果我们也删除元素后,会执行cursor=next ,而next=cursor->next (注意:next=cursor->next中的cursor是删除的那个元素),所以虽然删除了元素cursor,但是执行了cursor=next 后,cursor指向了正确的遍历位置,所以使用list_for_each_safe() 函数遍历list时并不会出现错误。 list_for_each_safe() 在遍历时之所以不会出现错误,是因为我们使用了next暂时保存了cursor(被删除元素所指向的下一个元素),所以用这个宏定义,不会出现遍历时删除元素的错误。
5.2 遍历“含有list_head项的结构体‘链表
#define list_for_each_entry(cursor, head, member) \
for (cursor = list_entry((head)->next, typeof(*cursor), member);\
&cursor->member != (head);\
cursor = list_entry(cursor->member.next, typeof(*cursor), member))
#define list_for_each_entry_safe(cursor, next, head, member) \
for (cursor = list_entry((head)->next, typeof(*cursor), member),\
next = list_entry(cursor->member.next, typeof(*cursor), member);\
&cursor->member != (head);\
cursor = next, next = list_entry(next->member.next, typeof(*cursor), member))
#include <stdio.h>
#include <linux/list.h>
LIST_HEAD(device_list);
typedef struct device_struct
{
unsigned char *devname;
struct list_head entry;
} device_struct_s;
int main(int argc, const char *argv[])
{
device_struct_s led, gpio, beep, *tmp;
led.devname = "led";
gpio.devname = "gpio";
beep.devname = "beep";
list_add(&led.entry, &device_list);
list_add(&gpio.entry, &device_list);
list_add(&beep.entry, &device_list);
struct list_head *cursor;
list_for_each(cursor, &device_list)
{
tmp = list_entry(cursor, device_struct_s, entry);
printf("tmp.devname = %s\n", tmp->devname);
}
device_struct_s *cursor2;
list_for_each_entry(cursor2, &device_list, entry)
{
printf("cursor2.devname = %s\n", cursor2->devname);
}
return 0;
}
另外:
- linux内核的链表中提供了反向遍历链表的宏
list_for_each_prev 和list_for_each_entry_reverse ,他们分别是list_for_each 和list_for_each_entry 的反方向的实现,使用方法完全一样。 - 如果遍历不是从链表头开始,而是从已知的某个节点pos开始,则要使用
list_for_each_entry_continue 宏(使用方法同list_for_each_entry宏)。 - 如果想实现如果pos有值则从pos开始遍历,如果没有则从链表的头开始遍历,为此,Linux专门提供了一个
list_prepare_entry(pos,head,member) 宏,将它的返回值作为list_for_each_entry_continue() 的pos参数,就可以满足这一要求。
- 我们将list_for_each_prev和list_for_each_entry_reverse的代码和执行结果也写下来:
printf("list_for_each_prev()\n");
struct list_head *cursor3;
list_for_each_prev(cursor3, &device_list)
{
tmp = list_entry(cursor3, device_struct_s, entry);
printf("tmp.devname = %s\n", tmp->devname);
}
printf("list_for_each_reverse()\n");
device_struct_s *cursor4;
list_for_each_entry_reverse(cursor4, &device_list, entry)
{
printf("cursor4.devname = %s\n", g->evname);
}
结合上边代码的执行结果如下:
list_for_each_prev() <--- 可以看到遍历结果是从尾部遍历到头部
tmp.devname = led
tmp.devname = gpio
tmp.devname = beep
list_for_each_reverse() <--- 可以看到遍历结果是从尾部遍历到头部
cursor4.devname = led
cursor4.devname = gpio
cursor4.devname = beep
; }
/* 4. 反向遍历含链表的入口的结构体的首地值 */ printf(“list_for_each_reverse()\n”); device_struct_s *cursor4; list_for_each_entry_reverse(cursor4, &device_list, entry) { printf(“cursor4.devname = %s\n”, g->evname); }
结合上边代码的执行结果如下:
```bash
list_for_each_prev() <--- 可以看到遍历结果是从尾部遍历到头部
tmp.devname = led
tmp.devname = gpio
tmp.devname = beep
list_for_each_reverse() <--- 可以看到遍历结果是从尾部遍历到头部
cursor4.devname = led
cursor4.devname = gpio
cursor4.devname = beep
参考文章:【内核数据结构】 内核链表分析_沧海一笑的技术博客_51CTO博客
|