]> git.ipfire.org Git - thirdparty/kernel/stable.git/commitdiff
bpf: Handle BPF_EXIST and BPF_NOEXIST for LPM trie
authorHou Tao <houtao1@huawei.com>
Fri, 6 Dec 2024 11:06:16 +0000 (19:06 +0800)
committerGreg Kroah-Hartman <gregkh@linuxfoundation.org>
Sat, 14 Dec 2024 18:51:31 +0000 (19:51 +0100)
[ Upstream commit eae6a075e9537dd69891cf77ca5a88fa8a28b4a1 ]

Add the currently missing handling for the BPF_EXIST and BPF_NOEXIST
flags. These flags can be specified by users and are relevant since LPM
trie supports exact matches during update.

Fixes: b95a5c4db09b ("bpf: add a longest prefix match trie map implementation")
Reviewed-by: Toke Høiland-Jørgensen <toke@redhat.com>
Acked-by: Daniel Borkmann <daniel@iogearbox.net>
Signed-off-by: Hou Tao <houtao1@huawei.com>
Link: https://lore.kernel.org/r/20241206110622.1161752-4-houtao@huaweicloud.com
Signed-off-by: Alexei Starovoitov <ast@kernel.org>
Signed-off-by: Sasha Levin <sashal@kernel.org>
kernel/bpf/lpm_trie.c

index 6b2bf71f8de4ba72fe76b03dbdc46ebc92b0b51e..0ab0ad989eaff1327ccaa5a3dda817bea9c00244 100644 (file)
@@ -366,6 +366,10 @@ static int trie_update_elem(struct bpf_map *map,
         * simply assign the @new_node to that slot and be done.
         */
        if (!node) {
+               if (flags == BPF_EXIST) {
+                       ret = -ENOENT;
+                       goto out;
+               }
                rcu_assign_pointer(*slot, new_node);
                goto out;
        }
@@ -374,18 +378,31 @@ static int trie_update_elem(struct bpf_map *map,
         * which already has the correct data array set.
         */
        if (node->prefixlen == matchlen) {
+               if (!(node->flags & LPM_TREE_NODE_FLAG_IM)) {
+                       if (flags == BPF_NOEXIST) {
+                               ret = -EEXIST;
+                               goto out;
+                       }
+                       trie->n_entries--;
+               } else if (flags == BPF_EXIST) {
+                       ret = -ENOENT;
+                       goto out;
+               }
+
                new_node->child[0] = node->child[0];
                new_node->child[1] = node->child[1];
 
-               if (!(node->flags & LPM_TREE_NODE_FLAG_IM))
-                       trie->n_entries--;
-
                rcu_assign_pointer(*slot, new_node);
                kfree_rcu(node, rcu);
 
                goto out;
        }
 
+       if (flags == BPF_EXIST) {
+               ret = -ENOENT;
+               goto out;
+       }
+
        /* If the new node matches the prefix completely, it must be inserted
         * as an ancestor. Simply insert it between @node and *@slot.
         */