mirror of
				https://github.com/RGBCube/serenity
				synced 2025-10-24 23:42:37 +00:00 
			
		
		
		
	 cf271183b4
			
		
	
	
		cf271183b4
		
	
	
	
	
		
			
			This has several benefits: 1) We no longer just blindly derefence a null pointer in various places 2) We will get nicer runtime error messages if the current process does turn out to be null in the call location 3) GCC no longer complains about possible nullptr dereferences when compiling without KUBSAN
		
			
				
	
	
		
			57 lines
		
	
	
	
		
			1.6 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			57 lines
		
	
	
	
		
			1.6 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /*
 | |
|  * Copyright (c) 2018-2020, Andreas Kling <kling@serenityos.org>
 | |
|  *
 | |
|  * SPDX-License-Identifier: BSD-2-Clause
 | |
|  */
 | |
| 
 | |
| #include <Kernel/Debug.h>
 | |
| #include <Kernel/FileSystem/FileDescription.h>
 | |
| #include <Kernel/Process.h>
 | |
| 
 | |
| namespace Kernel {
 | |
| 
 | |
| KResultOr<FlatPtr> Process::sys$fcntl(int fd, int cmd, u32 arg)
 | |
| {
 | |
|     VERIFY_PROCESS_BIG_LOCK_ACQUIRED(this);
 | |
|     REQUIRE_PROMISE(stdio);
 | |
|     dbgln_if(IO_DEBUG, "sys$fcntl: fd={}, cmd={}, arg={}", fd, cmd, arg);
 | |
|     auto description = fds().file_description(fd);
 | |
|     if (!description)
 | |
|         return EBADF;
 | |
|     // NOTE: The FD flags are not shared between FileDescription objects.
 | |
|     //       This means that dup() doesn't copy the FD_CLOEXEC flag!
 | |
|     switch (cmd) {
 | |
|     case F_DUPFD: {
 | |
|         int arg_fd = (int)arg;
 | |
|         if (arg_fd < 0)
 | |
|             return EINVAL;
 | |
|         auto new_fd_or_error = fds().allocate(arg_fd);
 | |
|         if (new_fd_or_error.is_error())
 | |
|             return new_fd_or_error.error();
 | |
|         auto new_fd = new_fd_or_error.release_value();
 | |
|         m_fds[new_fd.fd].set(*description);
 | |
|         return new_fd.fd;
 | |
|     }
 | |
|     case F_GETFD:
 | |
|         return m_fds[fd].flags();
 | |
|     case F_SETFD:
 | |
|         m_fds[fd].set_flags(arg);
 | |
|         break;
 | |
|     case F_GETFL:
 | |
|         return description->file_flags();
 | |
|     case F_SETFL:
 | |
|         description->set_file_flags(arg);
 | |
|         break;
 | |
|     case F_ISTTY:
 | |
|         return description->is_tty();
 | |
|     case F_GETLK:
 | |
|         return description->get_flock(Userspace<flock*>(arg));
 | |
|     case F_SETLK:
 | |
|         return description->apply_flock(Process::current(), Userspace<const flock*>(arg));
 | |
|     default:
 | |
|         return EINVAL;
 | |
|     }
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| }
 |