Loading...
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 | // SPDX-License-Identifier: GPL-2.0-or-later /* * Author: Aleksa Sarai <cyphar@cyphar.com> * Copyright (C) 2018-2019 SUSE LLC. */ #define _GNU_SOURCE #include <errno.h> #include <fcntl.h> #include <stdbool.h> #include <string.h> #include <syscall.h> #include <limits.h> #include "helpers.h" bool needs_openat2(const struct open_how *how) { return how->resolve != 0; } int raw_openat2(int dfd, const char *path, void *how, size_t size) { int ret = syscall(__NR_openat2, dfd, path, how, size); return ret >= 0 ? ret : -errno; } int sys_openat2(int dfd, const char *path, struct open_how *how) { return raw_openat2(dfd, path, how, sizeof(*how)); } int sys_openat(int dfd, const char *path, struct open_how *how) { int ret = openat(dfd, path, how->flags, how->mode); return ret >= 0 ? ret : -errno; } int sys_renameat2(int olddirfd, const char *oldpath, int newdirfd, const char *newpath, unsigned int flags) { int ret = syscall(__NR_renameat2, olddirfd, oldpath, newdirfd, newpath, flags); return ret >= 0 ? ret : -errno; } int touchat(int dfd, const char *path) { int fd = openat(dfd, path, O_CREAT, 0700); if (fd >= 0) close(fd); return fd; } char *fdreadlink(int fd) { char *target, *tmp; E_asprintf(&tmp, "/proc/self/fd/%d", fd); target = malloc(PATH_MAX); if (!target) ksft_exit_fail_msg("fdreadlink: malloc failed\n"); memset(target, 0, PATH_MAX); E_readlink(tmp, target, PATH_MAX); free(tmp); return target; } bool fdequal(int fd, int dfd, const char *path) { char *fdpath, *dfdpath, *other; bool cmp; fdpath = fdreadlink(fd); dfdpath = fdreadlink(dfd); if (!path) E_asprintf(&other, "%s", dfdpath); else if (*path == '/') E_asprintf(&other, "%s", path); else E_asprintf(&other, "%s/%s", dfdpath, path); cmp = !strcmp(fdpath, other); free(fdpath); free(dfdpath); free(other); return cmp; } bool openat2_supported = false; void __attribute__((constructor)) init(void) { struct open_how how = {}; int fd; BUILD_BUG_ON(sizeof(struct open_how) != OPEN_HOW_SIZE_VER0); /* Check openat2(2) support. */ fd = sys_openat2(AT_FDCWD, ".", &how); openat2_supported = (fd >= 0); if (fd >= 0) close(fd); } |