*     control plane functions.
  */
 struct nft_set_ops {
-       bool                            (*lookup)(const struct net *net,
+       const struct nft_set_ext *      (*lookup)(const struct net *net,
                                                  const struct nft_set *set,
-                                                 const u32 *key,
-                                                 const struct nft_set_ext **ext);
-       bool                            (*update)(struct nft_set *set,
+                                                 const u32 *key);
+       const struct nft_set_ext *      (*update)(struct nft_set *set,
                                                  const u32 *key,
                                                  struct nft_elem_priv *
                                                        (*new)(struct nft_set *,
                                                               const struct nft_expr *,
                                                               struct nft_regs *),
                                                  const struct nft_expr *expr,
-                                                 struct nft_regs *regs,
-                                                 const struct nft_set_ext **ext);
+                                                 struct nft_regs *regs);
        bool                            (*delete)(const struct nft_set *set,
                                                  const u32 *key);
 
 
 extern const struct nft_set_type nft_set_pipapo_avx2_type;
 
 #ifdef CONFIG_MITIGATION_RETPOLINE
-bool nft_rhash_lookup(const struct net *net, const struct nft_set *set,
-                     const u32 *key, const struct nft_set_ext **ext);
-bool nft_rbtree_lookup(const struct net *net, const struct nft_set *set,
-                      const u32 *key, const struct nft_set_ext **ext);
-bool nft_bitmap_lookup(const struct net *net, const struct nft_set *set,
-                      const u32 *key, const struct nft_set_ext **ext);
-bool nft_hash_lookup_fast(const struct net *net,
-                         const struct nft_set *set,
-                         const u32 *key, const struct nft_set_ext **ext);
-bool nft_hash_lookup(const struct net *net, const struct nft_set *set,
-                    const u32 *key, const struct nft_set_ext **ext);
-bool nft_set_do_lookup(const struct net *net, const struct nft_set *set,
-                      const u32 *key, const struct nft_set_ext **ext);
+const struct nft_set_ext *
+nft_rhash_lookup(const struct net *net, const struct nft_set *set,
+                const u32 *key);
+const struct nft_set_ext *
+nft_rbtree_lookup(const struct net *net, const struct nft_set *set,
+                 const u32 *key);
+const struct nft_set_ext *
+nft_bitmap_lookup(const struct net *net, const struct nft_set *set,
+                 const u32 *key);
+const struct nft_set_ext *
+nft_hash_lookup_fast(const struct net *net, const struct nft_set *set,
+                    const u32 *key);
+const struct nft_set_ext *
+nft_hash_lookup(const struct net *net, const struct nft_set *set,
+               const u32 *key);
+const struct nft_set_ext *
+nft_set_do_lookup(const struct net *net, const struct nft_set *set,
+                 const u32 *key);
 #else
-static inline bool
+static inline const struct nft_set_ext *
 nft_set_do_lookup(const struct net *net, const struct nft_set *set,
-                 const u32 *key, const struct nft_set_ext **ext)
+                 const u32 *key)
 {
-       return set->ops->lookup(net, set, key, ext);
+       return set->ops->lookup(net, set, key);
 }
 #endif
 
 /* called from nft_pipapo_avx2.c */
-bool nft_pipapo_lookup(const struct net *net, const struct nft_set *set,
-                      const u32 *key, const struct nft_set_ext **ext);
+const struct nft_set_ext *
+nft_pipapo_lookup(const struct net *net, const struct nft_set *set,
+                 const u32 *key);
 /* called from nft_set_pipapo.c */
-bool nft_pipapo_avx2_lookup(const struct net *net, const struct nft_set *set,
-                           const u32 *key, const struct nft_set_ext **ext);
+const struct nft_set_ext *
+nft_pipapo_avx2_lookup(const struct net *net, const struct nft_set *set,
+                       const u32 *key);
 
 void nft_counter_init_seqcount(void);
 
 
                return;
        }
 
-       if (set->ops->update(set, ®s->data[priv->sreg_key], nft_dynset_new,
-                            expr, regs, &ext)) {
+       ext = set->ops->update(set, ®s->data[priv->sreg_key], nft_dynset_new,
+                            expr, regs);
+       if (ext) {
                if (priv->op == NFT_DYNSET_OP_UPDATE &&
                    nft_set_ext_exists(ext, NFT_SET_EXT_TIMEOUT) &&
                    READ_ONCE(nft_set_ext_timeout(ext)->timeout) != 0) {
 
 };
 
 #ifdef CONFIG_MITIGATION_RETPOLINE
-bool nft_set_do_lookup(const struct net *net, const struct nft_set *set,
-                      const u32 *key, const struct nft_set_ext **ext)
+const struct nft_set_ext *
+nft_set_do_lookup(const struct net *net, const struct nft_set *set,
+                 const u32 *key)
 {
        if (set->ops == &nft_set_hash_fast_type.ops)
-               return nft_hash_lookup_fast(net, set, key, ext);
+               return nft_hash_lookup_fast(net, set, key);
        if (set->ops == &nft_set_hash_type.ops)
-               return nft_hash_lookup(net, set, key, ext);
+               return nft_hash_lookup(net, set, key);
 
        if (set->ops == &nft_set_rhash_type.ops)
-               return nft_rhash_lookup(net, set, key, ext);
+               return nft_rhash_lookup(net, set, key);
 
        if (set->ops == &nft_set_bitmap_type.ops)
-               return nft_bitmap_lookup(net, set, key, ext);
+               return nft_bitmap_lookup(net, set, key);
 
        if (set->ops == &nft_set_pipapo_type.ops)
-               return nft_pipapo_lookup(net, set, key, ext);
+               return nft_pipapo_lookup(net, set, key);
 #if defined(CONFIG_X86_64) && !defined(CONFIG_UML)
        if (set->ops == &nft_set_pipapo_avx2_type.ops)
-               return nft_pipapo_avx2_lookup(net, set, key, ext);
+               return nft_pipapo_avx2_lookup(net, set, key);
 #endif
 
        if (set->ops == &nft_set_rbtree_type.ops)
-               return nft_rbtree_lookup(net, set, key, ext);
+               return nft_rbtree_lookup(net, set, key);
 
        WARN_ON_ONCE(1);
-       return set->ops->lookup(net, set, key, ext);
+       return set->ops->lookup(net, set, key);
 }
 EXPORT_SYMBOL_GPL(nft_set_do_lookup);
 #endif
 {
        const struct nft_lookup *priv = nft_expr_priv(expr);
        const struct nft_set *set = priv->set;
-       const struct nft_set_ext *ext = NULL;
        const struct net *net = nft_net(pkt);
+       const struct nft_set_ext *ext;
        bool found;
 
-       found = nft_set_do_lookup(net, set, ®s->data[priv->sreg], &ext) ^
-                                 priv->invert;
+       ext = nft_set_do_lookup(net, set, ®s->data[priv->sreg]);
+       found = !!ext ^ priv->invert;
        if (!found) {
                ext = nft_set_catchall_lookup(net, set);
                if (!ext) {
 
        struct net *net = nft_net(pkt);
        const struct nft_set_ext *ext;
        struct nft_object *obj;
-       bool found;
 
-       found = nft_set_do_lookup(net, set, ®s->data[priv->sreg], &ext);
-       if (!found) {
+       ext = nft_set_do_lookup(net, set, ®s->data[priv->sreg]);
+       if (!ext) {
                ext = nft_set_catchall_lookup(net, set);
                if (!ext) {
                        regs->verdict.code = NFT_BREAK;
 
 }
 
 INDIRECT_CALLABLE_SCOPE
-bool nft_bitmap_lookup(const struct net *net, const struct nft_set *set,
-                      const u32 *key, const struct nft_set_ext **ext)
+const struct nft_set_ext *
+nft_bitmap_lookup(const struct net *net, const struct nft_set *set,
+                 const u32 *key)
 {
        const struct nft_bitmap *priv = nft_set_priv(set);
+       static const struct nft_set_ext found;
        u8 genmask = nft_genmask_cur(net);
        u32 idx, off;
 
        nft_bitmap_location(set, key, &idx, &off);
 
-       return nft_bitmap_active(priv->bitmap, idx, off, genmask);
+       if (nft_bitmap_active(priv->bitmap, idx, off, genmask))
+               return &found;
+
+       return NULL;
 }
 
 static struct nft_bitmap_elem *
 
 };
 
 INDIRECT_CALLABLE_SCOPE
-bool nft_rhash_lookup(const struct net *net, const struct nft_set *set,
-                     const u32 *key, const struct nft_set_ext **ext)
+const struct nft_set_ext *
+nft_rhash_lookup(const struct net *net, const struct nft_set *set,
+                const u32 *key)
 {
        struct nft_rhash *priv = nft_set_priv(set);
        const struct nft_rhash_elem *he;
 
        he = rhashtable_lookup(&priv->ht, &arg, nft_rhash_params);
        if (he != NULL)
-               *ext = &he->ext;
+               return &he->ext;
 
-       return !!he;
+       return NULL;
 }
 
 static struct nft_elem_priv *
        return ERR_PTR(-ENOENT);
 }
 
-static bool nft_rhash_update(struct nft_set *set, const u32 *key,
-                            struct nft_elem_priv *
-                                  (*new)(struct nft_set *,
-                                         const struct nft_expr *,
-                                         struct nft_regs *regs),
-                            const struct nft_expr *expr,
-                            struct nft_regs *regs,
-                            const struct nft_set_ext **ext)
+static const struct nft_set_ext *
+nft_rhash_update(struct nft_set *set, const u32 *key,
+                struct nft_elem_priv *(*new)(struct nft_set *, const struct nft_expr *,
+                struct nft_regs *regs),
+                const struct nft_expr *expr, struct nft_regs *regs)
 {
        struct nft_rhash *priv = nft_set_priv(set);
        struct nft_rhash_elem *he, *prev;
        }
 
 out:
-       *ext = &he->ext;
-       return true;
+       return &he->ext;
 
 err2:
        nft_set_elem_destroy(set, &he->priv, true);
        atomic_dec(&set->nelems);
 err1:
-       return false;
+       return NULL;
 }
 
 static int nft_rhash_insert(const struct net *net, const struct nft_set *set,
 };
 
 INDIRECT_CALLABLE_SCOPE
-bool nft_hash_lookup(const struct net *net, const struct nft_set *set,
-                    const u32 *key, const struct nft_set_ext **ext)
+const struct nft_set_ext *
+nft_hash_lookup(const struct net *net, const struct nft_set *set,
+               const u32 *key)
 {
        struct nft_hash *priv = nft_set_priv(set);
        u8 genmask = nft_genmask_cur(net);
        hash = reciprocal_scale(hash, priv->buckets);
        hlist_for_each_entry_rcu(he, &priv->table[hash], node) {
                if (!memcmp(nft_set_ext_key(&he->ext), key, set->klen) &&
-                   nft_set_elem_active(&he->ext, genmask)) {
-                       *ext = &he->ext;
-                       return true;
-               }
+                   nft_set_elem_active(&he->ext, genmask))
+                       return &he->ext;
        }
-       return false;
+       return NULL;
 }
 
 static struct nft_elem_priv *
 }
 
 INDIRECT_CALLABLE_SCOPE
-bool nft_hash_lookup_fast(const struct net *net,
-                         const struct nft_set *set,
-                         const u32 *key, const struct nft_set_ext **ext)
+const struct nft_set_ext *
+nft_hash_lookup_fast(const struct net *net, const struct nft_set *set,
+                    const u32 *key)
 {
        struct nft_hash *priv = nft_set_priv(set);
        u8 genmask = nft_genmask_cur(net);
        hlist_for_each_entry_rcu(he, &priv->table[hash], node) {
                k2 = *(u32 *)nft_set_ext_key(&he->ext)->data;
                if (k1 == k2 &&
-                   nft_set_elem_active(&he->ext, genmask)) {
-                       *ext = &he->ext;
-                       return true;
-               }
+                   nft_set_elem_active(&he->ext, genmask))
+                       return &he->ext;
        }
-       return false;
+       return NULL;
 }
 
 static u32 nft_jhash(const struct nft_set *set, const struct nft_hash *priv,
 
  *
  * Return: true on match, false otherwise.
  */
-bool nft_pipapo_lookup(const struct net *net, const struct nft_set *set,
-                      const u32 *key, const struct nft_set_ext **ext)
+const struct nft_set_ext *
+nft_pipapo_lookup(const struct net *net, const struct nft_set *set,
+                 const u32 *key)
 {
        struct nft_pipapo *priv = nft_set_priv(set);
        struct nft_pipapo_scratch *scratch;
                        scratch->map_index = map_index;
                        local_bh_enable();
 
-                       return false;
+                       return NULL;
                }
 
                if (last) {
-                       *ext = &f->mt[b].e->ext;
-                       if (unlikely(nft_set_elem_expired(*ext) ||
-                                    !nft_set_elem_active(*ext, genmask)))
+                       const struct nft_set_ext *ext;
+
+                       ext = &f->mt[b].e->ext;
+                       if (unlikely(nft_set_elem_expired(ext) ||
+                                    !nft_set_elem_active(ext, genmask)))
                                goto next_match;
 
                        /* Last field: we're just returning the key without
                        scratch->map_index = map_index;
                        local_bh_enable();
 
-                       return true;
+                       return ext;
                }
 
                /* Swap bitmap indices: res_map is the initial bitmap for the
 
 out:
        local_bh_enable();
-       return false;
+       return NULL;
 }
 
 /**
 
  *
  * Return: true on match, false otherwise.
  */
-bool nft_pipapo_avx2_lookup(const struct net *net, const struct nft_set *set,
-                           const u32 *key, const struct nft_set_ext **ext)
+const struct nft_set_ext *
+nft_pipapo_avx2_lookup(const struct net *net, const struct nft_set *set,
+                      const u32 *key)
 {
        struct nft_pipapo *priv = nft_set_priv(set);
        struct nft_pipapo_scratch *scratch;
        const struct nft_pipapo_match *m;
        const struct nft_pipapo_field *f;
        const u8 *rp = (const u8 *)key;
+       const struct nft_set_ext *ext;
        unsigned long *res, *fill;
        bool map_index;
-       int i, ret = 0;
+       int i;
 
        local_bh_disable();
 
        if (unlikely(!irq_fpu_usable())) {
-               bool fallback_res = nft_pipapo_lookup(net, set, key, ext);
+               ext = nft_pipapo_lookup(net, set, key);
 
                local_bh_enable();
-               return fallback_res;
+               return ext;
        }
 
        m = rcu_dereference(priv->match);
        if (unlikely(!scratch)) {
                kernel_fpu_end();
                local_bh_enable();
-               return false;
+               return NULL;
        }
 
        map_index = scratch->map_index;
 next_match:
        nft_pipapo_for_each_field(f, i, m) {
                bool last = i == m->field_count - 1, first = !i;
+               int ret = 0;
 
 #define NFT_SET_PIPAPO_AVX2_LOOKUP(b, n)                               \
                (ret = nft_pipapo_avx2_lookup_##b##b_##n(res, fill, f,  \
                        goto out;
 
                if (last) {
-                       *ext = &f->mt[ret].e->ext;
-                       if (unlikely(nft_set_elem_expired(*ext) ||
-                                    !nft_set_elem_active(*ext, genmask))) {
-                               ret = 0;
+                       ext = &f->mt[ret].e->ext;
+                       if (unlikely(nft_set_elem_expired(ext) ||
+                                    !nft_set_elem_active(ext, genmask))) {
+                               ext = NULL;
                                goto next_match;
                        }
 
        kernel_fpu_end();
        local_bh_enable();
 
-       return ret >= 0;
+       return ext;
 }
 
        return nft_set_elem_expired(&rbe->ext);
 }
 
-static bool __nft_rbtree_lookup(const struct net *net, const struct nft_set *set,
-                               const u32 *key, const struct nft_set_ext **ext,
-                               unsigned int seq)
+static const struct nft_set_ext *
+__nft_rbtree_lookup(const struct net *net, const struct nft_set *set,
+                   const u32 *key, unsigned int seq)
 {
        struct nft_rbtree *priv = nft_set_priv(set);
        const struct nft_rbtree_elem *rbe, *interval = NULL;
        parent = rcu_dereference_raw(priv->root.rb_node);
        while (parent != NULL) {
                if (read_seqcount_retry(&priv->count, seq))
-                       return false;
+                       return NULL;
 
                rbe = rb_entry(parent, struct nft_rbtree_elem, node);
 
                        }
 
                        if (nft_rbtree_elem_expired(rbe))
-                               return false;
+                               return NULL;
 
                        if (nft_rbtree_interval_end(rbe)) {
                                if (nft_set_is_anonymous(set))
-                                       return false;
+                                       return NULL;
                                parent = rcu_dereference_raw(parent->rb_left);
                                interval = NULL;
                                continue;
                        }
 
-                       *ext = &rbe->ext;
-                       return true;
+                       return &rbe->ext;
                }
        }
 
        if (set->flags & NFT_SET_INTERVAL && interval != NULL &&
            nft_set_elem_active(&interval->ext, genmask) &&
            !nft_rbtree_elem_expired(interval) &&
-           nft_rbtree_interval_start(interval)) {
-               *ext = &interval->ext;
-               return true;
-       }
+           nft_rbtree_interval_start(interval))
+               return &interval->ext;
 
-       return false;
+       return NULL;
 }
 
 INDIRECT_CALLABLE_SCOPE
-bool nft_rbtree_lookup(const struct net *net, const struct nft_set *set,
-                      const u32 *key, const struct nft_set_ext **ext)
+const struct nft_set_ext *
+nft_rbtree_lookup(const struct net *net, const struct nft_set *set,
+                 const u32 *key)
 {
        struct nft_rbtree *priv = nft_set_priv(set);
        unsigned int seq = read_seqcount_begin(&priv->count);
-       bool ret;
+       const struct nft_set_ext *ext;
 
-       ret = __nft_rbtree_lookup(net, set, key, ext, seq);
-       if (ret || !read_seqcount_retry(&priv->count, seq))
-               return ret;
+       ext = __nft_rbtree_lookup(net, set, key, seq);
+       if (ext || !read_seqcount_retry(&priv->count, seq))
+               return ext;
 
        read_lock_bh(&priv->lock);
        seq = read_seqcount_begin(&priv->count);
-       ret = __nft_rbtree_lookup(net, set, key, ext, seq);
+       ext = __nft_rbtree_lookup(net, set, key, seq);
        read_unlock_bh(&priv->lock);
 
-       return ret;
+       return ext;
 }
 
 static bool __nft_rbtree_get(const struct net *net, const struct nft_set *set,