/* Test trivial FD monitoring Copyright (C) Martin Schwenke, DataDirect Networks 2022 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 . */ #include "replace.h" #include "system/filesys.h" #include "system/network.h" #include "system/wait.h" #include #include #include #include "lib/util/tevent_unix.h" #include "common/tmon.h" #include "tests/src/test_backtrace.h" struct test_state { const char *label; unsigned long async_wait_time; unsigned long blocking_sleep_time; }; static void test_tmon_ping_done(struct tevent_req *subreq); static void test_async_wait_done(struct tevent_req *subreq); static struct tevent_req *test_send(TALLOC_CTX *mem_ctx, struct tevent_context *ev, const char *label, int fd, int direction, unsigned long timeout, unsigned long interval, unsigned long async_wait_time, unsigned long blocking_sleep_time) { struct tevent_req *req, *subreq; struct test_state *state; req = tevent_req_create(mem_ctx, &state, struct test_state); if (req == NULL) { return NULL; } state->label = label; state->async_wait_time = async_wait_time; state->blocking_sleep_time = blocking_sleep_time; subreq = tmon_ping_send(state, ev, fd, direction, timeout, interval); if (tevent_req_nomem(subreq, req)) { return tevent_req_post(req, ev); } tevent_req_set_callback(subreq, test_tmon_ping_done, req); if (state->async_wait_time != 0) { fprintf(stderr, "%s: async wait start %lu\n", state->label, state->async_wait_time); } subreq = tevent_wakeup_send(state, ev, tevent_timeval_current_ofs( (uint32_t)async_wait_time, 0)); if (tevent_req_nomem(subreq, req)) { return tevent_req_post(req, ev); } tevent_req_set_callback(subreq, test_async_wait_done, req); return req; } static void test_tmon_ping_done(struct tevent_req *subreq) { struct tevent_req *req = tevent_req_callback_data( subreq, struct tevent_req); struct test_state *state = tevent_req_data(req, struct test_state); bool status; int err; status = tmon_ping_recv(subreq, &err); TALLOC_FREE(subreq); if (!status) { switch(err) { case EPIPE: fprintf(stderr, "%s: pipe closed\n", state->label); break; case ETIMEDOUT: fprintf(stderr, "%s: ping timeout\n", state->label); break; default: fprintf(stderr, "%s: error (%d)\n", state->label, err); } tevent_req_error(req, err); return; } fprintf(stderr, "%s: done\n", state->label); tevent_req_done(req); } static void test_async_wait_done(struct tevent_req *subreq) { struct tevent_req *req = tevent_req_callback_data( subreq, struct tevent_req); struct test_state *state = tevent_req_data(req, struct test_state); unsigned int left; bool status; status = tevent_wakeup_recv(subreq); TALLOC_FREE(subreq); if (!status) { fprintf(stderr, "%s: tevent_wakeup_recv() failed\n", state->label); /* Ignore error */ } if (state->async_wait_time != 0) { fprintf(stderr, "%s: async wait end\n", state->label); } if (state->blocking_sleep_time == 0) { goto done; } fprintf(stderr, "%s: blocking sleep start %lu\n", state->label, state->blocking_sleep_time); left = sleep((unsigned int)state->blocking_sleep_time); fprintf(stderr, "%s: blocking sleep end\n", state->label); if (left != 0) { tevent_req_error(req, EINTR); return; } done: tevent_req_done(req); } static bool test_recv(struct tevent_req *req, int *perr) { if (tevent_req_is_unix_error(req, perr)) { return false; } return true; } static int test_one(bool is_parent, int sync_fd, int fd, int direction, unsigned long timeout, unsigned long interval, unsigned long async_wait_time, unsigned long blocking_sleep_time) { TALLOC_CTX *mem_ctx; struct tevent_context *ev; struct tevent_req *req; bool status; char buf[1] = ""; ssize_t count; int err; int ret; if (!is_parent) { count = read(sync_fd, buf, sizeof(buf)); assert(count == 1); assert(buf[0] == '\0'); close(sync_fd); } mem_ctx = talloc_new(NULL); if (mem_ctx == NULL) { ret = ENOMEM; goto done; } ev = tevent_context_init(mem_ctx); if (ev == NULL) { ret = ENOMEM; goto done; } req = test_send(mem_ctx, ev, is_parent ? "parent" : "child", fd, direction, timeout, interval, async_wait_time, blocking_sleep_time); if (req == NULL) { ret = ENOMEM; goto done; } if (is_parent) { count = write(sync_fd, buf, sizeof(buf)); assert(count == 1); } status = tevent_req_poll(req, ev); if (!status) { ret = EIO; goto done; } status = test_recv(req, &err); ret = status ? 0 : err; done: return ret; } static void test(unsigned long parent_timeout, unsigned long parent_interval, unsigned long parent_async_wait_time, unsigned long parent_blocking_sleep_time, int parent_result, unsigned long child_timeout, unsigned long child_interval, unsigned long child_async_wait_time, unsigned long child_blocking_sleep_time, int child_result) { int sync[2]; int fd[2]; pid_t pid; int wstatus; int ret; /* Pipe for synchronisation */ ret = pipe(sync); assert(ret == 0); ret = socketpair(AF_UNIX, SOCK_STREAM, 0, fd); assert(ret == 0); pid = fork(); assert(pid != -1); if (pid == 0) { /* child */ close(sync[1]); close(fd[0]); ret = test_one(false, sync[0], fd[1], TMON_FD_BOTH, child_timeout, child_interval, child_async_wait_time, child_blocking_sleep_time); _exit(ret); } /* Parent */ close(sync[0]); close(fd[1]); ret = test_one(true, sync[1], fd[0], TMON_FD_BOTH, parent_timeout, parent_interval, parent_async_wait_time, parent_blocking_sleep_time); assert(ret == parent_result); /* Close to mimick exit, so child status can be checked below */ close(fd[0]); /* Abort if child failed */ waitpid(pid, &wstatus, 0); if (WIFEXITED(wstatus)) { assert(WEXITSTATUS(wstatus) == child_result); } } struct test_inputs { unsigned int timeout; unsigned int interval; unsigned int async_wait_time; unsigned int blocking_sleep_time; int expected_result; }; static void get_test_inputs(const char **args, struct test_inputs *inputs) { if (strcmp(args[0], "false") == 0) { inputs->interval = 0; } else if (strcmp(args[0], "true") == 0) { inputs->interval = 1; } else { inputs->interval = strtoul(args[0], NULL, 0); } inputs->timeout = strtoul(args[1], NULL, 0); inputs->async_wait_time = (unsigned int)strtoul(args[2], NULL, 0); inputs->blocking_sleep_time = (unsigned int)strtoul(args[3], NULL, 0); inputs->expected_result = (int)strtoul(args[4], NULL, 0); } static void usage(const char *prog) { fprintf(stderr, "usage: %s " "\\\n\t" " " " " " " " " " " "\\\n\t" " " " " " " " " " " "\n", prog); exit(1); } int main(int argc, const char **argv) { struct test_inputs parent; struct test_inputs child; if (argc != 11) { usage(argv[0]); } test_backtrace_setup(); get_test_inputs(&argv[1], &parent); get_test_inputs(&argv[6], &child); test(parent.timeout, parent.interval, parent.async_wait_time, parent.blocking_sleep_time, parent.expected_result, child.timeout, child.interval, child.async_wait_time, child.blocking_sleep_time, child.expected_result); return 0; }