hc
2024-05-14 bedbef8ad3e75a304af6361af235302bcc61d06b
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);
}