write.cpp 3.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120
  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/FileDescription.h>
  9. #include <Kernel/Process.h>
  10. namespace Kernel {
  11. KResultOr<FlatPtr> Process::sys$writev(int fd, Userspace<const struct iovec*> iov, int iov_count)
  12. {
  13. REQUIRE_PROMISE(stdio);
  14. if (iov_count < 0)
  15. return EINVAL;
  16. // Arbitrary pain threshold.
  17. if (iov_count > (int)MiB)
  18. return EFAULT;
  19. u64 total_length = 0;
  20. Vector<iovec, 32> vecs;
  21. if (!vecs.try_resize(iov_count))
  22. return ENOMEM;
  23. if (!copy_n_from_user(vecs.data(), iov, iov_count))
  24. return EFAULT;
  25. for (auto& vec : vecs) {
  26. total_length += vec.iov_len;
  27. if (total_length > NumericLimits<i32>::max())
  28. return EINVAL;
  29. }
  30. auto description = file_description(fd);
  31. if (!description)
  32. return EBADF;
  33. if (!description->is_writable())
  34. return EBADF;
  35. int nwritten = 0;
  36. for (auto& vec : vecs) {
  37. auto buffer = UserOrKernelBuffer::for_user_buffer((u8*)vec.iov_base, vec.iov_len);
  38. if (!buffer.has_value())
  39. return EFAULT;
  40. auto result = do_write(*description, buffer.value(), vec.iov_len);
  41. if (result.is_error()) {
  42. if (nwritten == 0)
  43. return result.error();
  44. return nwritten;
  45. }
  46. nwritten += result.value();
  47. }
  48. return nwritten;
  49. }
  50. KResultOr<FlatPtr> Process::do_write(FileDescription& description, const UserOrKernelBuffer& data, size_t data_size)
  51. {
  52. size_t total_nwritten = 0;
  53. if (description.should_append() && description.file().is_seekable()) {
  54. auto seek_result = description.seek(0, SEEK_END);
  55. if (seek_result.is_error())
  56. return seek_result.error();
  57. }
  58. while (total_nwritten < data_size) {
  59. while (!description.can_write()) {
  60. if (!description.is_blocking()) {
  61. if (total_nwritten > 0)
  62. return total_nwritten;
  63. else
  64. return EAGAIN;
  65. }
  66. auto unblock_flags = Thread::FileBlocker::BlockFlags::None;
  67. if (Thread::current()->block<Thread::WriteBlocker>({}, description, unblock_flags).was_interrupted()) {
  68. if (total_nwritten == 0)
  69. return EINTR;
  70. }
  71. // TODO: handle exceptions in unblock_flags
  72. }
  73. auto nwritten_or_error = description.write(data.offset(total_nwritten), data_size - total_nwritten);
  74. if (nwritten_or_error.is_error()) {
  75. if (total_nwritten > 0)
  76. return total_nwritten;
  77. if (nwritten_or_error.error() == EAGAIN)
  78. continue;
  79. return nwritten_or_error.error();
  80. }
  81. VERIFY(nwritten_or_error.value() > 0);
  82. total_nwritten += nwritten_or_error.value();
  83. }
  84. return total_nwritten;
  85. }
  86. KResultOr<FlatPtr> Process::sys$write(int fd, Userspace<const u8*> data, size_t size)
  87. {
  88. REQUIRE_PROMISE(stdio);
  89. if (size == 0)
  90. return 0;
  91. if (size > NumericLimits<ssize_t>::max())
  92. return EINVAL;
  93. dbgln_if(IO_DEBUG, "sys$write({}, {}, {})", fd, data.ptr(), size);
  94. auto description = file_description(fd);
  95. if (!description)
  96. return EBADF;
  97. if (!description->is_writable())
  98. return EBADF;
  99. auto buffer = UserOrKernelBuffer::for_user_buffer(data, static_cast<size_t>(size));
  100. if (!buffer.has_value())
  101. return EFAULT;
  102. return do_write(*description, buffer.value(), size);
  103. }
  104. }