#pragma once #if defined(_WIN32) #include "wtr/watcher.hpp" #include #include #include #include #include namespace detail::wtr::watcher::adapter { namespace { /* Hold resources necessary to recieve and send filesystem events. */ class watch_event_proxy { public: /* Timeout for the completion port to wait for events, giving us some room to check if we're still alive. */ static constexpr auto delay_ms = static_cast(std::chrono::milliseconds(16).count()); /* I think the default page size in Windows is 64kb, so 65536 might also work well. */ static constexpr auto event_buf_len_max = 8192; bool is_valid = true; std::filesystem::path path = {}; wchar_t path_name[256] = {L""}; HANDLE path_handle = nullptr; HANDLE event_completion_token = nullptr; HANDLE event_token = CreateEventW(nullptr, true, false, nullptr); OVERLAPPED event_overlap = {}; FILE_NOTIFY_INFORMATION event_buf[event_buf_len_max] = {0}; DWORD event_buf_len_ready = 0; watch_event_proxy(std::filesystem::path const& path) noexcept : path{path} { memcpy(path_name, path.c_str(), path.string().size()); this->path_handle = CreateFileW( path.c_str(), FILE_LIST_DIRECTORY, FILE_SHARE_READ | FILE_SHARE_WRITE | FILE_SHARE_DELETE, nullptr, OPEN_EXISTING, FILE_FLAG_BACKUP_SEMANTICS | FILE_FLAG_OVERLAPPED, nullptr); if (path_handle) this->event_completion_token = CreateIoCompletionPort(INVALID_HANDLE_VALUE, nullptr, 0, 0); if (event_completion_token) this->is_valid = CreateIoCompletionPort( path_handle, event_completion_token, (ULONG_PTR)path_handle, 1) && ResetEvent(event_token); } ~watch_event_proxy() noexcept { if (event_token) CloseHandle(event_token); if (event_completion_token) CloseHandle(event_completion_token); } }; inline auto has_event(watch_event_proxy& w) noexcept -> bool { return w.event_buf_len_ready != 0; } inline auto do_event_recv( watch_event_proxy& w, ::wtr::watcher::event::callback const& callback) noexcept -> bool { w.event_buf_len_ready = 0; DWORD bytes_returned = 0; memset(&w.event_overlap, 0, sizeof(OVERLAPPED)); if (! w.is_valid) return false; auto read_ok = ReadDirectoryChangesW( w.path_handle, w.event_buf, w.event_buf_len_max, true, FILE_NOTIFY_CHANGE_SECURITY | FILE_NOTIFY_CHANGE_CREATION | FILE_NOTIFY_CHANGE_LAST_WRITE | FILE_NOTIFY_CHANGE_SIZE | FILE_NOTIFY_CHANGE_ATTRIBUTES | FILE_NOTIFY_CHANGE_DIR_NAME | FILE_NOTIFY_CHANGE_FILE_NAME, &bytes_returned, &w.event_overlap, nullptr); if (read_ok) { w.event_buf_len_ready = bytes_returned > 0 ? bytes_returned : 0; return true; } else { switch (GetLastError()) { case ERROR_IO_PENDING : w.event_buf_len_ready = 0; w.is_valid = false; callback( {"e/sys/read/pending", ::wtr::event::effect_type::other, ::wtr::event::path_type::watcher}); break; default : callback( {"e/sys/read", ::wtr::event::effect_type::other, ::wtr::event::path_type::watcher}); break; } return false; } } inline auto do_event_send( watch_event_proxy& w, ::wtr::watcher::event::callback const& callback) noexcept -> bool { using namespace ::wtr::watcher; struct RenameEventTracker { std::filesystem::path path_name; enum event::effect_type effect_type; enum event::path_type path_type; bool set = false; }; FILE_NOTIFY_INFORMATION* buf = w.event_buf; /* Rename events on Windows send two individual messages that correspond with the old data and the new data. While it is believed that these are sent sequentially with the old data first, there is no guarantee in the documentation. These trackers are used to ensure all data is available for the callback regardless of the order. */ RenameEventTracker old_tracker; RenameEventTracker new_tracker; auto const trigger_rename_callback = [&]() { auto renamed_from = event{ old_tracker.path_name, old_tracker.effect_type, old_tracker.path_type}; auto renamed_to = event{ new_tracker.path_name, new_tracker.effect_type, new_tracker.path_type}; callback({renamed_from, std::move(renamed_to)}); /* Reset for the possibility of more events. */ old_tracker = {}; new_tracker = {}; }; if (! w.is_valid) return false; while (buf + sizeof(FILE_NOTIFY_INFORMATION) <= buf + w.event_buf_len_ready) { if (buf->FileNameLength % 2 == 0) { auto path_name = w.path / std::wstring{buf->FileName, buf->FileNameLength / 2}; auto effect_type = [&buf]() noexcept { switch (buf->Action) { case FILE_ACTION_MODIFIED : return event::effect_type::modify; case FILE_ACTION_ADDED : return event::effect_type::create; case FILE_ACTION_REMOVED : return event::effect_type::destroy; case FILE_ACTION_RENAMED_OLD_NAME : return event::effect_type::rename; case FILE_ACTION_RENAMED_NEW_NAME : return event::effect_type::rename; default : return event::effect_type::other; } }(); auto path_type = [&path_name]() { try { return std::filesystem::is_directory(path_name) ? event::path_type::dir : event::path_type::file; } catch (...) { return event::path_type::other; } }(); if (buf->Action == FILE_ACTION_RENAMED_OLD_NAME) { old_tracker.path_name = path_name; old_tracker.effect_type = effect_type; old_tracker.path_type = path_type; old_tracker.set = true; if (new_tracker.set) trigger_rename_callback(); } else if (buf->Action == FILE_ACTION_RENAMED_NEW_NAME) { new_tracker.path_name = path_name; new_tracker.effect_type = effect_type; new_tracker.path_type = path_type; new_tracker.set = true; if (old_tracker.set) trigger_rename_callback(); } else { callback({path_name, effect_type, path_type}); } if (buf->NextEntryOffset == 0) break; else buf = (FILE_NOTIFY_INFORMATION*)((uint8_t*)buf + buf->NextEntryOffset); } } return true; } } // namespace /* while living watch for events return when dead true if no errors */ inline auto watch( std::filesystem::path const& path, ::wtr::watcher::event::callback const& callback, semabin const& living) noexcept -> bool { using namespace ::wtr::watcher; auto w = watch_event_proxy{path}; if (! w.is_valid) return false; do_event_recv(w, callback); while (w.is_valid && has_event(w)) do_event_send(w, callback); while (living.state() == semabin::state::pending) { ULONG_PTR completion_key = 0; LPOVERLAPPED overlap = nullptr; bool complete = GetQueuedCompletionStatus( w.event_completion_token, &w.event_buf_len_ready, &completion_key, &overlap, w.delay_ms); if (complete && overlap) { while (w.is_valid && has_event(w)) { do_event_send(w, callback); do_event_recv(w, callback); } } } return true; } } /* namespace detail::wtr::watcher::adapter */ #endif