Convert fib_trie to use Read Copy Update.  Since all updates to
FIB happen under RTNL mutex (semaphore), we don't need fib_lock
for writing.

This version is different from earlier RCU patches because:
        * only need hlist_add_before_rcu by recoding insert_leaf_info
        * only use RCU for read path, update stuff already is RTNL protected
        * need RCU assign for parent in update cases, but not when initializing
          unlinked node.
        * need synchronize_rcu in the replace case before deleting
          alias info

Signed-off-by: Stephen Hemminger <[EMAIL PROTECTED]>

---

 include/linux/list.h     |   10 +
 net/ipv4/fib_lookup.h    |    1 
 net/ipv4/fib_semantics.c |    3 
 net/ipv4/fib_trie.c      |  331 ++++++++++++++++++++++------------------------
 4 files changed, 173 insertions(+), 172 deletions(-)

diff --git a/include/linux/list.h b/include/linux/list.h
--- a/include/linux/list.h
+++ b/include/linux/list.h
@@ -634,6 +634,16 @@ static inline void hlist_add_after(struc
                next->next->pprev  = &next->next;
 }
 
+static inline void hlist_add_before_rcu(struct hlist_node *n,
+                                       struct hlist_node *next)
+{
+       n->pprev = next->pprev;
+       n->next = next;
+       smp_wmb();
+       next->pprev = &n->next;
+       *(n->pprev) = n;
+}
+
 #define hlist_entry(ptr, type, member) container_of(ptr,type,member)
 
 #define hlist_for_each(pos, head) \
diff --git a/net/ipv4/fib_lookup.h b/net/ipv4/fib_lookup.h
--- a/net/ipv4/fib_lookup.h
+++ b/net/ipv4/fib_lookup.h
@@ -7,6 +7,7 @@
 
 struct fib_alias {
        struct list_head        fa_list;
+       struct rcu_head rcu;
        struct fib_info         *fa_info;
        u8                      fa_tos;
        u8                      fa_type;
diff --git a/net/ipv4/fib_semantics.c b/net/ipv4/fib_semantics.c
--- a/net/ipv4/fib_semantics.c
+++ b/net/ipv4/fib_semantics.c
@@ -854,6 +854,7 @@ failure:
        return NULL;
 }
 
+/* Note! fib_semantic_match intentionally uses  RCU list functions. */
 int fib_semantic_match(struct list_head *head, const struct flowi *flp,
                       struct fib_result *res, __u32 zone, __u32 mask, 
                        int prefixlen)
@@ -861,7 +862,7 @@ int fib_semantic_match(struct list_head 
        struct fib_alias *fa;
        int nh_sel = 0;
 
-       list_for_each_entry(fa, head, fa_list) {
+       list_for_each_entry_rcu(fa, head, fa_list) {
                int err;
 
                if (fa->fa_tos &&
diff --git a/net/ipv4/fib_trie.c b/net/ipv4/fib_trie.c
--- a/net/ipv4/fib_trie.c
+++ b/net/ipv4/fib_trie.c
@@ -43,7 +43,7 @@
  *             2 of the License, or (at your option) any later version.
  */
 
-#define VERSION "0.325"
+#define VERSION "0.401"
 
 #include <linux/config.h>
 #include <asm/uaccess.h>
@@ -62,6 +62,7 @@
 #include <linux/netdevice.h>
 #include <linux/if_arp.h>
 #include <linux/proc_fs.h>
+#include <linux/rcupdate.h>
 #include <linux/skbuff.h>
 #include <linux/netlink.h>
 #include <linux/init.h>
@@ -81,22 +82,19 @@
 #define MASK_PFX(k, l) (((l)==0)?0:(k >> (KEYLENGTH-l)) << (KEYLENGTH-l))
 #define TKEY_GET_MASK(offset, bits) (((bits)==0)?0:((t_key)(-1) << (KEYLENGTH 
- bits) >> offset))
 
-static DEFINE_RWLOCK(fib_lock);
-
 typedef unsigned int t_key;
 
 #define T_TNODE 0
 #define T_LEAF  1
 #define NODE_TYPE_MASK 0x1UL
 #define NODE_PARENT(node) \
-       ((struct tnode *)((node)->parent & ~NODE_TYPE_MASK))
-#define NODE_SET_PARENT(node, ptr) \
-       ((node)->parent = (((unsigned long)(ptr)) | \
-                     ((node)->parent & NODE_TYPE_MASK)))
-#define NODE_INIT_PARENT(node, type) \
-       ((node)->parent = (type))
-#define NODE_TYPE(node) \
-       ((node)->parent & NODE_TYPE_MASK)
+       ((struct tnode *)rcu_dereference(((node)->parent & ~NODE_TYPE_MASK)))
+
+#define NODE_TYPE(node) ((node)->parent & NODE_TYPE_MASK)
+
+#define NODE_SET_PARENT(node, ptr)             \
+       rcu_assign_pointer((node)->parent,      \
+                          ((unsigned long)(ptr)) | NODE_TYPE(node))
 
 #define IS_TNODE(n) (!(n->parent & T_LEAF))
 #define IS_LEAF(n) (n->parent & T_LEAF)
@@ -110,10 +108,12 @@ struct leaf {
        t_key key;
        unsigned long parent;
        struct hlist_head list;
+       struct rcu_head rcu;
 };
 
 struct leaf_info {
        struct hlist_node hlist;
+       struct rcu_head rcu;
        int plen;
        struct list_head falh;
 };
@@ -125,6 +125,7 @@ struct tnode {
        unsigned short bits:5;          /* 2log(KEYLENGTH) bits needed */
        unsigned short full_children;   /* KEYLENGTH bits needed */
        unsigned short empty_children;  /* KEYLENGTH bits needed */
+       struct rcu_head rcu;
        struct node *child[0];
 };
 
@@ -172,7 +173,7 @@ static inline struct node *tnode_get_chi
 {
        BUG_ON(i >= 1 << tn->bits);
 
-       return tn->child[i];
+       return rcu_dereference(tn->child[i]);
 }
 
 static inline int tnode_child_length(const struct tnode *tn)
@@ -213,14 +214,6 @@ static inline int tkey_mismatch(t_key a,
        return i;
 }
 
-/* Candidate for fib_semantics */
-
-static void fn_free_alias(struct fib_alias *fa)
-{
-       fib_release_info(fa->fa_info);
-       kmem_cache_free(fn_alias_kmem, fa);
-}
-
 /*
   To understand this stuff, an understanding of keys and all their bits is 
   necessary. Every node in the trie has a key associated with it, but not 
@@ -292,53 +285,57 @@ static inline void check_tnode(const str
 static int halve_threshold = 25;
 static int inflate_threshold = 50;
 
-static struct leaf *leaf_new(void)
+
+static void __alias_free_mem(struct rcu_head *head)
 {
-       struct leaf *l = kmalloc(sizeof(struct leaf),  GFP_KERNEL);
-       if (l) {
-               NODE_INIT_PARENT(l, T_LEAF);
-               INIT_HLIST_HEAD(&l->list);
-       }
-       return l;
+       struct fib_alias *fa = container_of(head, struct fib_alias, rcu);
+       kmem_cache_free(fn_alias_kmem, fa);
 }
 
-static struct leaf_info *leaf_info_new(int plen)
+static inline void alias_free_mem_rcu(struct fib_alias *fa)
 {
-       struct leaf_info *li = kmalloc(sizeof(struct leaf_info),  GFP_KERNEL);
-
-       if (!li)
-               return NULL;
+       call_rcu(&fa->rcu, __alias_free_mem);
+}
 
-       li->plen = plen;
-       INIT_LIST_HEAD(&li->falh);
+static void __leaf_free_rcu(struct rcu_head *head)
+{
+       kfree(container_of(head, struct leaf, rcu));
+}
 
-       return li;
+static inline void free_leaf(struct leaf *leaf)
+{
+       call_rcu(&leaf->rcu, __leaf_free_rcu);
 }
 
-static inline void free_leaf(struct leaf *l)
+static void __leaf_info_free_rcu(struct rcu_head *head)
 {
-       kfree(l);
+       kfree(container_of(head, struct leaf_info, rcu));
 }
 
-static inline void free_leaf_info(struct leaf_info *li)
+static inline void free_leaf_info(struct leaf_info *leaf)
 {
-       kfree(li);
+       call_rcu(&leaf->rcu, __leaf_info_free_rcu);
 }
 
 static struct tnode *tnode_alloc(unsigned int size)
 {
-       if (size <= PAGE_SIZE) {
-               return kmalloc(size, GFP_KERNEL);
-       } else {
-               return (struct tnode *)
-                       __get_free_pages(GFP_KERNEL, get_order(size));
-       }
+       struct page *pages;
+
+       if (size <= PAGE_SIZE)
+               return kcalloc(size, 1, GFP_KERNEL);
+
+       pages = alloc_pages(GFP_KERNEL|__GFP_ZERO, get_order(size));
+       if (!pages)
+               return NULL;
+
+       return page_address(pages);
 }
 
-static void __tnode_free(struct tnode *tn)
+static void __tnode_free_rcu(struct rcu_head *head)
 {
+       struct tnode *tn = container_of(head, struct tnode, rcu);
        unsigned int size = sizeof(struct tnode) +
-                               (1 << tn->bits) * sizeof(struct node *);
+               (1 << tn->bits) * sizeof(struct node *);
 
        if (size <= PAGE_SIZE)
                kfree(tn);
@@ -346,6 +343,31 @@ static void __tnode_free(struct tnode *t
                free_pages((unsigned long)tn, get_order(size));
 }
 
+static inline void tnode_free(struct tnode *tn)
+{
+       call_rcu(&tn->rcu, __tnode_free_rcu);
+}
+
+static struct leaf *leaf_new(void)
+{
+       struct leaf *l = kmalloc(sizeof(struct leaf),  GFP_KERNEL);
+       if (l) {
+               l->parent = T_LEAF;
+               INIT_HLIST_HEAD(&l->list);
+       }
+       return l;
+}
+
+static struct leaf_info *leaf_info_new(int plen)
+{
+       struct leaf_info *li = kmalloc(sizeof(struct leaf_info),  GFP_KERNEL);
+       if (li) {
+               li->plen = plen;
+               INIT_LIST_HEAD(&li->falh);
+       }
+       return li;
+}
+
 static struct tnode* tnode_new(t_key key, int pos, int bits)
 {
        int nchildren = 1<<bits;
@@ -354,7 +376,7 @@ static struct tnode* tnode_new(t_key key
 
        if (tn) {
                memset(tn, 0, sz);
-               NODE_INIT_PARENT(tn, T_TNODE);
+               tn->parent = T_TNODE;
                tn->pos = pos;
                tn->bits = bits;
                tn->key = key;
@@ -367,17 +389,6 @@ static struct tnode* tnode_new(t_key key
        return tn;
 }
 
-static void tnode_free(struct tnode *tn)
-{
-       if (IS_LEAF(tn)) {
-               free_leaf((struct leaf *)tn);
-               pr_debug("FL %p \n", tn);
-       } else {
-               __tnode_free(tn);
-               pr_debug("FT %p \n", tn);
-       }
-}
-
 /*
  * Check whether a tnode 'n' is "full", i.e. it is an internal node
  * and no bits are skipped. See discussion in dyntree paper p. 6
@@ -403,13 +414,11 @@ static inline void put_child(struct trie
 
 static void tnode_put_child_reorg(struct tnode *tn, int i, struct node *n, int 
wasfull)
 {
-       struct node *chi;
+       struct node *chi = tn->child[i];
        int isfull;
 
        BUG_ON(i >= 1<<tn->bits);
 
-       write_lock_bh(&fib_lock);
-       chi = tn->child[i];
 
        /* update emptyChildren */
        if (n == NULL && chi != NULL)
@@ -430,8 +439,7 @@ static void tnode_put_child_reorg(struct
        if (n)
                NODE_SET_PARENT(n, tn);
 
-       tn->child[i] = n;
-       write_unlock_bh(&fib_lock);
+       rcu_assign_pointer(tn->child[i], n);
 }
 
 static struct node *resize(struct trie *t, struct tnode *tn)
@@ -456,17 +464,12 @@ static struct node *resize(struct trie *
                for (i = 0; i < tnode_child_length(tn); i++) {
                        struct node *n;
 
-                       write_lock_bh(&fib_lock);
                        n = tn->child[i];
-                       if (!n) {
-                               write_unlock_bh(&fib_lock);
+                       if (!n)
                                continue;
-                       }
 
                        /* compress one level */
-                       NODE_INIT_PARENT(n, NODE_TYPE(n));
-
-                       write_unlock_bh(&fib_lock);
+                       NODE_SET_PARENT(n, NULL);
                        tnode_free(tn);
                        return n;
                }
@@ -577,24 +580,17 @@ static struct node *resize(struct trie *
 
 
        /* Only one child remains */
-
        if (tn->empty_children == tnode_child_length(tn) - 1)
                for (i = 0; i < tnode_child_length(tn); i++) {
                        struct node *n;
 
-                       write_lock_bh(&fib_lock);
-
                        n = tn->child[i];
-                       if (!n) {
-                               write_unlock_bh(&fib_lock);
+                       if (!n)
                                continue;
-                       }
 
                        /* compress one level */
 
-                       NODE_INIT_PARENT(n, NODE_TYPE(n));
-
-                       write_unlock_bh(&fib_lock);
+                       NODE_SET_PARENT(n, NULL);
                        tnode_free(tn);
                        return n;
                }
@@ -843,7 +839,7 @@ static struct leaf_info *find_leaf_info(
        struct hlist_node *node;
        struct leaf_info *li;
 
-       hlist_for_each_entry(li, node, head, hlist)
+       hlist_for_each_entry_rcu(li, node, head, hlist)
                if (li->plen == plen)
                        return li;
 
@@ -862,26 +858,18 @@ static inline struct list_head * get_fa_
 
 static void insert_leaf_info(struct hlist_head *head, struct leaf_info *new)
 {
-       struct leaf_info *li = NULL, *last = NULL;
+       struct leaf_info *li = NULL;
        struct hlist_node *node;
 
-       write_lock_bh(&fib_lock);
-
-       if (hlist_empty(head)) {
-               hlist_add_head(&new->hlist, head);
-       } else {
+       if (hlist_empty(head))
+               hlist_add_head_rcu(&new->hlist, head);
+       else {
                hlist_for_each_entry(li, node, head, hlist) {
                        if (new->plen > li->plen)
                                break;
-
-                       last = li;
                }
-               if (last)
-                       hlist_add_after(&last->hlist, &new->hlist);
-               else
-                       hlist_add_before(&new->hlist, &li->hlist);
+               hlist_add_before_rcu(&new->hlist, &li->hlist);
        }
-       write_unlock_bh(&fib_lock);
 }
 
 static struct leaf *
@@ -892,7 +880,7 @@ fib_find_node(struct trie *t, u32 key)
        struct node *n;
 
        pos = 0;
-       n = t->trie;
+       n = rcu_dereference(t->trie);
 
        while (n != NULL &&  NODE_TYPE(n) == T_TNODE) {
                tn = (struct tnode *) n;
@@ -1081,7 +1069,7 @@ fib_insert_node(struct trie *t, int *err
                        cindex = tkey_extract_bits(key, tp->pos, tp->bits);
                        put_child(t, (struct tnode *)tp, cindex, (struct node 
*)tn);
                } else {
-                       t->trie = (struct node*) tn; /* First tnode */
+                       rcu_assign_pointer(t->trie, (struct node *)tn); /* 
First tnode */
                        tp = tn;
                }
        }
@@ -1091,7 +1079,8 @@ fib_insert_node(struct trie *t, int *err
                       tp, tp->pos, tp->bits, key, plen);
 
        /* Rebalance the trie */
-       t->trie = trie_rebalance(t, tp);
+
+       rcu_assign_pointer(t->trie, trie_rebalance(t, tp));
 done:
        t->revision++;
 err:
@@ -1166,7 +1155,9 @@ fn_trie_insert(struct fib_table *tb, str
                        struct fib_info *fi_drop;
                        u8 state;
 
-                       write_lock_bh(&fib_lock);
+                       /*
+                        * TODO: Make replace atomic
+                        */
 
                        fi_drop = fa->fa_info;
                        fa->fa_info = fi;
@@ -1175,7 +1166,7 @@ fn_trie_insert(struct fib_table *tb, str
                        state = fa->fa_state;
                        fa->fa_state &= ~FA_S_ACCESSED;
 
-                       write_unlock_bh(&fib_lock);
+                       synchronize_rcu();
 
                        fib_release_info(fi_drop);
                        if (state & FA_S_ACCESSED)
@@ -1227,11 +1218,8 @@ fn_trie_insert(struct fib_table *tb, str
                        goto out_free_new_fa;
        }
 
-       write_lock_bh(&fib_lock);
-
-       list_add_tail(&new_fa->fa_list, (fa ? &fa->fa_list : fa_head));
-
-       write_unlock_bh(&fib_lock);
+       list_add_tail_rcu(&new_fa->fa_list,
+                         (fa ? &fa->fa_list : fa_head));
 
        rt_cache_flush(-1);
        rtmsg_fib(RTM_NEWROUTE, htonl(key), new_fa, plen, tb->tb_id, nlhdr, 
req);
@@ -1256,7 +1244,7 @@ static inline int check_leaf(struct trie
        struct hlist_head *hhead = &l->list;
        struct hlist_node *node;
 
-       hlist_for_each_entry(li, node, hhead, hlist) {
+       hlist_for_each_entry_rcu(li, node, hhead, hlist) {
                i = li->plen;
                mask = ntohl(inet_make_mask(i));
                if (l->key != (key & mask))
@@ -1292,10 +1280,9 @@ fn_trie_lookup(struct fib_table *tb, con
        t_key node_prefix, key_prefix, pref_mismatch;
        int mp;
 
-       n = t->trie;
-
-       read_lock(&fib_lock);
+       rcu_read_lock();
 
+       n = rcu_dereference(t->trie);
        if (!n)
                goto failed;
 
@@ -1465,7 +1452,7 @@ backtrace:
 failed:
        ret = 1;
 found:
-       read_unlock(&fib_lock);
+       rcu_read_unlock();
        return ret;
 }
 
@@ -1503,15 +1490,17 @@ static int trie_leaf_remove(struct trie 
        t->revision++;
        t->size--;
 
+       preempt_disable();
        tp = NODE_PARENT(n);
        tnode_free((struct tnode *) n);
 
        if (tp) {
                cindex = tkey_extract_bits(key, tp->pos, tp->bits);
                put_child(t, (struct tnode *)tp, cindex, NULL);
-               t->trie = trie_rebalance(t, tp);
+               rcu_assign_pointer(t->trie, trie_rebalance(t, tp));
        } else
-               t->trie = NULL;
+               rcu_assign_pointer(t->trie, NULL);
+       preempt_enable();
 
        return 1;
 }
@@ -1527,7 +1516,6 @@ fn_trie_delete(struct fib_table *tb, str
        struct fib_alias *fa, *fa_to_delete;
        struct list_head *fa_head;
        struct leaf *l;
-       int kill_li = 0;
        struct leaf_info *li;
 
 
@@ -1587,18 +1575,12 @@ fn_trie_delete(struct fib_table *tb, str
        l = fib_find_node(t, key);
        li = find_leaf_info(&l->list, plen);
 
-       write_lock_bh(&fib_lock);
-
-       list_del(&fa->fa_list);
+       list_del_rcu(&fa->fa_list);
 
        if (list_empty(fa_head)) {
-               hlist_del(&li->hlist);
-               kill_li = 1;
-       }
-       write_unlock_bh(&fib_lock);
-
-       if (kill_li)
+               hlist_del_rcu(&li->hlist);
                free_leaf_info(li);
+       }
 
        if (hlist_empty(&l->list))
                trie_leaf_remove(t, key);
@@ -1606,7 +1588,8 @@ fn_trie_delete(struct fib_table *tb, str
        if (fa->fa_state & FA_S_ACCESSED)
                rt_cache_flush(-1);
 
-       fn_free_alias(fa);
+       fib_release_info(fa->fa_info);
+       alias_free_mem_rcu(fa);
        return 0;
 }
 
@@ -1618,12 +1601,10 @@ static int trie_flush_list(struct trie *
        list_for_each_entry_safe(fa, fa_node, head, fa_list) {
                struct fib_info *fi = fa->fa_info;
 
-               if (fi && (fi->fib_flags&RTNH_F_DEAD)) {
-                       write_lock_bh(&fib_lock);
-                       list_del(&fa->fa_list);
-                       write_unlock_bh(&fib_lock);
-
-                       fn_free_alias(fa);
+               if (fi && (fi->fib_flags & RTNH_F_DEAD)) {
+                       list_del_rcu(&fa->fa_list);
+                       fib_release_info(fa->fa_info);
+                       alias_free_mem_rcu(fa);
                        found++;
                }
        }
@@ -1641,10 +1622,7 @@ static int trie_flush_leaf(struct trie *
                found += trie_flush_list(t, &li->falh);
 
                if (list_empty(&li->falh)) {
-                       write_lock_bh(&fib_lock);
-                       hlist_del(&li->hlist);
-                       write_unlock_bh(&fib_lock);
-
+                       hlist_del_rcu(&li->hlist);
                        free_leaf_info(li);
                }
        }
@@ -1656,15 +1634,16 @@ static struct leaf *nextleaf(struct trie
        struct node *c = (struct node *) thisleaf;
        struct tnode *p;
        int idx;
+       struct node *trie = rcu_dereference(t->trie);
 
        if (c == NULL) {
-               if (t->trie == NULL)
+               if (trie == NULL)
                        return NULL;
 
-               if (IS_LEAF(t->trie))          /* trie w. just a leaf */
-                       return (struct leaf *) t->trie;
+               if (IS_LEAF(trie))          /* trie w. just a leaf */
+                       return (struct leaf *) trie;
 
-               p = (struct tnode*) t->trie;  /* Start */
+               p = (struct tnode*) trie;  /* Start */
        } else
                p = (struct tnode *) NODE_PARENT(c);
 
@@ -1679,23 +1658,26 @@ static struct leaf *nextleaf(struct trie
 
                last = 1 << p->bits;
                for (idx = pos; idx < last ; idx++) {
-                       if (!p->child[idx])
+                       c = rcu_dereference(p->child[idx]);
+
+                       if (!c)
                                continue;
 
                        /* Decend if tnode */
-                       while (IS_TNODE(p->child[idx])) {
-                               p = (struct tnode*) p->child[idx];
-                               idx = 0;
+                       while (IS_TNODE(c)) {
+                               p = (struct tnode *) c;
+                               idx = 0;
 
                                /* Rightmost non-NULL branch */
                                if (p && IS_TNODE(p))
-                                       while (p->child[idx] == NULL && idx < 
(1 << p->bits)) idx++;
+                                       while (!(c = 
rcu_dereference(p->child[idx]))
+                                              && idx < (1<<p->bits)) idx++;
 
                                /* Done with this tnode? */
-                               if (idx >= (1 << p->bits) || p->child[idx] == 
NULL)
+                               if (idx >= (1 << p->bits) || !c)
                                        goto up;
                        }
-                       return (struct leaf*) p->child[idx];
+                       return (struct leaf *) c;
                }
 up:
                /* No more children go up one step  */
@@ -1745,7 +1727,7 @@ fn_trie_select_default(struct fib_table 
        last_resort = NULL;
        order = -1;
 
-       read_lock(&fib_lock);
+       rcu_read_lock();
 
        l = fib_find_node(t, 0);
        if (!l)
@@ -1758,7 +1740,7 @@ fn_trie_select_default(struct fib_table 
        if (list_empty(fa_head))
                goto out;
 
-       list_for_each_entry(fa, fa_head, fa_list) {
+       list_for_each_entry_rcu(fa, fa_head, fa_list) {
                struct fib_info *next_fi = fa->fa_info;
 
                if (fa->fa_scope != res->scope ||
@@ -1809,7 +1791,7 @@ fn_trie_select_default(struct fib_table 
        }
        trie_last_dflt = last_idx;
  out:;
-       read_unlock(&fib_lock);
+       rcu_read_unlock();
 }
 
 static int fn_trie_dump_fa(t_key key, int plen, struct list_head *fah, struct 
fib_table *tb,
@@ -1823,7 +1805,7 @@ static int fn_trie_dump_fa(t_key key, in
        s_i = cb->args[3];
        i = 0;
 
-       list_for_each_entry(fa, fah, fa_list) {
+       list_for_each_entry_rcu(fa, fah, fa_list) {
                if (i < s_i) {
                        i++;
                        continue;
@@ -1898,7 +1880,7 @@ static int fn_trie_dump(struct fib_table
 
        s_m = cb->args[1];
 
-       read_lock(&fib_lock);
+       rcu_read_lock();
        for (m = 0; m <= 32; m++) {
                if (m < s_m)
                        continue;
@@ -1911,11 +1893,11 @@ static int fn_trie_dump(struct fib_table
                        goto out;
                }
        }
-       read_unlock(&fib_lock);
+       rcu_read_unlock();
        cb->args[1] = m;
        return skb->len;
 out:
-       read_unlock(&fib_lock);
+       rcu_read_unlock();
        return -1;
 }
 
@@ -2016,7 +1998,7 @@ static void printnode_seq(struct seq_fil
                                putspace_seq(seq, indent+2);
                                seq_printf(seq, "{/%d...dumping}\n", i);
 
-                               list_for_each_entry(fa, fa_head, fa_list) {
+                               list_for_each_entry_rcu(fa, fa_head, fa_list) {
                                        putspace_seq(seq, indent+2);
                                        if (fa->fa_info == NULL) {
                                                seq_printf(seq, "Error 
fa_info=NULL\n");
@@ -2056,28 +2038,28 @@ static void printnode_seq(struct seq_fil
 
 static void trie_dump_seq(struct seq_file *seq, struct trie *t)
 {
-       struct node *n = t->trie;
+       struct node *n = rcu_dereference(t->trie);
        int cindex = 0;
        int indent = 1;
        int pend = 0;
        int depth = 0;
        struct tnode *tn;
 
-       read_lock(&fib_lock);
+       rcu_read_lock();
 
        seq_printf(seq, "------ trie_dump of t=%p ------\n", t);
 
        if (!n) {
                seq_printf(seq, "------ trie is empty\n");
 
-               read_unlock(&fib_lock);
+               rcu_read_unlock();
                return;
        }
 
        printnode_seq(seq, indent, n, pend, cindex, 0);
 
        if (!IS_TNODE(n)) {
-               read_unlock(&fib_lock);
+               rcu_read_unlock();
                return;
        }
 
@@ -2088,26 +2070,32 @@ static void trie_dump_seq(struct seq_fil
        depth++;
 
        while (tn && cindex < (1 << tn->bits)) {
-               if (tn->child[cindex]) {
+               struct node *child = rcu_dereference(tn->child[cindex]);
+               if (!child)
+                       cindex++;
+               else {
                        /* Got a child */
+                       printnode_seq(seq, indent, child, pend,
+                                     cindex, tn->bits);
 
-                       printnode_seq(seq, indent, tn->child[cindex], pend, 
cindex, tn->bits);
-                       if (IS_LEAF(tn->child[cindex])) {
+                       if (IS_LEAF(child))
                                cindex++;
-                       } else {
+
+                       else {
                                /*
                                 * New tnode. Decend one level
                                 */
 
                                depth++;
-                               tn = (struct tnode *)tn->child[cindex];
-                               pend = tn->pos + tn->bits;
-                               putspace_seq(seq, indent); seq_printf(seq, 
"\\--\n");
+                               n = child;
+                               tn = (struct tnode *)n;
+                               pend = tn->pos+tn->bits;
+                               putspace_seq(seq, indent);
+                               seq_printf(seq, "\\--\n");
                                indent += 3;
                                cindex = 0;
                        }
-               } else
-                       cindex++;
+               }
 
                /*
                 * Test if we are done
@@ -2132,8 +2120,7 @@ static void trie_dump_seq(struct seq_fil
                        depth--;
                }
        }
-
-       read_unlock(&fib_lock);
+       rcu_read_unlock();
 }
 
 static struct trie_stat *trie_stat_new(void)
@@ -2159,7 +2146,7 @@ static struct trie_stat *trie_stat_new(v
 
 static struct trie_stat *trie_collect_stats(struct trie *t)
 {
-       struct node *n = t->trie;
+       struct node *n = rcu_dereference(t->trie);
        struct trie_stat *s = trie_stat_new();
        int cindex = 0;
        int pend = 0;
@@ -2170,7 +2157,7 @@ static struct trie_stat *trie_collect_st
        if (!n)
                return s;
 
-       read_lock(&fib_lock);
+       rcu_read_lock();
 
        if (IS_TNODE(n)) {
                struct tnode *tn = (struct tnode *)n;
@@ -2179,7 +2166,9 @@ static struct trie_stat *trie_collect_st
                depth++;
 
                while (tn && cindex < (1 << tn->bits)) {
-                       if (tn->child[cindex]) {
+                       struct node *ch = rcu_dereference(tn->child[cindex]);
+                       if (ch) {
+
                                /* Got a child */
 
                                if (IS_LEAF(tn->child[cindex])) {
@@ -2199,7 +2188,7 @@ static struct trie_stat *trie_collect_st
                                        s->nodesizes[tn->bits]++;
                                        depth++;
 
-                                       n = tn->child[cindex];
+                                       n = ch;
                                        tn = (struct tnode *)n;
                                        pend = tn->pos+tn->bits;
 
@@ -2236,7 +2225,7 @@ static struct trie_stat *trie_collect_st
                }
        }
 
-       read_unlock(&fib_lock);
+       rcu_read_unlock();
        return s;
 }
 
-
To unsubscribe from this list: send the line "unsubscribe netdev" in
the body of a message to [EMAIL PROTECTED]
More majordomo info at  http://vger.kernel.org/majordomo-info.html

Reply via email to