write.cpp 3.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109
  1. /*
  2. * Copyright (c) 2018-2020, Andreas Kling <kling@serenityos.org>
  3. *
  4. * SPDX-License-Identifier: BSD-2-Clause
  5. */
  6. #include <AK/NumericLimits.h>
  7. #include <Kernel/Debug.h>
  8. #include <Kernel/FileSystem/OpenFileDescription.h>
  9. #include <Kernel/Process.h>
  10. namespace Kernel {
  11. ErrorOr<FlatPtr> Process::sys$writev(int fd, Userspace<const struct iovec*> iov, int iov_count)
  12. {
  13. VERIFY_PROCESS_BIG_LOCK_ACQUIRED(this)
  14. REQUIRE_PROMISE(stdio);
  15. if (iov_count < 0)
  16. return EINVAL;
  17. // Arbitrary pain threshold.
  18. if (iov_count > (int)MiB)
  19. return EFAULT;
  20. u64 total_length = 0;
  21. Vector<iovec, 32> vecs;
  22. TRY(vecs.try_resize(iov_count));
  23. TRY(copy_n_from_user(vecs.data(), iov, iov_count));
  24. for (auto& vec : vecs) {
  25. total_length += vec.iov_len;
  26. if (total_length > NumericLimits<i32>::max())
  27. return EINVAL;
  28. }
  29. auto description = TRY(fds().open_file_description(fd));
  30. if (!description->is_writable())
  31. return EBADF;
  32. int nwritten = 0;
  33. for (auto& vec : vecs) {
  34. auto buffer = TRY(UserOrKernelBuffer::for_user_buffer((u8*)vec.iov_base, vec.iov_len));
  35. auto result = do_write(*description, buffer, vec.iov_len);
  36. if (result.is_error()) {
  37. if (nwritten == 0)
  38. return result.release_error();
  39. return nwritten;
  40. }
  41. nwritten += result.value();
  42. }
  43. return nwritten;
  44. }
  45. ErrorOr<FlatPtr> Process::do_write(OpenFileDescription& description, const UserOrKernelBuffer& data, size_t data_size)
  46. {
  47. size_t total_nwritten = 0;
  48. if (description.should_append() && description.file().is_seekable()) {
  49. TRY(description.seek(0, SEEK_END));
  50. }
  51. while (total_nwritten < data_size) {
  52. while (!description.can_write()) {
  53. if (!description.is_blocking()) {
  54. if (total_nwritten > 0)
  55. return total_nwritten;
  56. else
  57. return EAGAIN;
  58. }
  59. auto unblock_flags = Thread::FileBlocker::BlockFlags::None;
  60. if (Thread::current()->block<Thread::WriteBlocker>({}, description, unblock_flags).was_interrupted()) {
  61. if (total_nwritten == 0)
  62. return EINTR;
  63. }
  64. // TODO: handle exceptions in unblock_flags
  65. }
  66. auto nwritten_or_error = description.write(data.offset(total_nwritten), data_size - total_nwritten);
  67. if (nwritten_or_error.is_error()) {
  68. if (total_nwritten > 0)
  69. return total_nwritten;
  70. if (nwritten_or_error.error().code() == EAGAIN)
  71. continue;
  72. return nwritten_or_error.release_error();
  73. }
  74. VERIFY(nwritten_or_error.value() > 0);
  75. total_nwritten += nwritten_or_error.value();
  76. }
  77. return total_nwritten;
  78. }
  79. ErrorOr<FlatPtr> Process::sys$write(int fd, Userspace<const u8*> data, size_t size)
  80. {
  81. VERIFY_PROCESS_BIG_LOCK_ACQUIRED(this)
  82. REQUIRE_PROMISE(stdio);
  83. if (size == 0)
  84. return 0;
  85. if (size > NumericLimits<ssize_t>::max())
  86. return EINVAL;
  87. dbgln_if(IO_DEBUG, "sys$write({}, {}, {})", fd, data.ptr(), size);
  88. auto description = TRY(fds().open_file_description(fd));
  89. if (!description->is_writable())
  90. return EBADF;
  91. auto buffer = TRY(UserOrKernelBuffer::for_user_buffer(data, static_cast<size_t>(size)));
  92. return do_write(*description, buffer, size);
  93. }
  94. }