summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorMax Kellermann <max@musicpd.org>2018-08-02 11:15:35 +0200
committerMax Kellermann <max@musicpd.org>2018-08-02 11:15:35 +0200
commita31da51fd9bd215645b9cabce5fefa449ca14c90 (patch)
treee320d144db30fade309650e598930422824ac032
parentcf471e830ffd9dc417aeb5e85a9bbbfbd6d18b09 (diff)
lib/systemd/Watchdog: implement the systemd watchdog protocol
-rw-r--r--Makefile.am5
-rw-r--r--NEWS1
-rw-r--r--src/Instance.cxx3
-rw-r--r--src/Instance.hxx8
-rw-r--r--src/lib/systemd/Watchdog.cxx55
-rw-r--r--src/lib/systemd/Watchdog.hxx57
-rw-r--r--systemd/system/mpd.service.in5
-rw-r--r--systemd/user/mpd.service.in5
8 files changed, 139 insertions, 0 deletions
diff --git a/Makefile.am b/Makefile.am
index 6009d7ba6..71e1ae550 100644
--- a/Makefile.am
+++ b/Makefile.am
@@ -205,6 +205,11 @@ libmpd_a_SOURCES += \
src/unix/PidFile.hxx
endif
+if ENABLE_SYSTEMD_DAEMON
+libmpd_a_SOURCES += \
+ src/lib/systemd/Watchdog.cxx src/lib/systemd/Watchdog.hxx
+endif
+
endif
if ENABLE_DATABASE
diff --git a/NEWS b/NEWS
index 3c7630132..2abb4f0e1 100644
--- a/NEWS
+++ b/NEWS
@@ -36,6 +36,7 @@ ver 0.21 (not yet released)
- sndio: new mixer plugin
* encoder
- opus: support for sending metadata using ogg stream chaining
+* systemd watchdog support
* require GCC 6
ver 0.20.21 (not yet released)
diff --git a/src/Instance.cxx b/src/Instance.cxx
index b3595e995..f208ca194 100644
--- a/src/Instance.cxx
+++ b/src/Instance.cxx
@@ -41,6 +41,9 @@
Instance::Instance()
:rtio_thread(true),
+#ifdef ENABLE_SYSTEMD_DAEMON
+ systemd_watchdog(event_loop),
+#endif
idle_monitor(event_loop, BIND_THIS_METHOD(OnIdle))
{
}
diff --git a/src/Instance.hxx b/src/Instance.hxx
index 936923d73..6b1e6d87a 100644
--- a/src/Instance.hxx
+++ b/src/Instance.hxx
@@ -26,6 +26,10 @@
#include "event/MaskMonitor.hxx"
#include "Compiler.h"
+#ifdef ENABLE_SYSTEMD_DAEMON
+#include "lib/systemd/Watchdog.hxx"
+#endif
+
#ifdef ENABLE_CURL
#include "RemoteTagCacheHandler.hxx"
#endif
@@ -89,6 +93,10 @@ struct Instance final
*/
EventThread rtio_thread;
+#ifdef ENABLE_SYSTEMD_DAEMON
+ Systemd::Watchdog systemd_watchdog;
+#endif
+
MaskMonitor idle_monitor;
#ifdef ENABLE_NEIGHBOR_PLUGINS
diff --git a/src/lib/systemd/Watchdog.cxx b/src/lib/systemd/Watchdog.cxx
new file mode 100644
index 000000000..5cba23636
--- /dev/null
+++ b/src/lib/systemd/Watchdog.cxx
@@ -0,0 +1,55 @@
+/*
+ * Copyright (C) 2018 Max Kellermann <max.kellermann@gmail.com>
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ *
+ * - Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ *
+ * - Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the
+ * distribution.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+ * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+ * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
+ * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
+ * FOUNDATION OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
+ * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
+ * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
+ * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
+ * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
+ * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
+ * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED
+ * OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+#include "config.h"
+#include "Watchdog.hxx"
+
+#include <systemd/sd-daemon.h>
+
+namespace Systemd {
+
+Watchdog::Watchdog(EventLoop &_loop) noexcept
+ :timer(_loop, BIND_THIS_METHOD(OnTimer))
+{
+ uint64_t usec;
+ if (sd_watchdog_enabled(true, &usec) <= 0)
+ return;
+
+ interval = std::chrono::microseconds(usec) / 2;
+ timer.Schedule(interval);
+}
+
+void
+Watchdog::OnTimer() noexcept
+{
+ sd_notify(0, "WATCHDOG=1");
+ timer.Schedule(interval);
+}
+
+} // namespace Systemd
diff --git a/src/lib/systemd/Watchdog.hxx b/src/lib/systemd/Watchdog.hxx
new file mode 100644
index 000000000..9a0a8ed94
--- /dev/null
+++ b/src/lib/systemd/Watchdog.hxx
@@ -0,0 +1,57 @@
+/*
+ * Copyright (C) 2018 Max Kellermann <max.kellermann@gmail.com>
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ *
+ * - Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ *
+ * - Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the
+ * distribution.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+ * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+ * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
+ * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
+ * FOUNDATION OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
+ * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
+ * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
+ * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
+ * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
+ * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
+ * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED
+ * OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+#ifndef SYSTEMD_WATCHDOG_HXX
+#define SYSTEMD_WATCHDOG_HXX
+
+#include "check.h"
+#include "event/TimerEvent.hxx"
+
+namespace Systemd {
+
+/**
+ * This class implements the systemd watchdog protocol; see
+ * systemd.service(5) and sd_watchdog_enabled(3). If the watchdog is
+ * not enabled, this class does nothing.
+ */
+class Watchdog {
+ TimerEvent timer;
+
+ std::chrono::steady_clock::duration interval;
+
+public:
+ explicit Watchdog(EventLoop &_loop) noexcept;
+
+private:
+ void OnTimer() noexcept;
+};
+
+} // namespace Systemd
+
+#endif
diff --git a/systemd/system/mpd.service.in b/systemd/system/mpd.service.in
index 7b0218bf2..2c4007d25 100644
--- a/systemd/system/mpd.service.in
+++ b/systemd/system/mpd.service.in
@@ -7,6 +7,11 @@ After=network.target sound.target
Type=notify
ExecStart=@prefix@/bin/mpd --no-daemon
+# Enable this setting to ask systemd to watch over MPD, see
+# systemd.service(5). This is disabled by default because it causes
+# periodic wakeups which are unnecessary if MPD is not playing.
+#WatchdogSec=120
+
# allow MPD to use real-time priority 50
LimitRTPRIO=50
LimitRTTIME=infinity
diff --git a/systemd/user/mpd.service.in b/systemd/user/mpd.service.in
index 12b814c2d..eea52d951 100644
--- a/systemd/user/mpd.service.in
+++ b/systemd/user/mpd.service.in
@@ -7,6 +7,11 @@ After=network.target sound.target
Type=notify
ExecStart=@prefix@/bin/mpd --no-daemon
+# Enable this setting to ask systemd to watch over MPD, see
+# systemd.service(5). This is disabled by default because it causes
+# periodic wakeups which are unnecessary if MPD is not playing.
+#WatchdogSec=120
+
# allow MPD to use real-time priority 50
LimitRTPRIO=50
LimitRTTIME=infinity