#include "fmacros.h"
#include <sys/types.h>
#include <sys/socket.h>
#include <sys/select.h>
#include <sys/un.h>
#include <netinet/in.h>
#include <netinet/tcp.h>
#include <arpa/inet.h>
#include <unistd.h>
#include <fcntl.h>
#include <string.h>
#include <netdb.h>
#include <errno.h>
#include <stdarg.h>
#include <stdio.h>
#include <poll.h>
#include <limits.h>
#include "net.h"
#include "sds.h"
void
__redisSetError(redisContext *c,
int
type,
const
char
*str);
static
void
__redisSetErrorFromErrno(redisContext *c,
int
type,
const
char
*prefix) {
char
buf[128];
size_t
len = 0;
if
(prefix != NULL)
len = snprintf(buf,
sizeof
(buf),
"%s: "
,prefix);
strerror_r(
errno
,buf+len,
sizeof
(buf)-len);
__redisSetError(c,type,buf);
}
static
int
redisSetReuseAddr(redisContext *c,
int
fd) {
int
on = 1;
if
(setsockopt(fd, SOL_SOCKET, SO_REUSEADDR, &on,
sizeof
(on)) == -1) {
__redisSetErrorFromErrno(c,REDIS_ERR_IO,NULL);
close(fd);
return
REDIS_ERR;
}
return
REDIS_OK;
}
static
int
redisCreateSocket(redisContext *c,
int
type) {
int
s;
if
((s = socket(type, SOCK_STREAM, 0)) == -1) {
__redisSetErrorFromErrno(c,REDIS_ERR_IO,NULL);
return
REDIS_ERR;
}
if
(type == AF_INET) {
if
(redisSetReuseAddr(c,s) == REDIS_ERR) {
return
REDIS_ERR;
}
}
return
s;
}
static
int
redisSetBlocking(redisContext *c,
int
fd,
int
blocking) {
int
flags;
if
((flags = fcntl(fd, F_GETFL)) == -1) {
__redisSetErrorFromErrno(c,REDIS_ERR_IO,
"fcntl(F_GETFL)"
);
close(fd);
return
REDIS_ERR;
}
if
(blocking)
flags &= ~O_NONBLOCK;
else
flags |= O_NONBLOCK;
if
(fcntl(fd, F_SETFL, flags) == -1) {
__redisSetErrorFromErrno(c,REDIS_ERR_IO,
"fcntl(F_SETFL)"
);
close(fd);
return
REDIS_ERR;
}
return
REDIS_OK;
}
static
int
redisSetTcpNoDelay(redisContext *c,
int
fd) {
int
yes = 1;
if
(setsockopt(fd, IPPROTO_TCP, TCP_NODELAY, &yes,
sizeof
(yes)) == -1) {
__redisSetErrorFromErrno(c,REDIS_ERR_IO,
"setsockopt(TCP_NODELAY)"
);
close(fd);
return
REDIS_ERR;
}
return
REDIS_OK;
}
#define __MAX_MSEC (((LONG_MAX) - 999) / 1000)
static
int
redisContextWaitReady(redisContext *c,
int
fd,
const
struct
timeval *timeout) {
struct
pollfd wfd[1];
long
msec;
msec = -1;
wfd[0].fd = fd;
wfd[0].events = POLLOUT;
if
(timeout != NULL) {
if
(timeout->tv_usec > 1000000 || timeout->tv_sec > __MAX_MSEC) {
close(fd);
return
REDIS_ERR;
}
msec = (timeout->tv_sec * 1000) + ((timeout->tv_usec + 999) / 1000);
if
(msec < 0 || msec > INT_MAX) {
msec = INT_MAX;
}
}
if
(
errno
== EINPROGRESS) {
int
res;
if
((res = poll(wfd, 1, msec)) == -1) {
__redisSetErrorFromErrno(c, REDIS_ERR_IO,
"poll(2)"
);
close(fd);
return
REDIS_ERR;
}
else
if
(res == 0) {
errno
= ETIMEDOUT;
__redisSetErrorFromErrno(c,REDIS_ERR_IO,NULL);
close(fd);
return
REDIS_ERR;
}
if
(redisCheckSocketError(c, fd) != REDIS_OK)
return
REDIS_ERR;
return
REDIS_OK;
}
__redisSetErrorFromErrno(c,REDIS_ERR_IO,NULL);
close(fd);
return
REDIS_ERR;
}
int
redisCheckSocketError(redisContext *c,
int
fd) {
int
err = 0;
socklen_t errlen =
sizeof
(err);
if
(getsockopt(fd, SOL_SOCKET, SO_ERROR, &err, &errlen) == -1) {
__redisSetErrorFromErrno(c,REDIS_ERR_IO,
"getsockopt(SO_ERROR)"
);
close(fd);
return
REDIS_ERR;
}
if
(err) {
errno
= err;
__redisSetErrorFromErrno(c,REDIS_ERR_IO,NULL);
close(fd);
return
REDIS_ERR;
}
return
REDIS_OK;
}
int
redisContextSetTimeout(redisContext *c,
struct
timeval tv) {
if
(setsockopt(c->fd,SOL_SOCKET,SO_RCVTIMEO,&tv,
sizeof
(tv)) == -1) {
__redisSetErrorFromErrno(c,REDIS_ERR_IO,
"setsockopt(SO_RCVTIMEO)"
);
return
REDIS_ERR;
}
if
(setsockopt(c->fd,SOL_SOCKET,SO_SNDTIMEO,&tv,
sizeof
(tv)) == -1) {
__redisSetErrorFromErrno(c,REDIS_ERR_IO,
"setsockopt(SO_SNDTIMEO)"
);
return
REDIS_ERR;
}
return
REDIS_OK;
}
int
redisContextConnectTcp(redisContext *c,
const
char
*addr,
int
port,
struct
timeval *timeout) {
int
s, rv;
char
_port[6];
struct
addrinfo hints, *servinfo, *p;
int
blocking = (c->flags & REDIS_BLOCK);
snprintf(_port, 6,
"%d"
, port);
memset
(&hints,0,
sizeof
(hints));
hints.ai_family = AF_INET;
hints.ai_socktype = SOCK_STREAM;
if
((rv = getaddrinfo(addr,_port,&hints,&servinfo)) != 0) {
__redisSetError(c,REDIS_ERR_OTHER,gai_strerror(rv));
return
REDIS_ERR;
}
for
(p = servinfo; p != NULL; p = p->ai_next) {
if
((s = socket(p->ai_family,p->ai_socktype,p->ai_protocol)) == -1)
continue
;
if
(redisSetBlocking(c,s,0) != REDIS_OK)
goto
error;
if
(connect(s,p->ai_addr,p->ai_addrlen) == -1) {
if
(
errno
== EHOSTUNREACH) {
close(s);
continue
;
}
else
if
(
errno
== EINPROGRESS && !blocking) {
}
else
{
if
(redisContextWaitReady(c,s,timeout) != REDIS_OK)
goto
error;
}
}
if
(blocking && redisSetBlocking(c,s,1) != REDIS_OK)
goto
error;
if
(redisSetTcpNoDelay(c,s) != REDIS_OK)
goto
error;
c->fd = s;
c->flags |= REDIS_CONNECTED;
rv = REDIS_OK;
goto
end;
}
if
(p == NULL) {
char
buf[128];
snprintf(buf,
sizeof
(buf),
"Can't create socket: %s"
,
strerror
(
errno
));
__redisSetError(c,REDIS_ERR_OTHER,buf);
goto
error;
}
error:
rv = REDIS_ERR;
end:
freeaddrinfo(servinfo);
return
rv;
}
int
redisContextConnectUnix(redisContext *c,
const
char
*path,
struct
timeval *timeout) {
int
s;
int
blocking = (c->flags & REDIS_BLOCK);
struct
sockaddr_un sa;
if
((s = redisCreateSocket(c,AF_LOCAL)) < 0)
return
REDIS_ERR;
if
(redisSetBlocking(c,s,0) != REDIS_OK)
return
REDIS_ERR;
sa.sun_family = AF_LOCAL;
strncpy
(sa.sun_path,path,
sizeof
(sa.sun_path)-1);
if
(connect(s, (
struct
sockaddr*)&sa,
sizeof
(sa)) == -1) {
if
(
errno
== EINPROGRESS && !blocking) {
}
else
{
if
(redisContextWaitReady(c,s,timeout) != REDIS_OK)
return
REDIS_ERR;
}
}
if
(blocking && redisSetBlocking(c,s,1) != REDIS_OK)
return
REDIS_ERR;
c->fd = s;
c->flags |= REDIS_CONNECTED;
return
REDIS_OK;
}