sendfd.cpp 2.0 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768
  1. /*
  2. * Copyright (c) 2018-2020, Andreas Kling <kling@serenityos.org>
  3. *
  4. * SPDX-License-Identifier: BSD-2-Clause
  5. */
  6. #include <Kernel/FileSystem/FileDescription.h>
  7. #include <Kernel/Net/LocalSocket.h>
  8. #include <Kernel/Process.h>
  9. namespace Kernel {
  10. KResultOr<FlatPtr> Process::sys$sendfd(int sockfd, int fd)
  11. {
  12. VERIFY_PROCESS_BIG_LOCK_ACQUIRED(this)
  13. REQUIRE_PROMISE(sendfd);
  14. auto socket_description = fds().file_description(sockfd);
  15. if (!socket_description)
  16. return EBADF;
  17. if (!socket_description->is_socket())
  18. return ENOTSOCK;
  19. auto& socket = *socket_description->socket();
  20. if (!socket.is_local())
  21. return EAFNOSUPPORT;
  22. if (!socket.is_connected())
  23. return ENOTCONN;
  24. auto passing_descriptor = fds().file_description(fd);
  25. if (!passing_descriptor)
  26. return EBADF;
  27. auto& local_socket = static_cast<LocalSocket&>(socket);
  28. return local_socket.sendfd(*socket_description, *passing_descriptor);
  29. }
  30. KResultOr<FlatPtr> Process::sys$recvfd(int sockfd, int options)
  31. {
  32. VERIFY_PROCESS_BIG_LOCK_ACQUIRED(this)
  33. REQUIRE_PROMISE(recvfd);
  34. auto socket_description = fds().file_description(sockfd);
  35. if (!socket_description)
  36. return EBADF;
  37. if (!socket_description->is_socket())
  38. return ENOTSOCK;
  39. auto& socket = *socket_description->socket();
  40. if (!socket.is_local())
  41. return EAFNOSUPPORT;
  42. auto new_fd_or_error = m_fds.allocate();
  43. if (new_fd_or_error.is_error())
  44. return new_fd_or_error.error();
  45. auto new_fd = new_fd_or_error.release_value();
  46. auto& local_socket = static_cast<LocalSocket&>(socket);
  47. auto received_descriptor_or_error = local_socket.recvfd(*socket_description);
  48. if (received_descriptor_or_error.is_error())
  49. return received_descriptor_or_error.error();
  50. u32 fd_flags = 0;
  51. if (options & O_CLOEXEC)
  52. fd_flags |= FD_CLOEXEC;
  53. m_fds[new_fd.fd].set(*received_descriptor_or_error.value(), fd_flags);
  54. return new_fd.fd;
  55. }
  56. }