mirror of
https://github.com/samba-team/samba.git
synced 2025-01-11 05:18:09 +03:00
2ecdbcb22c
This will be generalised to do more I/O-based checks. Signed-off-by: Martin Schwenke <martin@meltin.net> Reviewed-by: Amitay Isaacs <amitay@gmail.com>
455 lines
9.5 KiB
C
455 lines
9.5 KiB
C
/*
|
|
CTDB mutex fcntl lock file helper
|
|
|
|
Copyright (C) Martin Schwenke 2015
|
|
|
|
wait_for_parent() code from ctdb_lock_helper.c:
|
|
|
|
Copyright (C) Amitay Isaacs 2013
|
|
|
|
This program is free software; you can redistribute it and/or modify
|
|
it under the terms of the GNU General Public License as published by
|
|
the Free Software Foundation; either version 3 of the License, or
|
|
(at your option) any later version.
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
GNU General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with this program; if not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#include "replace.h"
|
|
#include "system/filesys.h"
|
|
#include "system/network.h"
|
|
|
|
#include <tevent.h>
|
|
|
|
#include "lib/util/sys_rw.h"
|
|
#include "lib/util/tevent_unix.h"
|
|
#include "lib/util/util.h"
|
|
#include "lib/util/smb_strtox.h"
|
|
|
|
/* protocol.h is just needed for ctdb_sock_addr, which is used in system.h */
|
|
#include "protocol/protocol.h"
|
|
#include "common/system.h"
|
|
|
|
static char progpath[PATH_MAX];
|
|
static char *progname = NULL;
|
|
|
|
static char fcntl_lock(const char *file, int *outfd)
|
|
{
|
|
int fd;
|
|
struct flock lock;
|
|
|
|
fd = open(file, O_RDWR|O_CREAT, 0600);
|
|
if (fd == -1) {
|
|
fprintf(stderr, "%s: Unable to open %s - (%s)\n",
|
|
progname, file, strerror(errno));
|
|
return '3';
|
|
}
|
|
|
|
lock.l_type = F_WRLCK;
|
|
lock.l_whence = SEEK_SET;
|
|
lock.l_start = 0;
|
|
lock.l_len = 1;
|
|
lock.l_pid = 0;
|
|
|
|
if (fcntl(fd, F_SETLK, &lock) != 0) {
|
|
int saved_errno = errno;
|
|
close(fd);
|
|
if (saved_errno == EACCES ||
|
|
saved_errno == EAGAIN) {
|
|
/* Lock contention, fail silently */
|
|
return '1';
|
|
}
|
|
|
|
/* Log an error for any other failure */
|
|
fprintf(stderr,
|
|
"%s: Failed to get lock on '%s' - (%s)\n",
|
|
progname, file, strerror(saved_errno));
|
|
return '3';
|
|
}
|
|
|
|
*outfd = fd;
|
|
|
|
return '0';
|
|
}
|
|
|
|
/*
|
|
* Wait and see if the parent exits
|
|
*/
|
|
|
|
struct wait_for_parent_state {
|
|
struct tevent_context *ev;
|
|
pid_t ppid;
|
|
};
|
|
|
|
static void wait_for_parent_check(struct tevent_req *subreq);
|
|
|
|
static struct tevent_req *wait_for_parent_send(TALLOC_CTX *mem_ctx,
|
|
struct tevent_context *ev,
|
|
pid_t ppid)
|
|
{
|
|
struct tevent_req *req, *subreq;
|
|
struct wait_for_parent_state *state;
|
|
|
|
req = tevent_req_create(mem_ctx, &state, struct wait_for_parent_state);
|
|
if (req == NULL) {
|
|
return NULL;
|
|
}
|
|
|
|
state->ev = ev;
|
|
state->ppid = ppid;
|
|
|
|
if (ppid == 1) {
|
|
fprintf(stderr, "parent == 1\n");
|
|
tevent_req_done(req);
|
|
return tevent_req_post(req, ev);
|
|
}
|
|
|
|
subreq = tevent_wakeup_send(state, ev,
|
|
tevent_timeval_current_ofs(5,0));
|
|
if (tevent_req_nomem(subreq, req)) {
|
|
return tevent_req_post(req, ev);
|
|
}
|
|
tevent_req_set_callback(subreq, wait_for_parent_check, req);
|
|
|
|
return req;
|
|
}
|
|
|
|
static void wait_for_parent_check(struct tevent_req *subreq)
|
|
{
|
|
struct tevent_req *req = tevent_req_callback_data(
|
|
subreq, struct tevent_req);
|
|
struct wait_for_parent_state *state = tevent_req_data(
|
|
req, struct wait_for_parent_state);
|
|
bool status;
|
|
|
|
status = tevent_wakeup_recv(subreq);
|
|
TALLOC_FREE(subreq);
|
|
if (! status) {
|
|
/* Ignore error */
|
|
fprintf(stderr, "%s: tevent_wakeup_recv() failed\n", progname);
|
|
}
|
|
|
|
if (kill(state->ppid, 0) == -1 && errno == ESRCH) {
|
|
fprintf(stderr, "parent gone\n");
|
|
tevent_req_done(req);
|
|
return;
|
|
}
|
|
|
|
subreq = tevent_wakeup_send(state, state->ev,
|
|
tevent_timeval_current_ofs(5,0));
|
|
if (tevent_req_nomem(subreq, req)) {
|
|
return;
|
|
}
|
|
tevent_req_set_callback(subreq, wait_for_parent_check, req);
|
|
}
|
|
|
|
static bool wait_for_parent_recv(struct tevent_req *req, int *perr)
|
|
{
|
|
if (tevent_req_is_unix_error(req, perr)) {
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
/*
|
|
* Perform I/O on lock in a loop - complete when file removed or replaced
|
|
*/
|
|
|
|
struct lock_io_check_state {
|
|
struct tevent_context *ev;
|
|
const char *lock_file;
|
|
ino_t inode;
|
|
unsigned long recheck_interval;
|
|
};
|
|
|
|
static void lock_io_check_loop(struct tevent_req *subreq);
|
|
|
|
static struct tevent_req *lock_io_check_send(TALLOC_CTX *mem_ctx,
|
|
struct tevent_context *ev,
|
|
const char *lock_file,
|
|
int fd,
|
|
unsigned long recheck_interval)
|
|
{
|
|
struct tevent_req *req, *subreq;
|
|
struct lock_io_check_state *state;
|
|
struct stat sb;
|
|
int ret;
|
|
|
|
req = tevent_req_create(mem_ctx, &state, struct lock_io_check_state);
|
|
if (req == NULL) {
|
|
return NULL;
|
|
}
|
|
|
|
state->ev = ev;
|
|
state->lock_file = lock_file;
|
|
state->recheck_interval = recheck_interval;
|
|
|
|
ret = fstat(fd, &sb);
|
|
if (ret != 0) {
|
|
fprintf(stderr,
|
|
"%s: "
|
|
"lock lost - lock file \"%s\" check failed (ret=%d)\n",
|
|
progname,
|
|
state->lock_file,
|
|
errno);
|
|
tevent_req_done(req);
|
|
return tevent_req_post(req, ev);
|
|
}
|
|
state->inode = sb.st_ino;
|
|
|
|
subreq = tevent_wakeup_send(
|
|
state,
|
|
ev,
|
|
tevent_timeval_current_ofs(state->recheck_interval, 0));
|
|
if (tevent_req_nomem(subreq, req)) {
|
|
return tevent_req_post(req, ev);
|
|
}
|
|
tevent_req_set_callback(subreq, lock_io_check_loop, req);
|
|
|
|
return req;
|
|
}
|
|
|
|
static void lock_io_check_loop(struct tevent_req *subreq)
|
|
{
|
|
struct tevent_req *req = tevent_req_callback_data(
|
|
subreq, struct tevent_req);
|
|
struct lock_io_check_state *state = tevent_req_data(
|
|
req, struct lock_io_check_state);
|
|
bool status;
|
|
struct stat sb;
|
|
int ret;
|
|
|
|
status = tevent_wakeup_recv(subreq);
|
|
TALLOC_FREE(subreq);
|
|
if (! status) {
|
|
/* Ignore error */
|
|
fprintf(stderr, "%s: tevent_wakeup_recv() failed\n", progname);
|
|
}
|
|
|
|
ret = stat(state->lock_file, &sb);
|
|
if (ret != 0) {
|
|
fprintf(stderr,
|
|
"%s: "
|
|
"lock lost - lock file \"%s\" check failed (ret=%d)\n",
|
|
progname,
|
|
state->lock_file,
|
|
errno);
|
|
tevent_req_done(req);
|
|
return;
|
|
}
|
|
|
|
if (sb.st_ino != state->inode) {
|
|
fprintf(stderr,
|
|
"%s: lock lost - lock file \"%s\" inode changed\n",
|
|
progname,
|
|
state->lock_file);
|
|
tevent_req_done(req);
|
|
return;
|
|
}
|
|
|
|
subreq = tevent_wakeup_send(
|
|
state,
|
|
state->ev,
|
|
tevent_timeval_current_ofs(state->recheck_interval, 0));
|
|
if (tevent_req_nomem(subreq, req)) {
|
|
return;
|
|
}
|
|
tevent_req_set_callback(subreq, lock_io_check_loop, req);
|
|
}
|
|
|
|
static bool lock_io_check_recv(struct tevent_req *req, int *perr)
|
|
{
|
|
if (tevent_req_is_unix_error(req, perr)) {
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
/*
|
|
* Wait for a reason to exit, indicating that parent has exited or I/O
|
|
* on lock failed
|
|
*/
|
|
|
|
struct wait_for_exit_state {
|
|
};
|
|
|
|
static void wait_for_exit_parent_done(struct tevent_req *subreq);
|
|
static void wait_for_exit_io_check_done(struct tevent_req *subreq);
|
|
|
|
static struct tevent_req *wait_for_exit_send(TALLOC_CTX *mem_ctx,
|
|
struct tevent_context *ev,
|
|
pid_t ppid,
|
|
const char *lock_file,
|
|
int fd,
|
|
unsigned long recheck_interval)
|
|
{
|
|
struct tevent_req *req, *subreq;
|
|
struct wait_for_exit_state *state;
|
|
|
|
req = tevent_req_create(mem_ctx, &state, struct wait_for_exit_state);
|
|
if (req == NULL) {
|
|
return NULL;
|
|
}
|
|
|
|
subreq = wait_for_parent_send(state, ev, ppid);
|
|
if (tevent_req_nomem(subreq, req)) {
|
|
return tevent_req_post(req, ev);
|
|
}
|
|
tevent_req_set_callback(subreq, wait_for_exit_parent_done, req);
|
|
|
|
if (recheck_interval > 0) {
|
|
subreq = lock_io_check_send(state,
|
|
ev,
|
|
lock_file,
|
|
fd,
|
|
recheck_interval);
|
|
if (tevent_req_nomem(subreq, req)) {
|
|
return tevent_req_post(req, ev);
|
|
}
|
|
tevent_req_set_callback(subreq,
|
|
wait_for_exit_io_check_done,
|
|
req);
|
|
}
|
|
|
|
return req;
|
|
}
|
|
|
|
static void wait_for_exit_parent_done(struct tevent_req *subreq)
|
|
{
|
|
struct tevent_req *req = tevent_req_callback_data(
|
|
subreq, struct tevent_req);
|
|
bool status;
|
|
int err;
|
|
|
|
status = wait_for_parent_recv(subreq, &err);
|
|
TALLOC_FREE(subreq);
|
|
if (! status) {
|
|
/* Ignore error */
|
|
fprintf(stderr,
|
|
"%s: "
|
|
"wait_for_parent_recv() failed (%d)\n",
|
|
progname,
|
|
err);
|
|
}
|
|
|
|
tevent_req_done(req);
|
|
}
|
|
|
|
static void wait_for_exit_io_check_done(struct tevent_req *subreq)
|
|
{
|
|
struct tevent_req *req = tevent_req_callback_data(
|
|
subreq, struct tevent_req);
|
|
bool status;
|
|
int err;
|
|
|
|
status = lock_io_check_recv(subreq, &err);
|
|
TALLOC_FREE(subreq);
|
|
if (! status) {
|
|
/* Ignore error */
|
|
fprintf(stderr,
|
|
"%s: "
|
|
"wait_for_lost_recv() failed (%d)\n",
|
|
progname,
|
|
err);
|
|
}
|
|
|
|
tevent_req_done(req);
|
|
}
|
|
|
|
static bool wait_for_exit_recv(struct tevent_req *req, int *perr)
|
|
{
|
|
if (tevent_req_is_unix_error(req, perr)) {
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
static void usage(void)
|
|
{
|
|
fprintf(stderr, "Usage: %s <file> [recheck_interval]\n", progname);
|
|
}
|
|
|
|
int main(int argc, char *argv[])
|
|
{
|
|
struct tevent_context *ev;
|
|
char result;
|
|
int ppid;
|
|
const char *file = NULL;
|
|
unsigned long recheck_interval;
|
|
int ret;
|
|
int fd = -1;
|
|
struct tevent_req *req;
|
|
bool status;
|
|
|
|
strlcpy(progpath, argv[0], sizeof(progpath));
|
|
progname = basename(progpath);
|
|
|
|
if (argc < 2 || argc > 3) {
|
|
usage();
|
|
exit(1);
|
|
}
|
|
|
|
ev = tevent_context_init(NULL);
|
|
if (ev == NULL) {
|
|
fprintf(stderr, "locking: tevent_context_init() failed\n");
|
|
exit(1);
|
|
}
|
|
|
|
ppid = getppid();
|
|
|
|
file = argv[1];
|
|
|
|
recheck_interval = 5;
|
|
if (argc == 3) {
|
|
recheck_interval = smb_strtoul(argv[2],
|
|
NULL,
|
|
10,
|
|
&ret,
|
|
SMB_STR_STANDARD);
|
|
if (ret != 0) {
|
|
usage();
|
|
exit(1);
|
|
}
|
|
}
|
|
|
|
result = fcntl_lock(file, &fd);
|
|
sys_write(STDOUT_FILENO, &result, 1);
|
|
|
|
if (result != '0') {
|
|
return 0;
|
|
}
|
|
|
|
req = wait_for_exit_send(ev, ev, ppid, file, fd, recheck_interval);
|
|
if (req == NULL) {
|
|
fprintf(stderr,
|
|
"%s: wait_for_exit_send() failed\n",
|
|
progname);
|
|
exit(1);
|
|
}
|
|
|
|
tevent_req_poll(req, ev);
|
|
|
|
status = wait_for_exit_recv(req, &ret);
|
|
if (! status) {
|
|
fprintf(stderr,
|
|
"%s: wait_for_exit_recv() failed (%d)\n",
|
|
progname,
|
|
ret);
|
|
}
|
|
|
|
if (fd != -1) {
|
|
close(fd);
|
|
}
|
|
|
|
return 0;
|
|
}
|