All of lore.kernel.org
 help / color / mirror / Atom feed
* [RFC PATCH net] net: ipv6: make fib6_nh_init properly clean after itself on error
@ 2021-11-29 14:11 Nikolay Aleksandrov
  2021-11-29 21:23   ` kernel test robot
  2021-11-30 12:40 ` Ido Schimmel
  0 siblings, 2 replies; 11+ messages in thread
From: Nikolay Aleksandrov @ 2021-11-29 14:11 UTC (permalink / raw)
  To: netdev; +Cc: davem, kuba, dsahern, idosch, Nikolay Aleksandrov

From: Nikolay Aleksandrov <nikolay@nvidia.com>

Currently we have different cleanup expectations by different users of
fib6_nh_init:
 1. nh_create_ipv6
 - calls fib6_nh_release manually which does full cleanup

 2. ip6_route_info_create
 - calls fib6_info_release to drop refs to 0 and schedules rcu call
   for fib6_info_destroy_rcu() which also does full cleanup

 3. fib_check_nh_v6_gw
 - doesn't do any cleanup on error, expects fib6_nh_init to clean up
   after itself fully (nhc_pcpu_rth_output per-cpu memory leak on error)

We can alter fib6_nh_init to properly cleanup after itself so
expectations would be the same for everyone and noone would have to do
anything in such case. It is safe because the route is not inserted yet
so the fib6_nh should not be visible at fib6_nh_init point, thus it
should be possible to free up all resources in its error path. The
problems (and leaks) are because it doesn't free all resources in its
error path, the nhc_pcpu_rth_output per-cpu allocation done by
fib_nh_common_init is not cleaned up and it expects its callers to clean
up if an error occurred after it, e.g. the dst per-cpu allocation
might fail. This change allows us to fix the memory leak at 3. and also
to simplify nh_create_ipv6 and remove the special error handling.

Fixes: 717a8f5b2923 ("ipv4: Add fib_check_nh_v6_gw")
Signed-off-by: Nikolay Aleksandrov <nikolay@nvidia.com>
---
Sending as RFC to see what people think. I've tested this patch under
heavy load with replacing/traffic forwarding/nexthop add/del etc.
I've also tested error paths by adding artificial ENOMEM errors in
different fib6_nh_init stages while running kmemleak.

 net/ipv4/nexthop.c |  8 +-------
 net/ipv6/route.c   | 15 +++++++++------
 2 files changed, 10 insertions(+), 13 deletions(-)

diff --git a/net/ipv4/nexthop.c b/net/ipv4/nexthop.c
index 5dbd4b5505eb..a7debafe8b90 100644
--- a/net/ipv4/nexthop.c
+++ b/net/ipv4/nexthop.c
@@ -2565,14 +2565,8 @@ static int nh_create_ipv6(struct net *net,  struct nexthop *nh,
 	/* sets nh_dev if successful */
 	err = ipv6_stub->fib6_nh_init(net, fib6_nh, &fib6_cfg, GFP_KERNEL,
 				      extack);
-	if (err) {
-		/* IPv6 is not enabled, don't call fib6_nh_release */
-		if (err == -EAFNOSUPPORT)
-			goto out;
-		ipv6_stub->fib6_nh_release(fib6_nh);
-	} else {
+	if (!err)
 		nh->nh_flags = fib6_nh->fib_nh_flags;
-	}
 out:
 	return err;
 }
diff --git a/net/ipv6/route.c b/net/ipv6/route.c
index 42d60c76d30a..2107b13cc9ab 100644
--- a/net/ipv6/route.c
+++ b/net/ipv6/route.c
@@ -3635,7 +3635,9 @@ int fib6_nh_init(struct net *net, struct fib6_nh *fib6_nh,
 		in6_dev_put(idev);
 
 	if (err) {
-		lwtstate_put(fib6_nh->fib_nh_lws);
+		/* check if we failed after fib_nh_common_init() was called */
+		if (fib6_nh->nh_common.nhc_pcpu_rth_output)
+			fib_nh_common_release(&fib6_nh->nh_common);
 		fib6_nh->fib_nh_lws = NULL;
 		dev_put(dev);
 	}
@@ -3822,7 +3824,7 @@ static struct fib6_info *ip6_route_info_create(struct fib6_config *cfg,
 	} else {
 		err = fib6_nh_init(net, rt->fib6_nh, cfg, gfp_flags, extack);
 		if (err)
-			goto out;
+			goto out_free;
 
 		fib6_nh = rt->fib6_nh;
 
@@ -3841,7 +3843,7 @@ static struct fib6_info *ip6_route_info_create(struct fib6_config *cfg,
 		if (!ipv6_chk_addr(net, &cfg->fc_prefsrc, dev, 0)) {
 			NL_SET_ERR_MSG(extack, "Invalid source address");
 			err = -EINVAL;
-			goto out;
+			goto out_free;
 		}
 		rt->fib6_prefsrc.addr = cfg->fc_prefsrc;
 		rt->fib6_prefsrc.plen = 128;
@@ -3849,12 +3851,13 @@ static struct fib6_info *ip6_route_info_create(struct fib6_config *cfg,
 		rt->fib6_prefsrc.plen = 0;
 
 	return rt;
-out:
-	fib6_info_release(rt);
-	return ERR_PTR(err);
+
 out_free:
 	ip_fib_metrics_put(rt->fib6_metrics);
+	if (rt->nh)
+		nexthop_put(rt->nh);
 	kfree(rt);
+out:
 	return ERR_PTR(err);
 }
 
-- 
2.31.1


^ permalink raw reply related	[flat|nested] 11+ messages in thread
* Re: [RFC PATCH net] net: ipv6: make fib6_nh_init properly clean after itself on error
@ 2021-11-30  1:18 kernel test robot
  2021-12-08  3:21 ` kernel test robot
  0 siblings, 1 reply; 11+ messages in thread
From: kernel test robot @ 2021-11-30  1:18 UTC (permalink / raw)
  To: kbuild

[-- Attachment #1: Type: text/plain, Size: 4601 bytes --]

CC: kbuild-all(a)lists.01.org
In-Reply-To: <20211129141151.490533-1-razor@blackwall.org>
References: <20211129141151.490533-1-razor@blackwall.org>
TO: Nikolay Aleksandrov <razor@blackwall.org>

Hi Nikolay,

[FYI, it's a private test report for your RFC patch.]
[auto build test WARNING on net/master]

url:    https://github.com/0day-ci/linux/commits/Nikolay-Aleksandrov/net-ipv6-make-fib6_nh_init-properly-clean-after-itself-on-error/20211130-001132
base:   https://git.kernel.org/pub/scm/linux/kernel/git/davem/net.git 2191b1dfef7d45f44b5008d2148676d9f2c82874
:::::: branch date: 9 hours ago
:::::: commit date: 9 hours ago
config: x86_64-randconfig-s032-20211128 (https://download.01.org/0day-ci/archive/20211130/202111300910.7HWZTGBF-lkp(a)intel.com/config)
compiler: gcc-9 (Debian 9.3.0-22) 9.3.0
reproduce:
        # apt-get install sparse
        # sparse version: v0.6.4-dirty
        # https://github.com/0day-ci/linux/commit/2db1685345b4a1aecadba0a8197c79f5e49da8ec
        git remote add linux-review https://github.com/0day-ci/linux
        git fetch --no-tags linux-review Nikolay-Aleksandrov/net-ipv6-make-fib6_nh_init-properly-clean-after-itself-on-error/20211130-001132
        git checkout 2db1685345b4a1aecadba0a8197c79f5e49da8ec
        # save the config file to linux build tree
        make W=1 C=1 CF='-fdiagnostic-prefix -D__CHECK_ENDIAN__' O=build_dir ARCH=x86_64 SHELL=/bin/bash

If you fix the issue, kindly add following tag as appropriate
Reported-by: kernel test robot <lkp@intel.com>


sparse warnings: (new ones prefixed by >>)
>> net/ipv4/nexthop.c:2570:1: sparse: sparse: unused label 'out'
   net/ipv4/nexthop.c: note: in included file (through include/linux/sysctl.h, include/net/net_namespace.h, include/linux/netdevice.h, ...):
   include/linux/rbtree.h:74:9: sparse: sparse: incompatible types in comparison expression (different address spaces):
   include/linux/rbtree.h:74:9: sparse:    struct rb_node [noderef] __rcu *
   include/linux/rbtree.h:74:9: sparse:    struct rb_node *

vim +/out +2570 net/ipv4/nexthop.c

597cfe4fc3390a David Ahern         2019-05-24  2544  
53010f991a9f5e David Ahern         2019-05-24  2545  static int nh_create_ipv6(struct net *net,  struct nexthop *nh,
53010f991a9f5e David Ahern         2019-05-24  2546  			  struct nh_info *nhi, struct nh_config *cfg,
53010f991a9f5e David Ahern         2019-05-24  2547  			  struct netlink_ext_ack *extack)
53010f991a9f5e David Ahern         2019-05-24  2548  {
53010f991a9f5e David Ahern         2019-05-24  2549  	struct fib6_nh *fib6_nh = &nhi->fib6_nh;
53010f991a9f5e David Ahern         2019-05-24  2550  	struct fib6_config fib6_cfg = {
53010f991a9f5e David Ahern         2019-05-24  2551  		.fc_table = l3mdev_fib_table(cfg->dev),
53010f991a9f5e David Ahern         2019-05-24  2552  		.fc_ifindex = cfg->nh_ifindex,
53010f991a9f5e David Ahern         2019-05-24  2553  		.fc_gateway = cfg->gw.ipv6,
53010f991a9f5e David Ahern         2019-05-24  2554  		.fc_flags = cfg->nh_flags,
9aca491e0dccf8 Ryoga Saito         2021-09-02  2555  		.fc_nlinfo = cfg->nlinfo,
b513bd035f4044 David Ahern         2019-05-24  2556  		.fc_encap = cfg->nh_encap,
b513bd035f4044 David Ahern         2019-05-24  2557  		.fc_encap_type = cfg->nh_encap_type,
38428d68719c45 Roopa Prabhu        2020-05-21  2558  		.fc_is_fdb = cfg->nh_fdb,
53010f991a9f5e David Ahern         2019-05-24  2559  	};
6f43e5252833f3 Colin Ian King      2019-05-30  2560  	int err;
53010f991a9f5e David Ahern         2019-05-24  2561  
53010f991a9f5e David Ahern         2019-05-24  2562  	if (!ipv6_addr_any(&cfg->gw.ipv6))
53010f991a9f5e David Ahern         2019-05-24  2563  		fib6_cfg.fc_flags |= RTF_GATEWAY;
53010f991a9f5e David Ahern         2019-05-24  2564  
53010f991a9f5e David Ahern         2019-05-24  2565  	/* sets nh_dev if successful */
53010f991a9f5e David Ahern         2019-05-24  2566  	err = ipv6_stub->fib6_nh_init(net, fib6_nh, &fib6_cfg, GFP_KERNEL,
53010f991a9f5e David Ahern         2019-05-24  2567  				      extack);
2db1685345b4a1 Nikolay Aleksandrov 2021-11-29  2568  	if (!err)
53010f991a9f5e David Ahern         2019-05-24  2569  		nh->nh_flags = fib6_nh->fib_nh_flags;
1c743127cc54b1 Nikolay Aleksandrov 2021-11-23 @2570  out:
53010f991a9f5e David Ahern         2019-05-24  2571  	return err;
53010f991a9f5e David Ahern         2019-05-24  2572  }
53010f991a9f5e David Ahern         2019-05-24  2573  

---
0-DAY CI Kernel Test Service, Intel Corporation
https://lists.01.org/hyperkitty/list/kbuild-all(a)lists.01.org

^ permalink raw reply	[flat|nested] 11+ messages in thread

end of thread, other threads:[~2021-12-08  3:21 UTC | newest]

Thread overview: 11+ messages (download: mbox.gz follow: Atom feed
-- links below jump to the message on this page --
2021-11-29 14:11 [RFC PATCH net] net: ipv6: make fib6_nh_init properly clean after itself on error Nikolay Aleksandrov
2021-11-29 21:23 ` kernel test robot
2021-11-29 21:23   ` kernel test robot
2021-11-30 12:40 ` Ido Schimmel
2021-11-30 12:48   ` Nikolay Aleksandrov
2021-11-30 16:01   ` David Ahern
2021-11-30 16:45     ` Nikolay Aleksandrov
2021-11-30 17:18       ` David Ahern
2021-11-30 21:30         ` Nikolay Aleksandrov
  -- strict thread matches above, loose matches on Subject: below --
2021-11-30  1:18 kernel test robot
2021-12-08  3:21 ` kernel test robot

This is an external index of several public inboxes,
see mirroring instructions on how to clone and mirror
all data and code used by this external index.