Merge pull request #11560 from phire/EventHook_construct_on_first_use

HookableEvent: Switch to construct on first use
This commit is contained in:
Admiral H. Curtiss 2023-02-14 01:01:03 +01:00 committed by GitHub
commit 5f929d00eb
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
1 changed files with 28 additions and 15 deletions

View File

@ -67,36 +67,49 @@ private:
std::string m_name; std::string m_name;
}; };
struct Storage
{
std::mutex m_mutex;
std::vector<HookImpl*> m_listeners;
};
// We use the "Construct On First Use" idiom to avoid the static initialization order fiasco.
// https://isocpp.org/wiki/faq/ctors#static-init-order
static Storage& GetStorage()
{
static Storage storage;
return storage;
}
static void Remove(HookImpl* handle)
{
auto& storage = GetStorage();
std::lock_guard lock(storage.m_mutex);
std::erase(storage.m_listeners, handle);
}
public: public:
// Returns a handle that will unregister the listener when destroyed. // Returns a handle that will unregister the listener when destroyed.
static EventHook Register(CallbackType callback, std::string name) static EventHook Register(CallbackType callback, std::string name)
{ {
std::lock_guard lock(m_mutex); auto& storage = GetStorage();
std::lock_guard lock(storage.m_mutex);
DEBUG_LOG_FMT(COMMON, "Registering {} handler at {} event hook", name, EventName.value); DEBUG_LOG_FMT(COMMON, "Registering {} handler at {} event hook", name, EventName.value);
auto handle = std::make_unique<HookImpl>(callback, std::move(name)); auto handle = std::make_unique<HookImpl>(callback, std::move(name));
m_listeners.push_back(handle.get()); storage.m_listeners.push_back(handle.get());
return handle; return handle;
} }
static void Trigger(const CallbackArgs&... args) static void Trigger(const CallbackArgs&... args)
{ {
std::lock_guard lock(m_mutex); auto& storage = GetStorage();
std::lock_guard lock(storage.m_mutex);
for (const auto& handle : m_listeners) for (const auto& handle : storage.m_listeners)
handle->m_fn(args...); handle->m_fn(args...);
} }
private:
static void Remove(HookImpl* handle)
{
std::lock_guard lock(m_mutex);
std::erase(m_listeners, handle);
}
inline static std::vector<HookImpl*> m_listeners = {};
inline static std::mutex m_mutex;
}; };
} // namespace Common } // namespace Common