}
 
 /**
- * nft_pipapo_walk() - Walk over elements
+ * nft_pipapo_do_walk() - Walk over elements in m
  * @ctx:       nftables API context
  * @set:       nftables API set representation
+ * @m:         matching data pointing to key mapping array
  * @iter:      Iterator
  *
  * As elements are referenced in the mapping array for the last field, directly
  * scan that array: there's no need to follow rule mappings from the first
- * field.
+ * field. @m is protected either by RCU read lock or by transaction mutex.
  */
-static void nft_pipapo_walk(const struct nft_ctx *ctx, struct nft_set *set,
-                           struct nft_set_iter *iter)
+static void nft_pipapo_do_walk(const struct nft_ctx *ctx, struct nft_set *set,
+                              const struct nft_pipapo_match *m,
+                              struct nft_set_iter *iter)
 {
-       struct nft_pipapo *priv = nft_set_priv(set);
-       const struct nft_pipapo_match *m;
        const struct nft_pipapo_field *f;
        unsigned int i, r;
 
-       WARN_ON_ONCE(iter->type != NFT_ITER_READ &&
-                    iter->type != NFT_ITER_UPDATE);
-
-       rcu_read_lock();
-       if (iter->type == NFT_ITER_READ)
-               m = rcu_dereference(priv->match);
-       else
-               m = priv->clone;
-
-       if (unlikely(!m))
-               goto out;
-
        for (i = 0, f = m->f; i < m->field_count - 1; i++, f++)
                ;
 
 
                iter->err = iter->fn(ctx, set, iter, &e->priv);
                if (iter->err < 0)
-                       goto out;
+                       return;
 
 cont:
                iter->count++;
        }
+}
 
-out:
-       rcu_read_unlock();
+/**
+ * nft_pipapo_walk() - Walk over elements
+ * @ctx:       nftables API context
+ * @set:       nftables API set representation
+ * @iter:      Iterator
+ *
+ * Test if destructive action is needed or not, clone active backend if needed
+ * and call the real function to work on the data.
+ */
+static void nft_pipapo_walk(const struct nft_ctx *ctx, struct nft_set *set,
+                           struct nft_set_iter *iter)
+{
+       struct nft_pipapo *priv = nft_set_priv(set);
+       const struct nft_pipapo_match *m;
+
+       switch (iter->type) {
+       case NFT_ITER_UPDATE:
+               m = priv->clone;
+               nft_pipapo_do_walk(ctx, set, m, iter);
+               break;
+       case NFT_ITER_READ:
+               rcu_read_lock();
+               m = rcu_dereference(priv->match);
+               nft_pipapo_do_walk(ctx, set, m, iter);
+               rcu_read_unlock();
+               break;
+       default:
+               iter->err = -EINVAL;
+               WARN_ON_ONCE(1);
+               break;
+       }
 }
 
 /**