/* * Copyright © 2018 Alexey Dobriyan <adobriyan@gmail.com> * * Permission to use, copy, modify, and distribute this software for any * purpose with or without fee is hereby granted, provided that the above * copyright notice and this permission notice appear in all copies. * * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. */ // Test /proc/*/fd lookup. #undef NDEBUG #include <assert.h> #include <dirent.h> #include <errno.h> #include <limits.h> #include <sched.h> #include <stdio.h> #include <unistd.h> #include <sys/types.h> #include <sys/stat.h> #include <fcntl.h> #include "proc.h" /* lstat(2) has more "coverage" in case non-symlink pops up somehow. */ static void test_lookup_pass(const char *pathname) { struct stat st; ssize_t rv; memset(&st, 0, sizeof(struct stat)); rv = lstat(pathname, &st); assert(rv == 0); assert(S_ISLNK(st.st_mode)); } static void test_lookup_fail(const char *pathname) { struct stat st; ssize_t rv; rv = lstat(pathname, &st); assert(rv == -1 && errno == ENOENT); } static void test_lookup(unsigned int fd) { char buf[64]; unsigned int c; unsigned int u; int i; snprintf(buf, sizeof(buf), "/proc/self/fd/%u", fd); test_lookup_pass(buf); /* leading junk */ for (c = 1; c <= 255; c++) { if (c == '/') continue; snprintf(buf, sizeof(buf), "/proc/self/fd/%c%u", c, fd); test_lookup_fail(buf); } /* trailing junk */ for (c = 1; c <= 255; c++) { if (c == '/') continue; snprintf(buf, sizeof(buf), "/proc/self/fd/%u%c", fd, c); test_lookup_fail(buf); } for (i = INT_MIN; i < INT_MIN + 1024; i++) { snprintf(buf, sizeof(buf), "/proc/self/fd/%d", i); test_lookup_fail(buf); } for (i = -1024; i < 0; i++) { snprintf(buf, sizeof(buf), "/proc/self/fd/%d", i); test_lookup_fail(buf); } for (u = INT_MAX - 1024; u <= (unsigned int)INT_MAX + 1024; u++) { snprintf(buf, sizeof(buf), "/proc/self/fd/%u", u); test_lookup_fail(buf); } for (u = UINT_MAX - 1024; u != 0; u++) { snprintf(buf, sizeof(buf), "/proc/self/fd/%u", u); test_lookup_fail(buf); } } int main(void) { struct dirent *de; unsigned int fd, target_fd; if (unshare(CLONE_FILES) == -1) return 1; /* Wipe fdtable. */ do { DIR *d; d = opendir("/proc/self/fd"); if (!d) return 1; de = xreaddir(d); assert(de->d_type == DT_DIR); assert(streq(de->d_name, ".")); de = xreaddir(d); assert(de->d_type == DT_DIR); assert(streq(de->d_name, "..")); next: de = xreaddir(d); if (de) { unsigned long long fd_ull; unsigned int fd; char *end; assert(de->d_type == DT_LNK); fd_ull = xstrtoull(de->d_name, &end); assert(*end == '\0'); assert(fd_ull == (unsigned int)fd_ull); fd = fd_ull; if (fd == dirfd(d)) goto next; close(fd); } closedir(d); } while (de); /* Now fdtable is clean. */ fd = open("/", O_PATH|O_DIRECTORY); assert(fd == 0); test_lookup(fd); close(fd); /* Clean again! */ fd = open("/", O_PATH|O_DIRECTORY); assert(fd == 0); /* Default RLIMIT_NOFILE-1 */ target_fd = 1023; while (target_fd > 0) { if (dup2(fd, target_fd) == target_fd) break; target_fd /= 2; } assert(target_fd > 0); close(fd); test_lookup(target_fd); close(target_fd); return 0; }