2014-04-13 22:26:23 +00:00
|
|
|
// Copyright 2014 Dolphin Emulator Project
|
2015-05-17 23:08:10 +00:00
|
|
|
// Licensed under GPLv2+
|
2014-04-13 22:26:23 +00:00
|
|
|
// Refer to the license.txt file included.
|
|
|
|
|
|
|
|
// Abstraction for a simple flag that can be toggled in a multithreaded way.
|
2014-04-13 23:40:20 +00:00
|
|
|
//
|
|
|
|
// Simple API:
|
2014-04-13 22:26:23 +00:00
|
|
|
// * Set(bool = true): sets the Flag
|
|
|
|
// * IsSet(): tests if the flag is set
|
|
|
|
// * Clear(): clears the flag (equivalent to Set(false)).
|
2014-04-13 23:40:20 +00:00
|
|
|
//
|
|
|
|
// 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).
|
2014-04-13 22:26:23 +00:00
|
|
|
|
|
|
|
#pragma once
|
|
|
|
|
|
|
|
#include <atomic>
|
|
|
|
|
|
|
|
namespace Common {
|
|
|
|
|
|
|
|
class Flag final
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
// Declared as explicit since we do not want "= true" to work on a flag
|
|
|
|
// object - it should be made explicit that a flag is *not* a normal
|
|
|
|
// variable.
|
|
|
|
explicit Flag(bool initial_value = false) : m_val(initial_value) {}
|
|
|
|
|
|
|
|
void Set(bool val = true)
|
|
|
|
{
|
|
|
|
m_val.store(val);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Clear()
|
|
|
|
{
|
|
|
|
Set(false);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool IsSet() const
|
|
|
|
{
|
|
|
|
return m_val.load();
|
|
|
|
}
|
|
|
|
|
2014-04-13 23:40:20 +00:00
|
|
|
bool TestAndSet(bool val = true)
|
|
|
|
{
|
|
|
|
bool expected = !val;
|
|
|
|
return m_val.compare_exchange_strong(expected, val);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool TestAndClear()
|
|
|
|
{
|
|
|
|
return TestAndSet(false);
|
|
|
|
}
|
|
|
|
|
2014-04-13 22:26:23 +00:00
|
|
|
private:
|
|
|
|
// 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.
|
|
|
|
//
|
|
|
|
// Re-evaluate next time we upgrade that piece of shit.
|
|
|
|
std::atomic<bool> m_val;
|
|
|
|
};
|
|
|
|
|
|
|
|
} // namespace Common
|