summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorMark Nunberg <mnunberg@haskalah.org>2019-04-11 15:08:27 -0400
committerMark Nunberg <mnunberg@haskalah.org>2019-04-11 15:08:27 -0400
commit5aa2397f9ec494380774fb9155446f14ebbf3d1f (patch)
treea032bb704b4c22db1aae60b494ed51680b1b3dc0
parentdc3c6ce85cf17e1da653a4a17bf3cca69ed02732 (diff)
fix blocking timeouts on SSL reads/writes
-rw-r--r--sslio.c25
1 files changed, 24 insertions, 1 deletions
diff --git a/sslio.c b/sslio.c
index efbf06e..25cb5c2 100644
--- a/sslio.c
+++ b/sslio.c
@@ -4,6 +4,7 @@
#include <assert.h>
#ifdef HIREDIS_SSL
#include <pthread.h>
+#include <errno.h>
void __redisSetError(redisContext *c, int type, const char *str);
@@ -184,6 +185,28 @@ int redisSslRead(redisContext *c, char *buf, size_t bufcap) {
return -1;
} else {
int err = SSL_get_error(c->ssl->ssl, nread);
+ if (c->flags & REDIS_BLOCK) {
+ /**
+ * In blocking mode, we should never end up in a situation where
+ * we get an error without it being an actual error, except
+ * in the case of EINTR, which can be spuriously received from
+ * debuggers or whatever.
+ */
+ if (errno == EINTR) {
+ return 0;
+ } else {
+ const char *msg = NULL;
+ if (errno == EAGAIN) {
+ msg = "Timed out";
+ }
+ __redisSetError(c, REDIS_ERR_IO, msg);
+ return -1;
+ }
+ }
+
+ /**
+ * We can very well get an EWOULDBLOCK/EAGAIN, however
+ */
if (maybeCheckWant(c->ssl, err)) {
return 0;
} else {
@@ -203,7 +226,7 @@ int redisSslWrite(redisContext *c) {
c->ssl->lastLen = len;
int err = SSL_get_error(c->ssl->ssl, rv);
- if (maybeCheckWant(c->ssl, err)) {
+ if ((c->flags & REDIS_BLOCK) == 0 && maybeCheckWant(c->ssl, err)) {
return 0;
} else {
__redisSetError(c, REDIS_ERR_IO, NULL);