event/InotifyEvent: new class wrapping inotify
Replaces class InotifySource.
This commit is contained in:
		| @@ -43,7 +43,6 @@ db_glue_sources = [ | |||||||
| if enable_inotify | if enable_inotify | ||||||
|   db_glue_sources += [ |   db_glue_sources += [ | ||||||
|     'update/InotifyDomain.cxx', |     'update/InotifyDomain.cxx', | ||||||
|     'update/InotifySource.cxx', |  | ||||||
|     'update/InotifyQueue.cxx', |     'update/InotifyQueue.cxx', | ||||||
|     'update/InotifyUpdate.cxx', |     'update/InotifyUpdate.cxx', | ||||||
|   ] |   ] | ||||||
|   | |||||||
| @@ -1,115 +0,0 @@ | |||||||
| /* |  | ||||||
|  * Copyright 2003-2021 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 "InotifySource.hxx" |  | ||||||
| #include "InotifyDomain.hxx" |  | ||||||
| #include "io/FileDescriptor.hxx" |  | ||||||
| #include "system/Error.hxx" |  | ||||||
| #include "Log.hxx" |  | ||||||
|  |  | ||||||
| #include <cerrno> |  | ||||||
| #include <climits> |  | ||||||
| #include <cstdint> |  | ||||||
|  |  | ||||||
| #include <sys/inotify.h> |  | ||||||
|  |  | ||||||
| void |  | ||||||
| InotifySource::OnSocketReady([[maybe_unused]] unsigned flags) noexcept |  | ||||||
| { |  | ||||||
| 	uint8_t buffer[4096]; |  | ||||||
| 	static_assert(sizeof(buffer) >= sizeof(struct inotify_event) + NAME_MAX + 1, |  | ||||||
| 		      "inotify buffer too small"); |  | ||||||
|  |  | ||||||
| 	auto ifd = socket_event.GetFileDescriptor(); |  | ||||||
| 	ssize_t nbytes = ifd.Read(buffer, sizeof(buffer)); |  | ||||||
| 	if (nbytes <= 0) { |  | ||||||
| 		if (nbytes < 0) |  | ||||||
| 			FmtError(inotify_domain, |  | ||||||
| 				 "Failed to read from inotify: {}", |  | ||||||
| 				 strerror(errno)); |  | ||||||
| 		else |  | ||||||
| 			LogError(inotify_domain, |  | ||||||
| 				 "end of file from inotify"); |  | ||||||
| 		socket_event.Cancel(); |  | ||||||
| 		return; |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	const uint8_t *p = buffer, *const end = p + nbytes; |  | ||||||
|  |  | ||||||
| 	while (true) { |  | ||||||
| 		const size_t remaining = end - p; |  | ||||||
| 		const auto *event = |  | ||||||
| 			(const struct inotify_event *)p; |  | ||||||
| 		if (remaining < sizeof(*event) || |  | ||||||
| 		    remaining < sizeof(*event) + event->len) |  | ||||||
| 			break; |  | ||||||
|  |  | ||||||
| 		const char *name; |  | ||||||
| 		if (event->len > 0 && event->name[event->len - 1] == 0) |  | ||||||
| 			name = event->name; |  | ||||||
| 		else |  | ||||||
| 			name = nullptr; |  | ||||||
|  |  | ||||||
| 		callback(event->wd, event->mask, name, callback_ctx); |  | ||||||
| 		p += sizeof(*event) + event->len; |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
|  |  | ||||||
| static FileDescriptor |  | ||||||
| InotifyInit() |  | ||||||
| { |  | ||||||
| 	FileDescriptor fd; |  | ||||||
| 	if (!fd.CreateInotify()) |  | ||||||
| 		throw MakeErrno("inotify_init() has failed"); |  | ||||||
|  |  | ||||||
| 	return fd; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| InotifySource::InotifySource(EventLoop &_loop, |  | ||||||
| 			     mpd_inotify_callback_t _callback, void *_ctx) |  | ||||||
| 	:socket_event(_loop, BIND_THIS_METHOD(OnSocketReady), |  | ||||||
| 		      InotifyInit()), |  | ||||||
| 	 callback(_callback), callback_ctx(_ctx) |  | ||||||
| { |  | ||||||
| 	socket_event.ScheduleRead(); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| int |  | ||||||
| InotifySource::Add(const char *path_fs, unsigned mask) |  | ||||||
| { |  | ||||||
| 	auto ifd = socket_event.GetFileDescriptor(); |  | ||||||
| 	int wd = inotify_add_watch(ifd.Get(), path_fs, mask); |  | ||||||
| 	if (wd < 0) |  | ||||||
| 		throw MakeErrno("inotify_add_watch() has failed"); |  | ||||||
|  |  | ||||||
| 	return wd; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| void |  | ||||||
| InotifySource::Remove(unsigned wd) noexcept |  | ||||||
| { |  | ||||||
| 	auto ifd = socket_event.GetFileDescriptor(); |  | ||||||
| 	int ret = inotify_rm_watch(ifd.Get(), wd); |  | ||||||
| 	if (ret < 0 && errno != EINVAL) |  | ||||||
| 		FmtError(inotify_domain, "inotify_rm_watch() has failed: {}", |  | ||||||
| 			 strerror(errno)); |  | ||||||
|  |  | ||||||
| 	/* EINVAL may happen here when the file has been deleted; the |  | ||||||
| 	   kernel seems to auto-unregister deleted files */ |  | ||||||
| } |  | ||||||
| @@ -1,71 +0,0 @@ | |||||||
| /* |  | ||||||
|  * Copyright 2003-2021 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_INOTIFY_SOURCE_HXX |  | ||||||
| #define MPD_INOTIFY_SOURCE_HXX |  | ||||||
|  |  | ||||||
| #include "event/PipeEvent.hxx" |  | ||||||
|  |  | ||||||
| typedef void (*mpd_inotify_callback_t)(int wd, unsigned mask, |  | ||||||
| 				       const char *name, void *ctx); |  | ||||||
|  |  | ||||||
| class InotifySource final { |  | ||||||
| 	PipeEvent socket_event; |  | ||||||
|  |  | ||||||
| 	mpd_inotify_callback_t callback; |  | ||||||
| 	void *callback_ctx; |  | ||||||
|  |  | ||||||
| public: |  | ||||||
| 	/** |  | ||||||
| 	 * Creates a new inotify source and registers it in the |  | ||||||
| 	 * #EventLoop. |  | ||||||
| 	 * |  | ||||||
| 	 * Throws #std::system_error on error. |  | ||||||
| 	 * |  | ||||||
| 	 * @param callback a callback invoked for events received from |  | ||||||
| 	 * the kernel |  | ||||||
| 	 */ |  | ||||||
| 	InotifySource(EventLoop &_loop, |  | ||||||
| 		      mpd_inotify_callback_t callback, void *ctx); |  | ||||||
|  |  | ||||||
| 	~InotifySource() noexcept { |  | ||||||
| 		socket_event.Close(); |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	/** |  | ||||||
| 	 * Adds a path to the notify list. |  | ||||||
| 	 * |  | ||||||
| 	 * Throws #std::system_error on error. |  | ||||||
| 	 * |  | ||||||
| 	 * @return a watch descriptor |  | ||||||
| 	 */ |  | ||||||
| 	int Add(const char *path_fs, unsigned mask); |  | ||||||
|  |  | ||||||
| 	/** |  | ||||||
| 	 * Removes a path from the notify list. |  | ||||||
| 	 * |  | ||||||
| 	 * @param wd the watch descriptor returned by mpd_inotify_source_add() |  | ||||||
| 	 */ |  | ||||||
| 	void Remove(unsigned wd) noexcept; |  | ||||||
|  |  | ||||||
| private: |  | ||||||
| 	void OnSocketReady(unsigned flags) noexcept; |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| #endif |  | ||||||
| @@ -118,7 +118,7 @@ InotifyUpdate::Disable(WatchDirectory &directory) noexcept | |||||||
| 	for (WatchDirectory &child : directory.children) | 	for (WatchDirectory &child : directory.children) | ||||||
| 		Disable(child); | 		Disable(child); | ||||||
|  |  | ||||||
| 	source.Remove(directory.descriptor); | 	inotify_event.RemoveWatch(directory.descriptor); | ||||||
| } | } | ||||||
|  |  | ||||||
| void | void | ||||||
| @@ -199,7 +199,8 @@ try { | |||||||
| 			continue; | 			continue; | ||||||
|  |  | ||||||
| 		try { | 		try { | ||||||
| 			ret = source.Add(child_path_fs.c_str(), IN_MASK); | 			ret = inotify_event.AddWatch(child_path_fs.c_str(), | ||||||
|  | 						     IN_MASK); | ||||||
| 		} catch (...) { | 		} catch (...) { | ||||||
| 			FmtError(inotify_domain, | 			FmtError(inotify_domain, | ||||||
| 				 "Failed to register {}: {}", | 				 "Failed to register {}: {}", | ||||||
| @@ -240,7 +241,7 @@ WatchDirectory::GetDepth() const noexcept | |||||||
| inline | inline | ||||||
| InotifyUpdate::InotifyUpdate(EventLoop &loop, UpdateService &update, | InotifyUpdate::InotifyUpdate(EventLoop &loop, UpdateService &update, | ||||||
| 			     unsigned _max_depth) | 			     unsigned _max_depth) | ||||||
| 	:source(loop, InotifyCallback, this), | 	:inotify_event(loop, *this), | ||||||
| 	 queue(loop, update), | 	 queue(loop, update), | ||||||
| 	 max_depth(_max_depth) | 	 max_depth(_max_depth) | ||||||
| { | { | ||||||
| @@ -251,7 +252,7 @@ InotifyUpdate::~InotifyUpdate() noexcept = default; | |||||||
| inline void | inline void | ||||||
| InotifyUpdate::Start(Path path) | InotifyUpdate::Start(Path path) | ||||||
| { | { | ||||||
| 	int descriptor = source.Add(path.c_str(), IN_MASK); | 	int descriptor = inotify_event.AddWatch(path.c_str(), IN_MASK); | ||||||
|  |  | ||||||
| 	root = std::make_unique<WatchDirectory>(path, descriptor); | 	root = std::make_unique<WatchDirectory>(path, descriptor); | ||||||
| 	root->LoadExcludeList(path); | 	root->LoadExcludeList(path); | ||||||
| @@ -262,8 +263,7 @@ InotifyUpdate::Start(Path path) | |||||||
| } | } | ||||||
|  |  | ||||||
| void | void | ||||||
| InotifyUpdate::InotifyCallback(int wd, unsigned mask, | InotifyUpdate::OnInotify(int wd, unsigned mask, const char *) | ||||||
| 			       [[maybe_unused]] const char *name) noexcept |  | ||||||
| { | { | ||||||
| 	auto i = directories.find(wd); | 	auto i = directories.find(wd); | ||||||
| 	if (i == directories.end()) | 	if (i == directories.end()) | ||||||
| @@ -310,6 +310,12 @@ InotifyUpdate::InotifyCallback(int wd, unsigned mask, | |||||||
| 	} | 	} | ||||||
| } | } | ||||||
|  |  | ||||||
|  | void | ||||||
|  | InotifyUpdate::OnInotifyError(std::exception_ptr error) noexcept | ||||||
|  | { | ||||||
|  | 	LogError(error, "inotify error"); | ||||||
|  | } | ||||||
|  |  | ||||||
| std::unique_ptr<InotifyUpdate> | std::unique_ptr<InotifyUpdate> | ||||||
| mpd_inotify_init(EventLoop &loop, Storage &storage, UpdateService &update, | mpd_inotify_init(EventLoop &loop, Storage &storage, UpdateService &update, | ||||||
| 		 unsigned max_depth) | 		 unsigned max_depth) | ||||||
|   | |||||||
| @@ -20,8 +20,8 @@ | |||||||
| #ifndef MPD_INOTIFY_UPDATE_HXX | #ifndef MPD_INOTIFY_UPDATE_HXX | ||||||
| #define MPD_INOTIFY_UPDATE_HXX | #define MPD_INOTIFY_UPDATE_HXX | ||||||
|  |  | ||||||
| #include "InotifySource.hxx" |  | ||||||
| #include "InotifyQueue.hxx" | #include "InotifyQueue.hxx" | ||||||
|  | #include "event/InotifyEvent.hxx" | ||||||
|  |  | ||||||
| #include <map> | #include <map> | ||||||
| #include <memory> | #include <memory> | ||||||
| @@ -33,8 +33,8 @@ struct WatchDirectory; | |||||||
| /** | /** | ||||||
|  * Glue code between InotifySource and InotifyQueue. |  * Glue code between InotifySource and InotifyQueue. | ||||||
|  */ |  */ | ||||||
| class InotifyUpdate { | class InotifyUpdate final : InotifyHandler { | ||||||
| 	InotifySource source; | 	InotifyEvent inotify_event; | ||||||
| 	InotifyQueue queue; | 	InotifyQueue queue; | ||||||
|  |  | ||||||
| 	const unsigned max_depth; | 	const unsigned max_depth; | ||||||
| @@ -50,14 +50,6 @@ public: | |||||||
| 	void Start(Path path); | 	void Start(Path path); | ||||||
|  |  | ||||||
| private: | private: | ||||||
| 	void InotifyCallback(int wd, unsigned mask, const char *name) noexcept; |  | ||||||
|  |  | ||||||
| 	static void InotifyCallback(int wd, unsigned mask, |  | ||||||
| 				    const char *name, void *ctx) noexcept { |  | ||||||
| 		auto &iu = *(InotifyUpdate *)ctx; |  | ||||||
| 		iu.InotifyCallback(wd, mask, name); |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	void AddToMap(WatchDirectory &directory) noexcept; | 	void AddToMap(WatchDirectory &directory) noexcept; | ||||||
| 	void RemoveFromMap(WatchDirectory &directory) noexcept; | 	void RemoveFromMap(WatchDirectory &directory) noexcept; | ||||||
| 	void Disable(WatchDirectory &directory) noexcept; | 	void Disable(WatchDirectory &directory) noexcept; | ||||||
| @@ -66,6 +58,11 @@ private: | |||||||
| 	void RecursiveWatchSubdirectories(WatchDirectory &parent, | 	void RecursiveWatchSubdirectories(WatchDirectory &parent, | ||||||
| 					  Path path_fs, | 					  Path path_fs, | ||||||
| 					  unsigned depth) noexcept; | 					  unsigned depth) noexcept; | ||||||
|  |  | ||||||
|  | private: | ||||||
|  | 	/* virtual methods from class InotifyHandler */ | ||||||
|  | 	void OnInotify(int wd, unsigned mask, const char *name) override; | ||||||
|  | 	void OnInotifyError(std::exception_ptr error) noexcept override; | ||||||
| }; | }; | ||||||
|  |  | ||||||
| /** | /** | ||||||
|   | |||||||
							
								
								
									
										129
									
								
								src/event/InotifyEvent.cxx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										129
									
								
								src/event/InotifyEvent.cxx
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,129 @@ | |||||||
|  | /* | ||||||
|  |  * Copyright 2022 CM4all GmbH | ||||||
|  |  * All rights reserved. | ||||||
|  |  * | ||||||
|  |  * author: Max Kellermann <mk@cm4all.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 "InotifyEvent.hxx" | ||||||
|  | #include "system/Error.hxx" | ||||||
|  | #include "io/UniqueFileDescriptor.hxx" | ||||||
|  |  | ||||||
|  | #include <array> | ||||||
|  |  | ||||||
|  | #include <limits.h> | ||||||
|  | #include <sys/inotify.h> | ||||||
|  |  | ||||||
|  | static UniqueFileDescriptor | ||||||
|  | CreateInotify() | ||||||
|  | { | ||||||
|  | 	int fd = inotify_init1(IN_CLOEXEC|IN_NONBLOCK); | ||||||
|  | 	if (fd < 0) | ||||||
|  | 		throw MakeErrno("inotify_init1() failed"); | ||||||
|  |  | ||||||
|  | 	return UniqueFileDescriptor(fd); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | InotifyEvent::InotifyEvent(EventLoop &event_loop, InotifyHandler &_handler) | ||||||
|  | 	:event(event_loop, BIND_THIS_METHOD(OnInotifyReady), | ||||||
|  | 	       CreateInotify().Release()), | ||||||
|  | 	 handler(_handler) | ||||||
|  | { | ||||||
|  | 	Enable(); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | InotifyEvent::~InotifyEvent() noexcept | ||||||
|  | { | ||||||
|  | 	Close(); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | int | ||||||
|  | InotifyEvent::AddWatch(const char *pathname, uint32_t mask) | ||||||
|  | { | ||||||
|  | 	int wd = inotify_add_watch(event.GetFileDescriptor().Get(), | ||||||
|  | 				   pathname, mask); | ||||||
|  | 	if (wd < 0) | ||||||
|  | 		throw FormatErrno("inotify_add_watch('%s') failed", pathname); | ||||||
|  |  | ||||||
|  | 	return wd; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | int | ||||||
|  | InotifyEvent::AddModifyWatch(const char *pathname) | ||||||
|  | { | ||||||
|  | 	return AddWatch(pathname, IN_MODIFY); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | void | ||||||
|  | InotifyEvent::RemoveWatch(int wd) noexcept | ||||||
|  | { | ||||||
|  | 	inotify_rm_watch(event.GetFileDescriptor().Get(), wd); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | inline void | ||||||
|  | InotifyEvent::OnInotifyReady(unsigned) noexcept | ||||||
|  | try { | ||||||
|  | 	std::array<std::byte, 4096> buffer; | ||||||
|  | 	static_assert(sizeof(buffer) >= sizeof(struct inotify_event) + NAME_MAX + 1, | ||||||
|  | 		      "inotify buffer too small"); | ||||||
|  |  | ||||||
|  | 	ssize_t nbytes = event.GetFileDescriptor().Read(buffer.data(), | ||||||
|  | 							buffer.size()); | ||||||
|  | 	if (nbytes <= 0) [[unlikely]] { | ||||||
|  | 		if (nbytes == 0) | ||||||
|  | 			throw std::runtime_error{"EOF from inotify"}; | ||||||
|  |  | ||||||
|  | 		const int e = errno; | ||||||
|  | 		if (e == EAGAIN) | ||||||
|  | 			return; | ||||||
|  |  | ||||||
|  | 		throw MakeErrno(e, "Reading inotify failed"); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	const std::byte *p = buffer.data(), *const end = p + nbytes; | ||||||
|  |  | ||||||
|  | 	while (true) { | ||||||
|  | 		const size_t remaining = end - p; | ||||||
|  | 		const auto &ie = *(const struct inotify_event *)(const void *)p; | ||||||
|  | 		if (remaining < sizeof(ie) || | ||||||
|  | 		    remaining < sizeof(ie) + ie.len) | ||||||
|  | 			break; | ||||||
|  |  | ||||||
|  | 		const char *name; | ||||||
|  | 		if (ie.len > 0 && ie.name[ie.len - 1] == 0) | ||||||
|  | 			name = ie.name; | ||||||
|  | 		else | ||||||
|  | 			name = nullptr; | ||||||
|  |  | ||||||
|  | 		handler.OnInotify(ie.wd, ie.mask, name); | ||||||
|  | 		p += sizeof(ie) + ie.len; | ||||||
|  | 	} | ||||||
|  | } catch (...) { | ||||||
|  | 	Close(); | ||||||
|  | 	handler.OnInotifyError(std::current_exception()); | ||||||
|  | } | ||||||
							
								
								
									
										129
									
								
								src/event/InotifyEvent.hxx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										129
									
								
								src/event/InotifyEvent.hxx
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,129 @@ | |||||||
|  | /* | ||||||
|  |  * Copyright 2022 CM4all GmbH | ||||||
|  |  * All rights reserved. | ||||||
|  |  * | ||||||
|  |  * author: Max Kellermann <mk@cm4all.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. | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | #pragma once | ||||||
|  |  | ||||||
|  | #include "PipeEvent.hxx" | ||||||
|  |  | ||||||
|  | #include <exception> | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * Handler for #InotifyEvent. | ||||||
|  |  */ | ||||||
|  | class InotifyHandler { | ||||||
|  | public: | ||||||
|  | 	/** | ||||||
|  | 	 * An inotify event was received. | ||||||
|  | 	 * | ||||||
|  | 	 * @param wd the watch descriptor returned by | ||||||
|  | 	 * InotifyEvent::AddWatch(). | ||||||
|  | 	 */ | ||||||
|  | 	virtual void OnInotify(int wd, unsigned mask, const char *name) = 0; | ||||||
|  |  | ||||||
|  | 	/** | ||||||
|  | 	 * An (permanent) inotify error has occurred, and the | ||||||
|  | 	 * #InotifyEvent has been closed. | ||||||
|  | 	 */ | ||||||
|  | 	virtual void OnInotifyError(std::exception_ptr error) noexcept = 0; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * #EventLoop integration for Linux inotify. | ||||||
|  |  */ | ||||||
|  | class InotifyEvent final { | ||||||
|  | 	PipeEvent event; | ||||||
|  |  | ||||||
|  | 	InotifyHandler &handler; | ||||||
|  |  | ||||||
|  | public: | ||||||
|  | 	/** | ||||||
|  | 	 * Create an inotify file descriptor add register it in the | ||||||
|  | 	 * #EventLoop. | ||||||
|  | 	 * | ||||||
|  | 	 * Throws on error. | ||||||
|  | 	 */ | ||||||
|  | 	InotifyEvent(EventLoop &event_loop, InotifyHandler &_handler); | ||||||
|  |  | ||||||
|  | 	~InotifyEvent() noexcept; | ||||||
|  |  | ||||||
|  | 	EventLoop &GetEventLoop() const noexcept { | ||||||
|  | 		return event.GetEventLoop(); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	/** | ||||||
|  | 	 * Re-enable polling the inotify file descriptor after it was | ||||||
|  | 	 * disabled by Disable(). | ||||||
|  | 	 */ | ||||||
|  | 	void Enable() noexcept { | ||||||
|  | 		event.ScheduleRead(); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	/** | ||||||
|  | 	 * Disable polling the inotify file descriptor.  Can be | ||||||
|  | 	 * re-enabled by Enable(). | ||||||
|  | 	 */ | ||||||
|  | 	void Disable() noexcept { | ||||||
|  | 		event.Cancel(); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	/** | ||||||
|  | 	 * Permanently close the inotify file descriptor.  Further | ||||||
|  | 	 * method calls not allowed after that. | ||||||
|  | 	 */ | ||||||
|  | 	void Close() noexcept { | ||||||
|  | 		event.Close(); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	/** | ||||||
|  | 	 * Register a new path to be watched. | ||||||
|  | 	 * | ||||||
|  | 	 * Throws on error. | ||||||
|  | 	 * | ||||||
|  | 	 * @return a watch descriptor | ||||||
|  | 	 */ | ||||||
|  | 	int AddWatch(const char *pathname, uint32_t mask); | ||||||
|  |  | ||||||
|  | 	/** | ||||||
|  | 	 * Wrapper for AddWatch(pathname, IN_MODIFY). | ||||||
|  | 	 */ | ||||||
|  | 	int AddModifyWatch(const char *pathname); | ||||||
|  |  | ||||||
|  | 	/** | ||||||
|  | 	 * Stop watching the given watch descriptor. | ||||||
|  | 	 * | ||||||
|  | 	 * @param wd a watch descriptor returned by AddWatch() | ||||||
|  | 	 */ | ||||||
|  | 	void RemoveWatch(int wd) noexcept; | ||||||
|  |  | ||||||
|  | private: | ||||||
|  | 	void OnInotifyReady(unsigned) noexcept; | ||||||
|  | }; | ||||||
| @@ -19,6 +19,10 @@ else | |||||||
|   event_sources += 'PollBackend.cxx' |   event_sources += 'PollBackend.cxx' | ||||||
| endif | endif | ||||||
|  |  | ||||||
|  | if enable_inotify | ||||||
|  |   event_sources += 'InotifyEvent.cxx' | ||||||
|  | endif | ||||||
|  |  | ||||||
| event = static_library( | event = static_library( | ||||||
|   'event', |   'event', | ||||||
|   'SignalMonitor.cxx', |   'SignalMonitor.cxx', | ||||||
|   | |||||||
| @@ -44,7 +44,6 @@ | |||||||
| #ifdef __linux__ | #ifdef __linux__ | ||||||
| #include <sys/eventfd.h> | #include <sys/eventfd.h> | ||||||
| #include <sys/signalfd.h> | #include <sys/signalfd.h> | ||||||
| #include <sys/inotify.h> |  | ||||||
| #endif | #endif | ||||||
|  |  | ||||||
| #ifndef O_NOCTTY | #ifndef O_NOCTTY | ||||||
| @@ -283,17 +282,6 @@ FileDescriptor::CreateSignalFD(const sigset_t *mask) noexcept | |||||||
| 	return true; | 	return true; | ||||||
| } | } | ||||||
|  |  | ||||||
| bool |  | ||||||
| FileDescriptor::CreateInotify() noexcept |  | ||||||
| { |  | ||||||
| 	int new_fd = inotify_init1(IN_CLOEXEC|IN_NONBLOCK); |  | ||||||
| 	if (new_fd < 0) |  | ||||||
| 		return false; |  | ||||||
|  |  | ||||||
| 	fd = new_fd; |  | ||||||
| 	return true; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| #endif | #endif | ||||||
|  |  | ||||||
| bool | bool | ||||||
|   | |||||||
| @@ -209,7 +209,6 @@ public: | |||||||
| #ifdef __linux__ | #ifdef __linux__ | ||||||
| 	bool CreateEventFD(unsigned initval=0) noexcept; | 	bool CreateEventFD(unsigned initval=0) noexcept; | ||||||
| 	bool CreateSignalFD(const sigset_t *mask) noexcept; | 	bool CreateSignalFD(const sigset_t *mask) noexcept; | ||||||
| 	bool CreateInotify() noexcept; |  | ||||||
| #endif | #endif | ||||||
|  |  | ||||||
| 	/** | 	/** | ||||||
|   | |||||||
| @@ -133,8 +133,6 @@ if enable_inotify | |||||||
|     'run_inotify', |     'run_inotify', | ||||||
|     'run_inotify.cxx', |     'run_inotify.cxx', | ||||||
|     'ShutdownHandler.cxx', |     'ShutdownHandler.cxx', | ||||||
|     '../src/db/update/InotifyDomain.cxx', |  | ||||||
|     '../src/db/update/InotifySource.cxx', |  | ||||||
|     include_directories: inc, |     include_directories: inc, | ||||||
|     dependencies: [ |     dependencies: [ | ||||||
|       log_dep, |       log_dep, | ||||||
|   | |||||||
| @@ -18,7 +18,7 @@ | |||||||
|  */ |  */ | ||||||
|  |  | ||||||
| #include "ShutdownHandler.hxx" | #include "ShutdownHandler.hxx" | ||||||
| #include "db/update/InotifySource.hxx" | #include "event/InotifyEvent.hxx" | ||||||
| #include "event/Loop.hxx" | #include "event/Loop.hxx" | ||||||
| #include "Log.hxx" | #include "Log.hxx" | ||||||
|  |  | ||||||
| @@ -33,18 +33,23 @@ static constexpr unsigned IN_MASK = | |||||||
| 	IN_ATTRIB|IN_CLOSE_WRITE|IN_CREATE|IN_DELETE|IN_DELETE_SELF | 	IN_ATTRIB|IN_CLOSE_WRITE|IN_CREATE|IN_DELETE|IN_DELETE_SELF | ||||||
| 	|IN_MOVE|IN_MOVE_SELF; | 	|IN_MOVE|IN_MOVE_SELF; | ||||||
|  |  | ||||||
| static void | struct Instance final : InotifyHandler { | ||||||
| my_inotify_callback([[maybe_unused]] int wd, unsigned mask, |  | ||||||
| 		    const char *name, [[maybe_unused]] void *ctx) |  | ||||||
| { |  | ||||||
| 	printf("mask=0x%x name='%s'\n", mask, name); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| struct Instance { |  | ||||||
| 	EventLoop event_loop; | 	EventLoop event_loop; | ||||||
| 	const ShutdownHandler shutdown_handler{event_loop}; | 	const ShutdownHandler shutdown_handler{event_loop}; | ||||||
|  |  | ||||||
| 	InotifySource source{event_loop, my_inotify_callback, nullptr}; | 	InotifyEvent inotify_event{event_loop, *this}; | ||||||
|  |  | ||||||
|  | 	std::exception_ptr error; | ||||||
|  |  | ||||||
|  | 	/* virtual methods from class InotifyHandler */ | ||||||
|  | 	void OnInotify(int, unsigned mask, const char *name) override { | ||||||
|  | 		printf("mask=0x%x name='%s'\n", mask, name); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	void OnInotifyError(std::exception_ptr _error) noexcept override { | ||||||
|  | 		error = std::move(_error); | ||||||
|  | 		event_loop.Break(); | ||||||
|  | 	} | ||||||
| }; | }; | ||||||
|  |  | ||||||
| int main(int argc, char **argv) | int main(int argc, char **argv) | ||||||
| @@ -60,10 +65,13 @@ try { | |||||||
|  |  | ||||||
| 	Instance instance; | 	Instance instance; | ||||||
|  |  | ||||||
| 	instance.source.Add(path, IN_MASK); | 	instance.inotify_event.AddWatch(path, IN_MASK); | ||||||
|  |  | ||||||
| 	instance.event_loop.Run(); | 	instance.event_loop.Run(); | ||||||
|  |  | ||||||
|  | 	if (instance.error) | ||||||
|  | 		std::rethrow_exception(instance.error); | ||||||
|  |  | ||||||
| 	return EXIT_SUCCESS; | 	return EXIT_SUCCESS; | ||||||
| } catch (...) { | } catch (...) { | ||||||
| 	LogError(std::current_exception()); | 	LogError(std::current_exception()); | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user
	 Max Kellermann
					Max Kellermann