From 355dd5cb24f1d36cb13a2463ce7585186474adc8 Mon Sep 17 00:00:00 2001 From: Max Kellermann Date: Wed, 7 Aug 2013 18:59:42 +0200 Subject: [PATCH] event/DeferredMonitor: new class wrapping g_idle_add() --- Makefile.am | 1 + src/GlobalEvents.cxx | 30 +++++++++++-------- src/event/DeferredMonitor.cxx | 55 +++++++++++++++++++++++++++++++++++ src/event/DeferredMonitor.hxx | 54 ++++++++++++++++++++++++++++++++++ 4 files changed, 128 insertions(+), 12 deletions(-) create mode 100644 src/event/DeferredMonitor.cxx create mode 100644 src/event/DeferredMonitor.hxx diff --git a/Makefile.am b/Makefile.am index 667497d44..bcfe598c5 100644 --- a/Makefile.am +++ b/Makefile.am @@ -297,6 +297,7 @@ libevent_a_SOURCES = \ src/event/WakeFD.hxx \ src/event/SignalMonitor.hxx src/event/SignalMonitor.cxx \ src/event/TimeoutMonitor.hxx src/event/TimeoutMonitor.cxx \ + src/event/DeferredMonitor.hxx src/event/DeferredMonitor.cxx \ src/event/SocketMonitor.cxx src/event/SocketMonitor.hxx \ src/event/BufferedSocket.cxx src/event/BufferedSocket.hxx \ src/event/FullyBufferedSocket.cxx src/event/FullyBufferedSocket.hxx \ diff --git a/src/GlobalEvents.cxx b/src/GlobalEvents.cxx index e30090ef4..ade745535 100644 --- a/src/GlobalEvents.cxx +++ b/src/GlobalEvents.cxx @@ -19,6 +19,8 @@ #include "config.h" #include "GlobalEvents.hxx" +#include "util/Manual.hxx" +#include "event/DeferredMonitor.hxx" #include "gcc.h" #include @@ -30,7 +32,13 @@ #define G_LOG_DOMAIN "global_events" namespace GlobalEvents { - static guint source_id; + class Monitor final : public DeferredMonitor { + protected: + virtual void Run() override; + }; + + static Manual monitor; + static std::atomic_uint flags; static Handler handlers[MAX]; } @@ -47,29 +55,27 @@ InvokeGlobalEvent(GlobalEvents::Event event) GlobalEvents::handlers[event](); } -static gboolean -GlobalEventCallback(gcc_unused gpointer data) +void +GlobalEvents::Monitor::Run() { - const unsigned flags = GlobalEvents::flags.exchange(0); + const unsigned f = flags.exchange(0); - for (unsigned i = 0; i < GlobalEvents::MAX; ++i) - if (flags & (1u << i)) + for (unsigned i = 0; i < MAX; ++i) + if (f & (1u << i)) /* invoke the event handler */ - InvokeGlobalEvent(GlobalEvents::Event(i)); - - return false; + InvokeGlobalEvent(Event(i)); } void GlobalEvents::Initialize() { + monitor.Construct(); } void GlobalEvents::Deinitialize() { - if (source_id != 0) - g_source_remove(source_id); + monitor.Destruct(); } void @@ -88,5 +94,5 @@ GlobalEvents::Emit(Event event) const unsigned mask = 1u << unsigned(event); if (GlobalEvents::flags.fetch_or(mask) == 0) - source_id = g_idle_add(GlobalEventCallback, nullptr); + monitor->Schedule(); } diff --git a/src/event/DeferredMonitor.cxx b/src/event/DeferredMonitor.cxx new file mode 100644 index 000000000..10aa0d757 --- /dev/null +++ b/src/event/DeferredMonitor.cxx @@ -0,0 +1,55 @@ +/* + * Copyright (C) 2003-2013 The Music Player Daemon Project + * http://www.musicpd.org + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation; either version 2 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License along + * with this program; if not, write to the Free Software Foundation, Inc., + * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. + */ + +#include "config.h" +#include "DeferredMonitor.hxx" +#include "Loop.hxx" + +void +DeferredMonitor::Cancel() +{ + const auto id = source_id.exchange(0); + if (id != 0) + g_source_remove(id); +} + +void +DeferredMonitor::Schedule() +{ + const unsigned id = g_idle_add(Callback, this); + const auto old_id = source_id.exchange(id); + if (old_id != 0) + g_source_remove(old_id); +} + +void +DeferredMonitor::DoRun() +{ + const auto id = source_id.exchange(0); + if (id != 0) + Run(); +} + +gboolean +DeferredMonitor::Callback(gpointer data) +{ + DeferredMonitor &monitor = *(DeferredMonitor *)data; + monitor.Run(); + return false; +} diff --git a/src/event/DeferredMonitor.hxx b/src/event/DeferredMonitor.hxx new file mode 100644 index 000000000..8a08facba --- /dev/null +++ b/src/event/DeferredMonitor.hxx @@ -0,0 +1,54 @@ +/* + * Copyright (C) 2003-2013 The Music Player Daemon Project + * http://www.musicpd.org + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation; either version 2 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License along + * with this program; if not, write to the Free Software Foundation, Inc., + * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. + */ + +#ifndef MPD_SOCKET_DEFERRED_MONITOR_HXX +#define MPD_SOCKET_DEFERRED_MONITOR_HXX + +#include "check.h" + +#include + +#include + +/** + * Defer execution of an event into an #EventLoop. + */ +class DeferredMonitor { + std::atomic source_id; + +public: + DeferredMonitor() + :source_id(0) {} + + ~DeferredMonitor() { + Cancel(); + } + + void Schedule(); + void Cancel(); + +protected: + virtual void Run() = 0; + +private: + void DoRun(); + static gboolean Callback(gpointer data); +}; + +#endif /* MAIN_NOTIFY_H */