netfilter: nf_flow_table: hardware offload support
authorPablo Neira Ayuso <pablo@netfilter.org>
Mon, 11 Nov 2019 23:29:56 +0000 (00:29 +0100)
committerDavid S. Miller <davem@davemloft.net>
Wed, 13 Nov 2019 03:42:26 +0000 (19:42 -0800)
This patch adds the dataplane hardware offload to the flowtable
infrastructure. Three new flags represent the hardware state of this
flow:

* FLOW_OFFLOAD_HW: This flow entry resides in the hardware.
* FLOW_OFFLOAD_HW_DYING: This flow entry has been scheduled to be remove
  from hardware. This might be triggered by either packet path (via TCP
  RST/FIN packet) or via aging.
* FLOW_OFFLOAD_HW_DEAD: This flow entry has been already removed from
  the hardware, the software garbage collector can remove it from the
  software flowtable.

This patch supports for:

* IPv4 only.
* Aging via FLOW_CLS_STATS, no packet and byte counter synchronization
  at this stage.

This patch also adds the action callback that specifies how to convert
the flow entry into the flow_rule object that is passed to the driver.

Signed-off-by: Pablo Neira Ayuso <pablo@netfilter.org>
Signed-off-by: David S. Miller <davem@davemloft.net>
include/linux/netdevice.h
include/net/netfilter/nf_flow_table.h
net/ipv4/netfilter/nf_flow_table_ipv4.c
net/ipv6/netfilter/nf_flow_table_ipv6.c
net/netfilter/Makefile
net/netfilter/nf_flow_table_core.c
net/netfilter/nf_flow_table_inet.c
net/netfilter/nf_flow_table_offload.c [new file with mode: 0644]

index f857f01234f774d70d8c2425498e1fbc9909d88e..9e6fb8524d9181afed6343dcbcf70e41e88046ec 100644 (file)
@@ -848,6 +848,7 @@ enum tc_setup_type {
        TC_SETUP_ROOT_QDISC,
        TC_SETUP_QDISC_GRED,
        TC_SETUP_QDISC_TAPRIO,
+       TC_SETUP_FT,
 };
 
 /* These structures hold the attributes of bpf state that are being passed
index ece09d36c7a6aa18fefb51ac78408a344676310a..eea66de328d3ee9500c2020cde76d1abf2924f3d 100644 (file)
@@ -12,6 +12,9 @@
 #include <net/dst.h>
 
 struct nf_flowtable;
+struct nf_flow_rule;
+struct flow_offload;
+enum flow_offload_tuple_dir;
 
 struct nf_flowtable_type {
        struct list_head                list;
@@ -20,6 +23,10 @@ struct nf_flowtable_type {
        int                             (*setup)(struct nf_flowtable *ft,
                                                 struct net_device *dev,
                                                 enum flow_block_command cmd);
+       int                             (*action)(struct net *net,
+                                                 const struct flow_offload *flow,
+                                                 enum flow_offload_tuple_dir dir,
+                                                 struct nf_flow_rule *flow_rule);
        void                            (*free)(struct nf_flowtable *ft);
        nf_hookfn                       *hook;
        struct module                   *owner;
@@ -80,6 +87,9 @@ struct flow_offload_tuple_rhash {
 #define FLOW_OFFLOAD_DNAT      0x2
 #define FLOW_OFFLOAD_DYING     0x4
 #define FLOW_OFFLOAD_TEARDOWN  0x8
+#define FLOW_OFFLOAD_HW                0x10
+#define FLOW_OFFLOAD_HW_DYING  0x20
+#define FLOW_OFFLOAD_HW_DEAD   0x40
 
 enum flow_offload_type {
        NF_FLOW_OFFLOAD_UNSPEC  = 0,
@@ -142,11 +152,22 @@ unsigned int nf_flow_offload_ipv6_hook(void *priv, struct sk_buff *skb,
 #define MODULE_ALIAS_NF_FLOWTABLE(family)      \
        MODULE_ALIAS("nf-flowtable-" __stringify(family))
 
-static inline int nf_flow_table_offload_setup(struct nf_flowtable *flowtable,
-                                             struct net_device *dev,
-                                             enum flow_block_command cmd)
-{
-       return 0;
-}
+void nf_flow_offload_add(struct nf_flowtable *flowtable,
+                        struct flow_offload *flow);
+void nf_flow_offload_del(struct nf_flowtable *flowtable,
+                        struct flow_offload *flow);
+void nf_flow_offload_stats(struct nf_flowtable *flowtable,
+                          struct flow_offload *flow);
+
+void nf_flow_table_offload_flush(struct nf_flowtable *flowtable);
+int nf_flow_table_offload_setup(struct nf_flowtable *flowtable,
+                               struct net_device *dev,
+                               enum flow_block_command cmd);
+int nf_flow_rule_route(struct net *net, const struct flow_offload *flow,
+                      enum flow_offload_tuple_dir dir,
+                      struct nf_flow_rule *flow_rule);
+
+int nf_flow_table_offload_init(void);
+void nf_flow_table_offload_exit(void);
 
 #endif /* _NF_FLOW_TABLE_H */
index f3befddb5fdd25ad20efb5819d7801f8a699fcbe..168b72e18be05ec2c1d91770cdcc142444077261 100644 (file)
@@ -10,6 +10,7 @@ static struct nf_flowtable_type flowtable_ipv4 = {
        .family         = NFPROTO_IPV4,
        .init           = nf_flow_table_init,
        .setup          = nf_flow_table_offload_setup,
+       .action         = nf_flow_rule_route,
        .free           = nf_flow_table_free,
        .hook           = nf_flow_offload_ip_hook,
        .owner          = THIS_MODULE,
index 1c47f05eabd63a473fd583ea4b9c2efffd1b03ae..f069bc0dc0563897c0ba21d308b6e6ea89cb22e7 100644 (file)
@@ -11,6 +11,7 @@ static struct nf_flowtable_type flowtable_ipv6 = {
        .family         = NFPROTO_IPV6,
        .init           = nf_flow_table_init,
        .setup          = nf_flow_table_offload_setup,
+       .action         = nf_flow_rule_route,
        .free           = nf_flow_table_free,
        .hook           = nf_flow_offload_ipv6_hook,
        .owner          = THIS_MODULE,
index 4fc075b612feaaa1c528e47033dbdf5fa648078e..5e9b2eb243493307609fc5b42980786139a40e1b 100644 (file)
@@ -120,7 +120,8 @@ obj-$(CONFIG_NFT_FWD_NETDEV)        += nft_fwd_netdev.o
 
 # flow table infrastructure
 obj-$(CONFIG_NF_FLOW_TABLE)    += nf_flow_table.o
-nf_flow_table-objs := nf_flow_table_core.o nf_flow_table_ip.o
+nf_flow_table-objs             := nf_flow_table_core.o nf_flow_table_ip.o \
+                                  nf_flow_table_offload.o
 
 obj-$(CONFIG_NF_FLOW_TABLE_INET) += nf_flow_table_inet.o
 
index 139a5e074743e4166c47a2d54ab6632357fd8338..8468d2d02284f56ef5b641496615ac08b4090695 100644 (file)
@@ -250,6 +250,9 @@ int flow_offload_add(struct nf_flowtable *flow_table, struct flow_offload *flow)
                return err;
        }
 
+       if (flow_table->flags & NF_FLOWTABLE_HW_OFFLOAD)
+               nf_flow_offload_add(flow_table, flow);
+
        return 0;
 }
 EXPORT_SYMBOL_GPL(flow_offload_add);
@@ -350,9 +353,20 @@ static void nf_flow_offload_gc_step(struct flow_offload *flow, void *data)
 {
        struct nf_flowtable *flow_table = data;
 
+       if (flow->flags & FLOW_OFFLOAD_HW)
+               nf_flow_offload_stats(flow_table, flow);
+
        if (nf_flow_has_expired(flow) || nf_ct_is_dying(flow->ct) ||
-           (flow->flags & (FLOW_OFFLOAD_DYING | FLOW_OFFLOAD_TEARDOWN)))
-               flow_offload_del(flow_table, flow);
+           (flow->flags & (FLOW_OFFLOAD_DYING | FLOW_OFFLOAD_TEARDOWN))) {
+               if (flow->flags & FLOW_OFFLOAD_HW) {
+                       if (!(flow->flags & FLOW_OFFLOAD_HW_DYING))
+                               nf_flow_offload_del(flow_table, flow);
+                       else if (flow->flags & FLOW_OFFLOAD_HW_DEAD)
+                               flow_offload_del(flow_table, flow);
+               } else {
+                       flow_offload_del(flow_table, flow);
+               }
+       }
 }
 
 static void nf_flow_offload_work_gc(struct work_struct *work)
@@ -485,6 +499,7 @@ int nf_flow_table_init(struct nf_flowtable *flowtable)
        int err;
 
        INIT_DEFERRABLE_WORK(&flowtable->gc_work, nf_flow_offload_work_gc);
+       flow_block_init(&flowtable->flow_block);
 
        err = rhashtable_init(&flowtable->rhashtable,
                              &nf_flow_offload_rhash_params);
@@ -520,6 +535,7 @@ static void nf_flow_table_do_cleanup(struct flow_offload *flow, void *data)
 static void nf_flow_table_iterate_cleanup(struct nf_flowtable *flowtable,
                                          struct net_device *dev)
 {
+       nf_flow_table_offload_flush(flowtable);
        nf_flow_table_iterate(flowtable, nf_flow_table_do_cleanup, dev);
        flush_delayed_work(&flowtable->gc_work);
 }
@@ -547,5 +563,18 @@ void nf_flow_table_free(struct nf_flowtable *flow_table)
 }
 EXPORT_SYMBOL_GPL(nf_flow_table_free);
 
+static int __init nf_flow_table_module_init(void)
+{
+       return nf_flow_table_offload_init();
+}
+
+static void __exit nf_flow_table_module_exit(void)
+{
+       nf_flow_table_offload_exit();
+}
+
+module_init(nf_flow_table_module_init);
+module_exit(nf_flow_table_module_exit);
+
 MODULE_LICENSE("GPL");
 MODULE_AUTHOR("Pablo Neira Ayuso <pablo@netfilter.org>");
index 1e70fd504da3585ef58cfbacedba01c0763acef8..bfb910b874ce1832612b692d33f4b350e2a47c2a 100644 (file)
@@ -25,6 +25,7 @@ static struct nf_flowtable_type flowtable_inet = {
        .family         = NFPROTO_INET,
        .init           = nf_flow_table_init,
        .setup          = nf_flow_table_offload_setup,
+       .action         = nf_flow_rule_route,
        .free           = nf_flow_table_free,
        .hook           = nf_flow_offload_inet_hook,
        .owner          = THIS_MODULE,
diff --git a/net/netfilter/nf_flow_table_offload.c b/net/netfilter/nf_flow_table_offload.c
new file mode 100644 (file)
index 0000000..9be61f4
--- /dev/null
@@ -0,0 +1,758 @@
+#include <linux/kernel.h>
+#include <linux/init.h>
+#include <linux/module.h>
+#include <linux/netfilter.h>
+#include <linux/rhashtable.h>
+#include <linux/netdevice.h>
+#include <linux/tc_act/tc_csum.h>
+#include <net/flow_offload.h>
+#include <net/netfilter/nf_flow_table.h>
+#include <net/netfilter/nf_conntrack.h>
+#include <net/netfilter/nf_conntrack_core.h>
+#include <net/netfilter/nf_conntrack_tuple.h>
+
+static struct work_struct nf_flow_offload_work;
+static DEFINE_SPINLOCK(flow_offload_pending_list_lock);
+static LIST_HEAD(flow_offload_pending_list);
+
+struct flow_offload_work {
+       struct list_head        list;
+       enum flow_cls_command   cmd;
+       int                     priority;
+       struct nf_flowtable     *flowtable;
+       struct flow_offload     *flow;
+};
+
+struct nf_flow_key {
+       struct flow_dissector_key_control               control;
+       struct flow_dissector_key_basic                 basic;
+       union {
+               struct flow_dissector_key_ipv4_addrs    ipv4;
+       };
+       struct flow_dissector_key_tcp                   tcp;
+       struct flow_dissector_key_ports                 tp;
+} __aligned(BITS_PER_LONG / 8); /* Ensure that we can do comparisons as longs. */
+
+struct nf_flow_match {
+       struct flow_dissector   dissector;
+       struct nf_flow_key      key;
+       struct nf_flow_key      mask;
+};
+
+struct nf_flow_rule {
+       struct nf_flow_match    match;
+       struct flow_rule        *rule;
+};
+
+#define NF_FLOW_DISSECTOR(__match, __type, __field)    \
+       (__match)->dissector.offset[__type] =           \
+               offsetof(struct nf_flow_key, __field)
+
+static int nf_flow_rule_match(struct nf_flow_match *match,
+                             const struct flow_offload_tuple *tuple)
+{
+       struct nf_flow_key *mask = &match->mask;
+       struct nf_flow_key *key = &match->key;
+
+       NF_FLOW_DISSECTOR(match, FLOW_DISSECTOR_KEY_CONTROL, control);
+       NF_FLOW_DISSECTOR(match, FLOW_DISSECTOR_KEY_BASIC, basic);
+       NF_FLOW_DISSECTOR(match, FLOW_DISSECTOR_KEY_IPV4_ADDRS, ipv4);
+       NF_FLOW_DISSECTOR(match, FLOW_DISSECTOR_KEY_TCP, tcp);
+       NF_FLOW_DISSECTOR(match, FLOW_DISSECTOR_KEY_PORTS, tp);
+
+       switch (tuple->l3proto) {
+       case AF_INET:
+               key->control.addr_type = FLOW_DISSECTOR_KEY_IPV4_ADDRS;
+               key->basic.n_proto = htons(ETH_P_IP);
+               key->ipv4.src = tuple->src_v4.s_addr;
+               mask->ipv4.src = 0xffffffff;
+               key->ipv4.dst = tuple->dst_v4.s_addr;
+               mask->ipv4.dst = 0xffffffff;
+               break;
+       default:
+               return -EOPNOTSUPP;
+       }
+       mask->basic.n_proto = 0xffff;
+
+       switch (tuple->l4proto) {
+       case IPPROTO_TCP:
+               key->tcp.flags = 0;
+               mask->tcp.flags = TCP_FLAG_RST | TCP_FLAG_FIN;
+               match->dissector.used_keys |= BIT(FLOW_DISSECTOR_KEY_TCP);
+               break;
+       case IPPROTO_UDP:
+               break;
+       default:
+               return -EOPNOTSUPP;
+       }
+
+       key->basic.ip_proto = tuple->l4proto;
+       mask->basic.ip_proto = 0xff;
+
+       key->tp.src = tuple->src_port;
+       mask->tp.src = 0xffff;
+       key->tp.dst = tuple->dst_port;
+       mask->tp.dst = 0xffff;
+
+       match->dissector.used_keys |= BIT(FLOW_DISSECTOR_KEY_CONTROL) |
+                                     BIT(FLOW_DISSECTOR_KEY_BASIC) |
+                                     BIT(FLOW_DISSECTOR_KEY_IPV4_ADDRS) |
+                                     BIT(FLOW_DISSECTOR_KEY_PORTS);
+       return 0;
+}
+
+static void flow_offload_mangle(struct flow_action_entry *entry,
+                               enum flow_action_mangle_base htype,
+                               u32 offset, u8 *value, u8 *mask)
+{
+       entry->id = FLOW_ACTION_MANGLE;
+       entry->mangle.htype = htype;
+       entry->mangle.offset = offset;
+       memcpy(&entry->mangle.mask, mask, sizeof(u32));
+       memcpy(&entry->mangle.val, value, sizeof(u32));
+}
+
+static int flow_offload_eth_src(struct net *net,
+                               const struct flow_offload *flow,
+                               enum flow_offload_tuple_dir dir,
+                               struct flow_action_entry *entry0,
+                               struct flow_action_entry *entry1)
+{
+       const struct flow_offload_tuple *tuple = &flow->tuplehash[!dir].tuple;
+       struct net_device *dev;
+       u32 mask, val;
+       u16 val16;
+
+       dev = dev_get_by_index(net, tuple->iifidx);
+       if (!dev)
+               return -ENOENT;
+
+       mask = ~0xffff0000;
+       memcpy(&val16, dev->dev_addr, 2);
+       val = val16 << 16;
+       flow_offload_mangle(entry0, FLOW_ACT_MANGLE_HDR_TYPE_ETH, 4,
+                           (u8 *)&val, (u8 *)&mask);
+
+       mask = ~0xffffffff;
+       memcpy(&val, dev->dev_addr + 2, 4);
+       flow_offload_mangle(entry1, FLOW_ACT_MANGLE_HDR_TYPE_ETH, 8,
+                           (u8 *)&val, (u8 *)&mask);
+       dev_put(dev);
+
+       return 0;
+}
+
+static int flow_offload_eth_dst(struct net *net,
+                               const struct flow_offload *flow,
+                               enum flow_offload_tuple_dir dir,
+                               struct flow_action_entry *entry0,
+                               struct flow_action_entry *entry1)
+{
+       const struct flow_offload_tuple *tuple = &flow->tuplehash[dir].tuple;
+       struct neighbour *n;
+       u32 mask, val;
+       u16 val16;
+
+       n = dst_neigh_lookup(tuple->dst_cache, &tuple->dst_v4);
+       if (!n)
+               return -ENOENT;
+
+       mask = ~0xffffffff;
+       memcpy(&val, n->ha, 4);
+       flow_offload_mangle(entry0, FLOW_ACT_MANGLE_HDR_TYPE_ETH, 0,
+                           (u8 *)&val, (u8 *)&mask);
+
+       mask = ~0x0000ffff;
+       memcpy(&val16, n->ha + 4, 2);
+       val = val16;
+       flow_offload_mangle(entry1, FLOW_ACT_MANGLE_HDR_TYPE_ETH, 4,
+                           (u8 *)&val, (u8 *)&mask);
+       neigh_release(n);
+
+       return 0;
+}
+
+static void flow_offload_ipv4_snat(struct net *net,
+                                  const struct flow_offload *flow,
+                                  enum flow_offload_tuple_dir dir,
+                                  struct flow_action_entry *entry)
+{
+       u32 mask = ~htonl(0xffffffff);
+       __be32 addr;
+       u32 offset;
+
+       switch (dir) {
+       case FLOW_OFFLOAD_DIR_ORIGINAL:
+               addr = flow->tuplehash[FLOW_OFFLOAD_DIR_REPLY].tuple.dst_v4.s_addr;
+               offset = offsetof(struct iphdr, saddr);
+               break;
+       case FLOW_OFFLOAD_DIR_REPLY:
+               addr = flow->tuplehash[FLOW_OFFLOAD_DIR_ORIGINAL].tuple.src_v4.s_addr;
+               offset = offsetof(struct iphdr, daddr);
+               break;
+       default:
+               return;
+       }
+
+       flow_offload_mangle(entry, FLOW_ACT_MANGLE_HDR_TYPE_IP4, offset,
+                           (u8 *)&addr, (u8 *)&mask);
+}
+
+static void flow_offload_ipv4_dnat(struct net *net,
+                                  const struct flow_offload *flow,
+                                  enum flow_offload_tuple_dir dir,
+                                  struct flow_action_entry *entry)
+{
+       u32 mask = ~htonl(0xffffffff);
+       __be32 addr;
+       u32 offset;
+
+       switch (dir) {
+       case FLOW_OFFLOAD_DIR_ORIGINAL:
+               addr = flow->tuplehash[FLOW_OFFLOAD_DIR_REPLY].tuple.src_v4.s_addr;
+               offset = offsetof(struct iphdr, daddr);
+               break;
+       case FLOW_OFFLOAD_DIR_REPLY:
+               addr = flow->tuplehash[FLOW_OFFLOAD_DIR_ORIGINAL].tuple.dst_v4.s_addr;
+               offset = offsetof(struct iphdr, saddr);
+               break;
+       default:
+               return;
+       }
+
+       flow_offload_mangle(entry, FLOW_ACT_MANGLE_HDR_TYPE_IP4, offset,
+                           (u8 *)&addr, (u8 *)&mask);
+}
+
+static int flow_offload_l4proto(const struct flow_offload *flow)
+{
+       u8 protonum = flow->tuplehash[FLOW_OFFLOAD_DIR_ORIGINAL].tuple.l4proto;
+       u8 type = 0;
+
+       switch (protonum) {
+       case IPPROTO_TCP:
+               type = FLOW_ACT_MANGLE_HDR_TYPE_TCP;
+               break;
+       case IPPROTO_UDP:
+               type = FLOW_ACT_MANGLE_HDR_TYPE_UDP;
+               break;
+       default:
+               break;
+       }
+
+       return type;
+}
+
+static void flow_offload_port_snat(struct net *net,
+                                  const struct flow_offload *flow,
+                                  enum flow_offload_tuple_dir dir,
+                                  struct flow_action_entry *entry)
+{
+       u32 mask = ~htonl(0xffff0000);
+       __be16 port;
+       u32 offset;
+
+       switch (dir) {
+       case FLOW_OFFLOAD_DIR_ORIGINAL:
+               port = flow->tuplehash[FLOW_OFFLOAD_DIR_REPLY].tuple.dst_port;
+               offset = 0; /* offsetof(struct tcphdr, source); */
+               break;
+       case FLOW_OFFLOAD_DIR_REPLY:
+               port = flow->tuplehash[FLOW_OFFLOAD_DIR_ORIGINAL].tuple.src_port;
+               offset = 0; /* offsetof(struct tcphdr, dest); */
+               break;
+       default:
+               break;
+       }
+
+       flow_offload_mangle(entry, flow_offload_l4proto(flow), offset,
+                           (u8 *)&port, (u8 *)&mask);
+}
+
+static void flow_offload_port_dnat(struct net *net,
+                                  const struct flow_offload *flow,
+                                  enum flow_offload_tuple_dir dir,
+                                  struct flow_action_entry *entry)
+{
+       u32 mask = ~htonl(0xffff);
+       __be16 port;
+       u32 offset;
+
+       switch (dir) {
+       case FLOW_OFFLOAD_DIR_ORIGINAL:
+               port = flow->tuplehash[FLOW_OFFLOAD_DIR_REPLY].tuple.dst_port;
+               offset = 0; /* offsetof(struct tcphdr, source); */
+               break;
+       case FLOW_OFFLOAD_DIR_REPLY:
+               port = flow->tuplehash[FLOW_OFFLOAD_DIR_ORIGINAL].tuple.src_port;
+               offset = 0; /* offsetof(struct tcphdr, dest); */
+               break;
+       default:
+               break;
+       }
+
+       flow_offload_mangle(entry, flow_offload_l4proto(flow), offset,
+                           (u8 *)&port, (u8 *)&mask);
+}
+
+static void flow_offload_ipv4_checksum(struct net *net,
+                                      const struct flow_offload *flow,
+                                      struct flow_action_entry *entry)
+{
+       u8 protonum = flow->tuplehash[FLOW_OFFLOAD_DIR_ORIGINAL].tuple.l4proto;
+
+       entry->id = FLOW_ACTION_CSUM;
+       entry->csum_flags = TCA_CSUM_UPDATE_FLAG_IPV4HDR;
+
+       switch (protonum) {
+       case IPPROTO_TCP:
+               entry->csum_flags |= TCA_CSUM_UPDATE_FLAG_TCP;
+               break;
+       case IPPROTO_UDP:
+               entry->csum_flags |= TCA_CSUM_UPDATE_FLAG_UDP;
+               break;
+       }
+}
+
+static void flow_offload_redirect(const struct flow_offload *flow,
+                                 enum flow_offload_tuple_dir dir,
+                                 struct flow_action_entry *entry)
+{
+       struct rtable *rt;
+
+       rt = (struct rtable *)flow->tuplehash[dir].tuple.dst_cache;
+       entry->id = FLOW_ACTION_REDIRECT;
+       entry->dev = rt->dst.dev;
+       dev_hold(rt->dst.dev);
+}
+
+int nf_flow_rule_route(struct net *net, const struct flow_offload *flow,
+                      enum flow_offload_tuple_dir dir,
+                      struct nf_flow_rule *flow_rule)
+{
+       int i;
+
+       if (flow_offload_eth_src(net, flow, dir,
+                                &flow_rule->rule->action.entries[0],
+                                &flow_rule->rule->action.entries[1]) < 0)
+               return -1;
+
+       if (flow_offload_eth_dst(net, flow, dir,
+                                &flow_rule->rule->action.entries[2],
+                                &flow_rule->rule->action.entries[3]) < 0)
+               return -1;
+
+       i = 4;
+       if (flow->flags & FLOW_OFFLOAD_SNAT) {
+               flow_offload_ipv4_snat(net, flow, dir,
+                                      &flow_rule->rule->action.entries[i++]);
+               flow_offload_port_snat(net, flow, dir,
+                                      &flow_rule->rule->action.entries[i++]);
+       }
+       if (flow->flags & FLOW_OFFLOAD_DNAT) {
+               flow_offload_ipv4_dnat(net, flow, dir,
+                                      &flow_rule->rule->action.entries[i++]);
+               flow_offload_port_dnat(net, flow, dir,
+                                      &flow_rule->rule->action.entries[i++]);
+       }
+       if (flow->flags & FLOW_OFFLOAD_SNAT ||
+           flow->flags & FLOW_OFFLOAD_DNAT)
+               flow_offload_ipv4_checksum(net, flow,
+                                          &flow_rule->rule->action.entries[i++]);
+
+       flow_offload_redirect(flow, dir, &flow_rule->rule->action.entries[i++]);
+
+       return i;
+}
+EXPORT_SYMBOL_GPL(nf_flow_rule_route);
+
+static struct nf_flow_rule *
+nf_flow_offload_rule_alloc(struct net *net,
+                          const struct flow_offload_work *offload,
+                          enum flow_offload_tuple_dir dir)
+{
+       const struct nf_flowtable *flowtable = offload->flowtable;
+       const struct flow_offload *flow = offload->flow;
+       const struct flow_offload_tuple *tuple;
+       struct nf_flow_rule *flow_rule;
+       int err = -ENOMEM, num_actions;
+
+       flow_rule = kzalloc(sizeof(*flow_rule), GFP_KERNEL);
+       if (!flow_rule)
+               goto err_flow;
+
+       flow_rule->rule = flow_rule_alloc(10);
+       if (!flow_rule->rule)
+               goto err_flow_rule;
+
+       flow_rule->rule->match.dissector = &flow_rule->match.dissector;
+       flow_rule->rule->match.mask = &flow_rule->match.mask;
+       flow_rule->rule->match.key = &flow_rule->match.key;
+
+       tuple = &flow->tuplehash[dir].tuple;
+       err = nf_flow_rule_match(&flow_rule->match, tuple);
+       if (err < 0)
+               goto err_flow_match;
+
+       num_actions = flowtable->type->action(net, flow, dir, flow_rule);
+       if (num_actions < 0)
+               goto err_flow_match;
+
+       flow_rule->rule->action.num_entries = num_actions;
+
+       return flow_rule;
+
+err_flow_match:
+       kfree(flow_rule->rule);
+err_flow_rule:
+       kfree(flow_rule);
+err_flow:
+       return NULL;
+}
+
+static void __nf_flow_offload_destroy(struct nf_flow_rule *flow_rule)
+{
+       struct flow_action_entry *entry;
+       int i;
+
+       for (i = 0; i < flow_rule->rule->action.num_entries; i++) {
+               entry = &flow_rule->rule->action.entries[i];
+               if (entry->id != FLOW_ACTION_REDIRECT)
+                       continue;
+
+               dev_put(entry->dev);
+       }
+       kfree(flow_rule->rule);
+       kfree(flow_rule);
+}
+
+static void nf_flow_offload_destroy(struct nf_flow_rule *flow_rule[])
+{
+       int i;
+
+       for (i = 0; i < FLOW_OFFLOAD_DIR_MAX; i++)
+               __nf_flow_offload_destroy(flow_rule[i]);
+}
+
+static int nf_flow_offload_alloc(const struct flow_offload_work *offload,
+                                struct nf_flow_rule *flow_rule[])
+{
+       struct net *net = read_pnet(&offload->flowtable->net);
+
+       flow_rule[0] = nf_flow_offload_rule_alloc(net, offload,
+                                                 FLOW_OFFLOAD_DIR_ORIGINAL);
+       if (!flow_rule[0])
+               return -ENOMEM;
+
+       flow_rule[1] = nf_flow_offload_rule_alloc(net, offload,
+                                                 FLOW_OFFLOAD_DIR_REPLY);
+       if (!flow_rule[1]) {
+               __nf_flow_offload_destroy(flow_rule[0]);
+               return -ENOMEM;
+       }
+
+       return 0;
+}
+
+static void nf_flow_offload_init(struct flow_cls_offload *cls_flow,
+                                __be16 proto, int priority,
+                                enum flow_cls_command cmd,
+                                const struct flow_offload_tuple *tuple,
+                                struct netlink_ext_ack *extack)
+{
+       cls_flow->common.protocol = proto;
+       cls_flow->common.prio = priority;
+       cls_flow->common.extack = extack;
+       cls_flow->command = cmd;
+       cls_flow->cookie = (unsigned long)tuple;
+}
+
+static int flow_offload_tuple_add(struct flow_offload_work *offload,
+                                 struct nf_flow_rule *flow_rule,
+                                 enum flow_offload_tuple_dir dir)
+{
+       struct nf_flowtable *flowtable = offload->flowtable;
+       struct flow_cls_offload cls_flow = {};
+       struct flow_block_cb *block_cb;
+       struct netlink_ext_ack extack;
+       __be16 proto = ETH_P_ALL;
+       int err, i = 0;
+
+       nf_flow_offload_init(&cls_flow, proto, offload->priority,
+                            FLOW_CLS_REPLACE,
+                            &offload->flow->tuplehash[dir].tuple, &extack);
+       cls_flow.rule = flow_rule->rule;
+
+       list_for_each_entry(block_cb, &flowtable->flow_block.cb_list, list) {
+               err = block_cb->cb(TC_SETUP_FT, &cls_flow,
+                                  block_cb->cb_priv);
+               if (err < 0)
+                       continue;
+
+               i++;
+       }
+
+       return i;
+}
+
+static void flow_offload_tuple_del(struct flow_offload_work *offload,
+                                  enum flow_offload_tuple_dir dir)
+{
+       struct nf_flowtable *flowtable = offload->flowtable;
+       struct flow_cls_offload cls_flow = {};
+       struct flow_block_cb *block_cb;
+       struct netlink_ext_ack extack;
+       __be16 proto = ETH_P_ALL;
+
+       nf_flow_offload_init(&cls_flow, proto, offload->priority,
+                            FLOW_CLS_DESTROY,
+                            &offload->flow->tuplehash[dir].tuple, &extack);
+
+       list_for_each_entry(block_cb, &flowtable->flow_block.cb_list, list)
+               block_cb->cb(TC_SETUP_FT, &cls_flow, block_cb->cb_priv);
+
+       offload->flow->flags |= FLOW_OFFLOAD_HW_DEAD;
+}
+
+static int flow_offload_rule_add(struct flow_offload_work *offload,
+                                struct nf_flow_rule *flow_rule[])
+{
+       int ok_count = 0;
+
+       ok_count += flow_offload_tuple_add(offload, flow_rule[0],
+                                          FLOW_OFFLOAD_DIR_ORIGINAL);
+       ok_count += flow_offload_tuple_add(offload, flow_rule[1],
+                                          FLOW_OFFLOAD_DIR_REPLY);
+       if (ok_count == 0)
+               return -ENOENT;
+
+       return 0;
+}
+
+static int flow_offload_work_add(struct flow_offload_work *offload)
+{
+       struct nf_flow_rule *flow_rule[FLOW_OFFLOAD_DIR_MAX];
+       int err;
+
+       err = nf_flow_offload_alloc(offload, flow_rule);
+       if (err < 0)
+               return -ENOMEM;
+
+       err = flow_offload_rule_add(offload, flow_rule);
+
+       nf_flow_offload_destroy(flow_rule);
+
+       return err;
+}
+
+static void flow_offload_work_del(struct flow_offload_work *offload)
+{
+       flow_offload_tuple_del(offload, FLOW_OFFLOAD_DIR_ORIGINAL);
+       flow_offload_tuple_del(offload, FLOW_OFFLOAD_DIR_REPLY);
+}
+
+static void flow_offload_tuple_stats(struct flow_offload_work *offload,
+                                    enum flow_offload_tuple_dir dir,
+                                    struct flow_stats *stats)
+{
+       struct nf_flowtable *flowtable = offload->flowtable;
+       struct flow_cls_offload cls_flow = {};
+       struct flow_block_cb *block_cb;
+       struct netlink_ext_ack extack;
+       __be16 proto = ETH_P_ALL;
+
+       nf_flow_offload_init(&cls_flow, proto, offload->priority,
+                            FLOW_CLS_STATS,
+                            &offload->flow->tuplehash[dir].tuple, &extack);
+
+       list_for_each_entry(block_cb, &flowtable->flow_block.cb_list, list)
+               block_cb->cb(TC_SETUP_FT, &cls_flow, block_cb->cb_priv);
+       memcpy(stats, &cls_flow.stats, sizeof(*stats));
+}
+
+static void flow_offload_work_stats(struct flow_offload_work *offload)
+{
+       struct flow_stats stats[FLOW_OFFLOAD_DIR_MAX] = {};
+       u64 lastused;
+
+       flow_offload_tuple_stats(offload, FLOW_OFFLOAD_DIR_ORIGINAL, &stats[0]);
+       flow_offload_tuple_stats(offload, FLOW_OFFLOAD_DIR_REPLY, &stats[1]);
+
+       lastused = max_t(u64, stats[0].lastused, stats[1].lastused);
+       offload->flow->timeout = max_t(u64, offload->flow->timeout,
+                                      lastused + NF_FLOW_TIMEOUT);
+}
+
+static void flow_offload_work_handler(struct work_struct *work)
+{
+       struct flow_offload_work *offload, *next;
+       LIST_HEAD(offload_pending_list);
+       int ret;
+
+       spin_lock_bh(&flow_offload_pending_list_lock);
+       list_replace_init(&flow_offload_pending_list, &offload_pending_list);
+       spin_unlock_bh(&flow_offload_pending_list_lock);
+
+       list_for_each_entry_safe(offload, next, &offload_pending_list, list) {
+               switch (offload->cmd) {
+               case FLOW_CLS_REPLACE:
+                       ret = flow_offload_work_add(offload);
+                       if (ret < 0)
+                               offload->flow->flags &= ~FLOW_OFFLOAD_HW;
+                       break;
+               case FLOW_CLS_DESTROY:
+                       flow_offload_work_del(offload);
+                       break;
+               case FLOW_CLS_STATS:
+                       flow_offload_work_stats(offload);
+                       break;
+               default:
+                       WARN_ON_ONCE(1);
+               }
+               list_del(&offload->list);
+               kfree(offload);
+       }
+}
+
+static void flow_offload_queue_work(struct flow_offload_work *offload)
+{
+       spin_lock_bh(&flow_offload_pending_list_lock);
+       list_add_tail(&offload->list, &flow_offload_pending_list);
+       spin_unlock_bh(&flow_offload_pending_list_lock);
+
+       schedule_work(&nf_flow_offload_work);
+}
+
+void nf_flow_offload_add(struct nf_flowtable *flowtable,
+                        struct flow_offload *flow)
+{
+       struct flow_offload_work *offload;
+
+       offload = kmalloc(sizeof(struct flow_offload_work), GFP_ATOMIC);
+       if (!offload)
+               return;
+
+       offload->cmd = FLOW_CLS_REPLACE;
+       offload->flow = flow;
+       offload->priority = flowtable->priority;
+       offload->flowtable = flowtable;
+       flow->flags |= FLOW_OFFLOAD_HW;
+
+       flow_offload_queue_work(offload);
+}
+
+void nf_flow_offload_del(struct nf_flowtable *flowtable,
+                        struct flow_offload *flow)
+{
+       struct flow_offload_work *offload;
+
+       offload = kzalloc(sizeof(struct flow_offload_work), GFP_ATOMIC);
+       if (!offload)
+               return;
+
+       offload->cmd = FLOW_CLS_DESTROY;
+       offload->flow = flow;
+       offload->flow->flags |= FLOW_OFFLOAD_HW_DYING;
+       offload->flowtable = flowtable;
+
+       flow_offload_queue_work(offload);
+}
+
+void nf_flow_offload_stats(struct nf_flowtable *flowtable,
+                          struct flow_offload *flow)
+{
+       struct flow_offload_work *offload;
+       s64 delta;
+
+       delta = flow->timeout - jiffies;
+       if ((delta >= (9 * NF_FLOW_TIMEOUT) / 10) ||
+           flow->flags & FLOW_OFFLOAD_HW_DYING)
+               return;
+
+       offload = kzalloc(sizeof(struct flow_offload_work), GFP_ATOMIC);
+       if (!offload)
+               return;
+
+       offload->cmd = FLOW_CLS_STATS;
+       offload->flow = flow;
+       offload->flowtable = flowtable;
+
+       flow_offload_queue_work(offload);
+}
+
+void nf_flow_table_offload_flush(struct nf_flowtable *flowtable)
+{
+       if (flowtable->flags & NF_FLOWTABLE_HW_OFFLOAD)
+               flush_work(&nf_flow_offload_work);
+}
+
+static int nf_flow_table_block_setup(struct nf_flowtable *flowtable,
+                                    struct flow_block_offload *bo,
+                                    enum flow_block_command cmd)
+{
+       struct flow_block_cb *block_cb, *next;
+       int err = 0;
+
+       switch (cmd) {
+       case FLOW_BLOCK_BIND:
+               list_splice(&bo->cb_list, &flowtable->flow_block.cb_list);
+               break;
+       case FLOW_BLOCK_UNBIND:
+               list_for_each_entry_safe(block_cb, next, &bo->cb_list, list) {
+                       list_del(&block_cb->list);
+                       flow_block_cb_free(block_cb);
+               }
+               break;
+       default:
+               WARN_ON_ONCE(1);
+               err = -EOPNOTSUPP;
+       }
+
+       return err;
+}
+
+int nf_flow_table_offload_setup(struct nf_flowtable *flowtable,
+                               struct net_device *dev,
+                               enum flow_block_command cmd)
+{
+       struct netlink_ext_ack extack = {};
+       struct flow_block_offload bo = {};
+       int err;
+
+       if (!(flowtable->flags & NF_FLOWTABLE_HW_OFFLOAD))
+               return 0;
+
+       bo.net          = dev_net(dev);
+       bo.block        = &flowtable->flow_block;
+       bo.command      = cmd;
+       bo.binder_type  = FLOW_BLOCK_BINDER_TYPE_CLSACT_INGRESS;
+       bo.extack       = &extack;
+       INIT_LIST_HEAD(&bo.cb_list);
+
+       err = dev->netdev_ops->ndo_setup_tc(dev, TC_SETUP_BLOCK, &bo);
+       if (err < 0)
+               return err;
+
+       return nf_flow_table_block_setup(flowtable, &bo, cmd);
+}
+EXPORT_SYMBOL_GPL(nf_flow_table_offload_setup);
+
+int nf_flow_table_offload_init(void)
+{
+       INIT_WORK(&nf_flow_offload_work, flow_offload_work_handler);
+
+       return 0;
+}
+
+void nf_flow_table_offload_exit(void)
+{
+       struct flow_offload_work *offload, *next;
+       LIST_HEAD(offload_pending_list);
+
+       cancel_work_sync(&nf_flow_offload_work);
+
+       list_for_each_entry_safe(offload, next, &offload_pending_list, list) {
+               list_del(&offload->list);
+               kfree(offload);
+       }
+}