Stop applying a GCC-specific workaround on clang [BZ #30550]
[glibc.git] / posix / tst-chmod.c
blob470a696fe55c72041c3010ed0e1e9f7d1b603837
1 /* Test for chmod functions.
2 Copyright (C) 2000-2023 Free Software Foundation, Inc.
3 This file is part of the GNU C Library.
5 The GNU C Library is free software; you can redistribute it and/or
6 modify it under the terms of the GNU Lesser General Public
7 License as published by the Free Software Foundation; either
8 version 2.1 of the License, or (at your option) any later version.
10 The GNU C Library is distributed in the hope that it will be useful,
11 but WITHOUT ANY WARRANTY; without even the implied warranty of
12 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
13 Lesser General Public License for more details.
15 You should have received a copy of the GNU Lesser General Public
16 License along with the GNU C Library; if not, see
17 <https://www.gnu.org/licenses/>. */
19 #include <dirent.h>
20 #include <errno.h>
21 #include <error.h>
22 #include <fcntl.h>
23 #include <mcheck.h>
24 #include <stdio.h>
25 #include <stdlib.h>
26 #include <string.h>
27 #include <unistd.h>
28 #include <sys/stat.h>
30 #include <support/xunistd.h>
33 #define OUT_OF_MEMORY \
34 do { \
35 puts ("cannot allocate memory"); \
36 result = 1; \
37 goto fail; \
38 } while (0)
40 static int
41 do_test (int argc, char *argv[])
43 const char *builddir;
44 struct stat64 st1;
45 struct stat64 st2;
46 char *buf;
47 char *testdir;
48 char *testfile = NULL;
49 char *startdir;
50 size_t buflen;
51 int fd;
52 int result = 0;
53 DIR *dir;
55 mtrace ();
57 if (argc <= 1)
58 error (EXIT_FAILURE, 0, "no parameters");
60 /* This is where we will create the test files. */
61 builddir = argv[1];
62 buflen = strlen (builddir) + 50;
64 startdir = getcwd (NULL, 0);
65 if (startdir == NULL)
67 printf ("cannot get current directory: %m\n");
68 exit (EXIT_FAILURE);
71 /* A buffer large enough for everything we need. */
72 buf = (char *) alloca (buflen);
74 /* Create the directory name. */
75 snprintf (buf, buflen, "%s/chmoddirXXXXXX", builddir);
77 if (mkdtemp (buf) == NULL)
79 printf ("cannot create test directory: %m\n");
80 exit (EXIT_FAILURE);
83 if (chmod ("", 0600) == 0)
85 puts ("chmod(\"\", 0600 didn't fail");
86 result = 1;
88 else if (errno != ENOENT)
90 puts ("chmod(\"\",0600) does not set errno to ENOENT");
91 result = 1;
94 /* Create a duplicate. */
95 testdir = strdup (buf);
96 if (testdir == NULL)
97 OUT_OF_MEMORY;
99 if (stat64 (testdir, &st1) != 0)
101 printf ("cannot stat test directory: %m\n");
102 exit (1);
104 if (!S_ISDIR (st1.st_mode))
106 printf ("file not created as directory: %m\n");
107 exit (1);
110 /* We have to wait for a second to make sure the ctime changes. */
111 sleep (1);
113 /* Remove all access rights from the directory. */
114 if (chmod (testdir, 0) != 0)
116 printf ("cannot change mode of test directory: %m\n");
117 result = 1;
118 goto fail;
121 if (stat64 (testdir, &st2) != 0)
123 printf ("cannot stat test directory: %m\n");
124 result = 1;
125 goto fail;
128 /* Compare result. */
129 if ((st2.st_mode & ALLPERMS) != 0)
131 printf ("chmod(...,0) on directory left bits nonzero: %o\n",
132 st2.st_mode & ALLPERMS);
133 result = 1;
135 if (st1.st_ctime >= st2.st_ctime)
137 puts ("chmod(...,0) did not set ctime correctly");
138 result = 1;
141 /* Name of a file in the directory. */
142 snprintf (buf, buflen, "%s/file", testdir);
143 testfile = strdup (buf);
144 if (testfile == NULL)
145 OUT_OF_MEMORY;
147 fd = creat (testfile, 0);
148 if (fd != -1)
150 if (getuid () != 0)
152 puts ("managed to create test file in protected directory");
153 result = 1;
155 close (fd);
157 else if (errno != EACCES)
159 puts ("creat didn't generate correct errno value");
160 result = 1;
163 /* With this mode it still shouldn't be possible to create a file. */
164 if (chmod (testdir, 0600) != 0)
166 printf ("cannot change mode of test directory to 0600: %m\n");
167 result = 1;
168 goto fail;
171 fd = creat (testfile, 0);
172 if (fd != -1)
174 if (getuid () != 0)
176 puts ("managed to create test file in no-x protected directory");
177 result = 1;
179 close (fd);
181 else if (errno != EACCES)
183 puts ("creat didn't generate correct errno value");
184 result = 1;
187 /* Change the directory mode back to allow creating a file. This
188 time with fchmod. */
189 dir = opendir (testdir);
190 if (dir != NULL)
192 if (fchmod (dirfd (dir), 0700) != 0)
194 printf ("cannot change mode of test directory to 0700: %m\n");
195 result = 1;
196 closedir (dir);
197 goto fail;
200 closedir (dir);
202 else
204 printf ("cannot open directory: %m\n");
205 result = 1;
207 if (chmod (testdir, 0700) != 0)
209 printf ("cannot change mode of test directory to 0700: %m\n");
210 goto fail;
214 fd = creat (testfile, 0);
215 if (fd == -1)
217 puts ("still didn't manage to create test file in protected directory");
218 result = 1;
219 goto fail;
221 if (fstat64 (fd, &st1) != 0)
223 printf ("cannot stat new file: %m\n");
224 result = 1;
226 else if ((st1.st_mode & ALLPERMS) != 0)
228 puts ("file not created with access mode 0");
229 result = 1;
231 close (fd);
233 snprintf (buf, buflen, "%s/..", testdir);
234 xchdir (buf);
236 /* We are now in the directory above the one we create the test
237 directory in. */
239 sleep (1);
240 snprintf (buf, buflen, "./%s/../%s/file",
241 basename (testdir), basename (testdir));
242 if (chmod (buf, 0600) != 0)
244 printf ("cannot change mode of file to 0600: %m\n");
245 result = 1;
246 goto fail;
248 snprintf (buf, buflen, "./%s//file", basename (testdir));
249 if (stat64 (buf, &st2) != 0)
251 printf ("cannot stat new file: %m\n");
252 result = 1;
254 else if ((st2.st_mode & ALLPERMS) != 0600)
256 puts ("file mode not changed to 0600");
257 result = 1;
259 else if (st1.st_ctime >= st2.st_ctime)
261 puts ("chmod(\".../file\",0600) did not set ctime correctly");
262 result = 1;
265 if (chmod (buf, 0777 | S_ISUID | S_ISGID) != 0)
267 printf ("cannot change mode of file to %o: %m\n",
268 0777 | S_ISUID | S_ISGID);
269 result = 1;
271 if (stat64 (buf, &st2) != 0)
273 printf ("cannot stat test file: %m\n");
274 result = 1;
276 else if ((st2.st_mode & ALLPERMS) != (0777 | S_ISUID | S_ISGID))
278 puts ("file mode not changed to 0777 | S_ISUID | S_ISGID");
279 result = 1;
282 if (chmod (basename (testdir), 0777 | S_ISUID | S_ISGID | S_ISVTX) != 0)
284 printf ("cannot change mode of test directory to %o: %m\n",
285 0777 | S_ISUID | S_ISGID | S_ISVTX);
286 result = 1;
288 if (stat64 (basename (testdir), &st2) != 0)
290 printf ("cannot stat test directory: %m\n");
291 result = 1;
293 else if ((st2.st_mode & ALLPERMS) != (0777 | S_ISUID | S_ISGID | S_ISVTX))
295 puts ("directory mode not changed to 0777 | S_ISUID | S_ISGID | S_ISGID");
296 result = 1;
299 snprintf (buf, buflen, "./%s/no-such-file", basename (testdir));
300 if (chmod (buf, 0600) != -1)
302 puts ("chmod(\".../no-such-file\",0600) did not fail");
303 result = 1;
305 else if (errno != ENOENT)
307 puts ("chmod(\".../no-such-file\",0600) does not set errno to ENOENT");
308 result = 1;
311 snprintf (buf, buflen, "%s/", basename (testdir));
312 if (chmod (basename (testdir), 0677) != 0)
314 printf ("cannot change mode of test directory to 0677: %m\n");
315 result = 1;
317 else
319 snprintf (buf, buflen, "./%s/file", basename (testdir));
320 if (chmod (buf, 0600) == 0)
322 if (getuid () != 0)
324 puts ("chmod(\".../file\") with no-exec directory succeeded");
325 result = 1;
328 else if (errno != EACCES)
330 puts ("chmod(\".../file\") with no-exec directory didn't set EACCES");
331 result = 1;
335 if (chmod (basename (testdir), 0777) != 0)
337 printf ("cannot change mode of test directory to 0777: %m\n");
338 result = 1;
339 goto fail;
342 snprintf (buf, buflen, "%s/file/cannot-be", basename (testdir));
343 if (chmod (buf, 0600) == 0)
345 puts ("chmod(\".../file/cannot-be\",0600) did not fail");
346 result = 1;
348 else if (errno != ENOTDIR)
350 puts ("chmod(\".../file/cannot-be\",0600) does not set errno to ENOTDIR");
351 result = 1;
354 fail:
355 xchdir (startdir);
357 /* Remove all the files. */
358 chmod (testdir, 0700);
359 if (testfile != NULL)
361 chmod (testfile, 0700);
362 unlink (testfile);
364 rmdir (testdir);
366 /* Free the resources. */
367 free (testfile);
368 free (testdir);
369 free (startdir);
371 return result;
374 #include "../test-skeleton.c"