2018-03-15 15:21:41 +11:00
|
|
|
/*
|
2018-08-04 15:14:06 +10:00
|
|
|
* This Source Code Form is subject to the terms of the Mozilla Public
|
|
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
|
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/.
|
2018-03-15 15:21:41 +11:00
|
|
|
*
|
|
|
|
* Copyright 2018 Danny Robson <danny@nerdcruft.net>
|
|
|
|
*/
|
|
|
|
|
2024-05-29 16:29:08 +10:00
|
|
|
#include <cruft/util/thread/event.hpp>
|
|
|
|
#include <cruft/util/thread/monitor.hpp>
|
|
|
|
#include <cruft/util/thread/semaphore.hpp>
|
2019-06-22 15:46:34 +10:00
|
|
|
|
2024-05-29 16:29:08 +10:00
|
|
|
#include <cruft/util/tap.hpp>
|
2018-03-15 15:21:41 +11:00
|
|
|
|
|
|
|
#include <atomic>
|
2020-08-03 11:20:06 +10:00
|
|
|
#include <thread>
|
2018-03-15 15:21:41 +11:00
|
|
|
|
|
|
|
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
|
|
// a single atomic integer wrapper that allows precise ordering of thread
|
|
|
|
// entrance and exit for exact testing of thread interleavings.
|
|
|
|
struct foo {
|
|
|
|
foo ():
|
|
|
|
enter (0),
|
|
|
|
leave (0)
|
|
|
|
{ ; }
|
|
|
|
|
|
|
|
void
|
|
|
|
inc (void) {
|
|
|
|
enter.acquire ();
|
|
|
|
++value;
|
|
|
|
leave.release ();
|
|
|
|
}
|
|
|
|
|
|
|
|
std::atomic<int> value = 0;
|
2018-08-05 14:42:02 +10:00
|
|
|
cruft::thread::semaphore enter;
|
|
|
|
cruft::thread::semaphore leave;
|
2018-03-15 15:21:41 +11:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
|
|
int
|
|
|
|
main ()
|
|
|
|
{
|
2018-08-05 14:42:02 +10:00
|
|
|
cruft::TAP::logger tap;
|
2018-03-15 15:21:41 +11:00
|
|
|
|
2018-08-05 14:42:02 +10:00
|
|
|
cruft::thread::monitor<foo> obj;
|
2018-03-15 15:21:41 +11:00
|
|
|
const auto &value = obj->value;
|
|
|
|
auto &enter = obj->enter;
|
|
|
|
auto &leave = obj->leave;
|
|
|
|
|
2020-08-03 11:20:06 +10:00
|
|
|
std::thread t1 ([&] () { obj->inc (); });
|
|
|
|
std::thread t2 ([&] () { obj->inc (); });
|
2018-03-15 15:21:41 +11:00
|
|
|
|
|
|
|
tap.expect_eq (value, 0, "wrapped class was left initialised");
|
|
|
|
|
|
|
|
enter.release ();
|
|
|
|
leave.acquire ();
|
|
|
|
tap.expect_eq (value, 1, "only one thread had access");
|
|
|
|
|
|
|
|
enter.release ();
|
|
|
|
leave.acquire ();
|
|
|
|
tap.expect_eq (value, 2, "second thread got access");
|
|
|
|
|
|
|
|
t1.join ();
|
|
|
|
t2.join ();
|
|
|
|
|
|
|
|
return tap.status ();
|
|
|
|
}
|