From eeda958f332025a0cbdb5de8c49ba84836c9a10a Mon Sep 17 00:00:00 2001 From: Nekotekina Date: Sun, 25 Dec 2022 14:48:56 +0300 Subject: [PATCH] fs: fix read/read_at/write (Unix) It's known that too huge read can require multiple syscalls. --- Utilities/File.cpp | 43 +++++++++++++++++++++++++++++++++++++------ 1 file changed, 37 insertions(+), 6 deletions(-) diff --git a/Utilities/File.cpp b/Utilities/File.cpp index 046b252a33..8b0b90d52a 100644 --- a/Utilities/File.cpp +++ b/Utilities/File.cpp @@ -1382,24 +1382,55 @@ fs::file::file(const std::string& path, bs_t mode) u64 read(void* buffer, u64 count) override { - const auto result = ::read(m_fd, buffer, count); - ensure(result != -1); // "file::read" + u64 result = 0; + + // Loop because (huge?) read can be processed partially + while (auto r = ::read(m_fd, buffer, count)) + { + ensure(r > 0); // "file::read" + count -= r; + result += r; + buffer = static_cast(buffer) + r; + if (!count) + break; + } return result; } u64 read_at(u64 offset, void* buffer, u64 count) override { - const auto result = ::pread(m_fd, buffer, count, offset); - ensure(result != -1); + u64 result = 0; + + // For safety; see read() + while (auto r = ::pread(m_fd, buffer, count, offset)) + { + ensure(r > 0); // "file::read_at" + count -= r; + offset += r; + result += r; + buffer = static_cast(buffer) + r; + if (!count) + break; + } return result; } u64 write(const void* buffer, u64 count) override { - const auto result = ::write(m_fd, buffer, count); - ensure(result != -1); // "file::write" + u64 result = 0; + + // For safety; see read() + while (auto r = ::write(m_fd, buffer, count)) + { + ensure(r > 0); // "file::write" + count -= r; + result += r; + buffer = static_cast(buffer) + r; + if (!count) + break; + } return result; }