Linux kernel rb-tree (4)

这篇继续分析API的实现细节,本文讲rb_insert_color

  • 调用示例
static void __insert_vmap_area(struct vmap_area *va)
{
        struct rb_node **p = &vmap_area_root.rb_node;
        struct rb_node *parent = NULL;
        struct rb_node *tmp;

        // 遍历 rb_tree 
        while (*p) {
                struct vmap_area *tmp_va;

                parent = *p;
                tmp_va = rb_entry(parent, struct vmap_area, rb_node);

                // 比较,找到合适的插入位置 
                if (va->va_start < tmp_va->va_end)
                        p = &(*p)->rb_left;
                else if (va->va_end > tmp_va->va_start)
                        p = &(*p)->rb_right;
                else
                        BUG();
        }

        // 插入节点
        rb_link_node(&va->rb_node, parent, p);
        // 上色
        rb_insert_color(&va->rb_node, &vmap_area_root);

        //...
}
  • 实现细节
void rb_insert_color(struct rb_node *node, struct rb_root *root)
{
        __rb_insert(node, root, dummy_rotate);
}
EXPORT_SYMBOL(rb_insert_color);

static inline void dummy_rotate(struct rb_node *old, struct rb_node *new) {}


/**
 * 功能:
 *          没什么内容,简单的封装__rb_insert,多了个空函数dummy_rotate
 * 参数:
 *          node - 待插入的节点(struct rb_node *) 跟rb_link_node一样
 *          root - 待插入的树的根节点地址 (struct rb_root *)
 **/

进入__rb_insert

// lib/rbtree.c
static __always_inline void
__rb_insert(struct rb_node *node, struct rb_root *root,
            void (*augment_rotate)(struct rb_node *old, struct rb_node *new))
{
        struct rb_node *parent = rb_red_parent(node), *gparent, *tmp;
        // ...
}
/*
 * 参数:
 *      node - 待插入的节点(struct rb_node *) 跟 rb_link_node 一样
 *      root - 待插入的树的根节点地址 (struct rb_root *)
 *
 * 说明:
 *      - 上来先定义三个 rb_node 指针
 *      - rb_red_parent 并没有做什么事情,就是取 rb_node->__rb_parent_color
 *      - 由于参数 node 是 rb_link_node 处理过的 node, 所以取出的就是 parent 的地址
 *
 * */

static inline struct rb_node *rb_red_parent(struct rb_node *red)
{
        return (struct rb_node *)red->__rb_parent_color;
}


继续往下看

static __always_inline void
__rb_insert(struct rb_node *node, struct rb_root *root,
            void (*augment_rotate)(struct rb_node *old, struct rb_node *new))
{
        struct rb_node *parent = rb_red_parent(node), *gparent, *tmp;

        while (true) {
                /*
                 * Loop invariant: node is red
                 *
                 * If there is a black parent, we are done.
                 * Otherwise, take some corrective action as we don't
                 * want a red root or two consecutive red nodes.
                 */
                if (!parent) {
                        rb_set_parent_color(node, NULL, RB_BLACK);
                        break;
                } else if (rb_is_black(parent))
                        break;
                // ...
        }
}
/*
 * 说明:
 *      - 函数的主体是一个大循环体
 *      - 首先判断如果 parent 为空,表示 node 是插入的第一个节点,调用rb_set_parent_color 进行设置并退出循环
 *      - 如果 parent 不为空就判断 parent 的颜色是否是黑色,如果是就不需要调整直接退出循环(为什么呢?)
 *      - 回顾一下红黑树的几条原则,当 parent 为红色的时候,子节点必须是黑色, 而当parent为黑色的时候就没有限制
 *      - 而 rb_link_node 中设置 parent 地址的同时,实际上颜色已经设置为红色了(最低位是0),所以这里也没有再调用rb_set_parent_color
 *      - 大概这也是上一步中那个函数叫 rb_red_parent 的原因
 **/
static inline void rb_set_parent_color(struct rb_node *rb,
                                       struct rb_node *p, int color)
{
        rb->__rb_parent_color = (unsigned long)p | color;
}

/*
 * 参数:
 *      rb - 当前需要设置的 node
 *      p - 当前 node 的 parent 地址
 *      color - 当前 node 的颜色
 *
 * 说明:
 *      - rb_set_parent_color 把当前节点的 __rb_parent_color 设置为父节点地址和 color 的`或运算`结果
 *      - rb_set_parent_color 这个函数名的含义其实是设置当前node的parent和当前node的color,而不是设置当前node的parent的color
 *
 * */

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

推荐阅读更多精彩内容

  • 这篇开始分析API的实现细节,本文讲rb_link_node ,非常简单 调用示例(还是之前的) 实现 callb...
    _invincible_阅读 332评论 0 0
  • 写了个简单的Demo,使用内核提供的接口创建了一个红黑树。 API Demo code output visual...
    _invincible_阅读 298评论 0 0
  • 一、快捷键 ctr+b 执行ctr+/ 单行注释ctr+c ...
    o_8319阅读 5,803评论 2 16
  • 此排序没有任何优先级或者重要程度。此笔记只为记录平时开发中碰到的经常用到确不太注意的一些问题,每次用过就忘记,还要...
    闲庭阅读 3,280评论 0 37
  • 推荐指数: 6.0 书籍主旨关键词:特权、焦点、注意力、语言联想、情景联想 观点: 1.统计学现在叫数据分析,社会...
    Jenaral阅读 5,714评论 0 5