Commit 39673361 authored by Paolo Abeni's avatar Paolo Abeni

Merge tag 'nf-next-23-10-25' of git://git.kernel.org/pub/scm/linux/kernel/git/netfilter/nf-next

Pablo Neira Ayuso says:

====================
Netfilter updates for net-next

The following patchset contains Netfilter updates for net-next. Mostly
nf_tables updates with two patches for connlabel and br_netfilter.

1) Rename function name to perform on-demand GC for rbtree elements,
   and replace async GC in rbtree by sync GC. Patches from Florian Westphal.

2) Use commit_mutex for NFT_MSG_GETRULE_RESET to ensure that two
   concurrent threads invoking this command do not underrun stateful
   objects. Patches from Phil Sutter.

3) Use single hook to deal with IP and ARP packets in br_netfilter.
   Patch from Florian Westphal.

4) Use atomic_t in netns->connlabel use counter instead of using a
   spinlock, also patch from Florian.

5) Cleanups for stateful objects infrastructure in nf_tables.
   Patches from Phil Sutter.

6) Flush path uses opaque set element offered by the iterator, instead of
   calling pipapo_deactivate() which looks up for it again.

7) Set backend .flush interface always succeeds, make it return void
   instead.

8) Add struct nft_elem_priv placeholder structure and use it by replacing
   void * to pass opaque set element representation from backend to frontend
   which defeats compiler type checks.

9) Shrink memory consumption of set element transactions, by reducing
   struct nft_trans_elem object size and reducing stack memory usage.

10) Use struct nft_elem_priv also for set backend .insert operation too.

11) Carry reset flag in nft_set_dump_ctx structure, instead of passing it
    as a function argument, from Phil Sutter.

netfilter pull request 23-10-25

* tag 'nf-next-23-10-25' of git://git.kernel.org/pub/scm/linux/kernel/git/netfilter/nf-next:
  netfilter: nf_tables: Carry reset boolean in nft_set_dump_ctx
  netfilter: nf_tables: set->ops->insert returns opaque set element in case of EEXIST
  netfilter: nf_tables: shrink memory consumption of set elements
  netfilter: nf_tables: expose opaque set element as struct nft_elem_priv
  netfilter: nf_tables: set backend .flush always succeeds
  netfilter: nft_set_pipapo: no need to call pipapo_deactivate() from flush
  netfilter: nf_tables: Carry reset boolean in nft_obj_dump_ctx
  netfilter: nf_tables: nft_obj_filter fits into cb->ctx
  netfilter: nf_tables: Carry s_idx in nft_obj_dump_ctx
  netfilter: nf_tables: A better name for nft_obj_filter
  netfilter: nf_tables: Unconditionally allocate nft_obj_filter
  netfilter: nf_tables: Drop pointless memset in nf_tables_dump_obj
  netfilter: conntrack: switch connlabels to atomic_t
  br_netfilter: use single forward hook for ip and arp
  netfilter: nf_tables: Add locking for NFT_MSG_GETRULE_RESET requests
  netfilter: nf_tables: Introduce nf_tables_getrule_single()
  netfilter: nf_tables: Open-code audit log call in nf_tables_getrule()
  netfilter: nft_set_rbtree: prefer sync gc to async worker
  netfilter: nft_set_rbtree: rename gc deactivate+erase function
====================

Link: https://lore.kernel.org/r/20231025212555.132775-1-pablo@netfilter.orgSigned-off-by: default avatarPaolo Abeni <pabeni@redhat.com>
parents df3bc662 9cdee063
...@@ -39,7 +39,7 @@ static inline struct nf_conn_labels *nf_ct_labels_ext_add(struct nf_conn *ct) ...@@ -39,7 +39,7 @@ static inline struct nf_conn_labels *nf_ct_labels_ext_add(struct nf_conn *ct)
#ifdef CONFIG_NF_CONNTRACK_LABELS #ifdef CONFIG_NF_CONNTRACK_LABELS
struct net *net = nf_ct_net(ct); struct net *net = nf_ct_net(ct);
if (net->ct.labels_used == 0) if (atomic_read(&net->ct.labels_used) == 0)
return NULL; return NULL;
return nf_ct_ext_add(ct, NF_CT_EXT_LABELS, GFP_ATOMIC); return nf_ct_ext_add(ct, NF_CT_EXT_LABELS, GFP_ATOMIC);
......
...@@ -274,6 +274,9 @@ struct nft_userdata { ...@@ -274,6 +274,9 @@ struct nft_userdata {
unsigned char data[]; unsigned char data[];
}; };
/* placeholder structure for opaque set element backend representation. */
struct nft_elem_priv { };
/** /**
* struct nft_set_elem - generic representation of set elements * struct nft_set_elem - generic representation of set elements
* *
...@@ -294,9 +297,14 @@ struct nft_set_elem { ...@@ -294,9 +297,14 @@ struct nft_set_elem {
u32 buf[NFT_DATA_VALUE_MAXLEN / sizeof(u32)]; u32 buf[NFT_DATA_VALUE_MAXLEN / sizeof(u32)];
struct nft_data val; struct nft_data val;
} data; } data;
void *priv; struct nft_elem_priv *priv;
}; };
static inline void *nft_elem_priv_cast(const struct nft_elem_priv *priv)
{
return (void *)priv;
}
struct nft_set; struct nft_set;
struct nft_set_iter { struct nft_set_iter {
u8 genmask; u8 genmask;
...@@ -306,7 +314,7 @@ struct nft_set_iter { ...@@ -306,7 +314,7 @@ struct nft_set_iter {
int (*fn)(const struct nft_ctx *ctx, int (*fn)(const struct nft_ctx *ctx,
struct nft_set *set, struct nft_set *set,
const struct nft_set_iter *iter, const struct nft_set_iter *iter,
struct nft_set_elem *elem); struct nft_elem_priv *elem_priv);
}; };
/** /**
...@@ -430,7 +438,8 @@ struct nft_set_ops { ...@@ -430,7 +438,8 @@ struct nft_set_ops {
const struct nft_set_ext **ext); const struct nft_set_ext **ext);
bool (*update)(struct nft_set *set, bool (*update)(struct nft_set *set,
const u32 *key, const u32 *key,
void *(*new)(struct nft_set *, struct nft_elem_priv *
(*new)(struct nft_set *,
const struct nft_expr *, const struct nft_expr *,
struct nft_regs *), struct nft_regs *),
const struct nft_expr *expr, const struct nft_expr *expr,
...@@ -442,23 +451,23 @@ struct nft_set_ops { ...@@ -442,23 +451,23 @@ struct nft_set_ops {
int (*insert)(const struct net *net, int (*insert)(const struct net *net,
const struct nft_set *set, const struct nft_set *set,
const struct nft_set_elem *elem, const struct nft_set_elem *elem,
struct nft_set_ext **ext); struct nft_elem_priv **priv);
void (*activate)(const struct net *net, void (*activate)(const struct net *net,
const struct nft_set *set, const struct nft_set *set,
const struct nft_set_elem *elem); struct nft_elem_priv *elem_priv);
void * (*deactivate)(const struct net *net, struct nft_elem_priv * (*deactivate)(const struct net *net,
const struct nft_set *set, const struct nft_set *set,
const struct nft_set_elem *elem); const struct nft_set_elem *elem);
bool (*flush)(const struct net *net, void (*flush)(const struct net *net,
const struct nft_set *set, const struct nft_set *set,
void *priv); struct nft_elem_priv *priv);
void (*remove)(const struct net *net, void (*remove)(const struct net *net,
const struct nft_set *set, const struct nft_set *set,
const struct nft_set_elem *elem); struct nft_elem_priv *elem_priv);
void (*walk)(const struct nft_ctx *ctx, void (*walk)(const struct nft_ctx *ctx,
struct nft_set *set, struct nft_set *set,
struct nft_set_iter *iter); struct nft_set_iter *iter);
void * (*get)(const struct net *net, struct nft_elem_priv * (*get)(const struct net *net,
const struct nft_set *set, const struct nft_set *set,
const struct nft_set_elem *elem, const struct nft_set_elem *elem,
unsigned int flags); unsigned int flags);
...@@ -796,9 +805,9 @@ static inline bool nft_set_elem_expired(const struct nft_set_ext *ext) ...@@ -796,9 +805,9 @@ static inline bool nft_set_elem_expired(const struct nft_set_ext *ext)
} }
static inline struct nft_set_ext *nft_set_elem_ext(const struct nft_set *set, static inline struct nft_set_ext *nft_set_elem_ext(const struct nft_set *set,
void *elem) const struct nft_elem_priv *elem_priv)
{ {
return elem + set->ops->elemsize; return (void *)elem_priv + set->ops->elemsize;
} }
static inline struct nft_object **nft_set_ext_obj(const struct nft_set_ext *ext) static inline struct nft_object **nft_set_ext_obj(const struct nft_set_ext *ext)
...@@ -810,16 +819,19 @@ struct nft_expr *nft_set_elem_expr_alloc(const struct nft_ctx *ctx, ...@@ -810,16 +819,19 @@ struct nft_expr *nft_set_elem_expr_alloc(const struct nft_ctx *ctx,
const struct nft_set *set, const struct nft_set *set,
const struct nlattr *attr); const struct nlattr *attr);
void *nft_set_elem_init(const struct nft_set *set, struct nft_elem_priv *nft_set_elem_init(const struct nft_set *set,
const struct nft_set_ext_tmpl *tmpl, const struct nft_set_ext_tmpl *tmpl,
const u32 *key, const u32 *key_end, const u32 *data, const u32 *key, const u32 *key_end,
u64 timeout, u64 expiration, gfp_t gfp); const u32 *data,
u64 timeout, u64 expiration, gfp_t gfp);
int nft_set_elem_expr_clone(const struct nft_ctx *ctx, struct nft_set *set, int nft_set_elem_expr_clone(const struct nft_ctx *ctx, struct nft_set *set,
struct nft_expr *expr_array[]); struct nft_expr *expr_array[]);
void nft_set_elem_destroy(const struct nft_set *set, void *elem, void nft_set_elem_destroy(const struct nft_set *set,
const struct nft_elem_priv *elem_priv,
bool destroy_expr); bool destroy_expr);
void nf_tables_set_elem_destroy(const struct nft_ctx *ctx, void nf_tables_set_elem_destroy(const struct nft_ctx *ctx,
const struct nft_set *set, void *elem); const struct nft_set *set,
const struct nft_elem_priv *elem_priv);
struct nft_expr_ops; struct nft_expr_ops;
/** /**
...@@ -1061,7 +1073,7 @@ struct nft_chain { ...@@ -1061,7 +1073,7 @@ struct nft_chain {
int nft_chain_validate(const struct nft_ctx *ctx, const struct nft_chain *chain); int nft_chain_validate(const struct nft_ctx *ctx, const struct nft_chain *chain);
int nft_setelem_validate(const struct nft_ctx *ctx, struct nft_set *set, int nft_setelem_validate(const struct nft_ctx *ctx, struct nft_set *set,
const struct nft_set_iter *iter, const struct nft_set_iter *iter,
struct nft_set_elem *elem); struct nft_elem_priv *elem_priv);
int nft_set_catchall_validate(const struct nft_ctx *ctx, struct nft_set *set); int nft_set_catchall_validate(const struct nft_ctx *ctx, struct nft_set *set);
int nf_tables_bind_chain(const struct nft_ctx *ctx, struct nft_chain *chain); int nf_tables_bind_chain(const struct nft_ctx *ctx, struct nft_chain *chain);
void nf_tables_unbind_chain(const struct nft_ctx *ctx, struct nft_chain *chain); void nf_tables_unbind_chain(const struct nft_ctx *ctx, struct nft_chain *chain);
...@@ -1638,14 +1650,14 @@ struct nft_trans_table { ...@@ -1638,14 +1650,14 @@ struct nft_trans_table {
struct nft_trans_elem { struct nft_trans_elem {
struct nft_set *set; struct nft_set *set;
struct nft_set_elem elem; struct nft_elem_priv *elem_priv;
bool bound; bool bound;
}; };
#define nft_trans_elem_set(trans) \ #define nft_trans_elem_set(trans) \
(((struct nft_trans_elem *)trans->data)->set) (((struct nft_trans_elem *)trans->data)->set)
#define nft_trans_elem(trans) \ #define nft_trans_elem_priv(trans) \
(((struct nft_trans_elem *)trans->data)->elem) (((struct nft_trans_elem *)trans->data)->elem_priv)
#define nft_trans_elem_set_bound(trans) \ #define nft_trans_elem_set_bound(trans) \
(((struct nft_trans_elem *)trans->data)->bound) (((struct nft_trans_elem *)trans->data)->bound)
...@@ -1686,7 +1698,7 @@ struct nft_trans_gc { ...@@ -1686,7 +1698,7 @@ struct nft_trans_gc {
struct nft_set *set; struct nft_set *set;
u32 seq; u32 seq;
u16 count; u16 count;
void *priv[NFT_TRANS_GC_BATCHCOUNT]; struct nft_elem_priv *priv[NFT_TRANS_GC_BATCHCOUNT];
struct rcu_head rcu; struct rcu_head rcu;
}; };
...@@ -1709,7 +1721,7 @@ struct nft_trans_gc *nft_trans_gc_catchall_sync(struct nft_trans_gc *gc); ...@@ -1709,7 +1721,7 @@ struct nft_trans_gc *nft_trans_gc_catchall_sync(struct nft_trans_gc *gc);
void nft_setelem_data_deactivate(const struct net *net, void nft_setelem_data_deactivate(const struct net *net,
const struct nft_set *set, const struct nft_set *set,
struct nft_set_elem *elem); struct nft_elem_priv *elem_priv);
int __init nft_chain_filter_init(void); int __init nft_chain_filter_init(void);
void nft_chain_filter_fini(void); void nft_chain_filter_fini(void);
......
...@@ -107,7 +107,7 @@ struct netns_ct { ...@@ -107,7 +107,7 @@ struct netns_ct {
struct nf_ct_event_notifier __rcu *nf_conntrack_event_cb; struct nf_ct_event_notifier __rcu *nf_conntrack_event_cb;
struct nf_ip_net nf_ct_proto; struct nf_ip_net nf_ct_proto;
#if defined(CONFIG_NF_CONNTRACK_LABELS) #if defined(CONFIG_NF_CONNTRACK_LABELS)
unsigned int labels_used; atomic_t labels_used;
#endif #endif
}; };
#endif #endif
...@@ -570,18 +570,12 @@ static int br_nf_forward_finish(struct net *net, struct sock *sk, struct sk_buff ...@@ -570,18 +570,12 @@ static int br_nf_forward_finish(struct net *net, struct sock *sk, struct sk_buff
} }
/* This is the 'purely bridged' case. For IP, we pass the packet to static unsigned int br_nf_forward_ip(struct sk_buff *skb,
* netfilter with indev and outdev set to the bridge device, const struct nf_hook_state *state,
* but we are still able to filter on the 'real' indev/outdev u8 pf)
* because of the physdev module. For ARP, indev and outdev are the
* bridge ports. */
static unsigned int br_nf_forward_ip(void *priv,
struct sk_buff *skb,
const struct nf_hook_state *state)
{ {
struct nf_bridge_info *nf_bridge; struct nf_bridge_info *nf_bridge;
struct net_device *parent; struct net_device *parent;
u_int8_t pf;
nf_bridge = nf_bridge_info_get(skb); nf_bridge = nf_bridge_info_get(skb);
if (!nf_bridge) if (!nf_bridge)
...@@ -600,15 +594,6 @@ static unsigned int br_nf_forward_ip(void *priv, ...@@ -600,15 +594,6 @@ static unsigned int br_nf_forward_ip(void *priv,
if (!parent) if (!parent)
return NF_DROP_REASON(skb, SKB_DROP_REASON_DEV_READY, 0); return NF_DROP_REASON(skb, SKB_DROP_REASON_DEV_READY, 0);
if (IS_IP(skb) || is_vlan_ip(skb, state->net) ||
is_pppoe_ip(skb, state->net))
pf = NFPROTO_IPV4;
else if (IS_IPV6(skb) || is_vlan_ipv6(skb, state->net) ||
is_pppoe_ipv6(skb, state->net))
pf = NFPROTO_IPV6;
else
return NF_ACCEPT;
nf_bridge_pull_encap_header(skb); nf_bridge_pull_encap_header(skb);
if (skb->pkt_type == PACKET_OTHERHOST) { if (skb->pkt_type == PACKET_OTHERHOST) {
...@@ -620,19 +605,18 @@ static unsigned int br_nf_forward_ip(void *priv, ...@@ -620,19 +605,18 @@ static unsigned int br_nf_forward_ip(void *priv,
if (br_validate_ipv4(state->net, skb)) if (br_validate_ipv4(state->net, skb))
return NF_DROP_REASON(skb, SKB_DROP_REASON_IP_INHDR, 0); return NF_DROP_REASON(skb, SKB_DROP_REASON_IP_INHDR, 0);
IPCB(skb)->frag_max_size = nf_bridge->frag_max_size; IPCB(skb)->frag_max_size = nf_bridge->frag_max_size;
} skb->protocol = htons(ETH_P_IP);
} else if (pf == NFPROTO_IPV6) {
if (pf == NFPROTO_IPV6) {
if (br_validate_ipv6(state->net, skb)) if (br_validate_ipv6(state->net, skb))
return NF_DROP_REASON(skb, SKB_DROP_REASON_IP_INHDR, 0); return NF_DROP_REASON(skb, SKB_DROP_REASON_IP_INHDR, 0);
IP6CB(skb)->frag_max_size = nf_bridge->frag_max_size; IP6CB(skb)->frag_max_size = nf_bridge->frag_max_size;
skb->protocol = htons(ETH_P_IPV6);
} else {
WARN_ON_ONCE(1);
return NF_DROP;
} }
nf_bridge->physoutdev = skb->dev; nf_bridge->physoutdev = skb->dev;
if (pf == NFPROTO_IPV4)
skb->protocol = htons(ETH_P_IP);
else
skb->protocol = htons(ETH_P_IPV6);
NF_HOOK(pf, NF_INET_FORWARD, state->net, NULL, skb, NF_HOOK(pf, NF_INET_FORWARD, state->net, NULL, skb,
brnf_get_logical_dev(skb, state->in, state->net), brnf_get_logical_dev(skb, state->in, state->net),
...@@ -641,8 +625,7 @@ static unsigned int br_nf_forward_ip(void *priv, ...@@ -641,8 +625,7 @@ static unsigned int br_nf_forward_ip(void *priv,
return NF_STOLEN; return NF_STOLEN;
} }
static unsigned int br_nf_forward_arp(void *priv, static unsigned int br_nf_forward_arp(struct sk_buff *skb,
struct sk_buff *skb,
const struct nf_hook_state *state) const struct nf_hook_state *state)
{ {
struct net_bridge_port *p; struct net_bridge_port *p;
...@@ -659,11 +642,8 @@ static unsigned int br_nf_forward_arp(void *priv, ...@@ -659,11 +642,8 @@ static unsigned int br_nf_forward_arp(void *priv,
if (!brnet->call_arptables && !br_opt_get(br, BROPT_NF_CALL_ARPTABLES)) if (!brnet->call_arptables && !br_opt_get(br, BROPT_NF_CALL_ARPTABLES))
return NF_ACCEPT; return NF_ACCEPT;
if (!IS_ARP(skb)) { if (is_vlan_arp(skb, state->net))
if (!is_vlan_arp(skb, state->net))
return NF_ACCEPT;
nf_bridge_pull_encap_header(skb); nf_bridge_pull_encap_header(skb);
}
if (unlikely(!pskb_may_pull(skb, sizeof(struct arphdr)))) if (unlikely(!pskb_may_pull(skb, sizeof(struct arphdr))))
return NF_DROP_REASON(skb, SKB_DROP_REASON_PKT_TOO_SMALL, 0); return NF_DROP_REASON(skb, SKB_DROP_REASON_PKT_TOO_SMALL, 0);
...@@ -680,6 +660,28 @@ static unsigned int br_nf_forward_arp(void *priv, ...@@ -680,6 +660,28 @@ static unsigned int br_nf_forward_arp(void *priv,
return NF_STOLEN; return NF_STOLEN;
} }
/* This is the 'purely bridged' case. For IP, we pass the packet to
* netfilter with indev and outdev set to the bridge device,
* but we are still able to filter on the 'real' indev/outdev
* because of the physdev module. For ARP, indev and outdev are the
* bridge ports.
*/
static unsigned int br_nf_forward(void *priv,
struct sk_buff *skb,
const struct nf_hook_state *state)
{
if (IS_IP(skb) || is_vlan_ip(skb, state->net) ||
is_pppoe_ip(skb, state->net))
return br_nf_forward_ip(skb, state, NFPROTO_IPV4);
if (IS_IPV6(skb) || is_vlan_ipv6(skb, state->net) ||
is_pppoe_ipv6(skb, state->net))
return br_nf_forward_ip(skb, state, NFPROTO_IPV6);
if (IS_ARP(skb) || is_vlan_arp(skb, state->net))
return br_nf_forward_arp(skb, state);
return NF_ACCEPT;
}
static int br_nf_push_frag_xmit(struct net *net, struct sock *sk, struct sk_buff *skb) static int br_nf_push_frag_xmit(struct net *net, struct sock *sk, struct sk_buff *skb)
{ {
struct brnf_frag_data *data; struct brnf_frag_data *data;
...@@ -937,13 +939,7 @@ static const struct nf_hook_ops br_nf_ops[] = { ...@@ -937,13 +939,7 @@ static const struct nf_hook_ops br_nf_ops[] = {
.priority = NF_BR_PRI_BRNF, .priority = NF_BR_PRI_BRNF,
}, },
{ {
.hook = br_nf_forward_ip, .hook = br_nf_forward,
.pf = NFPROTO_BRIDGE,
.hooknum = NF_BR_FORWARD,
.priority = NF_BR_PRI_BRNF - 1,
},
{
.hook = br_nf_forward_arp,
.pf = NFPROTO_BRIDGE, .pf = NFPROTO_BRIDGE,
.hooknum = NF_BR_FORWARD, .hooknum = NF_BR_FORWARD,
.priority = NF_BR_PRI_BRNF, .priority = NF_BR_PRI_BRNF,
......
...@@ -11,8 +11,6 @@ ...@@ -11,8 +11,6 @@
#include <net/netfilter/nf_conntrack_ecache.h> #include <net/netfilter/nf_conntrack_ecache.h>
#include <net/netfilter/nf_conntrack_labels.h> #include <net/netfilter/nf_conntrack_labels.h>
static DEFINE_SPINLOCK(nf_connlabels_lock);
static int replace_u32(u32 *address, u32 mask, u32 new) static int replace_u32(u32 *address, u32 mask, u32 new)
{ {
u32 old, tmp; u32 old, tmp;
...@@ -60,23 +58,24 @@ EXPORT_SYMBOL_GPL(nf_connlabels_replace); ...@@ -60,23 +58,24 @@ EXPORT_SYMBOL_GPL(nf_connlabels_replace);
int nf_connlabels_get(struct net *net, unsigned int bits) int nf_connlabels_get(struct net *net, unsigned int bits)
{ {
int v;
if (BIT_WORD(bits) >= NF_CT_LABELS_MAX_SIZE / sizeof(long)) if (BIT_WORD(bits) >= NF_CT_LABELS_MAX_SIZE / sizeof(long))
return -ERANGE; return -ERANGE;
spin_lock(&nf_connlabels_lock);
net->ct.labels_used++;
spin_unlock(&nf_connlabels_lock);
BUILD_BUG_ON(NF_CT_LABELS_MAX_SIZE / sizeof(long) >= U8_MAX); BUILD_BUG_ON(NF_CT_LABELS_MAX_SIZE / sizeof(long) >= U8_MAX);
v = atomic_inc_return_relaxed(&net->ct.labels_used);
WARN_ON_ONCE(v <= 0);
return 0; return 0;
} }
EXPORT_SYMBOL_GPL(nf_connlabels_get); EXPORT_SYMBOL_GPL(nf_connlabels_get);
void nf_connlabels_put(struct net *net) void nf_connlabels_put(struct net *net)
{ {
spin_lock(&nf_connlabels_lock); int v = atomic_dec_return_relaxed(&net->ct.labels_used);
net->ct.labels_used--;
spin_unlock(&nf_connlabels_lock); WARN_ON_ONCE(v < 0);
} }
EXPORT_SYMBOL_GPL(nf_connlabels_put); EXPORT_SYMBOL_GPL(nf_connlabels_put);
This diff is collapsed.
...@@ -44,33 +44,34 @@ static int nft_dynset_expr_setup(const struct nft_dynset *priv, ...@@ -44,33 +44,34 @@ static int nft_dynset_expr_setup(const struct nft_dynset *priv,
return 0; return 0;
} }
static void *nft_dynset_new(struct nft_set *set, const struct nft_expr *expr, static struct nft_elem_priv *nft_dynset_new(struct nft_set *set,
struct nft_regs *regs) const struct nft_expr *expr,
struct nft_regs *regs)
{ {
const struct nft_dynset *priv = nft_expr_priv(expr); const struct nft_dynset *priv = nft_expr_priv(expr);
struct nft_set_ext *ext; struct nft_set_ext *ext;
void *elem_priv;
u64 timeout; u64 timeout;
void *elem;
if (!atomic_add_unless(&set->nelems, 1, set->size)) if (!atomic_add_unless(&set->nelems, 1, set->size))
return NULL; return NULL;
timeout = priv->timeout ? : set->timeout; timeout = priv->timeout ? : set->timeout;
elem = nft_set_elem_init(set, &priv->tmpl, elem_priv = nft_set_elem_init(set, &priv->tmpl,
&regs->data[priv->sreg_key], NULL, &regs->data[priv->sreg_key], NULL,
&regs->data[priv->sreg_data], &regs->data[priv->sreg_data],
timeout, 0, GFP_ATOMIC); timeout, 0, GFP_ATOMIC);
if (IS_ERR(elem)) if (IS_ERR(elem_priv))
goto err1; goto err1;
ext = nft_set_elem_ext(set, elem); ext = nft_set_elem_ext(set, elem_priv);
if (priv->num_exprs && nft_dynset_expr_setup(priv, ext) < 0) if (priv->num_exprs && nft_dynset_expr_setup(priv, ext) < 0)
goto err2; goto err2;
return elem; return elem_priv;
err2: err2:
nft_set_elem_destroy(set, elem, false); nft_set_elem_destroy(set, elem_priv, false);
err1: err1:
if (set->size) if (set->size)
atomic_dec(&set->nelems); atomic_dec(&set->nelems);
......
...@@ -13,6 +13,7 @@ ...@@ -13,6 +13,7 @@
#include <net/netfilter/nf_tables_core.h> #include <net/netfilter/nf_tables_core.h>
struct nft_bitmap_elem { struct nft_bitmap_elem {
struct nft_elem_priv priv;
struct list_head head; struct list_head head;
struct nft_set_ext ext; struct nft_set_ext ext;
}; };
...@@ -104,8 +105,9 @@ nft_bitmap_elem_find(const struct nft_set *set, struct nft_bitmap_elem *this, ...@@ -104,8 +105,9 @@ nft_bitmap_elem_find(const struct nft_set *set, struct nft_bitmap_elem *this,
return NULL; return NULL;
} }
static void *nft_bitmap_get(const struct net *net, const struct nft_set *set, static struct nft_elem_priv *
const struct nft_set_elem *elem, unsigned int flags) nft_bitmap_get(const struct net *net, const struct nft_set *set,
const struct nft_set_elem *elem, unsigned int flags)
{ {
const struct nft_bitmap *priv = nft_set_priv(set); const struct nft_bitmap *priv = nft_set_priv(set);
u8 genmask = nft_genmask_cur(net); u8 genmask = nft_genmask_cur(net);
...@@ -116,23 +118,23 @@ static void *nft_bitmap_get(const struct net *net, const struct nft_set *set, ...@@ -116,23 +118,23 @@ static void *nft_bitmap_get(const struct net *net, const struct nft_set *set,
!nft_set_elem_active(&be->ext, genmask)) !nft_set_elem_active(&be->ext, genmask))
continue; continue;
return be; return &be->priv;
} }
return ERR_PTR(-ENOENT); return ERR_PTR(-ENOENT);
} }
static int nft_bitmap_insert(const struct net *net, const struct nft_set *set, static int nft_bitmap_insert(const struct net *net, const struct nft_set *set,
const struct nft_set_elem *elem, const struct nft_set_elem *elem,
struct nft_set_ext **ext) struct nft_elem_priv **elem_priv)
{ {
struct nft_bitmap_elem *new = nft_elem_priv_cast(elem->priv), *be;
struct nft_bitmap *priv = nft_set_priv(set); struct nft_bitmap *priv = nft_set_priv(set);
struct nft_bitmap_elem *new = elem->priv, *be;
u8 genmask = nft_genmask_next(net); u8 genmask = nft_genmask_next(net);
u32 idx, off; u32 idx, off;
be = nft_bitmap_elem_find(set, new, genmask); be = nft_bitmap_elem_find(set, new, genmask);
if (be) { if (be) {
*ext = &be->ext; *elem_priv = &be->priv;
return -EEXIST; return -EEXIST;
} }
...@@ -144,12 +146,11 @@ static int nft_bitmap_insert(const struct net *net, const struct nft_set *set, ...@@ -144,12 +146,11 @@ static int nft_bitmap_insert(const struct net *net, const struct nft_set *set,
return 0; return 0;
} }
static void nft_bitmap_remove(const struct net *net, static void nft_bitmap_remove(const struct net *net, const struct nft_set *set,
const struct nft_set *set, struct nft_elem_priv *elem_priv)
const struct nft_set_elem *elem)
{ {
struct nft_bitmap_elem *be = nft_elem_priv_cast(elem_priv);
struct nft_bitmap *priv = nft_set_priv(set); struct nft_bitmap *priv = nft_set_priv(set);
struct nft_bitmap_elem *be = elem->priv;
u8 genmask = nft_genmask_next(net); u8 genmask = nft_genmask_next(net);
u32 idx, off; u32 idx, off;
...@@ -161,10 +162,10 @@ static void nft_bitmap_remove(const struct net *net, ...@@ -161,10 +162,10 @@ static void nft_bitmap_remove(const struct net *net,
static void nft_bitmap_activate(const struct net *net, static void nft_bitmap_activate(const struct net *net,
const struct nft_set *set, const struct nft_set *set,
const struct nft_set_elem *elem) struct nft_elem_priv *elem_priv)
{ {
struct nft_bitmap_elem *be = nft_elem_priv_cast(elem_priv);
struct nft_bitmap *priv = nft_set_priv(set); struct nft_bitmap *priv = nft_set_priv(set);
struct nft_bitmap_elem *be = elem->priv;
u8 genmask = nft_genmask_next(net); u8 genmask = nft_genmask_next(net);
u32 idx, off; u32 idx, off;
...@@ -174,28 +175,27 @@ static void nft_bitmap_activate(const struct net *net, ...@@ -174,28 +175,27 @@ static void nft_bitmap_activate(const struct net *net,
nft_set_elem_change_active(net, set, &be->ext); nft_set_elem_change_active(net, set, &be->ext);
} }
static bool nft_bitmap_flush(const struct net *net, static void nft_bitmap_flush(const struct net *net,
const struct nft_set *set, void *_be) const struct nft_set *set,
struct nft_elem_priv *elem_priv)
{ {
struct nft_bitmap_elem *be = nft_elem_priv_cast(elem_priv);
struct nft_bitmap *priv = nft_set_priv(set); struct nft_bitmap *priv = nft_set_priv(set);
u8 genmask = nft_genmask_next(net); u8 genmask = nft_genmask_next(net);
struct nft_bitmap_elem *be = _be;
u32 idx, off; u32 idx, off;
nft_bitmap_location(set, nft_set_ext_key(&be->ext), &idx, &off); nft_bitmap_location(set, nft_set_ext_key(&be->ext), &idx, &off);
/* Enter 10 state, similar to deactivation. */ /* Enter 10 state, similar to deactivation. */
priv->bitmap[idx] &= ~(genmask << off); priv->bitmap[idx] &= ~(genmask << off);
nft_set_elem_change_active(net, set, &be->ext); nft_set_elem_change_active(net, set, &be->ext);
return true;
} }
static void *nft_bitmap_deactivate(const struct net *net, static struct nft_elem_priv *
const struct nft_set *set, nft_bitmap_deactivate(const struct net *net, const struct nft_set *set,
const struct nft_set_elem *elem) const struct nft_set_elem *elem)
{ {
struct nft_bitmap_elem *this = nft_elem_priv_cast(elem->priv), *be;
struct nft_bitmap *priv = nft_set_priv(set); struct nft_bitmap *priv = nft_set_priv(set);
struct nft_bitmap_elem *this = elem->priv, *be;
u8 genmask = nft_genmask_next(net); u8 genmask = nft_genmask_next(net);
u32 idx, off; u32 idx, off;
...@@ -209,7 +209,7 @@ static void *nft_bitmap_deactivate(const struct net *net, ...@@ -209,7 +209,7 @@ static void *nft_bitmap_deactivate(const struct net *net,
priv->bitmap[idx] &= ~(genmask << off); priv->bitmap[idx] &= ~(genmask << off);
nft_set_elem_change_active(net, set, &be->ext); nft_set_elem_change_active(net, set, &be->ext);
return be; return &be->priv;
} }
static void nft_bitmap_walk(const struct nft_ctx *ctx, static void nft_bitmap_walk(const struct nft_ctx *ctx,
...@@ -218,7 +218,6 @@ static void nft_bitmap_walk(const struct nft_ctx *ctx, ...@@ -218,7 +218,6 @@ static void nft_bitmap_walk(const struct nft_ctx *ctx,
{ {
const struct nft_bitmap *priv = nft_set_priv(set); const struct nft_bitmap *priv = nft_set_priv(set);
struct nft_bitmap_elem *be; struct nft_bitmap_elem *be;
struct nft_set_elem elem;
list_for_each_entry_rcu(be, &priv->list, head) { list_for_each_entry_rcu(be, &priv->list, head) {
if (iter->count < iter->skip) if (iter->count < iter->skip)
...@@ -226,9 +225,7 @@ static void nft_bitmap_walk(const struct nft_ctx *ctx, ...@@ -226,9 +225,7 @@ static void nft_bitmap_walk(const struct nft_ctx *ctx,
if (!nft_set_elem_active(&be->ext, iter->genmask)) if (!nft_set_elem_active(&be->ext, iter->genmask))
goto cont; goto cont;
elem.priv = be; iter->err = iter->fn(ctx, set, iter, &be->priv);
iter->err = iter->fn(ctx, set, iter, &elem);
if (iter->err < 0) if (iter->err < 0)
return; return;
...@@ -265,6 +262,8 @@ static int nft_bitmap_init(const struct nft_set *set, ...@@ -265,6 +262,8 @@ static int nft_bitmap_init(const struct nft_set *set,
{ {
struct nft_bitmap *priv = nft_set_priv(set); struct nft_bitmap *priv = nft_set_priv(set);
BUILD_BUG_ON(offsetof(struct nft_bitmap_elem, priv) != 0);
INIT_LIST_HEAD(&priv->list); INIT_LIST_HEAD(&priv->list);
priv->bitmap_size = nft_bitmap_size(set->klen); priv->bitmap_size = nft_bitmap_size(set->klen);
...@@ -278,7 +277,7 @@ static void nft_bitmap_destroy(const struct nft_ctx *ctx, ...@@ -278,7 +277,7 @@ static void nft_bitmap_destroy(const struct nft_ctx *ctx,
struct nft_bitmap_elem *be, *n; struct nft_bitmap_elem *be, *n;
list_for_each_entry_safe(be, n, &priv->list, head) list_for_each_entry_safe(be, n, &priv->list, head)
nf_tables_set_elem_destroy(ctx, set, be); nf_tables_set_elem_destroy(ctx, set, &be->priv);
} }
static bool nft_bitmap_estimate(const struct nft_set_desc *desc, u32 features, static bool nft_bitmap_estimate(const struct nft_set_desc *desc, u32 features,
......
This diff is collapsed.
...@@ -599,11 +599,18 @@ static struct nft_pipapo_elem *pipapo_get(const struct net *net, ...@@ -599,11 +599,18 @@ static struct nft_pipapo_elem *pipapo_get(const struct net *net,
* @elem: nftables API element representation containing key data * @elem: nftables API element representation containing key data
* @flags: Unused * @flags: Unused
*/ */
static void *nft_pipapo_get(const struct net *net, const struct nft_set *set, static struct nft_elem_priv *
const struct nft_set_elem *elem, unsigned int flags) nft_pipapo_get(const struct net *net, const struct nft_set *set,
const struct nft_set_elem *elem, unsigned int flags)
{ {
return pipapo_get(net, set, (const u8 *)elem->key.val.data, static struct nft_pipapo_elem *e;
nft_genmask_cur(net));
e = pipapo_get(net, set, (const u8 *)elem->key.val.data,
nft_genmask_cur(net));
if (IS_ERR(e))
return ERR_CAST(e);
return &e->priv;
} }
/** /**
...@@ -1151,21 +1158,21 @@ static int pipapo_realloc_scratch(struct nft_pipapo_match *clone, ...@@ -1151,21 +1158,21 @@ static int pipapo_realloc_scratch(struct nft_pipapo_match *clone,
* @net: Network namespace * @net: Network namespace
* @set: nftables API set representation * @set: nftables API set representation
* @elem: nftables API element representation containing key data * @elem: nftables API element representation containing key data
* @ext2: Filled with pointer to &struct nft_set_ext in inserted element * @elem_priv: Filled with pointer to &struct nft_set_ext in inserted element
* *
* Return: 0 on success, error pointer on failure. * Return: 0 on success, error pointer on failure.
*/ */
static int nft_pipapo_insert(const struct net *net, const struct nft_set *set, static int nft_pipapo_insert(const struct net *net, const struct nft_set *set,
const struct nft_set_elem *elem, const struct nft_set_elem *elem,
struct nft_set_ext **ext2) struct nft_elem_priv **elem_priv)
{ {
const struct nft_set_ext *ext = nft_set_elem_ext(set, elem->priv); const struct nft_set_ext *ext = nft_set_elem_ext(set, elem->priv);
union nft_pipapo_map_bucket rulemap[NFT_PIPAPO_MAX_FIELDS]; union nft_pipapo_map_bucket rulemap[NFT_PIPAPO_MAX_FIELDS];
const u8 *start = (const u8 *)elem->key.val.data, *end; const u8 *start = (const u8 *)elem->key.val.data, *end;
struct nft_pipapo_elem *e = elem->priv, *dup;
struct nft_pipapo *priv = nft_set_priv(set); struct nft_pipapo *priv = nft_set_priv(set);
struct nft_pipapo_match *m = priv->clone; struct nft_pipapo_match *m = priv->clone;
u8 genmask = nft_genmask_next(net); u8 genmask = nft_genmask_next(net);
struct nft_pipapo_elem *e, *dup;
struct nft_pipapo_field *f; struct nft_pipapo_field *f;
const u8 *start_p, *end_p; const u8 *start_p, *end_p;
int i, bsize_max, err = 0; int i, bsize_max, err = 0;
...@@ -1188,7 +1195,7 @@ static int nft_pipapo_insert(const struct net *net, const struct nft_set *set, ...@@ -1188,7 +1195,7 @@ static int nft_pipapo_insert(const struct net *net, const struct nft_set *set,
if (!memcmp(start, dup_key->data, sizeof(*dup_key->data)) && if (!memcmp(start, dup_key->data, sizeof(*dup_key->data)) &&
!memcmp(end, dup_end->data, sizeof(*dup_end->data))) { !memcmp(end, dup_end->data, sizeof(*dup_end->data))) {
*ext2 = &dup->ext; *elem_priv = &dup->priv;
return -EEXIST; return -EEXIST;
} }
...@@ -1203,7 +1210,7 @@ static int nft_pipapo_insert(const struct net *net, const struct nft_set *set, ...@@ -1203,7 +1210,7 @@ static int nft_pipapo_insert(const struct net *net, const struct nft_set *set,
if (PTR_ERR(dup) != -ENOENT) { if (PTR_ERR(dup) != -ENOENT) {
if (IS_ERR(dup)) if (IS_ERR(dup))
return PTR_ERR(dup); return PTR_ERR(dup);
*ext2 = &dup->ext; *elem_priv = &dup->priv;
return -ENOTEMPTY; return -ENOTEMPTY;
} }
...@@ -1263,7 +1270,8 @@ static int nft_pipapo_insert(const struct net *net, const struct nft_set *set, ...@@ -1263,7 +1270,8 @@ static int nft_pipapo_insert(const struct net *net, const struct nft_set *set,
put_cpu_ptr(m->scratch); put_cpu_ptr(m->scratch);
} }
*ext2 = &e->ext; e = nft_elem_priv_cast(elem->priv);
*elem_priv = &e->priv;
pipapo_map(m, rulemap, e); pipapo_map(m, rulemap, e);
...@@ -1540,11 +1548,7 @@ static void nft_pipapo_gc_deactivate(struct net *net, struct nft_set *set, ...@@ -1540,11 +1548,7 @@ static void nft_pipapo_gc_deactivate(struct net *net, struct nft_set *set,
struct nft_pipapo_elem *e) struct nft_pipapo_elem *e)
{ {
struct nft_set_elem elem = { nft_setelem_data_deactivate(net, set, &e->priv);
.priv = e,
};
nft_setelem_data_deactivate(net, set, &elem);
} }
/** /**
...@@ -1731,7 +1735,7 @@ static void nft_pipapo_abort(const struct nft_set *set) ...@@ -1731,7 +1735,7 @@ static void nft_pipapo_abort(const struct nft_set *set)
* nft_pipapo_activate() - Mark element reference as active given key, commit * nft_pipapo_activate() - Mark element reference as active given key, commit
* @net: Network namespace * @net: Network namespace
* @set: nftables API set representation * @set: nftables API set representation
* @elem: nftables API element representation containing key data * @elem_priv: nftables API element representation containing key data
* *
* On insertion, elements are added to a copy of the matching data currently * On insertion, elements are added to a copy of the matching data currently
* in use for lookups, and not directly inserted into current lookup data. Both * in use for lookups, and not directly inserted into current lookup data. Both
...@@ -1740,9 +1744,9 @@ static void nft_pipapo_abort(const struct nft_set *set) ...@@ -1740,9 +1744,9 @@ static void nft_pipapo_abort(const struct nft_set *set)
*/ */
static void nft_pipapo_activate(const struct net *net, static void nft_pipapo_activate(const struct net *net,
const struct nft_set *set, const struct nft_set *set,
const struct nft_set_elem *elem) struct nft_elem_priv *elem_priv)
{ {
struct nft_pipapo_elem *e = elem->priv; struct nft_pipapo_elem *e = nft_elem_priv_cast(elem_priv);
nft_set_elem_change_active(net, set, &e->ext); nft_set_elem_change_active(net, set, &e->ext);
} }
...@@ -1782,9 +1786,9 @@ static void *pipapo_deactivate(const struct net *net, const struct nft_set *set, ...@@ -1782,9 +1786,9 @@ static void *pipapo_deactivate(const struct net *net, const struct nft_set *set,
* *
* Return: deactivated element if found, NULL otherwise. * Return: deactivated element if found, NULL otherwise.
*/ */
static void *nft_pipapo_deactivate(const struct net *net, static struct nft_elem_priv *
const struct nft_set *set, nft_pipapo_deactivate(const struct net *net, const struct nft_set *set,
const struct nft_set_elem *elem) const struct nft_set_elem *elem)
{ {
const struct nft_set_ext *ext = nft_set_elem_ext(set, elem->priv); const struct nft_set_ext *ext = nft_set_elem_ext(set, elem->priv);
...@@ -1795,7 +1799,7 @@ static void *nft_pipapo_deactivate(const struct net *net, ...@@ -1795,7 +1799,7 @@ static void *nft_pipapo_deactivate(const struct net *net,
* nft_pipapo_flush() - Call pipapo_deactivate() to make element inactive * nft_pipapo_flush() - Call pipapo_deactivate() to make element inactive
* @net: Network namespace * @net: Network namespace
* @set: nftables API set representation * @set: nftables API set representation
* @elem: nftables API element representation containing key data * @elem_priv: nftables API element representation containing key data
* *
* This is functionally the same as nft_pipapo_deactivate(), with a slightly * This is functionally the same as nft_pipapo_deactivate(), with a slightly
* different interface, and it's also called once for each element in a set * different interface, and it's also called once for each element in a set
...@@ -1809,13 +1813,12 @@ static void *nft_pipapo_deactivate(const struct net *net, ...@@ -1809,13 +1813,12 @@ static void *nft_pipapo_deactivate(const struct net *net,
* *
* Return: true if element was found and deactivated. * Return: true if element was found and deactivated.
*/ */
static bool nft_pipapo_flush(const struct net *net, const struct nft_set *set, static void nft_pipapo_flush(const struct net *net, const struct nft_set *set,
void *elem) struct nft_elem_priv *elem_priv)
{ {
struct nft_pipapo_elem *e = elem; struct nft_pipapo_elem *e = nft_elem_priv_cast(elem_priv);
return pipapo_deactivate(net, set, (const u8 *)nft_set_ext_key(&e->ext), nft_set_elem_change_active(net, set, &e->ext);
&e->ext);
} }
/** /**
...@@ -1938,7 +1941,7 @@ static bool pipapo_match_field(struct nft_pipapo_field *f, ...@@ -1938,7 +1941,7 @@ static bool pipapo_match_field(struct nft_pipapo_field *f,
* nft_pipapo_remove() - Remove element given key, commit * nft_pipapo_remove() - Remove element given key, commit
* @net: Network namespace * @net: Network namespace
* @set: nftables API set representation * @set: nftables API set representation
* @elem: nftables API element representation containing key data * @elem_priv: nftables API element representation containing key data
* *
* Similarly to nft_pipapo_activate(), this is used as commit operation by the * Similarly to nft_pipapo_activate(), this is used as commit operation by the
* API, but it's called once per element in the pending transaction, so we can't * API, but it's called once per element in the pending transaction, so we can't
...@@ -1946,14 +1949,15 @@ static bool pipapo_match_field(struct nft_pipapo_field *f, ...@@ -1946,14 +1949,15 @@ static bool pipapo_match_field(struct nft_pipapo_field *f,
* the matched element here, if any, and commit the updated matching data. * the matched element here, if any, and commit the updated matching data.
*/ */
static void nft_pipapo_remove(const struct net *net, const struct nft_set *set, static void nft_pipapo_remove(const struct net *net, const struct nft_set *set,
const struct nft_set_elem *elem) struct nft_elem_priv *elem_priv)
{ {
struct nft_pipapo *priv = nft_set_priv(set); struct nft_pipapo *priv = nft_set_priv(set);
struct nft_pipapo_match *m = priv->clone; struct nft_pipapo_match *m = priv->clone;
struct nft_pipapo_elem *e = elem->priv;
int rules_f0, first_rule = 0; int rules_f0, first_rule = 0;
struct nft_pipapo_elem *e;
const u8 *data; const u8 *data;
e = nft_elem_priv_cast(elem_priv);
data = (const u8 *)nft_set_ext_key(&e->ext); data = (const u8 *)nft_set_ext_key(&e->ext);
while ((rules_f0 = pipapo_rules_same_key(m->f, first_rule))) { while ((rules_f0 = pipapo_rules_same_key(m->f, first_rule))) {
...@@ -2030,7 +2034,6 @@ static void nft_pipapo_walk(const struct nft_ctx *ctx, struct nft_set *set, ...@@ -2030,7 +2034,6 @@ static void nft_pipapo_walk(const struct nft_ctx *ctx, struct nft_set *set,
for (r = 0; r < f->rules; r++) { for (r = 0; r < f->rules; r++) {
struct nft_pipapo_elem *e; struct nft_pipapo_elem *e;
struct nft_set_elem elem;
if (r < f->rules - 1 && f->mt[r + 1].e == f->mt[r].e) if (r < f->rules - 1 && f->mt[r + 1].e == f->mt[r].e)
continue; continue;
...@@ -2040,9 +2043,7 @@ static void nft_pipapo_walk(const struct nft_ctx *ctx, struct nft_set *set, ...@@ -2040,9 +2043,7 @@ static void nft_pipapo_walk(const struct nft_ctx *ctx, struct nft_set *set,
e = f->mt[r].e; e = f->mt[r].e;
elem.priv = e; iter->err = iter->fn(ctx, set, iter, &e->priv);
iter->err = iter->fn(ctx, set, iter, &elem);
if (iter->err < 0) if (iter->err < 0)
goto out; goto out;
...@@ -2114,6 +2115,8 @@ static int nft_pipapo_init(const struct nft_set *set, ...@@ -2114,6 +2115,8 @@ static int nft_pipapo_init(const struct nft_set *set,
struct nft_pipapo_field *f; struct nft_pipapo_field *f;
int err, i, field_count; int err, i, field_count;
BUILD_BUG_ON(offsetof(struct nft_pipapo_elem, priv) != 0);
field_count = desc->field_count ? : 1; field_count = desc->field_count ? : 1;
if (field_count > NFT_PIPAPO_MAX_FIELDS) if (field_count > NFT_PIPAPO_MAX_FIELDS)
...@@ -2208,7 +2211,7 @@ static void nft_set_pipapo_match_destroy(const struct nft_ctx *ctx, ...@@ -2208,7 +2211,7 @@ static void nft_set_pipapo_match_destroy(const struct nft_ctx *ctx,
e = f->mt[r].e; e = f->mt[r].e;
nf_tables_set_elem_destroy(ctx, set, e); nf_tables_set_elem_destroy(ctx, set, &e->priv);
} }
} }
......
...@@ -170,10 +170,12 @@ struct nft_pipapo_elem; ...@@ -170,10 +170,12 @@ struct nft_pipapo_elem;
/** /**
* struct nft_pipapo_elem - API-facing representation of single set element * struct nft_pipapo_elem - API-facing representation of single set element
* @priv: element placeholder
* @ext: nftables API extensions * @ext: nftables API extensions
*/ */
struct nft_pipapo_elem { struct nft_pipapo_elem {
struct nft_set_ext ext; struct nft_elem_priv priv;
struct nft_set_ext ext;
}; };
int pipapo_refill(unsigned long *map, int len, int rules, unsigned long *dst, int pipapo_refill(unsigned long *map, int len, int rules, unsigned long *dst,
......
This diff is collapsed.
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment