本笔记仅记录 include/linux/types.h
下定义的基本数据结构及其设计。
在C语言大型项目中,往往要使用类型想当繁多的链表。但是如果分别为每一个链表都加入增、删、改、查、搜索、排序等功能的话代码又显得相当繁琐。因此需要将链表的常用操作统一拆分出来。
在面向对象语言中,可以考虑定义一个链表基类即可实现上述需求。但是C语言结构体并不具备面向对象的继承关系,因此在C语言中实现一个通用的链表通常使用如下两种方式实现:
void *data
成员,随后 void *data
交由开发者自行控制和处理。使用这种方法的例子有clibs-list。
list_t
。*next
、 *prev
等指针,例如:C struct list_head { struct list_head *next, *prev; };
C struct my_data_list { int data; struct list_head list; };
.list
成员找到上一个或下一个链表节点的 .list
成员。但是此时的问题是如何使用 .list
成员完成获取链表的节点等操作。container_of
等方法解决上述问题,例如:struct list_head
的定义为:
struct list_head {
struct list_head *next, *prev;
};
本章节的宿主结构体统一定义如下:
struct mdata_list {
int data;
struct list_head list;
}
为了简化程序设计:
list_empty
等函数要求)LIST_HEAD_INIT
函数的定义为:
#define LIST_HEAD_INIT(name) { &(name), &(name) }
因此其参数类型为 struct list_head *list
。示例如下:
static struct mdata_list mdata_list_head = {
.data = -1;
.list = LIST_HEAD_INIT(mdata_list_head.list)
}
INIT_LIST_HEAD
函数的定义为:
static inline void INIT_LIST_HEAD(struct list_head *list)
{
list->next = list;
list->prev = list;
}
其使用示例如下:
static void func() {
struct mdata_list mdata_list_head = { 0 };
INIT_LIST_HEAD(&mdata_list_head.list);
}
获取宿主链表入口有两个函数 list_entry
和 container_of
,其本质相同互为别名,定义如下:
/**
* list_entry - get the struct for this entry
* @ptr: the &struct list_head pointer.
* @type: the type of the struct this is embedded in.
* @member: the name of the list_head within the struct.
*/
#define list_entry(ptr, type, member) \
container_of(ptr, type, member)
/**
* container_of - cast a member of a structure out to the containing structure
* @ptr: the pointer to the member.
* @type: the type of the container struct this is embedded in.
* @member: the name of the member within the struct.
*
*/
#define container_of(ptr, type, member) ({ \
const typeof( ((type *)0)->member ) *__mptr = (ptr); \
(type *)( (char *)__mptr - offsetof(type,member) );})
均为使用 list
链表锚点获取宿主结构体 entry
,使用demo如下:
struct my_data_list *entry = list_entry(
&(node.list),
struct my_data_list,
list
)
list_add
函数的定义为:
/**
* list_add - add a new entry
* @new: new entry to be added
* @head: list head to add it after
*
* Insert a new entry after the specified head.
* This is good for implementing stacks.
*/
static inline void list_add(struct list_head *new, struct list_head *head)
{
__list_add(new, head, head->next);
}
假设各宿主节点关系如下:
各宿主节点均遵从上方定义的统一的宿主数据结构,要在 node1
和 node2
之间增加 node4
,则示例如下:
list_add(&(node4.list), &(node1.list));
list_add_tail
函数的定义为:
/**
* list_add_tail - add a new entry
* @new: new entry to be added
* @head: list head to add it before
*
* Insert a new entry before the specified head.
* This is useful for implementing queues.
*/
static inline void list_add_tail(struct list_head *new, struct list_head *head)
{
__list_add(new, head->prev, head);
}
list_del
函数的定义为:
/**
* list_del - deletes entry from list.
* @entry: the element to delete from the list.
* Note: list_empty() on entry does not return true after this, the entry is
* in an undefined state.
*/
static inline void list_del(struct list_head *entry)
{
__list_del(entry->prev, entry->next);
entry->next = LIST_POISON1;
entry->prev = LIST_POISON2;
}
假设各宿主节点关系如下:
各宿主节点均遵从上方定义的统一的宿主数据结构,则删除node2节点的示例为:
list_del(node1);
list_for_each
函数的定义如下:
/**
* list_for_each - iterate over a list
* @pos: the &struct list_head to use as a loop cursor.
* @head: the head for your list.
*/
#define list_for_each(pos, head) \
for (pos = (head)->next; pos != (head); pos = pos->next)