diff options
author | Adhemerval Zanella <adhemerval.zanella@linaro.org> | 2021-08-24 16:12:24 -0300 |
---|---|---|
committer | Adhemerval Zanella <adhemerval.zanella@linaro.org> | 2021-08-26 17:13:45 -0300 |
commit | e814f4b04ee413a7bb3dfa43e74c8fb4abf58359 (patch) | |
tree | d83648696fae2c7a8ee71da52297bed7dafe06c6 /support/tst-support-open-dev-null-range.c | |
parent | 5aa359d33163bde660fec9b26e23cfb93d63ecde (diff) | |
download | glibc-e814f4b04ee413a7bb3dfa43e74c8fb4abf58359.tar.gz glibc-e814f4b04ee413a7bb3dfa43e74c8fb4abf58359.tar.xz glibc-e814f4b04ee413a7bb3dfa43e74c8fb4abf58359.zip |
support: Add support_open_dev_null_range
It returns a range of file descriptor referring to the '/dev/null' pathname. The function takes care of restarting the open range if a file descriptor is found within the specified range and also increases RLIMIT_NOFILE if required. Checked on x86_64-linux-gnu.
Diffstat (limited to 'support/tst-support-open-dev-null-range.c')
-rw-r--r-- | support/tst-support-open-dev-null-range.c | 155 |
1 files changed, 155 insertions, 0 deletions
diff --git a/support/tst-support-open-dev-null-range.c b/support/tst-support-open-dev-null-range.c new file mode 100644 index 0000000000..8e29def1ce --- /dev/null +++ b/support/tst-support-open-dev-null-range.c @@ -0,0 +1,155 @@ +/* Tests for support_open_dev_null_range. + Copyright (C) 2021 Free Software Foundation, Inc. + This file is part of the GNU C Library. + + The GNU C Library is free software; you can redistribute it and/or + modify it under the terms of the GNU Lesser General Public + License as published by the Free Software Foundation; either + version 2.1 of the License, or (at your option) any later version. + + The GNU C Library 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 + Lesser General Public License for more details. + + You should have received a copy of the GNU Lesser General Public + License along with the GNU C Library; if not, see + <https://www.gnu.org/licenses/>. */ + +#include <errno.h> +#include <dirent.h> +#include <fcntl.h> +#include <limits.h> +#include <support/check.h> +#include <support/support.h> +#include <support/xunistd.h> +#include <sys/resource.h> +#include <stdlib.h> + +#ifndef PATH_MAX +# define PATH_MAX 1024 +#endif + +#include <stdio.h> + +static void +check_path (int fd) +{ + char *proc_fd_path = xasprintf ("/proc/self/fd/%d", fd); + char file_path[PATH_MAX]; + ssize_t file_path_length + = readlink (proc_fd_path, file_path, sizeof (file_path)); + free (proc_fd_path); + if (file_path_length < 0) + FAIL_EXIT1 ("readlink (%s, %p, %zu)", proc_fd_path, file_path, + sizeof (file_path)); + file_path[file_path_length] = '\0'; + TEST_COMPARE_STRING (file_path, "/dev/null"); +} + +static int +number_of_opened_files (void) +{ + DIR *fds = opendir ("/proc/self/fd"); + if (fds == NULL) + FAIL_EXIT1 ("opendir (\"/proc/self/fd\"): %m"); + + int r = 0; + while (true) + { + errno = 0; + struct dirent64 *e = readdir64 (fds); + if (e == NULL) + { + if (errno != 0) + FAIL_EXIT1 ("readdir: %m"); + break; + } + + if (e->d_name[0] == '.') + continue; + + char *endptr; + long int fd = strtol (e->d_name, &endptr, 10); + if (*endptr != '\0' || fd < 0 || fd > INT_MAX) + FAIL_EXIT1 ("readdir: invalid file descriptor name: /proc/self/fd/%s", + e->d_name); + + /* Skip the descriptor which is used to enumerate the + descriptors. */ + if (fd == dirfd (fds)) + continue; + + r = r + 1; + } + + closedir (fds); + + return r; +} + +static int +do_test (void) +{ + const int nfds1 = 8; + int lowfd = support_open_dev_null_range (nfds1, O_RDONLY, 0600); + for (int i = 0; i < nfds1; i++) + { + TEST_VERIFY (fcntl (lowfd + i, F_GETFL) > -1); + check_path (lowfd + i); + } + + /* create some gaps. */ + xclose (lowfd + 1); + xclose (lowfd + 5); + xclose (lowfd + 6); + + const int nfds2 = 16; + int lowfd2 = support_open_dev_null_range (nfds2, O_RDONLY, 0600); + for (int i = 0; i < nfds2; i++) + { + TEST_VERIFY (fcntl (lowfd2 + i, F_GETFL) > -1); + check_path (lowfd2 + i); + } + + /* Decrease the maximum number of files. */ + { + struct rlimit rl; + if (getrlimit (RLIMIT_NOFILE, &rl) == -1) + FAIL_EXIT1 ("getrlimit (RLIMIT_NOFILE): %m"); + + rl.rlim_cur = number_of_opened_files (); + + if (setrlimit (RLIMIT_NOFILE, &rl) == 1) + FAIL_EXIT1 ("setrlimit (RLIMIT_NOFILE): %m"); + } + + const int nfds3 = 16; + int lowfd3 = support_open_dev_null_range (nfds3, O_RDONLY, 0600); + for (int i = 0; i < nfds3; i++) + { + TEST_VERIFY (fcntl (lowfd3 + i, F_GETFL) > -1); + check_path (lowfd3 + i); + } + + /* create a lot of gaps to trigger the range extension. */ + xclose (lowfd3 + 1); + xclose (lowfd3 + 3); + xclose (lowfd3 + 5); + xclose (lowfd3 + 7); + xclose (lowfd3 + 9); + xclose (lowfd3 + 11); + xclose (lowfd3 + 13); + + const int nfds4 = 16; + int lowfd4 = support_open_dev_null_range (nfds4, O_RDONLY, 0600); + for (int i = 0; i < nfds4; i++) + { + TEST_VERIFY (fcntl (lowfd4 + i, F_GETFL) > -1); + check_path (lowfd4 + i); + } + + return 0; +} + +#include <support/test-driver.c> |