#include "ppsspp_config.h" #ifdef _WIN32 #include #undef min #undef max #else #include #include #endif #include #include #ifndef MSG_NOSIGNAL // Default value to 0x00 (do nothing) in systems where it's not supported. #define MSG_NOSIGNAL 0x00 #endif #include "Common/File/FileDescriptor.h" #include "Common/Log.h" #include "Common/Net/NetBuffer.h" #include "Common/TimeUtil.h" namespace net { void RequestProgress::Update(int64_t downloaded, int64_t totalBytes, bool done) { if (totalBytes) { progress = (double)downloaded / (double)totalBytes; } else { progress = 0.01f; } if (callback) { callback(downloaded, totalBytes, done); } } bool Buffer::FlushSocket(uintptr_t sock, double timeout, bool *cancelled) { static constexpr float CANCEL_INTERVAL = 0.25f; for (size_t pos = 0, end = data_.size(); pos < end; ) { bool ready = false; double endTimeout = time_now_d() + timeout; while (!ready) { if (cancelled && *cancelled) return false; ready = fd_util::WaitUntilReady(sock, CANCEL_INTERVAL, true); if (!ready && time_now_d() > endTimeout) { ERROR_LOG(Log::IO, "FlushSocket timed out"); return false; } } int sent = send(sock, &data_[pos], end - pos, MSG_NOSIGNAL); // TODO: Do we need some retry logic here, instead of just giving up? if (sent < 0) { ERROR_LOG(Log::IO, "FlushSocket failed to send: %d", errno); return false; } pos += sent; } data_.resize(0); return true; } bool Buffer::ReadAllWithProgress(int fd, int knownSize, RequestProgress *progress) { static constexpr float CANCEL_INTERVAL = 0.25f; std::vector buf; // We're non-blocking and reading from an OS buffer, so try to read as much as we can at a time. if (knownSize >= 65536 * 16) { buf.resize(65536); } else if (knownSize >= 1024 * 16) { buf.resize(knownSize / 16); } else { buf.resize(1024); } double st = time_now_d(); int total = 0; while (true) { bool ready = false; // If we might need to cancel, check on a timer for it to be ready. // After this, we'll block on reading so we do this while first if we have a cancel pointer. while (!ready && progress && progress->cancelled) { if (*progress->cancelled) return false; ready = fd_util::WaitUntilReady(fd, CANCEL_INTERVAL, false); } int retval = recv(fd, &buf[0], buf.size(), MSG_NOSIGNAL); if (retval == 0) { return true; } else if (retval < 0) { #if PPSSPP_PLATFORM(WINDOWS) if (WSAGetLastError() != WSAEWOULDBLOCK) { #else if (errno != EWOULDBLOCK) { #endif ERROR_LOG(Log::IO, "Error reading from buffer: %i", retval); return false; } // Just try again on a would block error, not a real error. continue; } char *p = Append((size_t)retval); memcpy(p, &buf[0], retval); total += retval; if (progress) { progress->Update(total, knownSize, false); progress->kBps = (float)(total / (time_now_d() - st)) / 1024.0f; } } return true; } int Buffer::Read(int fd, size_t sz) { char buf[1024]; int retval; size_t received = 0; while ((retval = recv(fd, buf, std::min(sz, sizeof(buf)), MSG_NOSIGNAL)) > 0) { if (retval < 0) { return retval; } char *p = Append((size_t)retval); memcpy(p, buf, retval); sz -= retval; received += retval; if (sz == 0) return 0; } return (int)received; } } // namespace