From: Johannes Schindelin <johannes.schindelin@xxxxxx> This function will be used to make write accesses in trace_write() a bit safer. Note: this patch takes a very different approach for cross-platform support than Git is historically taking: the original approach is to first implement everything on Linux, using the functions available on Linux, and then trying to emulate these functions on platforms that do not have those functions such as Windows. This leads to all kinds of undesirable quirks in Git's source code (and performance characteristics) because of the lack of a proper abstraction layer: instead of declaring functions for the functionality we *actually* need, we abuse POSIX functions to say what we need, even if those functions serve much broader purposes (and do not make at all clear what the caller wants of them). For example, when Git wants to determine whether a path refers to a symbolic link, it calls lstat(). That POSIX function has to be emulated on Windows, painstakingly filling all the information lstat() would, only for the caller to throw away everything except that one bit of information, and all of the time figuring out the mtime/atime/ctime and file size and whatnot was spent in vain. If we were to follow that approach, we would extend the fcntl() emulation in compat/mingw.c after this commit, to support the function added in this commit. But fcntl() allows a lot more versatile file region locking that we actually need, so by necessity the fcntl() emulation would be quite complex: To support the `l_whence = SEEK_CUR` (which we would use, if it did not require so much book-keeping due to our writing between locking and unlocking the file), we would have to call `SetFilePointerEx()` (after obtaining the `HANDLE` on which all Win32 API functions work instead of the integer file descriptors used by all POSIX functions). Multiplying the number of Win32 API round-trips. Of course, we could implement an incomplete emulation of fcntl()'s F_WRLCK, but that would be unsatisfying. An alternative approach would be to use the `flock()` function, whose semantics are much closer to existing Win32 API. But `flock()` is not even a POSIX standard, so we would have to implement emulation of `flock()` for *other* platforms. And it would again be the wrong thing to do, as we would once again fail to have an abstraction that clearly says what *exact*functionality we want to use. This commit expressly tries to set a precedent for a better approach: Let's introduce a proper abstraction: a function that says in its name precisely what Git wants it to do (as opposed to *how* it does it on Linux): lock_or_unlock_fd_for_appending(). The next commit will provide a Windows-specific implementation of this function/functionality. Signed-off-by: Johannes Schindelin <johannes.schindelin@xxxxxx> --- git-compat-util.h | 2 ++ wrapper.c | 16 ++++++++++++++++ 2 files changed, 18 insertions(+) diff --git a/git-compat-util.h b/git-compat-util.h index 9a64998b2..13b83bade 100644 --- a/git-compat-util.h +++ b/git-compat-util.h @@ -1202,6 +1202,8 @@ struct tm *git_gmtime_r(const time_t *, struct tm *); #define getc_unlocked(fh) getc(fh) #endif +extern int lock_or_unlock_fd_for_appending(int fd, int lock_it); + /* * Our code often opens a path to an optional file, to work on its * contents when we can successfully open it. We can ignore a failure diff --git a/wrapper.c b/wrapper.c index e4fa9d84c..5aecbda34 100644 --- a/wrapper.c +++ b/wrapper.c @@ -690,3 +690,19 @@ int xgethostname(char *buf, size_t len) buf[len - 1] = 0; return ret; } + +#ifndef GIT_WINDOWS_NATIVE +int lock_or_unlock_fd_for_appending(int fd, int lock_it) +{ + struct flock flock; + + flock.l_type = lock_it ? F_WRLCK : F_UNLCK; + + /* (un-)lock the whole file */ + flock.l_whence = SEEK_SET; + flock.l_start = 0; + flock.l_len = 0; + + return fcntl(fd, F_SETLKW, &flock); +} +#endif -- gitgitgadget