主要移植了内核中的 list,rbtree。使得这2个数据结构在用户态程序中也能使用。

同时用 cpputest 对移植后的代码进行了测试。(测试代码其实也是使用这2个数据结构的方法)

内核代码的如下文件:(内核版本 v3.2 debian 7.5源码)

  1. include/linux/list.h  (删除了 hlist 相关内容)
  2. include/linux/rbtree.h
  3. lib/rbtree.c

对上面的代码进行了一些简化,只留了常用的函数。同时删除了其中和内核相关的部分。

主要内容:

  • list 介绍 (循环双向链表)
  • rbtree 介绍

1. list 介绍 (循环双向链表)

1.1 简介

Linux中的链表用法与一般数据结构书中介绍的用法有些不一样。

Linux内核中,为了保证链表的通用性,将链表的节点结构单独抽取了出来,也就是将链表的结构和链表的数据分开定义。

一般数据结构的书中介绍到的链表都是将链表的数据和链表的结构一起定义的。

注:具体介绍可以我之前的博客参见:http://www.cnblogs.com/wang_yb/archive/2013/04/16/3023892.html 中的 1.2节

里面很重要的一点就是:链表结构和数据分开后,是如何通过链表节点结构来获取数据的?

带有safe的函数或者宏都是可以用于多线程的

1.2 修改部分

  1. 删除了hlist相关内容
  2. 修改了 list_del 函数: 将 LIST_POISON1 和 LIST_POISON2 改成了 NULL
  3. 删除了 list_empty_careful: 用户空间用不上
  4. 删除 __list_for_each: 和 list_for_each 重复
  5. 删除 list_prepare_entry: 暂时不需要
  6. 删除 list_safe_reset_next: 暂时不需要
  7. 删除 list_rotate_left: 暂时不需要
  8. 所有变量 new 改为了 newnode: new 是 c++ 关键字,用CppUTest进行测试时无法编译

1.3 list.h 对外的接口

No.

主要 函数

说明

1. list_add 在 head 之后追加一个节点
2. list_add_tail 在 head 之前追加一个节点, 也就是在末尾追加一个节点
3. list_del 删除一个节点, 并将这个节点的next, prev 置为 NULL
4. list_del_init 删除一个节点并初始化删除的节点
5. list_replace 替换一个节点
6. list_replace_init 替换一个节点, 并初始化被替换的节点
7. list_move 移动节点到 head 之后
8. list_move_tail 移动节点到 head 之前
9. list_is_last 判断节点是否是链表中最后一个
10. list_empty 判断链表是否为空 (即, 是否只有 head 节点)
11. list_is_singular 判断链表中是否只有一个节点 (除了 head 之外)
12. list_cut_position 将1个链表截断为2个链表
13. list_splice 将2个链表合并为1个链表, @list中的所有节点(不包括list)加入到 head 之后
14. list_splice_tail 将2个链表合并为1个链表, @list中的所有节点(不包括list)加入到 head 之前
15. list_splice_init 同 list_splice, 最后会初始化 @list
16. list_splice_tail_init 同 list_splice_tail, 最后会初始化 @list

No.

主要 宏

说明

1. list_entry 获取包含此节点的 struct
2. list_first_entry 获取包含此节点的 首个 struct
3. list_for_each 从 head节点之后一个节点开始向后循环
4. list_for_each_prev 从 head节点之前一个节点开始向前循环
5. list_for_each_safe list_for_each 的安全版本, 即, 循环时即使有其它线程删除节点也可正常运行
6. list_for_each_prev_safe list_for_each_prev 的安全版本
7. list_for_each_entry 同 list_for_each, 只是参数不同
8. list_for_each_entry_reverse 同 list_for_each_prev, 只是参数不同
9. list_for_each_entry_continue 同 list_for_each_entry, 但不是从头(head)开始循环的
10. list_for_each_entry_continue_reverse 同 list_for_each_entry_reverse, 但不是从头(head)开始循环的
11. list_for_each_entry_from 从指定位置开始向后循环
12. list_for_each_entry_safe list_for_each_entry 的安全版本
13. list_for_each_entry_safe_continue list_for_each_entry_continue 的安全版本
14. list_for_each_entry_safe_from list_for_each_entry_from 的安全版本
15. list_for_each_entry_safe_reverse list_for_each_entry_reverse 的安全版本

1.4 使用示例 - 测试 list.h 中所有的list操作

构造如下场景,用来测试上述列出的所有的 list 操作:

1. 构造用来测试的 struct:(为了使得测试结果一目了然,struct尽量简单)

struct test_struct {
int num;
struct list_head head;
};

2. 逐个函数进行测试,使用测试框架 cppUTest

3. 宏 相关的暂时没有测试

4. 运行测试非常简单(前提是得安装 cpputest)

make
./test_list -v

2. rbtree 介绍

1.1  简介

红黑树是一种自平衡的二叉搜索树。红黑树是有序的。

注: 具体介绍可以我之前的博客参见:http://www.cnblogs.com/wang_yb/archive/2013/04/16/3023892.html 中的 第4节

这里只补充一点,红黑树虽然有些复杂,但是它的查找,插入,删除操作的效率还不错。查找,插入,删除的时间复杂度都是O(log n) n是树中元素数目

1.2 修改部分

为了是 rbtree 更加简单,暂时删除了以下内容:

  1. 删除了函数指针的定义 typedef rb_augment_f
  2. 删除了 rb_augment_insert
  3. 删除了 rb_augment_erase_begin
  4. 删除了 rb_augment_erase_end
  5. 删除了 rb_link_node

1.3 rbtree.h 对外接口

注意: rbtree 的对外接口中没有插入node的接口,只有在插入node之后改变node颜色的接口

可能是由于node的顺序因具体struct而异,所以没法统一实现

No.

主要 函数

说明

1. rb_set_parent 设置父节点的地址
2. rb_set_color 设置节点颜色
3. rb_init_node 初始化节点
4. rb_insert_color 设置新插入节点的颜色
5. rb_erase 删除一个节点
6. rb_next 返回当前节点的下一个节点
7. rb_prev 返回当前节点的上一个节点
8. rb_first 返回第一个叶子节点(也就是最左边的叶子节点)
9. rb_last 返回最后一个叶子节点(也就是最右边的叶子节点)
10. rb_replace_node 替换rbtree中的一个node(只是简单的替换,没有管替换的颜色对不对,数据的顺序对不对)

No.

主要 宏

说明

1. rb_parent 获取父节点的地址
2. rb_color 节点的颜色
3. rb_is_red 是否红节点
4. rb_is_black 是否黑节点
5. rb_set_red 设置节点为红色
6. rb_set_black 设置节点为黑色
7. RB_ROOT 初始化根节点
8. rb_entry 获取包含rbtree node的struct
9. RB_EMPTY_ROOT 判断是否只有根节点
10. RB_EMPTY_NODE 判断节点是否刚初始化,还没有加到树中
11. RB_CLEAR_NODE 设置节点的父节点也指向自己

1.4 rbtree.c 补充说明

rbtree.c 中函数都比较简单,比较复杂的是 rb_insert_color 和 rb_erase

这2个函数还涉及其它未公开的函数 __rb_rotate_left, __rb_rotate_right, __rb_erase_color

1. __rb_rotate_left  : 左旋,即,以参数 node 为中心点,逆时针旋转。左旋可以调整右子树的高度

下面的4副图演示了左旋时,struct rb_node 的 left 和 right 指针的变化。

下图是最复杂的一种情况,即所有相关节点的左右子树不为空的情况

2. __rb_rotate_right : 右旋,即,以参数 node 为中心点,顺时针旋转。右旋可以调整左子树的高度

下面的4副图演示了右旋时,struct rb_node 的 left 和 right 指针的变化。

下图是最复杂的一种情况,即所有相关节点的左右子树不为空的情况

3. rb_erase : 删除节点,调用 __rb_erase_color 调整颜色

下图演示删除节点时,struct rb_node 的 left 和 right 指针的变化。

下图是最复杂的一种情况,即所有相关节点的左右子树不为空的情况

4. __rb_erase_color  : 删除节点后,调整被删除节点后节点的颜色

被删除节点 A 的位置由被删除节点的下一个节点 B(即被删除节点的右子树中最左的节点)替换。

调整的颜色就是 B 节点的 child 和 parent

删除时各种情况的分析参见:http://zh.wikipedia.org/wiki/红黑树

5. rb_insert_color   : 设置新插入节点的颜色,调整rbtree的平衡

插入的位置需要自己定义,这个函数只是调整插入后节点的颜色

插入时各种情况的分析参见:http://zh.wikipedia.org/wiki/红黑树

1.5 使用示例

构造如下场景,用来测试上述列出的所有的 rbtree 操作:

1. 构造用来测试的 struct:(为了使得测试结果一目了然,struct尽量简单)

struct test_struct {
int num;
struct rb_node node;
};

2. 逐个函数进行测试,使用测试框架 cppUTest

3. 宏 相关的暂时没有测试

4. 运行测试非常简单(前提是得安装 cpputest)

make
./test_rbtree -v

相关测试代码下载

Kernel数据结构移植(list和rbtree)的更多相关文章

  1. kernel 4.4.12 移植 HUAWEI MU609 Mini PCIe Module

    首先请参考 http://www.cnblogs.com/chenfulin5/p/6951290.html 上一章刚讲了 kernel 3.2.0 移植 MU609 这一章记录新版kernel 的移 ...

  2. 红黑树(三)之 Linux内核中红黑树的经典实现

    概要 前面分别介绍了红黑树的理论知识 以及 通过C语言实现了红黑树.本章继续会红黑树进行介绍,下面将Linux 内核中的红黑树单独移植出来进行测试验证.若读者对红黑树的理论知识不熟悉,建立先学习红黑树 ...

  3. Linux数据对齐

    编写可移植代码而值得考虑的最后一个问题是如何存取不对齐的数据 -- 例如, 如何读取 一个存储于一个不是 4 字节倍数的地址的 4 字节值. i386 用户常常存取不对齐数据项, 但是不是所有的体系允 ...

  4. Nand Flash,Nor Flash,CFI Flash,SPI Flash 之间的关系

    前言:    在嵌入式开发中,如uboot的移植,kernel的移植都需要对Flash 有基本的了解.下面细说一下标题中的中Flash中的关系 一,Flash的内存存储结构    flash按照内部存 ...

  5. linux查看硬件信息及驱动设备相关整理

    查看声卡设备:cat /proc/asound/cards 查看USB设备:cat /proc/bus/usb/devices 常用命令整理如下:用硬件检测程序kuduz探测新硬件:service k ...

  6. 标准C++中的STL容器类简单介绍

    SGI -- Silicon Graphics[Computer System] Inc.硅图[计算机系统]公司. STL -- Standard Template Library 标准模板库.   ...

  7. 【linux】内核源代码下载与阅读

      原创,转载时请注明,谢谢.邮箱:tangzhongp@163.com 博客园地址:http://www.cnblogs.com/embedded-tzp Csdn博客地址:http://blog. ...

  8. Android内存分析命令(转)

    一.概述 1.1 内存指标概念 Item 全称 含义 等价 USS Unique Set Size 物理内存 进程独占的内存 PSS Proportional Set Size 物理内存 PSS= U ...

  9. DM368 UBL和u-boot的裁剪

    转载:http://blog.csdn.net/olei_oleitao/article/details/7919307   一.DM36X的BOOT过程介绍 DM36x的BOOT过程和DM6446. ...

随机推荐

  1. [React] react.js的一些库和用法

    React性能优化 记录一次利用 Timeline/Performance工具进行 React性能优化的真实案例 http://www.jianshu.com/p/9b0e9ef0a607 React ...

  2. 【转载】 IIS服务器防盗链设置

    在实际运行的服务器环境中,我们自己网站中的资源一般不希望被外部网站引用,被外部网站引用IIS网站中的资源文件,一是会加重了服务器的负担,二是占用了你自己服务器的外网带宽资源,因此我们希望防止盗链这种情 ...

  3. 数据库部分(MySql)_3

    表设计之关联关系 一对一:有两张表A和B,A表中有一条数据对应B表中的一条数据称为一对一: 应用场景:用户表和用户扩展表,商品表和商品信息扩展表: 如何建立关系:在从表中添加一个外键字段指向主表的主键 ...

  4. C#文件夹权限操作整理

    using System.Security.AccessControl; using System.IO; using System.Security.Principal; 取得目录的访问控制和审核安 ...

  5. oracle表空间大小的限制和DB_BLOCK_SIZE的概念

    之前接触的项目表空间最大也不超过10G,所以导入数据库时一直使用导入本地的oracle数据库文件的方法,即根据dmp文件大小设置一个数据文件,设定表空间最大值. --创建表空间,数据文件为'F:\ap ...

  6. JavaWeb学习日记----表单提交方式

    1.表单提交方式 (1) 使用input控件中的submit提交 代码如下: <!DOCTYPE html> <html lang="en"> <he ...

  7. Redirect all output to file

    That part is written to stderr, use 2> to redirect it. For example: foo > stdout.txt 2> std ...

  8. kafka指定partition的分区规则

    博客地址:https://www.cnblogs.com/gnivor/p/5318319.html

  9. Java自动内存管理机制学习(一):Java内存区域与内存溢出异常

    备注:本文引用自<深入理解Java虚拟机第二版> 2.1 运行时数据区域 Java虚拟机在执行Java程序的过程中把它所管理的内存划分为若干个不同的数据区域.这些区域都有各自的用途,以及创 ...

  10. 前端常见算法的JS实现

    1.冒泡排序 function bubbleSort(arr){ var i = 0, j = 0; for(i=1; i<arr.length; i++){ for(j=0; j<=ar ...