d06055501b
Create another thread to watch over the daemon process and automatically shut it down if necessary. This commit creates the basic framework for a "health" thread to monitor the daemon and/or the file system. Later commits will add platform-specific code to do the actual work. The "health" thread is intended to monitor conditions that would be difficult to track inside the IPC thread pool and/or the file system listener threads. For example, when there are file system events outside of the watched worktree root or if we want to have an idle-timeout auto-shutdown feature. This commit creates the health thread itself, defines the thread-proc and sets up the thread's event loop. It integrates this new thread into the existing IPC and Listener thread models. This commit defines the API to the platform-specific code where all of the monitoring will actually happen. The platform-specific code for MacOS is just stubs. Meaning that the health thread will immediately exit on MacOS, but that is OK and expected. Future work can define MacOS-specific monitoring. The platform-specific code for Windows sets up enough of the WaitForMultipleObjects() machinery to watch for system and/or custom events. Currently, the set of wait handles only includes our custom shutdown event (sent from our other theads). Later commits in this series will extend the set of wait handles to monitor other conditions. Signed-off-by: Jeff Hostetler <jeffhost@microsoft.com> Signed-off-by: Junio C Hamano <gitster@pobox.com>
73 lines
1.5 KiB
C
73 lines
1.5 KiB
C
#include "cache.h"
|
|
#include "config.h"
|
|
#include "fsmonitor.h"
|
|
#include "fsm-health.h"
|
|
#include "fsmonitor--daemon.h"
|
|
|
|
struct fsm_health_data
|
|
{
|
|
HANDLE hEventShutdown;
|
|
|
|
HANDLE hHandles[1]; /* the array does not own these handles */
|
|
#define HEALTH_SHUTDOWN 0
|
|
int nr_handles; /* number of active event handles */
|
|
};
|
|
|
|
int fsm_health__ctor(struct fsmonitor_daemon_state *state)
|
|
{
|
|
struct fsm_health_data *data;
|
|
|
|
CALLOC_ARRAY(data, 1);
|
|
|
|
data->hEventShutdown = CreateEvent(NULL, TRUE, FALSE, NULL);
|
|
|
|
data->hHandles[HEALTH_SHUTDOWN] = data->hEventShutdown;
|
|
data->nr_handles++;
|
|
|
|
state->health_data = data;
|
|
return 0;
|
|
}
|
|
|
|
void fsm_health__dtor(struct fsmonitor_daemon_state *state)
|
|
{
|
|
struct fsm_health_data *data;
|
|
|
|
if (!state || !state->health_data)
|
|
return;
|
|
|
|
data = state->health_data;
|
|
|
|
CloseHandle(data->hEventShutdown);
|
|
|
|
FREE_AND_NULL(state->health_data);
|
|
}
|
|
|
|
void fsm_health__loop(struct fsmonitor_daemon_state *state)
|
|
{
|
|
struct fsm_health_data *data = state->health_data;
|
|
|
|
for (;;) {
|
|
DWORD dwWait = WaitForMultipleObjects(data->nr_handles,
|
|
data->hHandles,
|
|
FALSE, INFINITE);
|
|
|
|
if (dwWait == WAIT_OBJECT_0 + HEALTH_SHUTDOWN)
|
|
goto clean_shutdown;
|
|
|
|
error(_("health thread wait failed [GLE %ld]"),
|
|
GetLastError());
|
|
goto force_error_stop;
|
|
}
|
|
|
|
force_error_stop:
|
|
state->health_error_code = -1;
|
|
ipc_server_stop_async(state->ipc_server_data);
|
|
clean_shutdown:
|
|
return;
|
|
}
|
|
|
|
void fsm_health__stop_async(struct fsmonitor_daemon_state *state)
|
|
{
|
|
SetEvent(state->health_data->hHandles[HEALTH_SHUTDOWN]);
|
|
}
|