// SPDX-License-Identifier: GPL-2.0-or-later // Copyright The Music Player Daemon Project #include "config.h" #include "Listen.hxx" #include "Log.hxx" #include "client/Listener.hxx" #include "config/Param.hxx" #include "config/Data.hxx" #include "config/Option.hxx" #include "config/Net.hxx" #include "lib/fmt/ExceptionFormatter.hxx" #include "lib/fmt/PathFormatter.hxx" #include "lib/fmt/RuntimeError.hxx" #include "net/AllocatedSocketAddress.hxx" #include "net/UniqueSocketDescriptor.hxx" #include "net/SocketUtil.hxx" #include "system/Error.hxx" #include "fs/AllocatedPath.hxx" #include "fs/glue/StandardDirectory.hxx" #include "fs/XDG.hxx" #include "util/Domain.hxx" #include #ifdef ENABLE_SYSTEMD_DAEMON #include #endif #define DEFAULT_PORT 6600 #if defined(USE_XDG) && defined(HAVE_UN) static constexpr Domain listen_domain("listen"); #endif int listen_port; #ifdef ENABLE_SYSTEMD_DAEMON static bool listen_systemd_activation(ClientListener &listener) { int n = sd_listen_fds(true); if (n <= 0) { if (n < 0) throw MakeErrno(-n, "sd_listen_fds() failed"); return false; } for (int i = SD_LISTEN_FDS_START, end = SD_LISTEN_FDS_START + n; i != end; ++i) listener.AddFD(UniqueSocketDescriptor(i)); return true; } #endif /** * Listen on "$XDG_RUNTIME_DIR/mpd/socket" (if applicable). * * @return true if a listener socket was added */ static bool ListenXdgRuntimeDir(ClientListener &listener) noexcept { #if defined(USE_XDG) && defined(HAVE_UN) if (geteuid() == 0) /* this MPD instance is a system-wide daemon; don't use $XDG_RUNTIME_DIR */ return false; const auto mpd_runtime_dir = GetAppRuntimeDir(); if (mpd_runtime_dir.IsNull()) return false; const auto socket_path = mpd_runtime_dir / Path::FromFS("socket"); unlink(socket_path.c_str()); AllocatedSocketAddress address; address.SetLocal(socket_path.c_str()); try { auto fd = socket_bind_listen(AF_LOCAL, SOCK_STREAM, 0, address, 5); chmod(socket_path.c_str(), 0600); listener.AddFD(std::move(fd), std::move(address)); return true; } catch (...) { FmtError(listen_domain, "Failed to listen on '{}' (not fatal): {}", socket_path, std::current_exception()); return false; } #else (void)listener; return false; #endif } void listen_global_init(const ConfigData &config, ClientListener &listener) { int port = config.GetPositive(ConfigOption::PORT, DEFAULT_PORT); #ifdef ENABLE_SYSTEMD_DAEMON if (listen_systemd_activation(listener)) return; #endif for (const auto ¶m : config.GetParamList(ConfigOption::BIND_TO_ADDRESS)) { try { ServerSocketAddGeneric(listener, param.value.c_str(), port); } catch (...) { std::throw_with_nested(FmtRuntimeError("Failed to listen on {} (line {})", param.value, param.line)); } } bool have_xdg_runtime_listener = false; if (listener.IsEmpty()) { /* no "bind_to_address" configured, bind the configured port on all interfaces */ have_xdg_runtime_listener = ListenXdgRuntimeDir(listener); try { listener.AddPort(port); } catch (...) { std::throw_with_nested(FmtRuntimeError("Failed to listen on *:{}", port)); } } try { listener.Open(); } catch (...) { if (have_xdg_runtime_listener) LogError(std::current_exception(), "Default TCP listener setup failed, but this is okay because we have a $XDG_RUNTIME_DIR listener"); else throw; } listen_port = port; }