Merge #18700: Fix locking on WSL using flock instead of fcntl

e8fa0a3d20 Fix WSL file locking by using flock instead of fcntl (Samuel Dobson)

Pull request description:

  Fixes #18622

  A bug in WSL means that fcntl does not exclusively lock files, allowing multiple instances of bitcoin to use the same datadir. If we instead use flock, it works correctly. Passes Travis, but testing on some OS variety would be sensible.

  From what I can tell, flock and fcntl don't work with each other on linux, so it would still be possible to run a node with this code change and a node before it with the same datadir (this isn't true for Mac/FreeBSD). flock also doesn't support NFS on MacOS and linux<2.6.12 while fcntl did. See here for example: https://gavv.github.io/articles/file-locks/

  If changing to flock for all systems is inadvisable, it would also be possible to just detect WSL and use flock when on that platform to avoid the bug.

ACKs for top commit:
  laanwj:
    Code review ACK e8fa0a3d20

Tree-SHA512: ca1009e171970101f1dc2332c5e998717aee00eebc80bb586b826927a74bd0d4c94712e46d1396821bc30533d76deac391b6e1c406c406865661f57fa062c702
This commit is contained in:
Wladimir J. van der Laan 2020-05-28 17:25:45 +02:00
commit ea3e9e0b84
No known key found for this signature in database
GPG key ID: 1E4AED62986CD25D

View file

@ -6,6 +6,9 @@
#ifndef WIN32 #ifndef WIN32
#include <fcntl.h> #include <fcntl.h>
#include <string>
#include <sys/file.h>
#include <sys/utsname.h>
#else #else
#ifndef NOMINMAX #ifndef NOMINMAX
#define NOMINMAX #define NOMINMAX
@ -47,20 +50,38 @@ FileLock::~FileLock()
} }
} }
static bool IsWSL()
{
struct utsname uname_data;
return uname(&uname_data) == 0 && std::string(uname_data.version).find("Microsoft") != std::string::npos;
}
bool FileLock::TryLock() bool FileLock::TryLock()
{ {
if (fd == -1) { if (fd == -1) {
return false; return false;
} }
struct flock lock;
lock.l_type = F_WRLCK; // Exclusive file locking is broken on WSL using fcntl (issue #18622)
lock.l_whence = SEEK_SET; // This workaround can be removed once the bug on WSL is fixed
lock.l_start = 0; static const bool is_wsl = IsWSL();
lock.l_len = 0; if (is_wsl) {
if (fcntl(fd, F_SETLK, &lock) == -1) { if (flock(fd, LOCK_EX | LOCK_NB) == -1) {
reason = GetErrorReason(); reason = GetErrorReason();
return false; return false;
}
} else {
struct flock lock;
lock.l_type = F_WRLCK;
lock.l_whence = SEEK_SET;
lock.l_start = 0;
lock.l_len = 0;
if (fcntl(fd, F_SETLK, &lock) == -1) {
reason = GetErrorReason();
return false;
}
} }
return true; return true;
} }
#else #else