diff --git a/lib/base/atomic.hpp b/lib/base/atomic.hpp index 62fa37638..c8f169c0b 100644 --- a/lib/base/atomic.hpp +++ b/lib/base/atomic.hpp @@ -4,6 +4,7 @@ #define ATOMIC_H #include +#include #include #include @@ -41,78 +42,49 @@ public: }; /** - * Wraps T into a std::atomic-like interface. + * Wraps any T into a std::atomic-like interface that locks using a mutex. + * + * In contrast to std::atomic, Locked is also valid for types that are not trivially copyable. + * In case T is trivially copyable, std::atomic is almost certainly the better choice. * * @ingroup base */ -template -class NotAtomic +template +class Locked { public: inline T load() const { + std::unique_lock lock(m_Mutex); + return m_Value; } inline void store(T desired) { + std::unique_lock lock(m_Mutex); + m_Value = std::move(desired); } +private: + mutable std::mutex m_Mutex; T m_Value; }; /** - * Tells whether to use std::atomic or NotAtomic. + * Type alias for std::atomic if possible, otherwise Locked is used as a fallback. * * @ingroup base */ -template -struct Atomicable -{ - // Doesn't work with too old compilers. - //static constexpr bool value = std::is_trivially_copyable::value && sizeof(T) <= sizeof(void*); - static constexpr bool value = (std::is_fundamental::value || std::is_pointer::value) && sizeof(T) <= sizeof(void*); -}; - -/** - * Uses either std::atomic or NotAtomic depending on atomicable. - * - * @ingroup base - */ -template -struct AtomicTemplate; - -template<> -struct AtomicTemplate -{ - template - struct tmplt - { - typedef NotAtomic type; - }; -}; - -template<> -struct AtomicTemplate -{ - template - struct tmplt - { - typedef std::atomic type; - }; -}; - -/** - * Uses either std::atomic or NotAtomic depending on T. - * - * @ingroup base - */ -template -struct EventuallyAtomic -{ - typedef typename AtomicTemplate::value>::template tmplt::type type; -}; +template +using AtomicOrLocked = +#if defined(__GNUC__) && __GNUC__ < 5 + // GCC does not implement std::is_trivially_copyable until version 5. + typename std::conditional::value || std::is_pointer::value, std::atomic, Locked>::type; +#else /* defined(__GNUC__) && __GNUC__ < 5 */ + typename std::conditional::value, std::atomic, Locked>::type; +#endif /* defined(__GNUC__) && __GNUC__ < 5 */ } diff --git a/lib/base/configobject.ti b/lib/base/configobject.ti index 3e5859c81..19b7cde83 100644 --- a/lib/base/configobject.ti +++ b/lib/base/configobject.ti @@ -59,10 +59,11 @@ abstract class ConfigObject : ConfigObjectBase < ConfigType [config, no_user_modify] String __name (Name); [config, no_user_modify] String "name" (ShortName) { get {{{ - if (m_ShortName.m_Value.IsEmpty()) + String shortName = m_ShortName.load(); + if (shortName.IsEmpty()) return GetName(); else - return m_ShortName.m_Value; + return shortName; }}} }; [config, no_user_modify] name(Zone) zone (ZoneName); diff --git a/lib/icinga/host.ti b/lib/icinga/host.ti index d69d92102..f6624e307 100644 --- a/lib/icinga/host.ti +++ b/lib/icinga/host.ti @@ -21,10 +21,11 @@ class Host : Checkable [config] String display_name { get {{{ - if (m_DisplayName.m_Value.IsEmpty()) + String displayName = m_DisplayName.load(); + if (displayName.IsEmpty()) return GetName(); else - return m_DisplayName.m_Value; + return displayName; }}} }; diff --git a/lib/icinga/hostgroup.ti b/lib/icinga/hostgroup.ti index efcb45e54..b679344aa 100644 --- a/lib/icinga/hostgroup.ti +++ b/lib/icinga/hostgroup.ti @@ -11,10 +11,11 @@ class HostGroup : CustomVarObject { [config] String display_name { get {{{ - if (m_DisplayName.m_Value.IsEmpty()) + String displayName = m_DisplayName.load(); + if (displayName.IsEmpty()) return GetName(); else - return m_DisplayName.m_Value; + return displayName; }}} }; diff --git a/lib/icinga/service.ti b/lib/icinga/service.ti index 4448c0aa8..bbd241241 100644 --- a/lib/icinga/service.ti +++ b/lib/icinga/service.ti @@ -33,10 +33,11 @@ class Service : Checkable < ServiceNameComposer [config] String display_name { get {{{ - if (m_DisplayName.m_Value.IsEmpty()) + String displayName = m_DisplayName.load(); + if (displayName.IsEmpty()) return GetShortName(); else - return m_DisplayName.m_Value; + return displayName; }}} }; [config, required] name(Host) host_name; diff --git a/lib/icinga/servicegroup.ti b/lib/icinga/servicegroup.ti index 69a0887b9..7daf9d419 100644 --- a/lib/icinga/servicegroup.ti +++ b/lib/icinga/servicegroup.ti @@ -11,10 +11,11 @@ class ServiceGroup : CustomVarObject { [config] String display_name { get {{{ - if (m_DisplayName.m_Value.IsEmpty()) + String displayName = m_DisplayName.load(); + if (displayName.IsEmpty()) return GetName(); else - return m_DisplayName.m_Value; + return displayName; }}} }; diff --git a/lib/icinga/timeperiod.ti b/lib/icinga/timeperiod.ti index 58a5e8c4d..bba272e81 100644 --- a/lib/icinga/timeperiod.ti +++ b/lib/icinga/timeperiod.ti @@ -12,10 +12,11 @@ class TimePeriod : CustomVarObject { [config] String display_name { get {{{ - if (m_DisplayName.m_Value.IsEmpty()) + String displayName = m_DisplayName.load(); + if (displayName.IsEmpty()) return GetName(); else - return m_DisplayName.m_Value; + return displayName; }}} }; [config, signal_with_old_value] Dictionary::Ptr ranges; diff --git a/lib/icinga/user.ti b/lib/icinga/user.ti index c69bac69c..8b8c43a14 100644 --- a/lib/icinga/user.ti +++ b/lib/icinga/user.ti @@ -13,10 +13,11 @@ class User : CustomVarObject { [config] String display_name { get {{{ - if (m_DisplayName.m_Value.IsEmpty()) + String displayName = m_DisplayName.load(); + if (displayName.IsEmpty()) return GetName(); else - return m_DisplayName.m_Value; + return displayName; }}} }; [config, no_user_modify, required, signal_with_old_value] array(name(UserGroup)) groups { diff --git a/lib/icinga/usergroup.ti b/lib/icinga/usergroup.ti index 35c7ccff4..e955c5e5e 100644 --- a/lib/icinga/usergroup.ti +++ b/lib/icinga/usergroup.ti @@ -11,10 +11,11 @@ class UserGroup : CustomVarObject { [config] String display_name { get {{{ - if (m_DisplayName.m_Value.IsEmpty()) + String displayName = m_DisplayName.load(); + if (displayName.IsEmpty()) return GetName(); else - return m_DisplayName.m_Value; + return displayName; }}} }; diff --git a/tools/mkclass/classcompiler.cpp b/tools/mkclass/classcompiler.cpp index c9502a240..af17c538d 100644 --- a/tools/mkclass/classcompiler.cpp +++ b/tools/mkclass/classcompiler.cpp @@ -1060,7 +1060,7 @@ void ClassCompiler::HandleClass(const Klass& klass, const ClassDebugInfo&) if (field.Attributes & FANoStorage) continue; - m_Header << "\tEventuallyAtomic<" << field.Type.GetRealType() << ">::type m_" << field.GetFriendlyName() << ";" << std::endl; + m_Header << "\tAtomicOrLocked<" << field.Type.GetRealType() << "> m_" << field.GetFriendlyName() << ";" << std::endl; } /* signal */