mirror of
https://gitea.wildfiregames.com/0ad/0ad
synced 2026-06-16 05:13:58 -07:00
Make include-what-you-use happy with some files in source/lib and fix what needs to be fixed. Ref: #8086 Signed-off-by: Ralph Sennhauser <ralph.sennhauser@gmail.com>
285 lines
7.2 KiB
C++
285 lines
7.2 KiB
C++
/* Copyright (C) 2025 Wildfire Games.
|
|
*
|
|
* Permission is hereby granted, free of charge, to any person obtaining
|
|
* a copy of this software and associated documentation files (the
|
|
* "Software"), to deal in the Software without restriction, including
|
|
* without limitation the rights to use, copy, modify, merge, publish,
|
|
* distribute, sublicense, and/or sell copies of the Software, and to
|
|
* permit persons to whom the Software is furnished to do so, subject to
|
|
* the following conditions:
|
|
*
|
|
* The above copyright notice and this permission notice shall be included
|
|
* in all copies or substantial portions of the Software.
|
|
*
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
|
* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
|
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
|
|
* IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY
|
|
* CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,
|
|
* TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE
|
|
* SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
|
*/
|
|
|
|
#include "precompiled.h"
|
|
|
|
#include "lib/sysdep/dir_watch.h"
|
|
|
|
#include "lib/debug.h"
|
|
#include "lib/os_path.h"
|
|
#include "lib/path.h"
|
|
#include "lib/posix/posix_pthread.h"
|
|
#include "lib/posix/posix_types.h"
|
|
#include "lib/status.h"
|
|
#include "ps/CLogger.h"
|
|
|
|
#include <cerrno>
|
|
#include <climits>
|
|
#include <cstddef>
|
|
#include <cstdint>
|
|
#include <cstdlib>
|
|
#include <map>
|
|
#include <memory>
|
|
#include <string>
|
|
#include <sys/inotify.h>
|
|
#include <sys/select.h>
|
|
#include <utility>
|
|
#include <vector>
|
|
|
|
struct NotificationEvent
|
|
{
|
|
std::string filename;
|
|
uint32_t code;
|
|
int wd;
|
|
};
|
|
|
|
// To avoid deadlocks and slow synchronous reads, it's necessary to use a
|
|
// separate thread for reading events from inotify.
|
|
// So we just spawn a thread to push events into this list, then swap it out
|
|
// when someone calls dir_watch_Poll.
|
|
// (We assume STL memory allocation is thread-safe.)
|
|
static std::vector<NotificationEvent> g_notifications;
|
|
static pthread_t g_event_loop_thread;
|
|
|
|
// Mutex must wrap all accesses of g_notifications
|
|
// while the event loop thread is running
|
|
static pthread_mutex_t g_mutex = PTHREAD_MUTEX_INITIALIZER;
|
|
|
|
// trool; -1 = init failed and all operations will be aborted silently.
|
|
// this is so that each dir_* call doesn't complain if the system's
|
|
// inotify is broken or unavailable.
|
|
static int initialized = 0;
|
|
|
|
// Inotify file descriptor
|
|
static int inotifyfd;
|
|
|
|
// With inotify, using a map seems to be a good alternative to FAM's userdata
|
|
typedef std::map<int, PDirWatch> DirWatchMap;
|
|
static DirWatchMap g_paths;
|
|
|
|
struct DirWatch
|
|
{
|
|
DirWatch()
|
|
: reqnum(-1)
|
|
{
|
|
}
|
|
|
|
~DirWatch()
|
|
{
|
|
ENSURE(initialized > 0);
|
|
inotify_rm_watch(inotifyfd, reqnum);
|
|
}
|
|
|
|
OsPath path;
|
|
int reqnum;
|
|
};
|
|
|
|
// for atexit
|
|
static void inotify_deinit()
|
|
{
|
|
close(inotifyfd);
|
|
|
|
#ifdef __BIONIC__
|
|
#warning TODO: pthread_cancel not supported on Bionic
|
|
#else
|
|
pthread_cancel(g_event_loop_thread);
|
|
#endif
|
|
// NOTE: POSIX threads are (by default) only cancellable inside particular
|
|
// functions (like 'select'), so this should safely terminate while it's
|
|
// in select/etc (and won't e.g. cancel while it's holding the
|
|
// mutex)
|
|
|
|
// Wait for the thread to finish
|
|
pthread_join(g_event_loop_thread, NULL);
|
|
}
|
|
|
|
static void inotify_event_loop_process_events()
|
|
{
|
|
// Buffer for reading the events.
|
|
// Need to be careful about overflow here.
|
|
char buffer[65535] = {0};
|
|
|
|
// Event iterator
|
|
ssize_t buffer_i = 0;
|
|
|
|
// Total size of all the events
|
|
ssize_t r;
|
|
|
|
// Size & struct for the current event
|
|
size_t event_size;
|
|
struct inotify_event *pevent;
|
|
|
|
r = read(inotifyfd, buffer, 65535);
|
|
if(r <= 0)
|
|
return;
|
|
|
|
while(buffer_i < r)
|
|
{
|
|
NotificationEvent ne;
|
|
pevent = (struct inotify_event *) &buffer[buffer_i];
|
|
|
|
event_size = offsetof(struct inotify_event, name) + pevent->len;
|
|
ne.wd = pevent->wd;
|
|
ne.filename = pevent->name;
|
|
ne.code = pevent->mask;
|
|
|
|
pthread_mutex_lock(&g_mutex);
|
|
g_notifications.push_back(ne);
|
|
pthread_mutex_unlock(&g_mutex);
|
|
|
|
buffer_i += event_size;
|
|
}
|
|
}
|
|
|
|
static void* inotify_event_loop(void*)
|
|
{
|
|
while(true)
|
|
{
|
|
fd_set fdrset;
|
|
FD_ZERO(&fdrset);
|
|
FD_SET(inotifyfd, &fdrset);
|
|
errno = 0;
|
|
// Block with select until there's events waiting
|
|
while(select(inotifyfd+1, &fdrset, NULL, NULL, NULL) < 0)
|
|
{
|
|
if(errno == EINTR)
|
|
{
|
|
// interrupted - try again
|
|
FD_ZERO(&fdrset);
|
|
FD_SET(inotifyfd, &fdrset);
|
|
}
|
|
else if(errno == EBADF)
|
|
{
|
|
// probably just lost the connection to inotify - kill the thread
|
|
debug_printf("inotify_event_loop: Invalid file descriptor inotifyfd=%d\n", inotifyfd);
|
|
return NULL;
|
|
}
|
|
else
|
|
{
|
|
// oops
|
|
debug_printf("inotify_event_loop: select error errno=%d\n", errno);
|
|
return NULL;
|
|
}
|
|
errno = 0;
|
|
}
|
|
if(FD_ISSET(inotifyfd, &fdrset))
|
|
inotify_event_loop_process_events();
|
|
}
|
|
}
|
|
|
|
Status dir_watch_Add(const OsPath& path, PDirWatch& dirWatch)
|
|
{
|
|
char resolved[PATH_MAX + 1];
|
|
|
|
// init already failed; don't try again or complain
|
|
if(initialized == -1)
|
|
return ERR::FAIL; // NOWARN
|
|
|
|
if(!initialized)
|
|
{
|
|
errno = 0;
|
|
if((inotifyfd = inotify_init()) < 0)
|
|
{
|
|
// Check for error ?
|
|
int err = errno;
|
|
initialized = -1;
|
|
LOGERROR("Error initializing inotify file descriptor; hotloading will be disabled, errno=%d", err);
|
|
errno = err;
|
|
return StatusFromErrno(); // NOWARN
|
|
}
|
|
|
|
errno = 0;
|
|
int ret = pthread_create(&g_event_loop_thread, NULL, &inotify_event_loop, NULL);
|
|
if (ret != 0)
|
|
{
|
|
initialized = -1;
|
|
LOGERROR("Error creating inotify event loop thread; hotloading will be disabled, err=%d", ret);
|
|
errno = ret;
|
|
return StatusFromErrno(); // NOWARN
|
|
}
|
|
|
|
initialized = 1;
|
|
atexit(inotify_deinit);
|
|
}
|
|
|
|
PDirWatch tmpDirWatch(new DirWatch);
|
|
errno = 0;
|
|
int wd = inotify_add_watch(inotifyfd, realpath(OsString(path).c_str(), resolved), IN_CREATE | IN_DELETE | IN_CLOSE_WRITE);
|
|
if (wd < 0)
|
|
WARN_RETURN(StatusFromErrno());
|
|
|
|
dirWatch.swap(tmpDirWatch);
|
|
dirWatch->path = path;
|
|
dirWatch->reqnum = wd;
|
|
g_paths.insert(std::make_pair(wd, dirWatch));
|
|
|
|
return INFO::OK;
|
|
}
|
|
|
|
Status dir_watch_Poll(DirWatchNotifications& notifications)
|
|
{
|
|
if(initialized == -1)
|
|
return ERR::FAIL; // NOWARN
|
|
if(!initialized) // XXX Fix Atlas instead of suppressing the warning
|
|
return ERR::FAIL; //WARN_RETURN(ERR::LOGIC);
|
|
|
|
std::vector<NotificationEvent> polled_notifications;
|
|
|
|
pthread_mutex_lock(&g_mutex);
|
|
g_notifications.swap(polled_notifications);
|
|
pthread_mutex_unlock(&g_mutex);
|
|
|
|
for(size_t i = 0; i < polled_notifications.size(); ++i)
|
|
{
|
|
DirWatchNotification::EType type;
|
|
// TODO: code is actually a bitmask, so this is slightly incorrect
|
|
switch(polled_notifications[i].code)
|
|
{
|
|
case IN_CLOSE_WRITE:
|
|
type = DirWatchNotification::Changed;
|
|
break;
|
|
case IN_CREATE:
|
|
type = DirWatchNotification::Created;
|
|
break;
|
|
case IN_DELETE:
|
|
type = DirWatchNotification::Deleted;
|
|
break;
|
|
default:
|
|
continue;
|
|
}
|
|
|
|
DirWatchMap::iterator it = g_paths.find(polled_notifications[i].wd);
|
|
if(it != g_paths.end())
|
|
{
|
|
OsPath filename = Path(OsString(it->second->path).append(polled_notifications[i].filename));
|
|
notifications.push_back(DirWatchNotification(filename, type));
|
|
}
|
|
else
|
|
{
|
|
debug_printf("dir_watch_Poll: Notification with invalid watch descriptor wd=%d\n", polled_notifications[i].wd);
|
|
}
|
|
}
|
|
|
|
// nothing new; try again later
|
|
return INFO::OK;
|
|
}
|
|
|