diff options
-rw-r--r-- | lib/ccan/failtest/_info | 3 | ||||
-rw-r--r-- | lib/ccan/failtest/failtest.c | 17 | ||||
-rw-r--r-- | lib/ccan/failtest/test/run-with-fdlimit.c | 51 |
3 files changed, 68 insertions, 3 deletions
diff --git a/lib/ccan/failtest/_info b/lib/ccan/failtest/_info index 14dcb783be..4a5b97c931 100644 --- a/lib/ccan/failtest/_info +++ b/lib/ccan/failtest/_info @@ -54,6 +54,9 @@ * * License: LGPL * Author: Rusty Russell <rusty@rustcorp.com.au> + * Ccanlint: + * // valgrind seems to mess up rlimit. + * tests_pass_valgrind test/run-with-fdlimit.c:FAIL */ int main(int argc, char *argv[]) { diff --git a/lib/ccan/failtest/failtest.c b/lib/ccan/failtest/failtest.c index 701586e9b8..5f68394518 100644 --- a/lib/ccan/failtest/failtest.c +++ b/lib/ccan/failtest/failtest.c @@ -14,6 +14,7 @@ #include <sys/stat.h> #include <sys/time.h> #include <sys/mman.h> +#include <sys/resource.h> #include <signal.h> #include <assert.h> #include <ccan/time/time.h> @@ -194,11 +195,21 @@ static struct failtest_call *add_history_(enum failtest_call_type type, static int move_fd_to_high(int fd) { int i; + struct rlimit lim; + int max; - for (i = FD_SETSIZE - 1; i >= 0; i--) { + if (getrlimit(RLIMIT_NOFILE, &lim) == 0) { + max = lim.rlim_cur; + printf("Max is %i\n", max); + } else + max = FD_SETSIZE; + + for (i = max - 1; i > fd; i--) { if (fcntl(i, F_GETFL) == -1 && errno == EBADF) { - if (dup2(fd, i) == -1) - err(1, "Failed to dup fd %i to %i", fd, i); + if (dup2(fd, i) == -1) { + warn("Failed to dup fd %i to %i", fd, i); + continue; + } close(fd); return i; } diff --git a/lib/ccan/failtest/test/run-with-fdlimit.c b/lib/ccan/failtest/test/run-with-fdlimit.c new file mode 100644 index 0000000000..6b4483f07e --- /dev/null +++ b/lib/ccan/failtest/test/run-with-fdlimit.c @@ -0,0 +1,51 @@ +/* Include the C files directly. */ +#include <ccan/failtest/failtest.c> +#include <stdlib.h> +#include <err.h> +#include <ccan/tap/tap.h> + +int main(void) +{ + int fd, pfd[2], ecode; + struct rlimit lim; + + if (getrlimit(RLIMIT_NOFILE, &lim) != 0) + err(1, "getrlimit RLIMIT_NOFILE fail?"); + + printf("rlimit = %lu/%lu (inf=%lu)\n", + (long)lim.rlim_cur, (long)lim.rlim_max, + (long)RLIM_INFINITY); + lim.rlim_cur /= 2; + if (lim.rlim_cur < 8) + errx(1, "getrlimit limit %li too low", (long)lim.rlim_cur); + if (setrlimit(RLIMIT_NOFILE, &lim) != 0) + err(1, "setrlimit RLIMIT_NOFILE (%li/%li)", + (long)lim.rlim_cur, (long)lim.rlim_max); + + plan_tests(2); + failtest_init(0, NULL); + + if (pipe(pfd)) + abort(); + + fd = failtest_open("run-with-fdlimit-scratch", "run-with_fdlimit.c", 1, + O_RDWR|O_CREAT, 0600); + if (fd == -1) { + /* We are the child: write error code for parent to check. */ + ecode = errno; + if (write(pfd[1], &ecode, sizeof(ecode)) != sizeof(ecode)) + abort(); + failtest_exit(0); + } + + /* Check child got correct errno. */ + ok1(read(pfd[0], &ecode, sizeof(ecode)) == sizeof(ecode)); + ok1(ecode == EACCES); + + /* Clean up. */ + failtest_close(fd, "run-open.c", 1); + close(pfd[0]); + close(pfd[1]); + + return exit_status(); +} |