| #ifndef FSMONITOR_DAEMON_H |
| #define FSMONITOR_DAEMON_H |
| |
| #ifdef HAVE_FSMONITOR_DAEMON_BACKEND |
| |
| #include "dir.h" |
| #include "run-command.h" |
| #include "simple-ipc.h" |
| #include "thread-utils.h" |
| #include "fsmonitor-path-utils.h" |
| |
| struct fsmonitor_batch; |
| struct fsmonitor_token_data; |
| |
| /* |
| * Create a new batch of path(s). The returned batch is considered |
| * private and not linked into the fsmonitor daemon state. The caller |
| * should fill this batch with one or more paths and then publish it. |
| */ |
| struct fsmonitor_batch *fsmonitor_batch__new(void); |
| |
| /* |
| * Free the list of batches starting with this one. |
| */ |
| void fsmonitor_batch__free_list(struct fsmonitor_batch *batch); |
| |
| /* |
| * Add this path to this batch of modified files. |
| * |
| * The batch should be private and NOT (yet) linked into the fsmonitor |
| * daemon state and therefore not yet visible to worker threads and so |
| * no locking is required. |
| */ |
| void fsmonitor_batch__add_path(struct fsmonitor_batch *batch, const char *path); |
| |
| struct fsm_listen_data; /* opaque platform-specific data for listener thread */ |
| struct fsm_health_data; /* opaque platform-specific data for health thread */ |
| |
| struct fsmonitor_daemon_state { |
| pthread_t listener_thread; |
| pthread_t health_thread; |
| pthread_mutex_t main_lock; |
| |
| struct strbuf path_worktree_watch; |
| struct strbuf path_gitdir_watch; |
| struct alias_info alias; |
| int nr_paths_watching; |
| |
| struct fsmonitor_token_data *current_token_data; |
| |
| struct strbuf path_cookie_prefix; |
| pthread_cond_t cookies_cond; |
| int cookie_seq; |
| struct hashmap cookies; |
| |
| int listen_error_code; |
| int health_error_code; |
| struct fsm_listen_data *listen_data; |
| struct fsm_health_data *health_data; |
| |
| struct ipc_server_data *ipc_server_data; |
| struct strbuf path_ipc; |
| |
| }; |
| |
| /* |
| * Pathname classifications. |
| * |
| * The daemon classifies the pathnames that it receives from file |
| * system notification events into the following categories and uses |
| * that to decide whether clients are told about them. (And to watch |
| * for file system synchronization events.) |
| * |
| * The daemon only collects and reports on the set of modified paths |
| * within the working directory (proper). |
| * |
| * The client should only care about paths within the working |
| * directory proper (inside the working directory and not ".git" nor |
| * inside of ".git/"). That is, the client has read the index and is |
| * asking for a list of any paths in the working directory that have |
| * been modified since the last token. The client does not care about |
| * file system changes within the ".git/" directory (such as new loose |
| * objects or packfiles). So the client will only receive paths that |
| * are classified as IS_WORKDIR_PATH. |
| * |
| * Note that ".git" is usually a directory and is therefore inside |
| * the cone of the FS watch that we have on the working directory root, |
| * so we will also get FS events for disk activity on and within ".git/" |
| * that we need to respond to or filter from the client. |
| * |
| * But Git also allows ".git" to be a *file* that points to a GITDIR |
| * outside of the working directory. When this happens, we need to |
| * create FS watches on both the working directory root *and* on the |
| * (external) GITDIR root. (The latter is required because we put |
| * cookie files inside it and use them to sync with the FS event |
| * stream.) |
| * |
| * Note that in the context of this discussion, I'm using "GITDIR" |
| * to only mean an external GITDIR referenced by a ".git" file. |
| * |
| * The platform FS event backends will receive watch-specific |
| * relative paths (except for those OSes that always emit absolute |
| * paths). We use the following enum and routines to classify each |
| * path so that we know how to handle it. There is a slight asymmetry |
| * here because ".git/" is inside the working directory and the |
| * (external) GITDIR is not, and therefore how we handle events may |
| * vary slightly, so I have different enums for "IS...DOT_GIT..." and |
| * "IS...GITDIR...". |
| * |
| * The daemon uses the IS_DOT_GIT and IS_GITDIR internally to mean the |
| * exact ".git" file/directory or GITDIR directory. If the daemon |
| * receives a delete event for either of these paths, it will |
| * automatically shutdown, for example. |
| * |
| * Note that the daemon DOES NOT explicitly watch nor special case the |
| * index. The daemon does not read the index nor have any internal |
| * index-relative state, so there are no "IS...INDEX..." enum values. |
| */ |
| enum fsmonitor_path_type { |
| IS_WORKDIR_PATH = 0, |
| |
| IS_DOT_GIT, |
| IS_INSIDE_DOT_GIT, |
| IS_INSIDE_DOT_GIT_WITH_COOKIE_PREFIX, |
| |
| IS_GITDIR, |
| IS_INSIDE_GITDIR, |
| IS_INSIDE_GITDIR_WITH_COOKIE_PREFIX, |
| |
| IS_OUTSIDE_CONE, |
| }; |
| |
| /* |
| * Classify a pathname relative to the root of the working directory. |
| */ |
| enum fsmonitor_path_type fsmonitor_classify_path_workdir_relative( |
| const char *relative_path); |
| |
| /* |
| * Classify a pathname relative to a <gitdir> that is external to the |
| * worktree directory. |
| */ |
| enum fsmonitor_path_type fsmonitor_classify_path_gitdir_relative( |
| const char *relative_path); |
| |
| /* |
| * Classify an absolute pathname received from a filesystem event. |
| */ |
| enum fsmonitor_path_type fsmonitor_classify_path_absolute( |
| struct fsmonitor_daemon_state *state, |
| const char *path); |
| |
| /* |
| * Prepend the this batch of path(s) onto the list of batches associated |
| * with the current token. This makes the batch visible to worker threads. |
| * |
| * The caller no longer owns the batch and must not free it. |
| * |
| * Wake up the client threads waiting on these cookies. |
| */ |
| void fsmonitor_publish(struct fsmonitor_daemon_state *state, |
| struct fsmonitor_batch *batch, |
| const struct string_list *cookie_names); |
| |
| /* |
| * If the platform-specific layer loses sync with the filesystem, |
| * it should call this to invalidate cached data and abort waiting |
| * threads. |
| */ |
| void fsmonitor_force_resync(struct fsmonitor_daemon_state *state); |
| |
| #endif /* HAVE_FSMONITOR_DAEMON_BACKEND */ |
| #endif /* FSMONITOR_DAEMON_H */ |