 4751fd5328
			
		
	
	
		4751fd5328
		
	
	
	
	
		
			
			This function has to ensure it doesn't follow a symlink that could be used to escape the virtfs directory. This could be easily achieved if fchmodat() on linux honored the AT_SYMLINK_NOFOLLOW flag as described in POSIX, but it doesn't. There was a tentative to implement a new fchmodat2() syscall with the correct semantics: https://patchwork.kernel.org/patch/9596301/ but it didn't gain much momentum. Also it was suggested to look at an O_PATH based solution in the first place. The current implementation covers most use-cases, but it notably fails if: - the target path has access rights equal to 0000 (openat() returns EPERM), => once you've done chmod(0000) on a file, you can never chmod() again - the target path is UNIX domain socket (openat() returns ENXIO) => bind() of UNIX domain sockets fails if the file is on 9pfs The solution is to use O_PATH: openat() now succeeds in both cases, and we can ensure the path isn't a symlink with fstat(). The associated entry in "/proc/self/fd" can hence be safely passed to the regular chmod() syscall. The previous behavior is kept for older systems that don't have O_PATH. Signed-off-by: Greg Kurz <groug@kaod.org> Reviewed-by: Eric Blake <eblake@redhat.com> Tested-by: Zhi Yong Wu <zhiyong.wu@ucloud.cn> Acked-by: Philippe Mathieu-Daudé <f4bug@amsat.org>
		
			
				
	
	
		
			65 lines
		
	
	
		
			1.5 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			65 lines
		
	
	
		
			1.5 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
|  * 9p utilities
 | |
|  *
 | |
|  * Copyright IBM, Corp. 2017
 | |
|  *
 | |
|  * Authors:
 | |
|  *  Greg Kurz <groug@kaod.org>
 | |
|  *
 | |
|  * This work is licensed under the terms of the GNU GPL, version 2 or later.
 | |
|  * See the COPYING file in the top-level directory.
 | |
|  */
 | |
| 
 | |
| #ifndef QEMU_9P_UTIL_H
 | |
| #define QEMU_9P_UTIL_H
 | |
| 
 | |
| #ifdef O_PATH
 | |
| #define O_PATH_9P_UTIL O_PATH
 | |
| #else
 | |
| #define O_PATH_9P_UTIL 0
 | |
| #endif
 | |
| 
 | |
| static inline void close_preserve_errno(int fd)
 | |
| {
 | |
|     int serrno = errno;
 | |
|     close(fd);
 | |
|     errno = serrno;
 | |
| }
 | |
| 
 | |
| static inline int openat_dir(int dirfd, const char *name)
 | |
| {
 | |
|     return openat(dirfd, name,
 | |
|                   O_DIRECTORY | O_RDONLY | O_NOFOLLOW | O_PATH_9P_UTIL);
 | |
| }
 | |
| 
 | |
| static inline int openat_file(int dirfd, const char *name, int flags,
 | |
|                               mode_t mode)
 | |
| {
 | |
|     int fd, serrno, ret;
 | |
| 
 | |
|     fd = openat(dirfd, name, flags | O_NOFOLLOW | O_NOCTTY | O_NONBLOCK,
 | |
|                 mode);
 | |
|     if (fd == -1) {
 | |
|         return -1;
 | |
|     }
 | |
| 
 | |
|     serrno = errno;
 | |
|     /* O_NONBLOCK was only needed to open the file. Let's drop it. We don't
 | |
|      * do that with O_PATH since fcntl(F_SETFL) isn't supported, and openat()
 | |
|      * ignored it anyway.
 | |
|      */
 | |
|     if (!(flags & O_PATH_9P_UTIL)) {
 | |
|         ret = fcntl(fd, F_SETFL, flags);
 | |
|         assert(!ret);
 | |
|     }
 | |
|     errno = serrno;
 | |
|     return fd;
 | |
| }
 | |
| 
 | |
| ssize_t fgetxattrat_nofollow(int dirfd, const char *path, const char *name,
 | |
|                              void *value, size_t size);
 | |
| int fsetxattrat_nofollow(int dirfd, const char *path, const char *name,
 | |
|                          void *value, size_t size, int flags);
 | |
| 
 | |
| #endif
 |