From 2f4a77ce63ca6221a776c13b567cd5d502d7567c Mon Sep 17 00:00:00 2001
From: Florian Fischer <florian.fl.fischer@fau.de>
Date: Tue, 17 Nov 2020 11:45:24 +0100
Subject: [PATCH] [test] add test using liburcu

The SimpleURCUTest creates a lock-less hash table and inserts values from
separate fibers and verifies the correct insertion.
---
 tests/SimpleURCUTest.cpp | 93 ++++++++++++++++++++++++++++++++++++++++
 tests/meson.build        | 17 +++++++-
 2 files changed, 108 insertions(+), 2 deletions(-)
 create mode 100644 tests/SimpleURCUTest.cpp

diff --git a/tests/SimpleURCUTest.cpp b/tests/SimpleURCUTest.cpp
new file mode 100644
index 00000000..dfd9f813
--- /dev/null
+++ b/tests/SimpleURCUTest.cpp
@@ -0,0 +1,93 @@
+// SPDX-License-Identifier: LGPL-3.0-or-later
+// Copyright © 2020 Florian Fischer
+#include <urcu.h>						 // for rcu_read_lock, rcu_read_unlock
+#include <urcu/rculfhash.h>	 // for RCU lock-free hash table
+
+#include <algorithm>	 // for find
+#include <cstdlib>		 // for exit, EXIT_FAILURE, EXIT_SUC...
+#include <functional>	 // for hash
+#include <iostream>		 // for hash
+#include <vector>			 // for vector
+
+#include "Common.hpp"										 // for die
+#include "CountingPrivateSemaphore.hpp"	 // for CPS
+#include "Fiber.hpp"										 // for Fiber
+#include "Runtime.hpp"									 // for Runtime
+#include "emper.hpp"										 // for spawn
+
+struct node {
+	int value;
+	struct cds_lfht_node node; /* Chaining in hash table */
+};
+
+using node_t = struct node;
+
+auto main() -> int {
+	Runtime runtime;
+	struct cds_lfht* ht;
+	std::vector<int> values = {
+			-5, 42, 42, 36, 24,
+	}; /* 42 is duplicated */
+
+	// Allocate new hash table.
+	ht = cds_lfht_new(1, 1, 0, CDS_LFHT_AUTO_RESIZE | CDS_LFHT_ACCOUNTING, nullptr);
+	if (!ht) {
+		die("Error allocating hash table", false);
+	}
+
+	Fiber* verifier = Fiber::from([&]() {
+		CPS cps;
+		for (auto& value : values) {
+			// add each value to the hash table
+			spawn(
+					[&ht, value] {
+						auto* node = reinterpret_cast<node_t*>(malloc(sizeof(node_t)));
+						if (!node) {
+							die("allocating node failed", true);
+						}
+
+						cds_lfht_node_init(&node->node);
+						node->value = value;
+						size_t hash = std::hash<int>{}(value);
+
+						rcu_read_lock();
+						cds_lfht_add(ht, hash, &node->node);
+						rcu_read_unlock();
+					},
+					cps);
+		}
+
+		// Wait for the adders to finish
+		cps.wait();
+
+		// Verify the content of the hash table.
+		// Iterate over each hash table node.
+		// Iteration needs to be performed within RCU read-side critical section.
+		struct cds_lfht_iter iter;
+		node_t* node;
+		rcu_read_lock();
+		size_t i = 0;
+		cds_lfht_for_each_entry(ht, &iter, node, node) {
+			++i;
+			auto it = std::find(values.begin(), values.end(), node->value);
+			if (it == values.end()) {
+				std::cerr << "value: " << node->value << " not found in cds_lfht" << std::endl;
+				exit(EXIT_FAILURE);
+			}
+		}
+		rcu_read_unlock();
+
+		if (i != values.size()) {
+			std::cerr << "number of values in cds_lfht: " << i
+								<< " differ number of added ones: " << values.size() << std::endl;
+			exit(EXIT_FAILURE);
+		}
+
+		exit(EXIT_SUCCESS);
+	});
+
+	runtime.schedule(*verifier);
+	runtime.waitUntilFinished();
+
+	return EXIT_FAILURE;
+}
diff --git a/tests/meson.build b/tests/meson.build
index 2ac875c0..e1c2f709 100644
--- a/tests/meson.build
+++ b/tests/meson.build
@@ -1,3 +1,7 @@
+cc = meson.get_compiler('c')
+liburcu_memb = cc.find_library('urcu-memb')
+liburcu_cds = cc.find_library('urcu-cds')
+
 tests = {
 		  'SimpleFibTest.cpp':
 		  {
@@ -29,10 +33,14 @@ tests = {
 		  {
 			'description': 'Simple LAWS scheduling strategy test',
 		  },
+		  'SimpleURCUTest.cpp':
+		  {
+			'description': 'Simple userspace-rcu hash table test',
+			'dependencies': [liburcu_memb, liburcu_cds]
+		  },
 		}
 
 undef_ndebug = '-UNDEBUG'
-test_dep = [thread_dep]
 test_env = environment(
   {
 	# Set glibc's MALLOC_PERTURB to 1. This means that newly allocated
@@ -50,12 +58,17 @@ foreach source, test_dict : tests
   # The test_name is the name of the source file without the file suffix.
   test_name = source.split('.')[0]
 
+  test_dep = [thread_dep]
+  if test_dict.has_key('dependencies')
+  	test_dep += test_dict['dependencies']
+  endif
+
   test_exe = executable(test_name,
 						source,
 						include_directories: emper_all_include,
 						c_args: undef_ndebug,
 						cpp_args: undef_ndebug,
-						dependencies: test_dep,
+						dependencies: emper_dependencies + test_dep,
 						link_with: [emper, emper_c],
 					   )
 
-- 
GitLab