sstream.cpp: do a better job at handling the EINTR signal in recv() and send()

metadata
Wenzel Jakob 2013-12-04 16:50:03 +01:00
parent 065974cd83
commit 4f61786b71
3 changed files with 19 additions and 8 deletions

View File

@ -78,7 +78,7 @@ public:
std::string toString() const; std::string toString() const;
/// Handle the last socket-specific error (looks up the appropriate OS description) /// Handle the last socket-specific error (looks up the appropriate OS description)
static void handleError(const std::string &cmd, ELogLevel level = EError); static bool handleError(const std::string &cmd, ELogLevel level = EError);
//! @} //! @}
// ============================================================= // =============================================================

View File

@ -169,8 +169,10 @@ void SocketStream::read(void *ptr, size_t size) {
if (n == 0) { if (n == 0) {
throw EOFException("Connection closed while reading!", throw EOFException("Connection closed while reading!",
(size_t) (data - (char *) ptr)); (size_t) (data - (char *) ptr));
} else if (n == -1) { } else if (n == SOCKET_ERROR) {
handleError("recv", EWarn); if (!handleError("recv", EWarn))
continue; /* Wasn't an error after all -- continue */
throw EOFException("Connection closed while reading!", throw EOFException("Connection closed while reading!",
(size_t) (data - (char *) ptr)); (size_t) (data - (char *) ptr));
} }
@ -194,7 +196,9 @@ void SocketStream::write(const void *ptr, size_t size) {
int n = send(m_socket, data, size, 0); int n = send(m_socket, data, size, 0);
#endif #endif
if (n == SOCKET_ERROR) { if (n == SOCKET_ERROR) {
handleError("send", EWarn); if (!handleError("send", EWarn))
continue; /* Wasn't an error after all -- continue */
throw EOFException("Connection closed while writing!", throw EOFException("Connection closed while writing!",
(size_t) (data - (char *) ptr)); (size_t) (data - (char *) ptr));
} }
@ -242,7 +246,7 @@ void SocketStream::flush() {
/* Ignore */ /* Ignore */
} }
void SocketStream::handleError(const std::string &cmd, ELogLevel level) { bool SocketStream::handleError(const std::string &cmd, ELogLevel level) {
#ifndef WIN32 #ifndef WIN32
if (cmd.find("(") == std::string::npos) if (cmd.find("(") == std::string::npos)
Log(level, "Error in %s(): %s!", cmd.c_str(), strerror(errno)); Log(level, "Error in %s(): %s!", cmd.c_str(), strerror(errno));
@ -252,8 +256,13 @@ void SocketStream::handleError(const std::string &cmd, ELogLevel level) {
std::string err; std::string err;
int error = WSAGetLastError(); int error = WSAGetLastError();
switch (error) { switch (error) {
case WSABASEERR: err = "No Error"; break; case WSABASEERR: err = "Internal error (no reason given)"; break;
case WSAEINTR: err = "Interrupted system call"; break; case WSAEINTR:
if (level == EWarn) /* This is not really a warning -- just retry the operation. */
return false;
else
err = "Interrupted system call";
break;
case WSAEBADF: err = "Bad file number"; break; case WSAEBADF: err = "Bad file number"; break;
case WSAEACCES: err = "Permission denied"; break; case WSAEACCES: err = "Permission denied"; break;
case WSAEFAULT: err = "Bad address"; break; case WSAEFAULT: err = "Bad address"; break;
@ -309,6 +318,7 @@ void SocketStream::handleError(const std::string &cmd, ELogLevel level) {
else else
Log(level, "Error %i in %s: %s!", error, cmd.c_str(), err.c_str()); Log(level, "Error %i in %s: %s!", error, cmd.c_str(), err.c_str());
#endif #endif
return true;
} }
MTS_IMPLEMENT_CLASS(SocketStream, false, Stream) MTS_IMPLEMENT_CLASS(SocketStream, false, Stream)

View File

@ -34,6 +34,7 @@
#include <arpa/inet.h> #include <arpa/inet.h>
#include <sys/wait.h> #include <sys/wait.h>
#include <signal.h> #include <signal.h>
#define INVALID_SOCKET -1 #define INVALID_SOCKET -1
#define SOCKET int #define SOCKET int
#endif #endif