-
Notifications
You must be signed in to change notification settings - Fork 10
/
Copy pathpre_connector.hpp
199 lines (153 loc) · 4.92 KB
/
pre_connector.hpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
#pragma once
#include "logging.hpp"
#include <netdb.h> //hostent
#include <arpa/inet.h>
#include <fcntl.h>
#include <sys/time.h>
#include <unistd.h>
#include <string.h>
#include <errno.h>
#include <map>
namespace rdmaio {
constexpr struct timeval default_timeout = {0,8000};
constexpr struct timeval no_timeout = {0,0}; // it means forever
inline __attribute__ ((always_inline)) // inline to avoid multiple-definiations
int64_t diff_time(const struct timeval &end, const struct timeval &start) {
int64_t diff = (end.tv_sec > start.tv_sec)?(end.tv_sec - start.tv_sec) * 1000:0;
if (end.tv_usec > start.tv_usec) {
diff += (end.tv_usec - start.tv_usec);
} else {
diff -= (start.tv_usec - end.tv_usec);
}
return diff;
}
class PreConnector { // helper class used to exchange QP information using TCP/IP
public:
static int get_listen_socket(const std::string &addr,int port) {
struct sockaddr_in serv_addr;
auto sockfd = socket(AF_INET, SOCK_STREAM, 0);
RDMA_ASSERT(sockfd >= 0) << "ERROR opening listen socket: " << strerror(errno);
/* setup the host_addr structure for use in bind call */
// server byte order
serv_addr.sin_family = AF_INET;
serv_addr.sin_addr.s_addr = INADDR_ANY;
// port
serv_addr.sin_port = htons(port);
RDMA_ASSERT(bind(sockfd, (struct sockaddr *) &serv_addr,
sizeof(serv_addr)) == 0) << "ERROR on binding: " << strerror(errno);
return sockfd;
}
static int get_send_socket(const std::string &addr,int port,struct timeval timeout = default_timeout) {
int sockfd;
struct sockaddr_in serv_addr;
RDMA_ASSERT((sockfd = socket(AF_INET, SOCK_STREAM, 0)) >= 0) << "Error open socket for send!";
fcntl(sockfd, F_SETFL, O_NONBLOCK);
serv_addr.sin_family = AF_INET;
serv_addr.sin_port = htons(port);
auto ip = host_to_ip(addr);
if(ip == "") {
close(sockfd);
return -1;
}
serv_addr.sin_addr.s_addr = inet_addr(ip.c_str());
if(connect(sockfd, (struct sockaddr *) &serv_addr, sizeof(serv_addr)) == -1) {
if (errno == EINPROGRESS) {
goto PROGRESS;
}
close(sockfd);
return -1;
}
PROGRESS:
// check return status
fd_set fdset;
FD_ZERO(&fdset);
FD_SET(sockfd, &fdset);
if(select(sockfd + 1, NULL, &fdset, NULL, &timeout) == 1)
{
int so_error;
socklen_t len = sizeof so_error;
getsockopt(sockfd, SOL_SOCKET, SO_ERROR, &so_error, &len);
if (so_error == 0) {
// success
} else {
close(sockfd);
return -1;
}
}
return sockfd;
}
// timeout in microsend
static bool wait_recv(int socket, uint32_t timeout = 2000) {
while(true) {
fd_set rfds;
FD_ZERO(&rfds);
FD_SET(socket, &rfds);
struct timeval s_timeout = {0,timeout};
int ready = select(socket + 1, &rfds, NULL, NULL, &s_timeout);
RDMA_ASSERT(ready != -1);
if(ready == 0) { // no file descriptor found
continue;
}
if(ready < 0) { // error case
RDMA_ASSERT(false) << "select error " << strerror(errno);
}
if (FD_ISSET(socket, &rfds)) {
break; // ready
}
}
return true;
}
static void wait_close(int socket) {
shutdown(socket, SHUT_WR);
char buf[2];
struct timeval timeout={1,0};
auto ret = setsockopt(socket,SOL_SOCKET,SO_RCVTIMEO,(const char*)&timeout,sizeof(timeout));
RDMA_ASSERT(ret == 0);
recv(socket,buf,2,0);
close(socket);
}
static int send_to(int fd, char *usrbuf, size_t n) {
size_t nleft = n;
ssize_t nwritten;
char *bufp = usrbuf;
while (nleft > 0) {
if ((nwritten = write(fd, bufp, nleft)) <= 0) {
if (errno == EINTR) /* Interrupted by sig handler return */
nwritten = 0; /* and call write() again */
else
return -1; /* errno set by write() */
}
nleft -= nwritten;
bufp += nwritten;
}
return n;
}
typedef std::map<std::string,std::string> ipmap_t;
static ipmap_t &local_ip_cache() {
static __thread ipmap_t cache;
return cache;
}
static std::string host_to_ip(const std::string &host) {
ipmap_t cache = local_ip_cache();
if(cache.find(host) != cache.end())
return cache[host];
std::string res = "";
struct addrinfo hints, *infoptr;
memset(&hints, 0, sizeof hints);
hints.ai_family = AF_INET; // AF_INET means IPv4 only addresses
int result = getaddrinfo(host.c_str(), NULL, &hints, &infoptr);
if (result) {
fprintf(stderr, "getaddrinfo: %s at %s\n", gai_strerror(result),host.c_str());
return "";
}
char ip[64]; memset(ip,0,sizeof(ip));
for(struct addrinfo *p = infoptr; p != NULL; p = p->ai_next) {
getnameinfo(p->ai_addr, p->ai_addrlen, ip, sizeof(ip), NULL, 0, NI_NUMERICHOST);
}
res = std::string(ip);
if(res != "")
cache.insert(std::make_pair(host,res));
return res;
}
};
}; // namespace rdmaio