mirror of git://sourceware.org/git/glibc.git
				
				
				
			
		
			
				
	
	
		
			319 lines
		
	
	
		
			8.8 KiB
		
	
	
	
		
			C
		
	
	
	
			
		
		
	
	
			319 lines
		
	
	
		
			8.8 KiB
		
	
	
	
		
			C
		
	
	
	
| /* Test open and openat with O_TMPFILE.
 | |
|    Copyright (C) 2016-2025 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/>.  */
 | |
| 
 | |
| /* This test verifies that open and openat work as expected, i.e. they
 | |
|    create a deleted file with the requested file mode.  */
 | |
| 
 | |
| #include <errno.h>
 | |
| #include <fcntl.h>
 | |
| #include <stdbool.h>
 | |
| #include <stdio.h>
 | |
| #include <stdlib.h>
 | |
| #include <string.h>
 | |
| #include <sys/stat.h>
 | |
| #include <unistd.h>
 | |
| 
 | |
| #include <support/support.h>
 | |
| 
 | |
| #ifdef O_TMPFILE
 | |
| typedef int (*wrapper_func) (const char *, int, mode_t);
 | |
| 
 | |
| /* Error-checking wrapper for the open function, compatible with the
 | |
|    wrapper_func type.  */
 | |
| static int
 | |
| wrap_open (const char *path, int flags, mode_t mode)
 | |
| {
 | |
|   int ret = open (path, flags, mode);
 | |
|   if (ret < 0)
 | |
|     {
 | |
|       printf ("error: open (\"%s\", 0x%x, 0%03o): %m\n", path, flags, mode);
 | |
|       exit (1);
 | |
|     }
 | |
|   return ret;
 | |
| }
 | |
| 
 | |
| /* Error-checking wrapper for the openat function, compatible with the
 | |
|    wrapper_func type.  */
 | |
| static int
 | |
| wrap_openat (const char *path, int flags, mode_t mode)
 | |
| {
 | |
|   int ret = openat (AT_FDCWD, path, flags, mode);
 | |
|   if (ret < 0)
 | |
|     {
 | |
|       printf ("error: openat (\"%s\", 0x%x, 0%03o): %m\n", path, flags, mode);
 | |
|       exit (1);
 | |
|     }
 | |
|   return ret;
 | |
| }
 | |
| 
 | |
| /* Error-checking wrapper for the open64 function, compatible with the
 | |
|    wrapper_func type.  */
 | |
| static int
 | |
| wrap_open64 (const char *path, int flags, mode_t mode)
 | |
| {
 | |
|   int ret = open64 (path, flags, mode);
 | |
|   if (ret < 0)
 | |
|     {
 | |
|       printf ("error: open64 (\"%s\", 0x%x, 0%03o): %m\n", path, flags, mode);
 | |
|       exit (1);
 | |
|     }
 | |
|   return ret;
 | |
| }
 | |
| 
 | |
| /* Error-checking wrapper for the openat64 function, compatible with the
 | |
|    wrapper_func type.  */
 | |
| static int
 | |
| wrap_openat64 (const char *path, int flags, mode_t mode)
 | |
| {
 | |
|   int ret = openat64 (AT_FDCWD, path, flags, mode);
 | |
|   if (ret < 0)
 | |
|     {
 | |
|       printf ("error: openat64 (\"%s\", 0x%x, 0%03o): %m\n", path, flags, mode);
 | |
|       exit (1);
 | |
|     }
 | |
|   return ret;
 | |
| }
 | |
| 
 | |
| /* Return true if FD is flagged as deleted in /proc/self/fd, false if
 | |
|    not.  */
 | |
| static bool
 | |
| is_file_deleted (int fd)
 | |
| {
 | |
|   char *proc_fd_path = xasprintf ("/proc/self/fd/%d", fd);
 | |
|   char file_path[4096];
 | |
|   ssize_t file_path_length
 | |
|     = readlink (proc_fd_path, file_path, sizeof (file_path));
 | |
|   if (file_path_length < 0)
 | |
|     {
 | |
|       printf ("error: readlink (\"%s\"): %m", proc_fd_path);
 | |
|       free (proc_fd_path);
 | |
|       exit (1);
 | |
|     }
 | |
|   free (proc_fd_path);
 | |
|   if (file_path_length == sizeof (file_path))
 | |
|     {
 | |
|       printf ("error: path in /proc resolves to overlong file name: %.*s\n",
 | |
|               (int) file_path_length, file_path);
 | |
|       exit (1);
 | |
|     }
 | |
|   const char *deleted = " (deleted)";
 | |
|   if (file_path_length < strlen (deleted))
 | |
|     {
 | |
|       printf ("error: path in /proc is too short: %.*s\n",
 | |
|               (int) file_path_length, file_path);
 | |
|       exit (1);
 | |
|     }
 | |
|   return memcmp (file_path + file_path_length - strlen (deleted),
 | |
|               deleted, strlen (deleted)) == 0;
 | |
| }
 | |
| 
 | |
| /* Obtain a file name which is difficult to guess.  */
 | |
| static char *
 | |
| get_random_name (void)
 | |
| {
 | |
|   unsigned long long bytes[2];
 | |
|   int random_device = open ("/dev/urandom", O_RDONLY);
 | |
|   if (random_device < 0)
 | |
|     {
 | |
|       printf ("error: open (\"/dev/urandom\"): %m\n");
 | |
|       exit (1);
 | |
|     }
 | |
|   ssize_t ret = read (random_device, bytes, sizeof (bytes));
 | |
|   if (ret < 0)
 | |
|     {
 | |
|       printf ("error: read (\"/dev/urandom\"): %m\n");
 | |
|       exit (1);
 | |
|     }
 | |
|   if (ret != sizeof (bytes))
 | |
|     {
 | |
|       printf ("error: short read from /dev/urandom: %zd\n", ret);
 | |
|       exit (1);
 | |
|     }
 | |
|   close (random_device);
 | |
|   return xasprintf ("tst-open-tmpfile-%08llx%08llx.tmp", bytes[0], bytes[1]);
 | |
| }
 | |
| 
 | |
| /* Check open/openat (as specified by OP and WRAPPER) with a specific
 | |
|    PATH/FLAGS/MODE combination.  */
 | |
| static void
 | |
| check_wrapper_flags_mode (const char *op, wrapper_func wrapper,
 | |
|                           const char *path, int flags, mode_t mode)
 | |
| {
 | |
|   int fd = wrapper (path, flags | O_TMPFILE, mode);
 | |
|   struct stat64 st;
 | |
|   if (fstat64 (fd, &st) != 0)
 | |
|     {
 | |
|       printf ("error: fstat64: %m\n");
 | |
|       exit (1);
 | |
|     }
 | |
| 
 | |
|   /* Verify that the mode was correctly processed.  */
 | |
|   int actual_mode = st.st_mode & 0777;
 | |
|   if (actual_mode != mode)
 | |
|     {
 | |
|       printf ("error: unexpected mode; expected 0%03o, actual 0%03o\n",
 | |
|               mode, actual_mode);
 | |
|       exit (1);
 | |
|     }
 | |
| 
 | |
|   /* Check that the file is marked as deleted in /proc.  */
 | |
|   if (!is_file_deleted (fd))
 | |
|     {
 | |
|       printf ("error: path in /proc is not marked as deleted\n");
 | |
|       exit (1);
 | |
|     }
 | |
| 
 | |
|   /* Check that the file can be turned into a regular file with
 | |
|      linkat.  Open a file descriptor for the directory at PATH.  Use
 | |
|      AT_FDCWD if PATH is ".", to exercise that functionality as
 | |
|      well.  */
 | |
|   int path_fd;
 | |
|   if (strcmp (path, ".") == 0)
 | |
|     path_fd = AT_FDCWD;
 | |
|   else
 | |
|     {
 | |
|       path_fd = open (path, O_RDONLY | O_DIRECTORY);
 | |
|       if (path_fd < 0)
 | |
|         {
 | |
|           printf ("error: open (\"%s\"): %m\n", path);
 | |
|           exit (1);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|   /* Use a hard-to-guess name for the new directory entry.  */
 | |
|   char *new_name = get_random_name ();
 | |
| 
 | |
|   /* linkat does not require privileges if the path in /proc/self/fd
 | |
|      is used.  */
 | |
|   char *proc_fd_path = xasprintf ("/proc/self/fd/%d", fd);
 | |
|   if (linkat (AT_FDCWD, proc_fd_path, path_fd, new_name,
 | |
|               AT_SYMLINK_FOLLOW) == 0)
 | |
|     {
 | |
|       if (unlinkat (path_fd, new_name, 0) != 0 && errno != ENOENT)
 | |
|         {
 | |
|           printf ("error: unlinkat (\"%s/%s\"): %m\n", path, new_name);
 | |
|           exit (1);
 | |
|         }
 | |
|     }
 | |
|   else
 | |
|     {
 | |
|       /* linkat failed.  This is expected if O_EXCL was specified.  */
 | |
|       if ((flags & O_EXCL) == 0)
 | |
|         {
 | |
|           printf ("error: linkat failed after %s (\"%s\", 0x%x, 0%03o): %m\n",
 | |
|                   op, path, flags, mode);
 | |
|           exit (1);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|   free (proc_fd_path);
 | |
|   free (new_name);
 | |
|   if (path_fd != AT_FDCWD)
 | |
|     close (path_fd);
 | |
|   close (fd);
 | |
| }
 | |
| 
 | |
| /* Check OP/WRAPPER with various flags at a specific PATH and
 | |
|    MODE.  */
 | |
| static void
 | |
| check_wrapper_mode (const char *op, wrapper_func wrapper,
 | |
|                     const char *path, mode_t mode)
 | |
| {
 | |
|   check_wrapper_flags_mode (op, wrapper, path, O_WRONLY, mode);
 | |
|   check_wrapper_flags_mode (op, wrapper, path, O_WRONLY | O_EXCL, mode);
 | |
|   check_wrapper_flags_mode (op, wrapper, path, O_RDWR, mode);
 | |
|   check_wrapper_flags_mode (op, wrapper, path, O_RDWR | O_EXCL, mode);
 | |
| }
 | |
| 
 | |
| /* Check open/openat with varying permissions.  */
 | |
| static void
 | |
| check_wrapper (const char *op, wrapper_func wrapper,
 | |
|                     const char *path)
 | |
| {
 | |
|   printf ("info: testing %s at: %s\n", op, path);
 | |
|   check_wrapper_mode (op, wrapper, path, 0);
 | |
|   check_wrapper_mode (op, wrapper, path, 0640);
 | |
|   check_wrapper_mode (op, wrapper, path, 0600);
 | |
|   check_wrapper_mode (op, wrapper, path, 0755);
 | |
|   check_wrapper_mode (op, wrapper, path, 0750);
 | |
| }
 | |
| 
 | |
| /* Verify that the directory at PATH supports O_TMPFILE.  Exit with
 | |
|    status 77 (unsupported) if the kernel does not support O_TMPFILE.
 | |
|    Even with kernel support, not all file systems O_TMPFILE, so return
 | |
|    true if the directory supports O_TMPFILE, false if not.  */
 | |
| static bool
 | |
| probe_path (const char *path)
 | |
| {
 | |
|   int fd = openat (AT_FDCWD, path, O_TMPFILE | O_RDWR, 0);
 | |
|   if (fd < 0)
 | |
|     {
 | |
|       if (errno == EISDIR)
 | |
|         /* The system does not support O_TMPFILE.  */
 | |
|         {
 | |
|           printf ("info: kernel does not support O_TMPFILE\n");
 | |
|           exit (77);
 | |
|         }
 | |
|       if (errno == EOPNOTSUPP)
 | |
|         {
 | |
|           printf ("info: path does not support O_TMPFILE: %s\n", path);
 | |
|           return false;
 | |
|         }
 | |
|       printf ("error: openat (\"%s\", O_TMPFILE | O_RDWR): %m\n", path);
 | |
|       exit (1);
 | |
|     }
 | |
|   close (fd);
 | |
|   return true;
 | |
| }
 | |
| 
 | |
| static int
 | |
| do_test (void)
 | |
| {
 | |
|   umask (0);
 | |
|   const char *paths[] = { ".", "/dev/shm", "/tmp",
 | |
|                           getenv ("TEST_TMPFILE_PATH"),
 | |
|                           NULL };
 | |
|   bool supported = false;
 | |
|   for (int i = 0; paths[i] != NULL; ++i)
 | |
|     if (probe_path (paths[i]))
 | |
|       {
 | |
|         supported = true;
 | |
|         check_wrapper ("open", wrap_open, paths[i]);
 | |
|         check_wrapper ("openat", wrap_openat, paths[i]);
 | |
|         check_wrapper ("open64", wrap_open64, paths[i]);
 | |
|         check_wrapper ("openat64", wrap_openat64, paths[i]);
 | |
|       }
 | |
| 
 | |
|   if (!supported)
 | |
|     return 77;
 | |
| 
 | |
|   return 0;
 | |
| }
 | |
| 
 | |
| #else  /* !O_TMPFILE */
 | |
| 
 | |
| static int
 | |
| do_test (void)
 | |
| {
 | |
|   return 77;
 | |
| }
 | |
| 
 | |
| #endif  /* O_TMPFILE */
 | |
| 
 | |
| #include <support/test-driver.c>
 |