Common::Flag: Add support for TestAndSet + test by implementing basic spinlocks.

This commit is contained in:
Pierre Bourdon 2014-04-14 01:40:20 +02:00
parent 6bdcbad3e4
commit e24cad0780
2 changed files with 50 additions and 1 deletions

View File

@ -3,10 +3,17 @@
// Refer to the license.txt file included. // Refer to the license.txt file included.
// Abstraction for a simple flag that can be toggled in a multithreaded way. // Abstraction for a simple flag that can be toggled in a multithreaded way.
// It exposes a very simple API: //
// Simple API:
// * Set(bool = true): sets the Flag // * Set(bool = true): sets the Flag
// * IsSet(): tests if the flag is set // * IsSet(): tests if the flag is set
// * Clear(): clears the flag (equivalent to Set(false)). // * Clear(): clears the flag (equivalent to Set(false)).
//
// More advanced features:
// * TestAndSet(bool = true): sets the flag to the given value. If a change was
// needed (the flag did not already have this value)
// the function returns true. Else, false.
// * TestAndClear(): alias for TestAndSet(false).
#pragma once #pragma once
@ -37,6 +44,17 @@ public:
return m_val.load(); return m_val.load();
} }
bool TestAndSet(bool val = true)
{
bool expected = !val;
return m_val.compare_exchange_strong(expected, val);
}
bool TestAndClear()
{
return TestAndSet(false);
}
private: private:
// We are not using std::atomic_bool here because MSVC sucks as of VC++ // We are not using std::atomic_bool here because MSVC sucks as of VC++
// 2013 and does not implement the std::atomic_bool(bool) constructor. // 2013 and does not implement the std::atomic_bool(bool) constructor.

View File

@ -2,6 +2,7 @@
// Licensed under GPLv2 // Licensed under GPLv2
// Refer to the license.txt file included. // Refer to the license.txt file included.
#include <array>
#include <gtest/gtest.h> #include <gtest/gtest.h>
#include <thread> #include <thread>
@ -23,6 +24,9 @@ TEST(Flag, Simple)
f.Set(false); f.Set(false);
EXPECT_FALSE(f.IsSet()); EXPECT_FALSE(f.IsSet());
EXPECT_TRUE(f.TestAndSet());
EXPECT_TRUE(f.TestAndClear());
Flag f2(true); Flag f2(true);
EXPECT_TRUE(f2.IsSet()); EXPECT_TRUE(f2.IsSet());
} }
@ -58,3 +62,30 @@ TEST(Flag, MultiThreaded)
EXPECT_EQ(ITERATIONS_COUNT, count); EXPECT_EQ(ITERATIONS_COUNT, count);
} }
TEST(Flag, SpinLock)
{
// Uses a flag to implement basic spinlocking using TestAndSet.
Flag f;
int count = 0;
const int ITERATIONS_COUNT = 5000;
const int THREADS_COUNT = 50;
auto adder_func = [&]() {
for (int i = 0; i < ITERATIONS_COUNT; ++i)
{
// Acquire the spinlock.
while (!f.TestAndSet());
count++;
f.Clear();
}
};
std::array<std::thread, THREADS_COUNT> threads;
for (auto& th : threads)
th = std::thread(adder_func);
for (auto& th : threads)
th.join();
EXPECT_EQ(ITERATIONS_COUNT * THREADS_COUNT, count);
}