Linux內(nèi)核中的radix tree小記

基樹(shù)

內(nèi)核中的基樹(shù)的節(jié)點(diǎn)蹲盘,使用struct radix_tree_node來(lái)表示股毫,其源代碼如下:

struct radix_tree_node {
    unsigned int    height;     /* Height from the bottom */
    unsigned int    count;      /* 子節(jié)點(diǎn)的個(gè)數(shù) */
    union {
        struct radix_tree_node *parent; /* Used when ascending tree */
        struct rcu_head rcu_head;   /* Used when freeing node */
    };
    void __rcu  *slots[RADIX_TREE_MAP_SIZE];
    unsigned long   tags[RADIX_TREE_MAX_TAGS][RADIX_TREE_TAG_LONGS];
};

/* root tags are stored in gfp_mask, shifted by __GFP_BITS_SHIFT */
struct radix_tree_root {
    unsigned int        height;
    gfp_t           gfp_mask;
    struct radix_tree_node  __rcu *rnode;
};

slots是指向各個(gè)孩子節(jié)點(diǎn)的指針,RADIX_TREE_MAP_SIZE通常為64(跟宏定義有關(guān)召衔,我們以64為例來(lái)說(shuō)明)铃诬。

tags顧名思義是標(biāo)簽,代表此節(jié)點(diǎn)的所有孩子節(jié)點(diǎn)的標(biāo)簽苍凛。tags是二維數(shù)組趣席,RADIX_TREE_MAX_TAGS定義為3,即最多支持3種標(biāo)簽醇蝴。RADIX_TREE_TAG_LONGS的長(zhǎng)度使得可以放下所有子節(jié)點(diǎn)的tag(一個(gè)tag占1位)宣肚。

一個(gè)典型的基樹(shù)如下圖所示:


height=3的基樹(shù)

不得不說(shuō),本來(lái)簡(jiǎn)單的數(shù)據(jù)結(jié)構(gòu)悠栓,加上了RCU的機(jī)制后霉涨,總是有點(diǎn)難以理解。

初始化

跟Linux內(nèi)核中其它的數(shù)據(jù)結(jié)構(gòu)類(lèi)似闸迷,兩種初始化方式:

#define RADIX_TREE(name, mask) \
    struct radix_tree_root name = RADIX_TREE_INIT(mask)

#define INIT_RADIX_TREE(root, mask)                 \
do {                                    \
    (root)->height = 0;                     \
    (root)->gfp_mask = (mask);                  \
    (root)->rnode = NULL;                       \
} while (0)

查找

/*
 * is_slot == 1 : search for the slot.
 * is_slot == 0 : search for the node.
 */
static void *radix_tree_lookup_element(struct radix_tree_root *root,
                unsigned long index, int is_slot)
{
    unsigned int height, shift;
    struct radix_tree_node *node, **slot;

    node = rcu_dereference_raw(root->rnode);
    if (node == NULL)
        return NULL;

    if (!radix_tree_is_indirect_ptr(node)) {
        if (index > 0)
            return NULL;
        return is_slot ? (void *)&root->rnode : node;
    }
    node = indirect_to_ptr(node);

    height = node->height;
    if (index > radix_tree_maxindex(height))
        return NULL;

    shift = (height-1) * RADIX_TREE_MAP_SHIFT;

    do {
        slot = (struct radix_tree_node **)
            (node->slots + ((index>>shift) & RADIX_TREE_MAP_MASK));
        node = rcu_dereference_raw(*slot);
        if (node == NULL)
            return NULL;

        shift -= RADIX_TREE_MAP_SHIFT;
        height--;
    } while (height > 0);

    return is_slot ? (void *)slot : indirect_to_ptr(node);
}

/**
 *  radix_tree_lookup    -    perform lookup operation on a radix tree
 *  @root:      radix tree root
 *  @index:     index key
 *
 *  Lookup the item at the position @index in the radix tree @root.
 *
 *  This function can be called under rcu_read_lock, however the caller
 *  must manage lifetimes of leaf nodes (eg. RCU may also be used to free
 *  them safely). No RCU barriers are required to access or modify the
 *  returned item, however.
 */
void *radix_tree_lookup(struct radix_tree_root *root, unsigned long index)
{
    return radix_tree_lookup_element(root, index, 0);
}

查找的過(guò)程嵌纲,就是把index從高位開(kāi)始,每6位(以64位機(jī)器為例)為一個(gè)單位腥沽,分隔成一個(gè)個(gè)的slot index逮走,然后從radix樹(shù)的根往下搜索的過(guò)程,整體還是比較好理解的今阳。暫時(shí)忽略indirect_to_prt师溅,rcu_dereference_raw這幾個(gè)函數(shù)茅信,這不影響對(duì)整體流程的理解。

插入

/**
 *  radix_tree_insert    -    insert into a radix tree
 *  @root:      radix tree root
 *  @index:     index key
 *  @item:      item to insert
 *
 *  Insert an item into the radix tree at position @index.
 */
int radix_tree_insert(struct radix_tree_root *root,
            unsigned long index, void *item)
{
    struct radix_tree_node *node = NULL, *slot;
    unsigned int height, shift;
    int offset;
    int error;

    BUG_ON(radix_tree_is_indirect_ptr(item));

    /* Make sure the tree is high enough.  */
    if (index > radix_tree_maxindex(root->height)) {
        error = radix_tree_extend(root, index);
        if (error)
            return error;
    }

    slot = indirect_to_ptr(root->rnode);

    height = root->height;
    shift = (height-1) * RADIX_TREE_MAP_SHIFT;

    offset = 0;         /* uninitialised var warning */
    while (height > 0) {
        if (slot == NULL) {
            /* Have to add a child node.  */
            if (!(slot = radix_tree_node_alloc(root)))
                return -ENOMEM;
            slot->height = height;
            slot->parent = node;
            if (node) {
                rcu_assign_pointer(node->slots[offset], slot);
                node->count++;
            } else
                rcu_assign_pointer(root->rnode, ptr_to_indirect(slot));
        }

        /* Go a level down */
        offset = (index >> shift) & RADIX_TREE_MAP_MASK;
        node = slot;
        slot = node->slots[offset];
        shift -= RADIX_TREE_MAP_SHIFT;
        height--;
    }

    if (slot != NULL)
        return -EEXIST;

    if (node) {
        node->count++;
        rcu_assign_pointer(node->slots[offset], item);
        BUG_ON(tag_get(node, 0, offset));
        BUG_ON(tag_get(node, 1, offset));
    } else {
        rcu_assign_pointer(root->rnode, item);
        BUG_ON(root_tag_get(root, 0));
        BUG_ON(root_tag_get(root, 1));
    }

    return 0;
}

插入的過(guò)程跟查找的過(guò)程非常相似其實(shí)墓臭,就是沿著樹(shù)從上到下地找蘸鲸,某個(gè)位置沒(méi)有元素就創(chuàng)建元素,直到出錯(cuò)或者把元素放到指定的slot中窿锉。

Tag

內(nèi)核的radix tree支持Tag功能酌摇,就是給某個(gè)元素打一個(gè)tag。這個(gè)tag會(huì)影響該元素到基樹(shù)根上所有元素嗡载。這樣通過(guò)查看根元素是否有這個(gè)tag窑多,就能判斷根元素下的子元素中是否存在這種tag的元素。另外內(nèi)核的基樹(shù)提供了迭代所有設(shè)置過(guò)tag的元素的方法:radix_tree_for_each_tagged洼滚。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末埂息,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子遥巴,更是在濱河造成了極大的恐慌千康,老刑警劉巖,帶你破解...
    沈念sama閱讀 221,576評(píng)論 6 515
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件铲掐,死亡現(xiàn)場(chǎng)離奇詭異拾弃,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī)迹炼,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 94,515評(píng)論 3 399
  • 文/潘曉璐 我一進(jìn)店門(mén)砸彬,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái),“玉大人斯入,你說(shuō)我怎么就攤上這事砂碉。” “怎么了刻两?”我有些...
    開(kāi)封第一講書(shū)人閱讀 168,017評(píng)論 0 360
  • 文/不壞的土叔 我叫張陵增蹭,是天一觀的道長(zhǎng)。 經(jīng)常有香客問(wèn)我磅摹,道長(zhǎng)滋迈,這世上最難降的妖魔是什么? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 59,626評(píng)論 1 296
  • 正文 為了忘掉前任户誓,我火速辦了婚禮饼灿,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘帝美。我一直安慰自己碍彭,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 68,625評(píng)論 6 397
  • 文/花漫 我一把揭開(kāi)白布。 她就那樣靜靜地躺著庇忌,像睡著了一般舞箍。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上皆疹,一...
    開(kāi)封第一講書(shū)人閱讀 52,255評(píng)論 1 308
  • 那天疏橄,我揣著相機(jī)與錄音,去河邊找鬼略就。 笑死捎迫,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的残制。 我是一名探鬼主播立砸,決...
    沈念sama閱讀 40,825評(píng)論 3 421
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼掖疮,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼初茶!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起浊闪,我...
    開(kāi)封第一講書(shū)人閱讀 39,729評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤恼布,失蹤者是張志新(化名)和其女友劉穎,沒(méi)想到半個(gè)月后搁宾,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體折汞,經(jīng)...
    沈念sama閱讀 46,271評(píng)論 1 320
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 38,363評(píng)論 3 340
  • 正文 我和宋清朗相戀三年盖腿,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了爽待。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 40,498評(píng)論 1 352
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡翩腐,死狀恐怖鸟款,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情茂卦,我是刑警寧澤何什,帶...
    沈念sama閱讀 36,183評(píng)論 5 350
  • 正文 年R本政府宣布,位于F島的核電站等龙,受9級(jí)特大地震影響处渣,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜蛛砰,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,867評(píng)論 3 333
  • 文/蒙蒙 一罐栈、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧泥畅,春花似錦荠诬、人聲如沸。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 32,338評(píng)論 0 24
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)望迎。三九已至,卻和暖如春凌外,著一層夾襖步出監(jiān)牢的瞬間辩尊,已是汗流浹背。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 33,458評(píng)論 1 272
  • 我被黑心中介騙來(lái)泰國(guó)打工康辑, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留摄欲,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 48,906評(píng)論 3 376
  • 正文 我出身青樓疮薇,卻偏偏與公主長(zhǎng)得像胸墙,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子按咒,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,507評(píng)論 2 359