Received: by 2002:ac0:8c8e:0:0:0:0:0 with SMTP id r14csp118041ima; Wed, 6 Feb 2019 18:28:01 -0800 (PST) X-Google-Smtp-Source: AHgI3IblGTbuIkjVwdUprxAbw2m27tylS4MdkO4yMbo1s2WnWe76LPTRvXfr/8aMepIfkmtvUc9/ X-Received: by 2002:a17:902:1a2:: with SMTP id b31mr14225421plb.87.1549506481590; Wed, 06 Feb 2019 18:28:01 -0800 (PST) ARC-Seal: i=1; a=rsa-sha256; t=1549506481; cv=none; d=google.com; s=arc-20160816; b=oOgvvgeUrb5j2KAMyI3yEK8rKwrb1xeMarmGZwFR/vnfnoOARdkpjONLaO7No/VvEI t1BlTf3ZH/rWdPeUF9MjXsETWDZHtPkkGJ+JYRk+1pC3eDpvonsW9PjWSP68imrbAUL9 pJiwkWejvaM3WeXWG1D1Ndex7l7Q+JBg0rSyAngVC8gLK2XB492CDMBjfFbCKNxRvZrT Yi1DDpuxByA9GDzNG+SCwoLqXX4M/eLdPzLj0PleiannS3edkNJdthIeyP0Ik62L7os4 cCh/+ydUBdZTg+83ijIrXJ2F3qVm5Z3zpVtzZ9wZQJi0218RAWzG6M02QOHe+44D47qf WDbw== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:sender:content-transfer-encoding:mime-version :references:in-reply-to:message-id:date:subject:cc:to:from :dkim-signature; bh=7/IXmKXx1uHiE+qFthPkn/ChskgOixZa8oJYp7CgSz0=; b=i6NSZEGCR0DATyfPcDeaeP4N3mdus+bFlVeTgZn7jtctzeKd78VQnc/j9CLnEHCgSO JvDVqS9ti8ZVnwmJDRlk3hF3f/wXgBYPdfo7LZ1Cdh73cDHacVovh10dYGwDZLigdpUp bJ5eYY2QVMA+MUq0XcgtnMswKqMuTuG53/abtfsrx+og1vs1PV0qx8TkHYoZoXivZ6Be PHSUsN+DGP/kQgwTadRK5kENXgpqNCmo9AWuQCwhdZ8PHTAqmcuMxblwbDV/IukpShCX UKDd/6Dy4Tdsbb4mAtTEEqnOqnNo79CaCVXdC8lE4y9ic2a5rRNwZnkwTl6thAXH7mJo 0goA== ARC-Authentication-Results: i=1; mx.google.com; dkim=pass header.i=@alliedtelesis.co.nz header.s=mail181024 header.b=sh27nDG1; spf=pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=pass (p=NONE sp=NONE dis=NONE) header.from=alliedtelesis.co.nz Return-Path: Received: from vger.kernel.org (vger.kernel.org. [209.132.180.67]) by mx.google.com with ESMTP id e35si4886349pgb.548.2019.02.06.18.27.45; Wed, 06 Feb 2019 18:28:01 -0800 (PST) Received-SPF: pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) client-ip=209.132.180.67; Authentication-Results: mx.google.com; dkim=pass header.i=@alliedtelesis.co.nz header.s=mail181024 header.b=sh27nDG1; spf=pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=pass (p=NONE sp=NONE dis=NONE) header.from=alliedtelesis.co.nz Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S1726978AbfBGCIj (ORCPT + 99 others); Wed, 6 Feb 2019 21:08:39 -0500 Received: from gate2.alliedtelesis.co.nz ([202.36.163.20]:33508 "EHLO gate2.alliedtelesis.co.nz" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S1726649AbfBGCIh (ORCPT ); Wed, 6 Feb 2019 21:08:37 -0500 Received: from mmarshal3.atlnz.lc (mmarshal3.atlnz.lc [10.32.18.43]) (using TLSv1.2 with cipher ECDHE-RSA-AES256-GCM-SHA384 (256/256 bits)) (Client did not present a certificate) by gate2.alliedtelesis.co.nz (Postfix) with ESMTPS id 7E37E8364B; Thu, 7 Feb 2019 15:08:34 +1300 (NZDT) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=alliedtelesis.co.nz; s=mail181024; t=1549505314; bh=7/IXmKXx1uHiE+qFthPkn/ChskgOixZa8oJYp7CgSz0=; h=From:To:Cc:Subject:Date:In-Reply-To:References; b=sh27nDG1GXzgl+B2zxsSH/5jlrgJ4As2B1FlZshPK+bbEyX/4/eYkC8Q9aY3twR5b XBJ36aqF1XGs1bl3llZRwNIPyYePVOyPmREmqoOI1psgtl1HOf4uJ1CZ9+pHjUtbk6 MW8CEkX+jznZeMMeteX6OSO43KEVl87j+3seqEqkFUy2klkF7ywlC+lomMw4UUi/mu ZvkVumSEgF/3sbXzk3qCZ5Pp+U8nKX9QwcVKHj3ygdxiJG96SAsS8cerVJxDJYbNcM UJgyBHBri7bybrYV7p5820anWYtYdtKsCwKj83jD/SRU1w19oqqiEA3spR7TpP/Qnv +Hl6yZmM3YYIQ== Received: from smtp (Not Verified[10.32.16.33]) by mmarshal3.atlnz.lc with Trustwave SEG (v7,5,8,10121) id ; Thu, 07 Feb 2019 15:08:35 +1300 Received: from callums-dl.ws.atlnz.lc (callums-dl.ws.atlnz.lc [10.33.22.12]) by smtp (Postfix) with ESMTP id 78EB913EF53; Thu, 7 Feb 2019 15:08:37 +1300 (NZDT) Received: by callums-dl.ws.atlnz.lc (Postfix, from userid 1764) id 7B1513A9BF1; Thu, 7 Feb 2019 15:08:31 +1300 (NZDT) From: Callum Sinclair To: davem@davemloft.net, kuznet@ms2.inr.ac.ru, yoshfuji@linux-ipv6.org, nikolay@cumulusnetworks.com, netdev@vger.kernel.org, linux-kernel@vger.kernel.org Cc: Callum Sinclair Subject: [PATCH] ipmr: ip6mr: Create new sockopt to clear mfc cache only Date: Thu, 7 Feb 2019 15:08:28 +1300 Message-Id: <20190207020828.21854-2-callum.sinclair@alliedtelesis.co.nz> X-Mailer: git-send-email 2.20.1 In-Reply-To: <20190207020828.21854-1-callum.sinclair@alliedtelesis.co.nz> References: <20190207020828.21854-1-callum.sinclair@alliedtelesis.co.nz> MIME-Version: 1.0 Content-Transfer-Encoding: quoted-printable x-atlnz-ls: pat Sender: linux-kernel-owner@vger.kernel.org Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org Currently the only way to clear the mfc cache was to delete the entries one by one using the MRT_DEL_MFC socket option or to destroy and recreate the socket. Create a new socket option which will clear the multicast forwarding cache on the socket without destroying the socket. Signed-off-by: Callum Sinclair --- include/uapi/linux/mroute.h | 7 +++- include/uapi/linux/mroute6.h | 7 +++- net/ipv4/ipmr.c | 69 +++++++++++++++++++++------------- net/ipv6/ip6mr.c | 73 ++++++++++++++++++++++-------------- 4 files changed, 99 insertions(+), 57 deletions(-) diff --git a/include/uapi/linux/mroute.h b/include/uapi/linux/mroute.h index 5d37a9ccce63..2d475edc3ec3 100644 --- a/include/uapi/linux/mroute.h +++ b/include/uapi/linux/mroute.h @@ -28,12 +28,17 @@ #define MRT_TABLE (MRT_BASE+9) /* Specify mroute table ID */ #define MRT_ADD_MFC_PROXY (MRT_BASE+10) /* Add a (*,*|G) mfc entry */ #define MRT_DEL_MFC_PROXY (MRT_BASE+11) /* Del a (*,*|G) mfc entry */ -#define MRT_MAX (MRT_BASE+11) +#define MRT_FLUSH (MRT_BASE+12) /* Flush all multicast entries and vifs = */ +#define MRT_MAX (MRT_BASE+12) =20 #define SIOCGETVIFCNT SIOCPROTOPRIVATE /* IP protocol privates */ #define SIOCGETSGCNT (SIOCPROTOPRIVATE+1) #define SIOCGETRPF (SIOCPROTOPRIVATE+2) =20 +/* Flags used for MRT_FLUSH */ +#define MRT_FLUSH_ENTRIES 1 /* For flushing all multicast entries */ +#define MRT_FLUSH_VIFS 2 /* For flushing all multicast vifs */ + #define MAXVIFS 32 typedef unsigned long vifbitmap_t; /* User mode code depends on this lot= */ typedef unsigned short vifi_t; diff --git a/include/uapi/linux/mroute6.h b/include/uapi/linux/mroute6.h index 9999cc006390..b04094d997c8 100644 --- a/include/uapi/linux/mroute6.h +++ b/include/uapi/linux/mroute6.h @@ -31,12 +31,17 @@ #define MRT6_TABLE (MRT6_BASE+9) /* Specify mroute table ID */ #define MRT6_ADD_MFC_PROXY (MRT6_BASE+10) /* Add a (*,*|G) mfc entry */ #define MRT6_DEL_MFC_PROXY (MRT6_BASE+11) /* Del a (*,*|G) mfc entry */ -#define MRT6_MAX (MRT6_BASE+11) +#define MRT6_FLUSH (MRT6_BASE+12) /* Flush all multicast entries and vif= s */ +#define MRT6_MAX (MRT6_BASE+12) =20 #define SIOCGETMIFCNT_IN6 SIOCPROTOPRIVATE /* IP protocol privates */ #define SIOCGETSGCNT_IN6 (SIOCPROTOPRIVATE+1) #define SIOCGETRPF (SIOCPROTOPRIVATE+2) =20 +/* Flags used for MRT6_FLUSH*/ +#define MRT6_FLUSH_ENTRIES 1 /* For flushing all multicast entries */ +#define MRT6_FLUSH_VIFS 2 /* For flushing all multicast vifs */ + #define MAXMIFS 32 typedef unsigned long mifbitmap_t; /* User mode code depends on this lot= */ typedef unsigned short mifi_t; diff --git a/net/ipv4/ipmr.c b/net/ipv4/ipmr.c index ddbf8c9a1abb..2eb569138569 100644 --- a/net/ipv4/ipmr.c +++ b/net/ipv4/ipmr.c @@ -416,7 +416,7 @@ static struct mr_table *ipmr_new_table(struct net *ne= t, u32 id) static void ipmr_free_table(struct mr_table *mrt) { del_timer_sync(&mrt->ipmr_expire_timer); - mroute_clean_tables(mrt, true); + mroute_clean_tables(mrt, true, MRT_FLUSH_VIFS | MRT_FLUSH_ENTRIES); rhltable_destroy(&mrt->mfc_hash); kfree(mrt); } @@ -1299,44 +1299,48 @@ static int ipmr_mfc_add(struct net *net, struct m= r_table *mrt, } =20 /* Close the multicast socket, and clear the vif tables etc */ -static void mroute_clean_tables(struct mr_table *mrt, bool all) +static void mroute_clean_tables(struct mr_table *mrt, bool all, int flag= s) { struct net *net =3D read_pnet(&mrt->net); - struct mr_mfc *c, *tmp; struct mfc_cache *cache; + struct mr_mfc *c, *tmp; LIST_HEAD(list); int i; =20 /* Shut down all active vif entries */ - for (i =3D 0; i < mrt->maxvif; i++) { - if (!all && (mrt->vif_table[i].flags & VIFF_STATIC)) - continue; - vif_delete(mrt, i, 0, &list); + if (flags & MRT_FLUSH_VIFS) { + for (i =3D 0; i < mrt->maxvif; i++) { + if (!all && (mrt->vif_table[i].flags & VIFF_STATIC)) + continue; + vif_delete(mrt, i, 0, &list); + } + unregister_netdevice_many(&list); } - unregister_netdevice_many(&list); =20 /* Wipe the cache */ - list_for_each_entry_safe(c, tmp, &mrt->mfc_cache_list, list) { - if (!all && (c->mfc_flags & MFC_STATIC)) - continue; - rhltable_remove(&mrt->mfc_hash, &c->mnode, ipmr_rht_params); - list_del_rcu(&c->list); - cache =3D (struct mfc_cache *)c; - call_ipmr_mfc_entry_notifiers(net, FIB_EVENT_ENTRY_DEL, cache, - mrt->id); - mroute_netlink_event(mrt, cache, RTM_DELROUTE); - mr_cache_put(c); - } - - if (atomic_read(&mrt->cache_resolve_queue_len) !=3D 0) { - spin_lock_bh(&mfc_unres_lock); - list_for_each_entry_safe(c, tmp, &mrt->mfc_unres_queue, list) { - list_del(&c->list); + if (flags & MRT_FLUSH_ENTRIES) { + list_for_each_entry_safe(c, tmp, &mrt->mfc_cache_list, list) { + if (!all && (c->mfc_flags & MFC_STATIC)) + continue; + rhltable_remove(&mrt->mfc_hash, &c->mnode, ipmr_rht_params); + list_del_rcu(&c->list); cache =3D (struct mfc_cache *)c; + call_ipmr_mfc_entry_notifiers(net, FIB_EVENT_ENTRY_DEL, cache, + mrt->id); mroute_netlink_event(mrt, cache, RTM_DELROUTE); - ipmr_destroy_unres(mrt, cache); + mr_cache_put(c); + } + + if (atomic_read(&mrt->cache_resolve_queue_len) !=3D 0) { + spin_lock_bh(&mfc_unres_lock); + list_for_each_entry_safe(c, tmp, &mrt->mfc_unres_queue, list) { + list_del(&c->list); + cache =3D (struct mfc_cache *)c; + mroute_netlink_event(mrt, cache, RTM_DELROUTE); + ipmr_destroy_unres(mrt, cache); + } + spin_unlock_bh(&mfc_unres_lock); } - spin_unlock_bh(&mfc_unres_lock); } } =20 @@ -1357,7 +1361,7 @@ static void mrtsock_destruct(struct sock *sk) NETCONFA_IFINDEX_ALL, net->ipv4.devconf_all); RCU_INIT_POINTER(mrt->mroute_sk, NULL); - mroute_clean_tables(mrt, false); + mroute_clean_tables(mrt, false, MRT_FLUSH_VIFS | MRT_FLUSH_ENTRIES); } } rtnl_unlock(); @@ -1482,6 +1486,17 @@ int ip_mroute_setsockopt(struct sock *sk, int optn= ame, char __user *optval, sk =3D=3D rtnl_dereference(mrt->mroute_sk), parent); break; + case MRT_FLUSH: + if (get_user(val, (int __user *)optval)) { + ret =3D -EFAULT; + break; + } + rtnl_lock(); + ipmr_for_each_table(mrt, net) { + mroute_clean_tables(mrt, true, val); + } + rtnl_unlock(); + break; /* Control PIM assert. */ case MRT_ASSERT: if (optlen !=3D sizeof(val)) { diff --git a/net/ipv6/ip6mr.c b/net/ipv6/ip6mr.c index 30337b38274b..473c83d197fe 100644 --- a/net/ipv6/ip6mr.c +++ b/net/ipv6/ip6mr.c @@ -393,7 +393,7 @@ static struct mr_table *ip6mr_new_table(struct net *n= et, u32 id) static void ip6mr_free_table(struct mr_table *mrt) { del_timer_sync(&mrt->ipmr_expire_timer); - mroute_clean_tables(mrt, true); + mroute_clean_tables(mrt, true, MRT6_FLUSH_VIFS | MRT6_FLUSH_ENTRIES); rhltable_destroy(&mrt->mfc_hash); kfree(mrt); } @@ -1496,43 +1496,47 @@ static int ip6mr_mfc_add(struct net *net, struct = mr_table *mrt, * Close the multicast socket, and clear the vif tables etc */ =20 -static void mroute_clean_tables(struct mr_table *mrt, bool all) +static void mroute_clean_tables(struct mr_table *mrt, bool all, int flag= s) { struct mr_mfc *c, *tmp; LIST_HEAD(list); int i; =20 /* Shut down all active vif entries */ - for (i =3D 0; i < mrt->maxvif; i++) { - if (!all && (mrt->vif_table[i].flags & VIFF_STATIC)) - continue; - mif6_delete(mrt, i, 0, &list); + if (flags & MRT6_FLUSH_VIFS) { + for (i =3D 0; i < mrt->maxvif; i++) { + if (!all && (mrt->vif_table[i].flags & VIFF_STATIC)) + continue; + mif6_delete(mrt, i, 0, &list); + } + unregister_netdevice_many(&list); } - unregister_netdevice_many(&list); =20 /* Wipe the cache */ - list_for_each_entry_safe(c, tmp, &mrt->mfc_cache_list, list) { - if (!all && (c->mfc_flags & MFC_STATIC)) - continue; - rhltable_remove(&mrt->mfc_hash, &c->mnode, ip6mr_rht_params); - list_del_rcu(&c->list); - mr6_netlink_event(mrt, (struct mfc6_cache *)c, RTM_DELROUTE); - mr_cache_put(c); - } + if (flags & MRT6_FLUSH_ENTRIES) { + list_for_each_entry_safe(c, tmp, &mrt->mfc_cache_list, list) { + if (!all && (c->mfc_flags & MFC_STATIC)) + continue; + rhltable_remove(&mrt->mfc_hash, &c->mnode, ip6mr_rht_params); + list_del_rcu(&c->list); + mr6_netlink_event(mrt, (struct mfc6_cache *)c, RTM_DELROUTE); + mr_cache_put(c); + } =20 - if (atomic_read(&mrt->cache_resolve_queue_len) !=3D 0) { - spin_lock_bh(&mfc_unres_lock); - list_for_each_entry_safe(c, tmp, &mrt->mfc_unres_queue, list) { - list_del(&c->list); - call_ip6mr_mfc_entry_notifiers(read_pnet(&mrt->net), - FIB_EVENT_ENTRY_DEL, - (struct mfc6_cache *)c, - mrt->id); - mr6_netlink_event(mrt, (struct mfc6_cache *)c, - RTM_DELROUTE); - ip6mr_destroy_unres(mrt, (struct mfc6_cache *)c); + if (atomic_read(&mrt->cache_resolve_queue_len) !=3D 0) { + spin_lock_bh(&mfc_unres_lock); + list_for_each_entry_safe(c, tmp, &mrt->mfc_unres_queue, list) { + list_del(&c->list); + call_ip6mr_mfc_entry_notifiers(read_pnet(&mrt->net), + FIB_EVENT_ENTRY_DEL, + (struct mfc6_cache *)c, + mrt->id); + mr6_netlink_event(mrt, (struct mfc6_cache *)c, + RTM_DELROUTE); + ip6mr_destroy_unres(mrt, (struct mfc6_cache *)c); + } + spin_unlock_bh(&mfc_unres_lock); } - spin_unlock_bh(&mfc_unres_lock); } } =20 @@ -1588,7 +1592,7 @@ int ip6mr_sk_done(struct sock *sk) NETCONFA_IFINDEX_ALL, net->ipv6.devconf_all); =20 - mroute_clean_tables(mrt, false); + mroute_clean_tables(mrt, false, MRT6_FLUSH_VIFS | MRT6_FLUSH_ENTRIES)= ; err =3D 0; break; } @@ -1703,6 +1707,19 @@ int ip6_mroute_setsockopt(struct sock *sk, int opt= name, char __user *optval, uns parent); rtnl_unlock(); return ret; + case MRT6_DEL_MFC_ALL: + { + int flags; + + if (get_user(flags, (int __user *)optval)) + return -EFAULT; + rtnl_lock(); + ip6mr_for_each_table(mrt, net) { + mroute_clean_tables(mrt, true, flags); + } + rtnl_unlock(); + return 0; + } =20 /* * Control PIM assert (to activate pim will activate assert) --=20 2.20.1