Linux内核list_head使用示例

#include<linux/init.h>
#include<linux/slab.h>
#include<linux/module.h>
#include<linux/kernel.h>
#include<linux/list.h>

struct student
{
	int id;
	char *name;
	struct list_head list;
};

static void testlist_exit(void)
{
    printk("*************************\n");
    printk("testlist is exited!\n");
    printk("*************************\n");
}

void print_student(struct student *stu)
{
    printk("======================\n");
    printk("id  =%d\n", stu->id);
    printk("name=%s\n", stu->name);
    printk("======================\n");
}

static int testlist_init(void)
{

	struct student *stu1, *stu2, *stu3, *stu4;
    struct student *stu;

	// init a list head
    LIST_HEAD(stu_head);

	// init four list nodes
    stu1 = kmalloc(sizeof(*stu1), GFP_KERNEL);
    stu1->id = 1;
    stu1->name = "st1";
    INIT_LIST_HEAD(&stu1->list);

	stu2 = kmalloc(sizeof(*stu2), GFP_KERNEL);
    stu2->id = 2;
    stu2->name = "st2";
    INIT_LIST_HEAD(&stu2->list);

    stu3 = kmalloc(sizeof(*stu3), GFP_KERNEL);
    stu3->id = 3;
    stu3->name = "st3";
    INIT_LIST_HEAD(&stu3->list);

    stu4 = kmalloc(sizeof(*stu4), GFP_KERNEL);
    stu4->id = 4;
    stu4->name = "st4";
    INIT_LIST_HEAD(&stu4->list);

	list_add(&stu1->list, &stu_head);
    list_add(&stu2->list, &stu_head);
    list_add(&stu3->list, &stu_head);
    list_add(&stu4->list, &stu_head);

	// print each student from 4 to 1:[4,3,2,1]
	list_for_each_entry(stu, &stu_head, list)
    {
        print_student(stu);
    }

	// print each student from 1 to 4:[1,2,3,4]
    list_for_each_entry_reverse(stu, &stu_head, list)
    {
        print_student(stu);
    }

	// delete a entry stu2
    list_del(&stu2->list);
	// print each student from 4 to 1:[4,3,1]
    list_for_each_entry(stu, &stu_head, list)
    {
        print_student(stu);
    }

	// replace stu3 with stu2
    list_replace(&stu3->list, &stu2->list);
	// print each student from 4 to 1:[4,2,1]
    list_for_each_entry(stu, &stu_head, list)
    {
        print_student(stu);
    }

	return 0;
}

module_init(testlist_init);
module_exit(testlist_exit);
MODULE_LICENSE("GPL");
obj-m +=test_list.o

KDIR :=/lib/modules/$(shell uname -r)/build
PWD=$(shell pwd)

modules:
        $(MAKE) -C $(KDIR) M=$(PWD) modules
clean:
        $(MAKE) -C $(KDIR) M=$(PWD) clean

list_head原理参考:

内核双链表遍历:带 safe 和不带 safe 的区别内核双链表遍历:带 safe 和不带 safe 的区别.icon-default.png?t=M1L8https://biscuitos.github.io/blog/LIST_ADV_safe/

一文搞懂 Linux 内核链表(深度分析) - 云+社区 - 腾讯云在 Linux 内核中使用最多的数据结构就是链表了,其中就包含了许多高级思想。 比如面向对象、类似C++模板的实现、堆和栈的实现。https://cloud.tencent.com/developer/article/1805773

  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值