Linux 内核 hlist 详解
在Linux内核中,hlist(哈希链表)使用非常广泛。本文将对其数据结构和核心函数进行分析。 和hlist相关的数据结构有两个:hlist_head 和 hlist_node
结构如下图所示: ? hlist_head结构体只有一个域,即first。 first指针指向该hlist链表的第一个节点。 如果要删除hash桶对应链表中的第一个普通结点 对应的程序代码如下:
如果要删除hash桶对应链表中的非第一个结点 对应的程序代码如下:
可以看到删除第一个普通结点和删除非第一个普通结点的代码是一样的。 下面再来看看如果hlist_node中包含两个分别指向前驱结点和后继结点的指针 很明显删除hash桶对应链表中的非第一个普通结点,只需要如下两行代码:
可是,如果是删除的hash桶对应链表中的第一个普通结点: 附一张在hash桶头结点之后,插入第一个普通结点的图: ? 在遍历上,如果使用hlist_hode,list_node指针进行遍历,两者过程大致相似。 #define list_for_each(pos,head) 如果使用其寄生结构的指针进行遍历,则hlist与list也略有不同,hlist在遍历时需要一个指向hlist_node的临时指针,该指针的引入,一是为了遍历,而list的遍历在list_entry的参数中实现了,更主要的目的在于判断结束,因为hlist最后一个节点的next为NULL,只有hlist_node指向NULL时才算结束,而这个NULL不包含在任何寄生结构内,不能通过tpos->member的方式访问到,故临时变量pos的引入时必须的。 #define list_for_each_entry(pos,head,member) 另外,list和hlist的遍历都实现了safe版本,因在遍历时,没有任何特别的东西来阻止对链表执行删除操作(通常在使用链表时使用锁来保护并发访问)。安全版本的遍历函数使用临时存放的方法使得检索链表时能不被删除操作所影响。 #define list_for_each_safe(pos,n,head) 附上linux内核中与hlist相关的完整代码: //hash桶的头结点 struct hlist_head { struct hlist_node *first;//指向每一个hash桶的第一个结点的指针 }; //hash桶的普通结点 struct hlist_node { struct hlist_node *next;//指向下一个结点的指针 struct hlist_node **pprev;//指向上一个结点的next指针的地址 }; //以下三种方法都是初始化hash桶的头结点 #define HLIST_HEAD_INIT { .first = NULL } #define HLIST_HEAD(name) struct hlist_head name = { .first = NULL } #define INIT_HLIST_HEAD(ptr) ((ptr)->first = NULL) //初始化hash桶的普通结点 static inline void INIT_HLIST_NODE(struct hlist_node *h) { h->next = NULL; h->pprev = NULL; } //判断一个结点是否已经存在于hash桶中 static inline int hlist_unhashed(const struct hlist_node *h) { return !h->pprev; } //判断一个hash桶是否为空 static inline int hlist_empty(const struct hlist_head *h) { return !h->first; } static inline void __hlist_del(struct hlist_node *n) { struct hlist_node *next = n->next;//获取指向待删除结点的下一个结点的指针 struct hlist_node **pprev = n->pprev;//保留待删除结点的pprev域 *pprev = next;//修改待删除结点的pprev域,逻辑上使待删除结点的前驱结点指向待删除结点的后继结点 if (next) next->pprev = pprev;//设置待删除结点的下一个结点的pprev域,保持hlist的结构 } static inline void hlist_del(struct hlist_node *n) { __hlist_del(n);//删除结点之后,需要将其next域和pprev域设置为无用值 n->next = LIST_POISON1; n->pprev = LIST_POISON2; } static inline void hlist_del_init(struct hlist_node *n) { if (!hlist_unhashed(n)) { __hlist_del(n); INIT_HLIST_NODE(n); } } //将普通结点n插入到头结点h对应的hash桶的第一个结点的位置 static inline void hlist_add_head(struct hlist_node *n,struct hlist_head *h) { struct hlist_node *first = h->first; n->next = first; if (first) first->pprev = &n->next; h->first = n; n->pprev = &h->first; } /* next must be != NULL */ //在next结点之前插入结点n,即使next结点是hash桶中的第一个结点也可以 static inline void hlist_add_before(struct hlist_node *n,struct hlist_node *next) { n->pprev = next->pprev; n->next = next; next->pprev = &n->next; *(n->pprev) = n; } //在结点n之后插入结点next static inline void hlist_add_after(struct hlist_node *n,struct hlist_node *next) { next->next = n->next; n->next = next; next->pprev = &n->next; if(next->next) next->next->pprev = &next->next; } /* * Move a list from one list head to another. Fixup the pprev * reference of the first entry if it exists. */ static inline void hlist_move_list(struct hlist_head *old,struct hlist_head *new) { new->first = old->first; if (new->first) new->first->pprev = &new->first; old->first = NULL; } //通过一个结构体内部一个成员的地址获取结构体的首地址 #define hlist_entry(ptr,type,member) container_of(ptr,member) #define hlist_for_each(pos,head) for (pos = (head)->first; pos && ({ prefetch(pos->next); 1; }); pos = pos->next) #define hlist_for_each_safe(pos,head) for (pos = (head)->first; pos && ({ n = pos->next; 1; }); pos = n) /** * hlist_for_each_entry - iterate over list of given type * @tpos: the type * to use as a loop cursor. * @pos: the &struct hlist_node to use as a loop cursor. * @head: the head for your list. * @member: the name of the hlist_node within the struct. */ #define hlist_for_each_entry(tpos,member) for (pos = (head)->first; pos && ({ prefetch(pos->next); 1;}) && ({ tpos = hlist_entry(pos,member); 1;}); pos = pos->next) /** * hlist_for_each_entry_continue - iterate over a hlist continuing after current point * @tpos: the type * to use as a loop cursor. * @pos: the &struct hlist_node to use as a loop cursor. * @member: the name of the hlist_node within the struct. */ #define hlist_for_each_entry_continue(tpos,member) for (pos = (pos)->next; pos && ({ prefetch(pos->next); 1;}) && ({ tpos = hlist_entry(pos,member); 1;}); pos = pos->next) /** * hlist_for_each_entry_from - iterate over a hlist continuing from current point * @tpos: the type * to use as a loop cursor. * @pos: the &struct hlist_node to use as a loop cursor. * @member: the name of the hlist_node within the struct. */ #define hlist_for_each_entry_from(tpos,member) for (; pos && ({ prefetch(pos->next); 1;}) && ({ tpos = hlist_entry(pos,member); 1;}); pos = pos->next) /** * hlist_for_each_entry_safe - iterate over list of given type safe against removal of list entry * @tpos: the type * to use as a loop cursor. * @pos: the &struct hlist_node to use as a loop cursor. * @n: another &struct hlist_node to use as temporary storage * @head: the head for your list. * @member: the name of the hlist_node within the struct. */ #define hlist_for_each_entry_safe(tpos,member) for (pos = (head)->first; pos && ({ n = pos->next; 1; }) && ({ tpos = hlist_entry(pos,member); 1;}); pos = n) (编辑:李大同) 【声明】本站内容均来自网络,其相关言论仅代表作者个人观点,不代表本站立场。若无意侵犯到您的权利,请及时与联系站长删除相关内容! |