2017-11-18 17:09:20 +01:00
|
|
|
/* SPDX-License-Identifier: LGPL-2.1+ */
|
2015-05-04 22:12:46 +02:00
|
|
|
|
|
|
|
#include <errno.h>
|
2015-11-30 21:43:37 +01:00
|
|
|
#include <fcntl.h>
|
2015-11-16 22:09:36 +01:00
|
|
|
#include <stdio.h>
|
|
|
|
#include <string.h>
|
2015-05-04 22:12:46 +02:00
|
|
|
#include <sys/file.h>
|
2015-11-30 21:43:37 +01:00
|
|
|
#include <sys/stat.h>
|
2015-05-04 22:12:46 +02:00
|
|
|
|
2015-10-27 03:01:06 +01:00
|
|
|
#include "alloc-util.h"
|
2015-10-25 13:14:12 +01:00
|
|
|
#include "fd-util.h"
|
2015-10-26 21:16:26 +01:00
|
|
|
#include "fs-util.h"
|
2015-10-25 13:14:12 +01:00
|
|
|
#include "lockfile-util.h"
|
2015-11-30 21:43:37 +01:00
|
|
|
#include "macro.h"
|
2019-01-11 17:30:32 +01:00
|
|
|
#include "missing_fcntl.h"
|
2015-12-01 23:22:03 +01:00
|
|
|
#include "path-util.h"
|
2015-05-04 22:12:46 +02:00
|
|
|
|
|
|
|
int make_lock_file(const char *p, int operation, LockFile *ret) {
|
|
|
|
_cleanup_close_ int fd = -1;
|
|
|
|
_cleanup_free_ char *t = NULL;
|
|
|
|
int r;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* We use UNPOSIX locks if they are available. They have nice
|
|
|
|
* semantics, and are mostly compatible with NFS. However,
|
|
|
|
* they are only available on new kernels. When we detect we
|
|
|
|
* are running on an older kernel, then we fall back to good
|
|
|
|
* old BSD locks. They also have nice semantics, but are
|
|
|
|
* slightly problematic on NFS, where they are upgraded to
|
|
|
|
* POSIX locks, even though locally they are orthogonal to
|
|
|
|
* POSIX locks.
|
|
|
|
*/
|
|
|
|
|
|
|
|
t = strdup(p);
|
|
|
|
if (!t)
|
|
|
|
return -ENOMEM;
|
|
|
|
|
|
|
|
for (;;) {
|
|
|
|
struct flock fl = {
|
|
|
|
.l_type = (operation & ~LOCK_NB) == LOCK_EX ? F_WRLCK : F_RDLCK,
|
|
|
|
.l_whence = SEEK_SET,
|
|
|
|
};
|
|
|
|
struct stat st;
|
|
|
|
|
|
|
|
fd = open(p, O_CREAT|O_RDWR|O_NOFOLLOW|O_CLOEXEC|O_NOCTTY, 0600);
|
|
|
|
if (fd < 0)
|
|
|
|
return -errno;
|
|
|
|
|
|
|
|
r = fcntl(fd, (operation & LOCK_NB) ? F_OFD_SETLK : F_OFD_SETLKW, &fl);
|
|
|
|
if (r < 0) {
|
|
|
|
|
|
|
|
/* If the kernel is too old, use good old BSD locks */
|
|
|
|
if (errno == EINVAL)
|
|
|
|
r = flock(fd, operation);
|
|
|
|
|
|
|
|
if (r < 0)
|
|
|
|
return errno == EAGAIN ? -EBUSY : -errno;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* If we acquired the lock, let's check if the file
|
|
|
|
* still exists in the file system. If not, then the
|
|
|
|
* previous exclusive owner removed it and then closed
|
|
|
|
* it. In such a case our acquired lock is worthless,
|
|
|
|
* hence try again. */
|
|
|
|
|
|
|
|
r = fstat(fd, &st);
|
|
|
|
if (r < 0)
|
|
|
|
return -errno;
|
|
|
|
if (st.st_nlink > 0)
|
|
|
|
break;
|
|
|
|
|
|
|
|
fd = safe_close(fd);
|
|
|
|
}
|
|
|
|
|
|
|
|
ret->path = t;
|
|
|
|
ret->fd = fd;
|
|
|
|
ret->operation = operation;
|
|
|
|
|
|
|
|
fd = -1;
|
|
|
|
t = NULL;
|
|
|
|
|
|
|
|
return r;
|
|
|
|
}
|
|
|
|
|
|
|
|
int make_lock_file_for(const char *p, int operation, LockFile *ret) {
|
|
|
|
const char *fn;
|
|
|
|
char *t;
|
|
|
|
|
|
|
|
assert(p);
|
|
|
|
assert(ret);
|
|
|
|
|
|
|
|
fn = basename(p);
|
|
|
|
if (!filename_is_valid(fn))
|
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
t = newa(char, strlen(p) + 2 + 4 + 1);
|
|
|
|
stpcpy(stpcpy(stpcpy(mempcpy(t, p, fn - p), ".#"), fn), ".lck");
|
|
|
|
|
|
|
|
return make_lock_file(t, operation, ret);
|
|
|
|
}
|
|
|
|
|
|
|
|
void release_lock_file(LockFile *f) {
|
|
|
|
int r;
|
|
|
|
|
|
|
|
if (!f)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (f->path) {
|
|
|
|
|
|
|
|
/* If we are the exclusive owner we can safely delete
|
|
|
|
* the lock file itself. If we are not the exclusive
|
|
|
|
* owner, we can try becoming it. */
|
|
|
|
|
|
|
|
if (f->fd >= 0 &&
|
|
|
|
(f->operation & ~LOCK_NB) == LOCK_SH) {
|
|
|
|
static const struct flock fl = {
|
|
|
|
.l_type = F_WRLCK,
|
|
|
|
.l_whence = SEEK_SET,
|
|
|
|
};
|
|
|
|
|
|
|
|
r = fcntl(f->fd, F_OFD_SETLK, &fl);
|
|
|
|
if (r < 0 && errno == EINVAL)
|
|
|
|
r = flock(f->fd, LOCK_EX|LOCK_NB);
|
|
|
|
|
|
|
|
if (r >= 0)
|
|
|
|
f->operation = LOCK_EX|LOCK_NB;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((f->operation & ~LOCK_NB) == LOCK_EX)
|
|
|
|
unlink_noerrno(f->path);
|
|
|
|
|
2015-09-08 18:43:11 +02:00
|
|
|
f->path = mfree(f->path);
|
2015-05-04 22:12:46 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
f->fd = safe_close(f->fd);
|
|
|
|
f->operation = 0;
|
|
|
|
}
|