32_Linux内核链表剖析

关键词:

0. 课程目标

  • 移植Linux内核链表,使其适用于非GNU编译器
  • 分析Linux内核中链表的基本实现

1. 移植时的注意事项

  • 清除文件间的依赖:剥离依赖文件中与链表实现相关的代码
  • 清楚平台相关代码(GNU C)
    ({ })
    typeof
    __builtin_prefetch
    static inline

2. Linux内核链表的实现

  • 带头结点的双向循环链表,且头结点为表中成员
  • 头结点的next指针指向首结点
  • 头结点的prev指针指向尾结点
    Linux内核链表原理图

3. Linux内核链表的结点定义

struct list_head {
    struct list_head *next, *prev;
};

在Linux内核链表结点定义中,结点的数据域放在哪儿?
使用Linux内核链表时,需要用户自定义Node结点,结点中包含了指针域和数据域。如下

struct Node
{
    struct list_head head;      // 指针域
    int value;                  // 数据域
    //...
};

4. Linux内核链表的创建及其初始化

#include <stdio.h>
#include "LinuxList.h"

struct Node
{
    struct list_head head;      // 指针域
    int value;                  // 数据域
};

int main(void)
{
    struct Node l = {0};    // 创建链表头结点
    struct list_head* list = (struct list_head*)&l;

    INIT_LIST_HEAD(list);   // 初始化头结点

    return 0;
}

5. INIT_LIST_HEAD()源码:

static void INIT_LIST_HEAD(struct list_head *list)
{
    list->next = list;
    list->prev = list;
}
INIT_LIST_HEAD源码示意图

6. 插入操作

  • 在链表头部插入:list_add(node, head)
  • 在链表尾部插入:list_add_tail(node, head)
static void __list_add(struct list_head *node,
                  struct list_head *prev,
                  struct list_head *next)
{
    next->prev = node;
    node->next = next;
    node->prev = prev;
    prev->next = node;
}

static void list_add(struct list_head *node, struct list_head *head)
{
    __list_add(node, head, head->next);
}

static void list_add_tail(struct list_head *node, struct list_head *head)
{
    __list_add(node, head->prev, head);
}

7. 删除操作

static void __list_del(struct list_head * prev, struct list_head * next)
{
    next->prev = prev;
    prev->next = next;
}

static void list_del(struct list_head *entry)
{
    __list_del(entry->prev, entry->next);
    entry->next = LIST_POISON1;
    entry->prev = LIST_POISON2;
}

8. 遍历操作

  • 正向遍历:list_for_each(pos, head)
  • 逆向遍历:list_for_each_pre(pos, head)
/**
 * 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; prefetch(pos->next), pos != (head); \
            pos = pos->next)

/**
 * list_for_each_prev   -   iterate over a list backwards
 * @pos:    the &struct list_head to use as a loop cursor.
 * @head:   the head for your list.
 */
#define list_for_each_prev(pos, head) \
    for (pos = (head)->prev; prefetch(pos->prev), pos != (head); \
            pos = pos->prev)

Linux内核中遍历操作是通过宏定义来实现的,式中关键字prefetch是gnu中特有的关键字,为了在不同的操作系统中使用,需要重定义这个关键字为#define prefetch(x) ((void)x)

9. list_entry

#define list_entry(ptr, type, member) \
    container_of(ptr, type, member)

10. 相关测试

#include <stdio.h>
#include "LinuxList.h"

void test_demo1()
{
    struct Node
    {
        struct list_head head;      // 指针域
        int value;                  // 数据域
    };

    struct Node l = {0};    // 创建链表头结点
    struct list_head* list = (struct list_head*)&l;
    struct list_head* slide = NULL;
    struct list_head* toDel = NULL;
    int value = 0;
    int i = 0;

    INIT_LIST_HEAD(list);   // 初始化头结点

    printf("Insert begin...\n");

    for(i=0; i<5; ++i)
    {
        struct Node* n = ( struct Node*)malloc(sizeof(struct Node));
        n->value = i;
        list_add_tail((struct list_head*)n, list);
    }

    list_for_each(slide, list)
    {
        printf("%d\n", ((struct Node*)slide)->value);
    }

    printf("Insert end...\n");

    printf("Delete begin...\n");

    list_for_each(slide, list)
    {
        value = ((struct Node*)slide)->value;
        if( value == 3 )
        {
            list_del(slide);
            free(slide);
            break;
        }
    }

    list_for_each(slide, list)
    {
        printf("%d\n", ((struct Node*)slide)->value);
    }

    printf("Delete end...\n");
}

void test_demo2()
{
    struct Node
    {
        int value;                  // 数据域
        struct list_head head;      // 指针域
    };

    struct Node l = {0};    // 创建链表头结点
    struct list_head* list = &l.head;
    struct list_head* slide = NULL;
    int i = 0;

    INIT_LIST_HEAD(list);   // 初始化头结点

    printf("Insert begin...\n");

    for(i=0; i<5; ++i)
    {
        struct Node* n = ( struct Node*)malloc(sizeof(struct Node));
        n->value = i;
        list_add(&n->head, list);
    }

    list_for_each(slide, list)
    {
        printf("%d\n", (list_entry(slide, struct Node, head)->value));
    }

    printf("Insert end...\n");

    printf("Delete begin...\n");

    list_for_each(slide, list)
    {
        struct Node* n = list_entry(slide, struct Node, head);
        if( n->value == 3 )
        {
            list_del(slide);
            free(n);
            break;
        }
    }

    list_for_each(slide, list)
    {
        printf("%d\n", list_entry(slide, struct Node, head)->value);
    }

    printf("Delete end...\n");
}

int main(void)
{
    test_demo1();
    test_demo2();

    return 0;
}

11. 小结

  • Linux内核链表移植时需要剔除依赖以及平台相关代码
  • Linux内核链表是带头结点的双向循环链表
  • 使用Linux内核链表时需要自定义链表结点:
    (1) 将struct list_head作为结点结构体的第一个成员或最后一个成员
    (2) struct list_head作为最后一个成员时,需要使用list_entry
    (3) list_entry的定义中使用了container_of

声明:此文章仅是本人在学习狄泰学院《数据结构实战开发教程》所做的笔记,文章中包含狄泰软件资料内容,一切版权归狄泰软件所有!
实验环境:ubuntu10 + Qt Creator2.4.1 + Qt SDK 4.7.4

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,904评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,581评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,527评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,463评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,546评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,572评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,582评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,330评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,776评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,087评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,257评论 1 344
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,923评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,571评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,192评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,436评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,145评论 2 366
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,127评论 2 352

推荐阅读更多精彩内容