Skip to content

Commit 2313a21

Browse files
author
Ivan Walulya
committedMay 17, 2021
8266637: CHT: Add insert_and_get method
Reviewed-by: tschatzl, rehn
1 parent 7b736ec commit 2313a21

File tree

3 files changed

+38
-8
lines changed

3 files changed

+38
-8
lines changed
 

‎src/hotspot/share/utilities/concurrentHashTable.hpp

+16-5
Original file line numberDiff line numberDiff line change
@@ -308,10 +308,10 @@ class ConcurrentHashTable : public CHeapObj<F> {
308308
VALUE* internal_get(Thread* thread, LOOKUP_FUNC& lookup_f,
309309
bool* grow_hint = NULL);
310310

311-
// Plain insert.
312-
template <typename LOOKUP_FUNC>
313-
bool internal_insert(Thread* thread, LOOKUP_FUNC& lookup_f, const VALUE& value,
314-
bool* grow_hint, bool* clean_hint);
311+
// Insert and get current value.
312+
template <typename LOOKUP_FUNC, typename FOUND_FUNC>
313+
bool internal_insert_get(Thread* thread, LOOKUP_FUNC& lookup_f, const VALUE& value,
314+
FOUND_FUNC& foundf, bool* grow_hint, bool* clean_hint);
315315

316316
// Returns true if an item matching LOOKUP_FUNC is removed.
317317
// Calls DELETE_FUNC before destroying the node.
@@ -410,7 +410,18 @@ class ConcurrentHashTable : public CHeapObj<F> {
410410
template <typename LOOKUP_FUNC>
411411
bool insert(Thread* thread, LOOKUP_FUNC& lookup_f, const VALUE& value,
412412
bool* grow_hint = NULL, bool* clean_hint = NULL) {
413-
return internal_insert(thread, lookup_f, value, grow_hint, clean_hint);
413+
struct NOP {
414+
void operator()(...) const {}
415+
} nop;
416+
return internal_insert_get(thread, lookup_f, value, nop, grow_hint, clean_hint);
417+
}
418+
419+
// Returns true if the item was inserted, duplicates are found with
420+
// LOOKUP_FUNC then FOUND_FUNC is called.
421+
template <typename LOOKUP_FUNC, typename FOUND_FUNC>
422+
bool insert_get(Thread* thread, LOOKUP_FUNC& lookup_f, VALUE& value, FOUND_FUNC& foundf,
423+
bool* grow_hint = NULL, bool* clean_hint = NULL) {
424+
return internal_insert_get(thread, lookup_f, value, foundf, grow_hint, clean_hint);
414425
}
415426

416427
// This does a fast unsafe insert and can thus only be used when there is no

‎src/hotspot/share/utilities/concurrentHashTable.inline.hpp

+5-3
Original file line numberDiff line numberDiff line change
@@ -880,10 +880,10 @@ inline typename CONFIG::Value* ConcurrentHashTable<CONFIG, F>::
880880
}
881881

882882
template <typename CONFIG, MEMFLAGS F>
883-
template <typename LOOKUP_FUNC>
883+
template <typename LOOKUP_FUNC, typename FOUND_FUNC>
884884
inline bool ConcurrentHashTable<CONFIG, F>::
885-
internal_insert(Thread* thread, LOOKUP_FUNC& lookup_f, const VALUE& value,
886-
bool* grow_hint, bool* clean_hint)
885+
internal_insert_get(Thread* thread, LOOKUP_FUNC& lookup_f, const VALUE& value,
886+
FOUND_FUNC& foundf, bool* grow_hint, bool* clean_hint)
887887
{
888888
bool ret = false;
889889
bool clean = false;
@@ -902,6 +902,7 @@ inline bool ConcurrentHashTable<CONFIG, F>::
902902
if (old == NULL) {
903903
new_node->set_next(first_at_start);
904904
if (bucket->cas_first(new_node, first_at_start)) {
905+
foundf(new_node->value());
905906
JFR_ONLY(_stats_rate.add();)
906907
new_node = NULL;
907908
ret = true;
@@ -911,6 +912,7 @@ inline bool ConcurrentHashTable<CONFIG, F>::
911912
locked = bucket->is_locked();
912913
} else {
913914
// There is a duplicate.
915+
foundf(old->value());
914916
break; /* leave critical section */
915917
}
916918
} /* leave critical section */

‎test/hotspot/gtest/utilities/test_concurrentHashtable.cpp

+17
Original file line numberDiff line numberDiff line change
@@ -147,6 +147,19 @@ static void cht_insert(Thread* thr) {
147147
delete cht;
148148
}
149149

150+
static void cht_insert_get(Thread* thr) {
151+
uintptr_t val = 0x2;
152+
SimpleTestLookup stl(val);
153+
SimpleTestTable* cht = new SimpleTestTable();
154+
ValueGet vg;
155+
EXPECT_TRUE(cht->insert_get(thr, stl, val, vg)) << "Insert unique value failed.";
156+
EXPECT_EQ(val, vg.get_value()) << "Getting an inserted value failed.";
157+
ValueGet vg_dup;
158+
EXPECT_FALSE(cht->insert_get(thr, stl, val, vg_dup)) << "Insert duplicate value succeeded.";
159+
EXPECT_EQ(val, vg_dup.get_value()) << "Getting an existing value failed.";
160+
delete cht;
161+
}
162+
150163
static void cht_get_insert(Thread* thr) {
151164
uintptr_t val = 0x2;
152165
SimpleTestLookup stl(val);
@@ -449,6 +462,10 @@ TEST_VM(ConcurrentHashTable, basic_get_insert) {
449462
nomt_test_doer(cht_get_insert);
450463
}
451464

465+
TEST_VM(ConcurrentHashTable, basic_insert_get) {
466+
nomt_test_doer(cht_insert_get);
467+
}
468+
452469
TEST_VM(ConcurrentHashTable, basic_scope) {
453470
nomt_test_doer(cht_scope);
454471
}

0 commit comments

Comments
 (0)
Please sign in to comment.