From 63c2ade6e6f034229b23ec508a7c19071fdc2ea7 Mon Sep 17 00:00:00 2001
From: Kirill Shcherbatov <kshcherbatov@tarantool.org>
Date: Tue, 30 Apr 2019 16:55:45 +0300
Subject: [PATCH] salad: introduce bps_tree_delete_identical routine

A new routine bps_tree_delete_identical performs an element
deletion if and only if the found element is identical
to the routine argument.

Needed for #1257
---
 src/box/vy_cache.h        |  4 +--
 src/box/vy_mem.h          |  4 +--
 src/lib/salad/bps_tree.h  | 35 +++++++++++++++++++-
 test/unit/bps_tree.cc     | 68 +++++++++++++++++++++++++++++++++++++++
 test/unit/bps_tree.result |  2 ++
 5 files changed, 108 insertions(+), 5 deletions(-)

diff --git a/src/box/vy_cache.h b/src/box/vy_cache.h
index a042904757..1f0143059b 100644
--- a/src/box/vy_cache.h
+++ b/src/box/vy_cache.h
@@ -97,7 +97,7 @@ vy_cache_tree_key_cmp(struct vy_cache_node *a, struct vy_entry b,
 #define bps_tree_elem_t struct vy_cache_node *
 #define bps_tree_key_t struct vy_entry
 #define bps_tree_arg_t struct key_def *
-#define BPS_TREE_NO_DEBUG
+#define BPS_TREE_IDENTICAL(a, b) vy_entry_is_equal(a->entry, b->entry)
 
 #include "salad/bps_tree.h"
 
@@ -109,7 +109,7 @@ vy_cache_tree_key_cmp(struct vy_cache_node *a, struct vy_entry b,
 #undef bps_tree_elem_t
 #undef bps_tree_key_t
 #undef bps_tree_arg_t
-#undef BPS_TREE_NO_DEBUG
+#undef BPS_TREE_IDENTICAL
 
 /**
  * Environment of the cache
diff --git a/src/box/vy_mem.h b/src/box/vy_mem.h
index 7df9a18172..35cca63340 100644
--- a/src/box/vy_mem.h
+++ b/src/box/vy_mem.h
@@ -121,10 +121,10 @@ vy_mem_tree_cmp_key(struct vy_entry entry, struct vy_mem_tree_key *key,
 #define BPS_TREE_EXTENT_SIZE VY_MEM_TREE_EXTENT_SIZE
 #define BPS_TREE_COMPARE(a, b, cmp_def) vy_mem_tree_cmp(a, b, cmp_def)
 #define BPS_TREE_COMPARE_KEY(a, b, cmp_def) vy_mem_tree_cmp_key(a, b, cmp_def)
+#define BPS_TREE_IDENTICAL(a, b) vy_entry_is_equal(a, b)
 #define bps_tree_elem_t struct vy_entry
 #define bps_tree_key_t struct vy_mem_tree_key *
 #define bps_tree_arg_t struct key_def *
-#define BPS_TREE_NO_DEBUG
 
 #include <salad/bps_tree.h>
 
@@ -136,7 +136,7 @@ vy_mem_tree_cmp_key(struct vy_entry entry, struct vy_mem_tree_key *key,
 #undef bps_tree_elem_t
 #undef bps_tree_key_t
 #undef bps_tree_arg_t
-#undef BPS_TREE_NO_DEBUG
+#undef BPS_TREE_IDENTICAL
 
 /** @endcond false */
 
diff --git a/src/lib/salad/bps_tree.h b/src/lib/salad/bps_tree.h
index 29eb8b81aa..58d69bf66a 100644
--- a/src/lib/salad/bps_tree.h
+++ b/src/lib/salad/bps_tree.h
@@ -279,7 +279,7 @@
  * same data.
  */
 #ifndef BPS_TREE_IDENTICAL
-#define BPS_TREE_IDENTICAL(a, b) (a == b)
+#error "BPS_TREE_IDENTICAL must be defined"
 #endif
 
 /**
@@ -360,6 +360,7 @@ typedef uint32_t bps_tree_block_id_t;
 #define bps_tree_insert _api_name(insert)
 #define bps_tree_insert_get_iterator _api_name(insert_get_iterator)
 #define bps_tree_delete _api_name(delete)
+#define bps_tree_delete_identical _api_name(delete_identical)
 #define bps_tree_size _api_name(size)
 #define bps_tree_mem_used _api_name(mem_used)
 #define bps_tree_random _api_name(random)
@@ -4516,6 +4517,36 @@ bps_tree_delete(struct bps_tree *tree, bps_tree_elem_t elem)
 	return 0;
 }
 
+/**
+ * @brief Delete an identical element from a tree (unlike
+ * bps_tree_delete this routine relies on BPS_TREE_IDENTICAL
+ * instead of BPS_TREE_COMPARE).
+ * @param tree - pointer to a tree
+ * @param elem - the element tot delete
+ * @return - true on success or false if the element was not
+ *           found in tree or is not identical.
+ */
+static inline int
+bps_tree_delete_identical(struct bps_tree *tree, bps_tree_elem_t elem)
+{
+	if (tree->root_id == (bps_tree_block_id_t)(-1))
+		return -1;
+	struct bps_inner_path_elem path[BPS_TREE_MAX_DEPTH];
+	struct bps_leaf_path_elem leaf_path_elem;
+	bool exact;
+	bps_tree_collect_path(tree, elem, path, &leaf_path_elem, &exact);
+
+	if (!exact)
+		return -1;
+
+	struct bps_leaf *leaf = leaf_path_elem.block;
+	if (!BPS_TREE_IDENTICAL(elem,
+				leaf->elems[leaf_path_elem.insertion_point]))
+		return -1;
+	bps_tree_process_delete_leaf(tree, &leaf_path_elem);
+	return 0;
+}
+
 /**
  * @brief Recursively find a maximum element in subtree.
  * Used only for debug purposes
@@ -6054,6 +6085,7 @@ bps_tree_debug_check_internal_functions(bool assertme)
 #undef bps_tree_find
 #undef bps_tree_insert
 #undef bps_tree_delete
+#undef bps_tree_delete_identical
 #undef bps_tree_size
 #undef bps_tree_mem_used
 #undef bps_tree_random
@@ -6155,4 +6187,5 @@ bps_tree_debug_check_internal_functions(bool assertme)
 #undef bps_tree_debug_check_move_to_left_inner
 #undef bps_tree_debug_check_insert_and_move_to_right_inner
 #undef bps_tree_debug_check_insert_and_move_to_left_inner
+
 /* }}} */
diff --git a/test/unit/bps_tree.cc b/test/unit/bps_tree.cc
index 191dce95a3..a622ecc876 100644
--- a/test/unit/bps_tree.cc
+++ b/test/unit/bps_tree.cc
@@ -25,6 +25,7 @@ compare(type_t a, type_t b);
 #define BPS_TREE_NAME testtest
 #define BPS_TREE_BLOCK_SIZE 512 /* value is to low specially for tests */
 #define BPS_TREE_EXTENT_SIZE 16*1024 /* value is to low specially for tests */
+#define BPS_TREE_IDENTICAL(a, b) (a == b)
 #define BPS_TREE_COMPARE(a, b, arg) compare(a, b)
 #define BPS_TREE_COMPARE_KEY(a, b, arg) compare(a, b)
 #define bps_tree_elem_t char
@@ -34,6 +35,7 @@ compare(type_t a, type_t b);
 #undef BPS_TREE_NAME
 #undef BPS_TREE_BLOCK_SIZE
 #undef BPS_TREE_EXTENT_SIZE
+#undef BPS_TREE_IDENTICAL
 #undef BPS_TREE_COMPARE
 #undef BPS_TREE_COMPARE_KEY
 #undef bps_tree_elem_t
@@ -44,6 +46,7 @@ compare(type_t a, type_t b);
 #define BPS_TREE_NAME test
 #define BPS_TREE_BLOCK_SIZE 128 /* value is to low specially for tests */
 #define BPS_TREE_EXTENT_SIZE 2048 /* value is to low specially for tests */
+#define BPS_TREE_IDENTICAL(a, b) (a == b)
 #define BPS_TREE_COMPARE(a, b, arg) compare(a, b)
 #define BPS_TREE_COMPARE_KEY(a, b, arg) compare(a, b)
 #define bps_tree_elem_t type_t
@@ -54,16 +57,59 @@ compare(type_t a, type_t b);
 #undef BPS_TREE_NAME
 #undef BPS_TREE_BLOCK_SIZE
 #undef BPS_TREE_EXTENT_SIZE
+#undef BPS_TREE_IDENTICAL
 #undef BPS_TREE_COMPARE
 #undef BPS_TREE_COMPARE_KEY
 #undef bps_tree_elem_t
 #undef bps_tree_key_t
 #undef bps_tree_arg_t
 
+struct elem_t {
+	long info;
+	long marker;
+};
+
+static bool
+equal(const elem_t &a, const elem_t &b)
+{
+	return a.info == b.info && a.marker == b.marker;
+}
+
+static int compare(const elem_t &a, const elem_t &b)
+{
+	return a.info < b.info ? -1 : a.info > b.info ? 1 : 0;
+}
+
+static int compare_key(const elem_t &a, long b)
+{
+	return a.info < b ? -1 : a.info > b ? 1 : 0;
+}
+
+#define BPS_TREE_NAME struct_tree
+#define BPS_TREE_BLOCK_SIZE 128 /* value is to low specially for tests */
+#define BPS_TREE_EXTENT_SIZE 2048 /* value is to low specially for tests */
+#define BPS_TREE_IDENTICAL(a, b) equal(a, b)
+#define BPS_TREE_COMPARE(a, b, arg) compare(a, b)
+#define BPS_TREE_COMPARE_KEY(a, b, arg) compare_key(a, b)
+#define bps_tree_elem_t struct elem_t
+#define bps_tree_key_t long
+#define bps_tree_arg_t int
+#include "salad/bps_tree.h"
+#undef BPS_TREE_NAME
+#undef BPS_TREE_BLOCK_SIZE
+#undef BPS_TREE_EXTENT_SIZE
+#undef BPS_TREE_COMPARE
+#undef BPS_TREE_COMPARE_KEY
+#undef BPS_TREE_IDENTICAL
+#undef bps_tree_elem_t
+#undef bps_tree_key_t
+#undef bps_tree_arg_t
+
 /* tree for approximate_count test */
 #define BPS_TREE_NAME approx
 #define BPS_TREE_BLOCK_SIZE 128 /* value is to low specially for tests */
 #define BPS_TREE_EXTENT_SIZE 2048 /* value is to low specially for tests */
+#define BPS_TREE_IDENTICAL(a, b) (a == b)
 #define BPS_TREE_COMPARE(a, b, arg) ((a) < (b) ? -1 : (a) > (b) ? 1 : 0)
 #define BPS_TREE_COMPARE_KEY(a, b, arg) (((a) >> 32) < (b) ? -1 : ((a) >> 32) > (b) ? 1 : 0)
 #define bps_tree_elem_t uint64_t
@@ -792,6 +838,27 @@ insert_get_iterator()
 	footer();
 }
 
+static void
+delete_identical_check()
+{
+	header();
+	struct_tree tree;
+	struct_tree_create(&tree, 0, extent_alloc, extent_free, &extents_count);
+	struct elem_t e1 = {1, 1};
+	struct_tree_insert(&tree, e1, NULL);
+	struct elem_t e2 = {1, 2};
+	if (struct_tree_delete_identical(&tree, e2) == 0)
+		fail("deletion of the non-identical element must fail", "false");
+	if (struct_tree_find(&tree, 1) == NULL)
+		fail("test non-identical element deletion failure", "false");
+	if (struct_tree_delete_identical(&tree, e1) != 0)
+		fail("deletion of the identical element must not fail", "false");
+	if (struct_tree_find(&tree, 1) != NULL)
+		fail("test identical element deletion completion", "false");
+	struct_tree_destroy(&tree);
+	footer();
+}
+
 int
 main(void)
 {
@@ -806,4 +873,5 @@ main(void)
 	if (extents_count != 0)
 		fail("memory leak!", "true");
 	insert_get_iterator();
+	delete_identical_check();
 }
diff --git a/test/unit/bps_tree.result b/test/unit/bps_tree.result
index 8e1c3d4b60..dc21bf3403 100644
--- a/test/unit/bps_tree.result
+++ b/test/unit/bps_tree.result
@@ -280,3 +280,5 @@ Count: 10575
 	*** approximate_count: done ***
 	*** insert_get_iterator ***
 	*** insert_get_iterator: done ***
+	*** delete_identical_check ***
+	*** delete_identical_check: done ***
-- 
GitLab