ipv6: check fn before doing FIB6_SUBTREE(fn)
authorWei Wang <weiwan@google.com>
Fri, 13 Oct 2017 22:01:08 +0000 (15:01 -0700)
committerDavid S. Miller <davem@davemloft.net>
Mon, 16 Oct 2017 20:06:12 +0000 (21:06 +0100)
In fib6_locate(), we need to first make sure fn is not NULL before doing
FIB6_SUBTREE(fn) to avoid crash.

This fixes the following static checker warning:
net/ipv6/ip6_fib.c:1462 fib6_locate()
         warn: variable dereferenced before check 'fn' (see line 1459)

net/ipv6/ip6_fib.c
  1458          if (src_len) {
  1459                  struct fib6_node *subtree = FIB6_SUBTREE(fn);
                                                    ^^^^^^^^^^^^^^^^
We shifted this dereference

  1460
  1461                  WARN_ON(saddr == NULL);
  1462                  if (fn && subtree)
                            ^^
before the check for NULL.

  1463                          fn = fib6_locate_1(subtree, saddr, src_len,
  1464                                             offsetof(struct rt6_info, rt6i_src)

Fixes: 66f5d6ce53e6 ("ipv6: replace rwlock with rcu and spinlock in fib6_table")
Reported-by: Dan Carpenter <dan.carpenter@oracle.com>
Signed-off-by: Wei Wang <weiwan@google.com>
Acked-by: Eric Dumazet <edumazet@google.com>
Acked-by: Martin KaFai Lau <kafai@fb.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
net/ipv6/ip6_fib.c

index c2ecd5ec638aa98c97644935c723f97f0000edef..548af48212fcb575e8e57a8047ffa06d4ca07851 100644 (file)
@@ -1456,13 +1456,16 @@ struct fib6_node *fib6_locate(struct fib6_node *root,
 
 #ifdef CONFIG_IPV6_SUBTREES
        if (src_len) {
-               struct fib6_node *subtree = FIB6_SUBTREE(fn);
-
                WARN_ON(saddr == NULL);
-               if (fn && subtree)
-                       fn = fib6_locate_1(subtree, saddr, src_len,
+               if (fn) {
+                       struct fib6_node *subtree = FIB6_SUBTREE(fn);
+
+                       if (subtree) {
+                               fn = fib6_locate_1(subtree, saddr, src_len,
                                           offsetof(struct rt6_info, rt6i_src),
                                           exact_match);
+                       }
+               }
        }
 #endif