jdk/test/hotspot/gtest/utilities/test_waitBarrier.cpp
Kim Barrett e27309f10d 8374350: Convert hotspot gtests to use Atomic<T>
Reviewed-by: aboldtch, iwalulya
2026-01-06 15:00:45 +00:00

152 lines
5.2 KiB
C++

/*
* Copyright (c) 2019, 2026, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
*
* This code is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License version 2 only, as
* published by the Free Software Foundation.
*
* This code is distributed in the hope that it will be useful, but WITHOUT
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
* version 2 for more details (a copy is included in the LICENSE file that
* accompanied this code).
*
* You should have received a copy of the GNU General Public License version
* 2 along with this work; if not, write to the Free Software Foundation,
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
*
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
* or visit www.oracle.com if you need additional information or have any
* questions.
*/
#include "runtime/atomic.hpp"
#include "runtime/orderAccess.hpp"
#include "runtime/os.hpp"
#include "utilities/spinYield.hpp"
#include "utilities/waitBarrier.hpp"
#include "threadHelper.inline.hpp"
static Atomic<int> wait_tag{0};
static Atomic<int> valid_value{0};
template <typename WaitBarrierImpl>
class WBThread : public JavaTestThread {
public:
static Atomic<bool> _exit;
WaitBarrierType<WaitBarrierImpl>* _wait_barrier;
Semaphore* _wrt_start;
Atomic<int> _on_barrier;
WBThread(Semaphore* post, WaitBarrierType<WaitBarrierImpl>* wb, Semaphore* wrt_start)
: JavaTestThread(post), _wait_barrier(wb), _wrt_start(wrt_start) {};
virtual ~WBThread(){}
void main_run() {
_wrt_start->signal();
int vv, tag;
// Similar to how a JavaThread would stop in a safepoint.
while (!_exit.load_relaxed()) {
// Load the published tag.
tag = wait_tag.load_acquire();
// Publish the tag this thread is going to wait for.
_on_barrier.release_store(tag);
if (_on_barrier.load_relaxed() == 0) {
SpinPause();
continue;
}
OrderAccess::storeload(); // Loads in WB must not float up.
// Wait until we are woken.
_wait_barrier->wait(tag);
// Verify that we do not see an invalid value.
vv = valid_value.load_acquire();
ASSERT_EQ((vv & 0x1), 0);
_on_barrier.release_store(0);
}
}
};
template <typename WaitBarrierImpl>
Atomic<bool> WBThread<WaitBarrierImpl>::_exit{false};
template <typename WaitBarrierImpl>
class WBArmerThread : public JavaTestThread {
public:
WBArmerThread(Semaphore* post) : JavaTestThread(post) {
};
virtual ~WBArmerThread(){}
void main_run() {
static const int NUMBER_OF_READERS = 4;
Semaphore post;
Semaphore wrt_start;
WaitBarrierType<WaitBarrierImpl> wb(this);
WBThread<WaitBarrierImpl>* reader1 = new WBThread<WaitBarrierImpl>(&post, &wb, &wrt_start);
WBThread<WaitBarrierImpl>* reader2 = new WBThread<WaitBarrierImpl>(&post, &wb, &wrt_start);
WBThread<WaitBarrierImpl>* reader3 = new WBThread<WaitBarrierImpl>(&post, &wb, &wrt_start);
WBThread<WaitBarrierImpl>* reader4 = new WBThread<WaitBarrierImpl>(&post, &wb, &wrt_start);
reader1->doit();
reader2->doit();
reader3->doit();
reader4->doit();
int nw = NUMBER_OF_READERS;
while (nw > 0) {
wrt_start.wait();
--nw;
}
jlong stop_ms = os::javaTimeMillis() + 1000; // 1 seconds max test time
int next_tag = 1;
// Similar to how the VM thread would use a WaitBarrier in a safepoint.
while (stop_ms > os::javaTimeMillis()) {
// Arm next tag.
wb.arm(next_tag);
// Publish tag.
wait_tag.release_store_fence(next_tag);
// Wait until threads picked up new tag.
while (reader1->_on_barrier.load_relaxed() != wait_tag.load_relaxed() ||
reader2->_on_barrier.load_relaxed() != wait_tag.load_relaxed() ||
reader3->_on_barrier.load_relaxed() != wait_tag.load_relaxed() ||
reader4->_on_barrier.load_relaxed() != wait_tag.load_relaxed()) {
SpinPause();
}
// Set an invalid value.
valid_value.release_store(valid_value.load_relaxed() + 1); // odd
os::naked_yield();
// Set a valid value.
valid_value.release_store(valid_value.load_relaxed() + 1); // even
// Publish inactive tag.
wait_tag.release_store_fence(0); // Stores in WB must not float up.
wb.disarm();
// Wait until threads done valid_value verification.
while (reader1->_on_barrier.load_relaxed() != 0 ||
reader2->_on_barrier.load_relaxed() != 0 ||
reader3->_on_barrier.load_relaxed() != 0 ||
reader4->_on_barrier.load_relaxed() != 0) {
SpinPause();
}
++next_tag;
}
WBThread<WaitBarrierImpl>::_exit.store_relaxed(true);
for (int i = 0; i < NUMBER_OF_READERS; i++) {
post.wait();
}
}
};
TEST_VM(WaitBarrier, default_wb) {
WBThread<WaitBarrierDefault>::_exit.store_relaxed(false);
mt_test_doer<WBArmerThread<WaitBarrierDefault> >();
}
#if defined(LINUX)
TEST_VM(WaitBarrier, generic_wb) {
WBThread<GenericWaitBarrier>::_exit.store_relaxed(false);
mt_test_doer<WBArmerThread<GenericWaitBarrier> >();
}
#endif