Source-Changes-HG archive

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index][Old Index]

[src/trunk]: src Backport selected NPF fixes from the upstream (to be pulled ...



details:   https://anonhg.NetBSD.org/src/rev/d1a01e97552f
branches:  trunk
changeset: 1010356:d1a01e97552f
user:      rmind <rmind%NetBSD.org@localhost>
date:      Sat May 23 19:56:00 2020 +0000

description:
Backport selected NPF fixes from the upstream (to be pulled up):

- npf_conndb_lookup: protect the connection lookup with pserialize(9),
  instead of incorrectly assuming that the handler always runs at IPL_SOFNET.
  Should fix crashes reported on high load (PR/55182).

- npf_config_destroy: handle partially initialized config; fixes crashes
  with some invalid configurations.

- NAT policy creation / destruction: set the initial reference and do not
  wait for reference draining on destruction; destroy the policy on the
  last reference drop instead.  Fixes a lockup with the dynamic NAT rules.

- npf_nat_{export,import}: fix a regression since dynamic NAT rules.

- npfctl: fix a regression and restore the default group behaviour.

- Add npf_cache_tcp() and validate the TCP data offset (from maxv@).

diffstat:

 sys/net/npf/npf_conf.c          |  18 +++++--
 sys/net/npf/npf_conn.c          |   6 +-
 sys/net/npf/npf_conn.h          |   2 +-
 sys/net/npf/npf_conndb.c        |   8 ++-
 sys/net/npf/npf_inet.c          |  22 +++++++-
 sys/net/npf/npf_nat.c           |  93 ++++++++++++++++++++++++++++------------
 usr.sbin/npf/npfctl/npf_build.c |  48 +++++++++++++++++---
 usr.sbin/npf/npfctl/npf_show.c  |   4 +-
 usr.sbin/npf/npfctl/npfctl.h    |   1 +
 9 files changed, 149 insertions(+), 53 deletions(-)

diffs (truncated from 506 to 300 lines):

diff -r 23d7067316ac -r d1a01e97552f sys/net/npf/npf_conf.c
--- a/sys/net/npf/npf_conf.c    Sat May 23 19:52:12 2020 +0000
+++ b/sys/net/npf/npf_conf.c    Sat May 23 19:56:00 2020 +0000
@@ -47,7 +47,7 @@
 
 #ifdef _KERNEL
 #include <sys/cdefs.h>
-__KERNEL_RCSID(0, "$NetBSD: npf_conf.c,v 1.15 2019/08/25 13:21:03 rmind Exp $");
+__KERNEL_RCSID(0, "$NetBSD: npf_conf.c,v 1.16 2020/05/23 19:56:00 rmind Exp $");
 
 #include <sys/param.h>
 #include <sys/types.h>
@@ -94,10 +94,18 @@
         * Note: the rulesets must be destroyed first, in order to drop
         * any references to the tableset.
         */
-       npf_ruleset_destroy(nc->ruleset);
-       npf_ruleset_destroy(nc->nat_ruleset);
-       npf_rprocset_destroy(nc->rule_procs);
-       npf_tableset_destroy(nc->tableset);
+       if (nc->ruleset) {
+               npf_ruleset_destroy(nc->ruleset);
+       }
+       if (nc->nat_ruleset) {
+               npf_ruleset_destroy(nc->nat_ruleset);
+       }
+       if (nc->rule_procs) {
+               npf_rprocset_destroy(nc->rule_procs);
+       }
+       if (nc->tableset) {
+               npf_tableset_destroy(nc->tableset);
+       }
        kmem_free(nc, sizeof(npf_config_t));
 }
 
diff -r 23d7067316ac -r d1a01e97552f sys/net/npf/npf_conn.c
--- a/sys/net/npf/npf_conn.c    Sat May 23 19:52:12 2020 +0000
+++ b/sys/net/npf/npf_conn.c    Sat May 23 19:56:00 2020 +0000
@@ -107,7 +107,7 @@
 
 #ifdef _KERNEL
 #include <sys/cdefs.h>
-__KERNEL_RCSID(0, "$NetBSD: npf_conn.c,v 1.30 2019/09/29 17:00:29 rmind Exp $");
+__KERNEL_RCSID(0, "$NetBSD: npf_conn.c,v 1.31 2020/05/23 19:56:00 rmind Exp $");
 
 #include <sys/param.h>
 #include <sys/types.h>
@@ -311,7 +311,7 @@
        if (!npf_conn_conkey(npc, &key, true)) {
                return NULL;
        }
-       con = npf_conndb_lookup(npf->conn_db, &key, forw);
+       con = npf_conndb_lookup(npf, &key, forw);
        if (con == NULL) {
                return NULL;
        }
@@ -908,7 +908,7 @@
        if (!kdict || !npf_connkey_import(kdict, &key)) {
                return EINVAL;
        }
-       con = npf_conndb_lookup(npf->conn_db, &key, &forw);
+       con = npf_conndb_lookup(npf, &key, &forw);
        if (con == NULL) {
                return ESRCH;
        }
diff -r 23d7067316ac -r d1a01e97552f sys/net/npf/npf_conn.h
--- a/sys/net/npf/npf_conn.h    Sat May 23 19:52:12 2020 +0000
+++ b/sys/net/npf/npf_conn.h    Sat May 23 19:56:00 2020 +0000
@@ -157,7 +157,7 @@
 npf_conndb_t * npf_conndb_create(void);
 void           npf_conndb_destroy(npf_conndb_t *);
 
-npf_conn_t *   npf_conndb_lookup(npf_conndb_t *, const npf_connkey_t *, bool *);
+npf_conn_t *   npf_conndb_lookup(npf_t *, const npf_connkey_t *, bool *);
 bool           npf_conndb_insert(npf_conndb_t *, const npf_connkey_t *,
                    npf_conn_t *, bool);
 npf_conn_t *   npf_conndb_remove(npf_conndb_t *, npf_connkey_t *);
diff -r 23d7067316ac -r d1a01e97552f sys/net/npf/npf_conndb.c
--- a/sys/net/npf/npf_conndb.c  Sat May 23 19:52:12 2020 +0000
+++ b/sys/net/npf/npf_conndb.c  Sat May 23 19:56:00 2020 +0000
@@ -46,7 +46,7 @@
 
 #ifdef _KERNEL
 #include <sys/cdefs.h>
-__KERNEL_RCSID(0, "$NetBSD: npf_conndb.c,v 1.7 2019/12/14 15:21:51 riastradh Exp $");
+__KERNEL_RCSID(0, "$NetBSD: npf_conndb.c,v 1.8 2020/05/23 19:56:00 rmind Exp $");
 
 #include <sys/param.h>
 #include <sys/types.h>
@@ -143,8 +143,9 @@
  * npf_conndb_lookup: find a connection given the key.
  */
 npf_conn_t *
-npf_conndb_lookup(npf_conndb_t *cd, const npf_connkey_t *ck, bool *forw)
+npf_conndb_lookup(npf_t *npf, const npf_connkey_t *ck, bool *forw)
 {
+       npf_conndb_t *cd = atomic_load_relaxed(&npf->conn_db);
        const unsigned keylen = NPF_CONNKEY_LEN(ck);
        npf_conn_t *con;
        void *val;
@@ -152,8 +153,10 @@
        /*
         * Lookup the connection key in the key-value map.
         */
+       int s = npf_config_read_enter(npf);
        val = thmap_get(cd->cd_map, ck->ck_key, keylen);
        if (!val) {
+               npf_config_read_exit(npf, s);
                return NULL;
        }
 
@@ -169,6 +172,7 @@
         * Acquire a reference and return the connection.
         */
        atomic_inc_uint(&con->c_refcnt);
+       npf_config_read_exit(npf, s);
        return con;
 }
 
diff -r 23d7067316ac -r d1a01e97552f sys/net/npf/npf_inet.c
--- a/sys/net/npf/npf_inet.c    Sat May 23 19:52:12 2020 +0000
+++ b/sys/net/npf/npf_inet.c    Sat May 23 19:56:00 2020 +0000
@@ -38,7 +38,7 @@
 
 #ifdef _KERNEL
 #include <sys/cdefs.h>
-__KERNEL_RCSID(0, "$NetBSD: npf_inet.c,v 1.55 2019/08/11 20:26:34 rmind Exp $");
+__KERNEL_RCSID(0, "$NetBSD: npf_inet.c,v 1.56 2020/05/23 19:56:00 rmind Exp $");
 
 #include <sys/param.h>
 #include <sys/types.h>
@@ -562,6 +562,22 @@
        return flags;
 }
 
+static inline int
+npf_cache_tcp(npf_cache_t *npc, nbuf_t *nbuf, unsigned hlen)
+{
+       struct tcphdr *th;
+
+       th = nbuf_advance(nbuf, hlen, sizeof(struct tcphdr));
+       if (__predict_false(th == NULL)) {
+               return NPC_FMTERR;
+       }
+       if (__predict_false(th->th_off < 5)) {
+               return NPC_FMTERR;
+       }
+       npc->npc_l4.tcp = th;
+       return NPC_LAYER4 | NPC_TCP;
+}
+
 /*
  * npf_cache_all: general routine to cache all relevant IP (v4 or v6)
  * and TCP, UDP or ICMP headers.
@@ -601,9 +617,7 @@
        switch (npc->npc_proto) {
        case IPPROTO_TCP:
                /* Cache: layer 4 - TCP. */
-               npc->npc_l4.tcp = nbuf_advance(nbuf, hlen,
-                   sizeof(struct tcphdr));
-               l4flags = NPC_LAYER4 | NPC_TCP;
+               l4flags = npf_cache_tcp(npc, nbuf, hlen);
                break;
        case IPPROTO_UDP:
                /* Cache: layer 4 - UDP. */
diff -r 23d7067316ac -r d1a01e97552f sys/net/npf/npf_nat.c
--- a/sys/net/npf/npf_nat.c     Sat May 23 19:52:12 2020 +0000
+++ b/sys/net/npf/npf_nat.c     Sat May 23 19:56:00 2020 +0000
@@ -67,7 +67,7 @@
 
 #ifdef _KERNEL
 #include <sys/cdefs.h>
-__KERNEL_RCSID(0, "$NetBSD: npf_nat.c,v 1.48 2019/08/25 13:21:03 rmind Exp $");
+__KERNEL_RCSID(0, "$NetBSD: npf_nat.c,v 1.49 2020/05/23 19:56:00 rmind Exp $");
 
 #include <sys/param.h>
 #include <sys/types.h>
@@ -182,6 +182,7 @@
        size_t len;
 
        np = kmem_zalloc(sizeof(npf_natpolicy_t), KM_SLEEP);
+       atomic_store_relaxed(&np->n_refcnt, 1);
        np->n_npfctx = npf;
 
        /* The translation type, flags and policy ID. */
@@ -271,37 +272,53 @@
        return 0;
 }
 
+static void
+npf_natpolicy_release(npf_natpolicy_t *np)
+{
+       KASSERT(atomic_load_relaxed(&np->n_refcnt) > 0);
+
+       if (atomic_dec_uint_nv(&np->n_refcnt) != 0) {
+               return;
+       }
+       KASSERT(LIST_EMPTY(&np->n_nat_list));
+       mutex_destroy(&np->n_lock);
+       kmem_free(np, sizeof(npf_natpolicy_t));
+}
+
 /*
  * npf_nat_freepolicy: free the NAT policy.
  *
  * => Called from npf_rule_free() during the reload via npf_ruleset_destroy().
+ * => At this point, NAT policy cannot acquire new references.
  */
 void
 npf_nat_freepolicy(npf_natpolicy_t *np)
 {
-       npf_conn_t *con;
-       npf_nat_t *nt;
+       /*
+        * Drain the references.  If there are active NAT connections,
+        * then expire them and kick the worker.
+        */
+       if (atomic_load_relaxed(&np->n_refcnt) > 1) {
+               npf_nat_t *nt;
 
-       /*
-        * Disassociate all entries from the policy.  At this point,
-        * new entries can no longer be created for this policy.
-        */
-       while (np->n_refcnt) {
                mutex_enter(&np->n_lock);
                LIST_FOREACH(nt, &np->n_nat_list, nt_entry) {
-                       con = nt->nt_conn;
+                       npf_conn_t *con = nt->nt_conn;
                        KASSERT(con != NULL);
                        npf_conn_expire(con);
                }
                mutex_exit(&np->n_lock);
+               npf_worker_signal(np->n_npfctx);
+       }
+       KASSERT(atomic_load_relaxed(&np->n_refcnt) >= 1);
 
-               /* Kick the worker - all references should be going away. */
-               npf_worker_signal(np->n_npfctx);
-               kpause("npfgcnat", false, 1, NULL);
-       }
-       KASSERT(LIST_EMPTY(&np->n_nat_list));
-       mutex_destroy(&np->n_lock);
-       kmem_free(np, sizeof(npf_natpolicy_t));
+       /*
+        * Drop the initial reference, but it might not be the last one.
+        * If so, the last reference will be triggered via:
+        *
+        * npf_conn_destroy() -> npf_nat_destroy() -> npf_natpolicy_release()
+        */
+       npf_natpolicy_release(np);
 }
 
 void
@@ -649,7 +666,7 @@
                        npf_recache(npc);
                }
                error = npf_nat_algo(npc, np, forw);
-               atomic_dec_uint(&np->n_refcnt);
+               npf_natpolicy_release(np);
                return error;
        }
 
@@ -662,7 +679,7 @@
        if (con == NULL) {
                ncon = npf_conn_establish(npc, di, true);
                if (ncon == NULL) {
-                       atomic_dec_uint(&np->n_refcnt);
+                       npf_natpolicy_release(np);
                        return ENOMEM;
                }
                con = ncon;
@@ -674,7 +691,7 @@
         */
        nt = npf_nat_create(npc, np, con);
        if (nt == NULL) {
-               atomic_dec_uint(&np->n_refcnt);
+               npf_natpolicy_release(np);
                error = ENOMEM;
                goto out;
        }
@@ -757,11 +774,15 @@
        }
        npf_stats_inc(np->n_npfctx, NPF_STAT_NAT_DESTROY);
 
+       /*
+        * Remove the connection from the list and drop the reference on
+        * the NAT policy.  Note: this might trigger its destruction.
+        */
        mutex_enter(&np->n_lock);
        LIST_REMOVE(nt, nt_entry);
-       KASSERT(np->n_refcnt > 0);
-       atomic_dec_uint(&np->n_refcnt);
        mutex_exit(&np->n_lock);
+       npf_natpolicy_release(np);
+
        pool_cache_put(nat_cache, nt);
 }
 
@@ -772,10 +793,13 @@



Home | Main Index | Thread Index | Old Index