lwtarantool 0.0.2
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- checksums.yaml +7 -0
- data/README.md +84 -0
- data/ext/lwtarantool/conn.c +314 -0
- data/ext/lwtarantool/depend +23 -0
- data/ext/lwtarantool/errors.c +24 -0
- data/ext/lwtarantool/extconf.rb +8 -0
- data/ext/lwtarantool/lwtarantool.c +12 -0
- data/ext/lwtarantool/lwtarantool.h +38 -0
- data/ext/lwtarantool/request.c +124 -0
- data/ext/lwtarantool/vendor/msgpuck/CMakeLists.txt +73 -0
- data/ext/lwtarantool/vendor/msgpuck/hints.c +674 -0
- data/ext/lwtarantool/vendor/msgpuck/msgpuck.c +375 -0
- data/ext/lwtarantool/vendor/msgpuck/msgpuck.h +2195 -0
- data/ext/lwtarantool/vendor/msgpuck/test/CMakeLists.txt +25 -0
- data/ext/lwtarantool/vendor/msgpuck/test/msgpuck.c +1126 -0
- data/ext/lwtarantool/vendor/msgpuck/test/test.c +105 -0
- data/ext/lwtarantool/vendor/msgpuck/test/test.h +123 -0
- data/ext/lwtarantool/vendor/tarantool-c/CMakeLists.txt +89 -0
- data/ext/lwtarantool/vendor/tarantool-c/cmake/FindMsgPuck.cmake +18 -0
- data/ext/lwtarantool/vendor/tarantool-c/cmake/FindSphinx.cmake +10 -0
- data/ext/lwtarantool/vendor/tarantool-c/doc/CMakeLists.txt +15 -0
- data/ext/lwtarantool/vendor/tarantool-c/doc/source/example.c +340 -0
- data/ext/lwtarantool/vendor/tarantool-c/include/CMakeLists.txt +6 -0
- data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tarantool.h +67 -0
- data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_auth.h +72 -0
- data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_buf.h +88 -0
- data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_call.h +83 -0
- data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_delete.h +52 -0
- data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_execute.h +24 -0
- data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_insert.h +62 -0
- data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_io.h +67 -0
- data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_iob.h +62 -0
- data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_iter.h +301 -0
- data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_mem.h +102 -0
- data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_net.h +242 -0
- data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_object.h +248 -0
- data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_opt.h +138 -0
- data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_ping.h +49 -0
- data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_proto.h +295 -0
- data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_reply.h +204 -0
- data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_request.h +377 -0
- data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_schema.h +165 -0
- data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_select.h +59 -0
- data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_stream.h +79 -0
- data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_update.h +226 -0
- data/ext/lwtarantool/vendor/tarantool-c/include/tp.h +1998 -0
- data/ext/lwtarantool/vendor/tarantool-c/test/CMakeLists.txt +36 -0
- data/ext/lwtarantool/vendor/tarantool-c/test/common.c +233 -0
- data/ext/lwtarantool/vendor/tarantool-c/test/common.h +28 -0
- data/ext/lwtarantool/vendor/tarantool-c/test/plain_test.c +152 -0
- data/ext/lwtarantool/vendor/tarantool-c/test/tarantool_call.c +57 -0
- data/ext/lwtarantool/vendor/tarantool-c/test/tarantool_disconnect.c +31 -0
- data/ext/lwtarantool/vendor/tarantool-c/test/tarantool_tcp.c +840 -0
- data/ext/lwtarantool/vendor/tarantool-c/test/tarantool_unix.c +384 -0
- data/ext/lwtarantool/vendor/tarantool-c/test/test.c +72 -0
- data/ext/lwtarantool/vendor/tarantool-c/test/test.h +94 -0
- data/ext/lwtarantool/vendor/tarantool-c/test/tnt_assoc.c +6 -0
- data/ext/lwtarantool/vendor/tarantool-c/test/tnt_assoc.h +67 -0
- data/ext/lwtarantool/vendor/tarantool-c/third_party/PMurHash.c +317 -0
- data/ext/lwtarantool/vendor/tarantool-c/third_party/PMurHash.h +68 -0
- data/ext/lwtarantool/vendor/tarantool-c/third_party/base64.c +300 -0
- data/ext/lwtarantool/vendor/tarantool-c/third_party/base64.h +91 -0
- data/ext/lwtarantool/vendor/tarantool-c/third_party/mhash.h +589 -0
- data/ext/lwtarantool/vendor/tarantool-c/third_party/sha1.c +219 -0
- data/ext/lwtarantool/vendor/tarantool-c/third_party/sha1.h +24 -0
- data/ext/lwtarantool/vendor/tarantool-c/third_party/uri.c +6528 -0
- data/ext/lwtarantool/vendor/tarantool-c/third_party/uri.h +81 -0
- data/ext/lwtarantool/vendor/tarantool-c/tnt/CMakeLists.txt +83 -0
- data/ext/lwtarantool/vendor/tarantool-c/tnt/pmatomic.h +494 -0
- data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_assoc.c +9 -0
- data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_assoc.h +172 -0
- data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_auth.c +118 -0
- data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_buf.c +171 -0
- data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_call.c +79 -0
- data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_delete.c +52 -0
- data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_execute.c +60 -0
- data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_insert.c +60 -0
- data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_io.c +520 -0
- data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_iob.c +75 -0
- data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_iter.c +293 -0
- data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_mem.c +82 -0
- data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_net.c +336 -0
- data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_object.c +476 -0
- data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_opt.c +117 -0
- data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_ping.c +38 -0
- data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_proto_internal.h +43 -0
- data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_reply.c +300 -0
- data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_request.c +336 -0
- data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_schema.c +334 -0
- data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_select.c +58 -0
- data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_stream.c +71 -0
- data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_update.c +247 -0
- data/ext/lwtarantool/vendor/tarantool-c/tntrpl/CMakeLists.txt +69 -0
- data/ext/lwtarantool/vendor/tarantool-c/tntrpl/tnt_dir.c +173 -0
- data/ext/lwtarantool/vendor/tarantool-c/tntrpl/tnt_dir.h +58 -0
- data/ext/lwtarantool/vendor/tarantool-c/tntrpl/tnt_log.c +329 -0
- data/ext/lwtarantool/vendor/tarantool-c/tntrpl/tnt_log.h +119 -0
- data/ext/lwtarantool/vendor/tarantool-c/tntrpl/tnt_rpl.c +189 -0
- data/ext/lwtarantool/vendor/tarantool-c/tntrpl/tnt_rpl.h +47 -0
- data/ext/lwtarantool/vendor/tarantool-c/tntrpl/tnt_snapshot.c +163 -0
- data/ext/lwtarantool/vendor/tarantool-c/tntrpl/tnt_snapshot.h +50 -0
- data/ext/lwtarantool/vendor/tarantool-c/tntrpl/tnt_xlog.c +163 -0
- data/ext/lwtarantool/vendor/tarantool-c/tntrpl/tnt_xlog.h +50 -0
- data/lib/lwtarantool/connection.rb +84 -0
- data/lib/lwtarantool/request.rb +58 -0
- data/lib/lwtarantool.rb +23 -0
- metadata +164 -0
@@ -0,0 +1,520 @@
|
|
1
|
+
|
2
|
+
/*
|
3
|
+
* Redistribution and use in source and binary forms, with or
|
4
|
+
* without modification, are permitted provided that the following
|
5
|
+
* conditions are met:
|
6
|
+
*
|
7
|
+
* 1. Redistributions of source code must retain the above
|
8
|
+
* copyright notice, this list of conditions and the
|
9
|
+
* following disclaimer.
|
10
|
+
*
|
11
|
+
* 2. Redistributions in binary form must reproduce the above
|
12
|
+
* copyright notice, this list of conditions and the following
|
13
|
+
* disclaimer in the documentation and/or other materials
|
14
|
+
* provided with the distribution.
|
15
|
+
*
|
16
|
+
* THIS SOFTWARE IS PROVIDED BY <COPYRIGHT HOLDER> ``AS IS'' AND
|
17
|
+
* ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
|
18
|
+
* TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
19
|
+
* A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL
|
20
|
+
* <COPYRIGHT HOLDER> OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT,
|
21
|
+
* INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
22
|
+
* DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
|
23
|
+
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR
|
24
|
+
* BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
|
25
|
+
* LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
26
|
+
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF
|
27
|
+
* THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
|
28
|
+
* SUCH DAMAGE.
|
29
|
+
*/
|
30
|
+
/* need this to get IOV_MAX on some platforms. */
|
31
|
+
#ifndef __need_IOV_MAX
|
32
|
+
#define __need_IOV_MAX
|
33
|
+
#endif
|
34
|
+
#include <limits.h>
|
35
|
+
|
36
|
+
#include <stdlib.h>
|
37
|
+
#include <stdio.h>
|
38
|
+
#include <stdarg.h>
|
39
|
+
#include <string.h>
|
40
|
+
#include <stdbool.h>
|
41
|
+
|
42
|
+
#include <sys/time.h>
|
43
|
+
#include <sys/types.h>
|
44
|
+
#include <sys/socket.h>
|
45
|
+
#include <sys/uio.h>
|
46
|
+
#include <netinet/in.h>
|
47
|
+
#include <sys/un.h>
|
48
|
+
#include <netinet/tcp.h>
|
49
|
+
#include <netdb.h>
|
50
|
+
#include <unistd.h>
|
51
|
+
#include <fcntl.h>
|
52
|
+
#include <errno.h>
|
53
|
+
|
54
|
+
#include <tarantool/tnt_net.h>
|
55
|
+
#include <tarantool/tnt_io.h>
|
56
|
+
|
57
|
+
#include <uri.h>
|
58
|
+
|
59
|
+
#if !defined(MIN)
|
60
|
+
# define MIN(a, b) (a) < (b) ? (a) : (b)
|
61
|
+
#endif /* !defined(MIN) */
|
62
|
+
|
63
|
+
static enum tnt_error
|
64
|
+
tnt_io_resolve(struct sockaddr_in *addr,
|
65
|
+
const char *hostname, unsigned short port)
|
66
|
+
{
|
67
|
+
memset(addr, 0, sizeof(struct sockaddr_in));
|
68
|
+
addr->sin_family = AF_INET;
|
69
|
+
addr->sin_port = htons(port);
|
70
|
+
struct addrinfo *addr_info = NULL;
|
71
|
+
if (getaddrinfo(hostname, NULL, NULL, &addr_info) == 0 &&
|
72
|
+
addr_info != NULL) {
|
73
|
+
memcpy(&addr->sin_addr,
|
74
|
+
(void*)&((struct sockaddr_in *)addr_info->ai_addr)->sin_addr,
|
75
|
+
sizeof(addr->sin_addr));
|
76
|
+
freeaddrinfo(addr_info);
|
77
|
+
return TNT_EOK;
|
78
|
+
}
|
79
|
+
if (addr_info)
|
80
|
+
freeaddrinfo(addr_info);
|
81
|
+
return TNT_ERESOLVE;
|
82
|
+
}
|
83
|
+
|
84
|
+
static enum tnt_error
|
85
|
+
tnt_io_nonblock(struct tnt_stream_net *s, int set)
|
86
|
+
{
|
87
|
+
int flags = fcntl(s->fd, F_GETFL);
|
88
|
+
if (flags == -1) {
|
89
|
+
s->errno_ = errno;
|
90
|
+
return TNT_ESYSTEM;
|
91
|
+
}
|
92
|
+
if (set)
|
93
|
+
flags |= O_NONBLOCK;
|
94
|
+
else
|
95
|
+
flags &= ~O_NONBLOCK;
|
96
|
+
if (fcntl(s->fd, F_SETFL, flags) == -1) {
|
97
|
+
s->errno_ = errno;
|
98
|
+
return TNT_ESYSTEM;
|
99
|
+
}
|
100
|
+
return TNT_EOK;
|
101
|
+
}
|
102
|
+
|
103
|
+
static enum tnt_error
|
104
|
+
tnt_io_connect_do(struct tnt_stream_net *s, struct sockaddr *addr,
|
105
|
+
socklen_t addr_size)
|
106
|
+
{
|
107
|
+
/* setting nonblock */
|
108
|
+
enum tnt_error result = tnt_io_nonblock(s, 1);
|
109
|
+
if (result != TNT_EOK)
|
110
|
+
return result;
|
111
|
+
|
112
|
+
if (connect(s->fd, (struct sockaddr*)addr, addr_size) != -1)
|
113
|
+
return TNT_EOK;
|
114
|
+
if (errno == EINPROGRESS) {
|
115
|
+
/** waiting for connection while handling signal events */
|
116
|
+
const int64_t micro = 1000000;
|
117
|
+
int64_t tmout_usec = s->opt.tmout_connect.tv_sec * micro;
|
118
|
+
/* get start connect time */
|
119
|
+
struct timeval start_connect;
|
120
|
+
if (gettimeofday(&start_connect, NULL) == -1) {
|
121
|
+
s->errno_ = errno;
|
122
|
+
return TNT_ESYSTEM;
|
123
|
+
}
|
124
|
+
/* set initial timer */
|
125
|
+
struct timeval tmout;
|
126
|
+
memcpy(&tmout, &s->opt.tmout_connect, sizeof(tmout));
|
127
|
+
while (1) {
|
128
|
+
fd_set fds;
|
129
|
+
FD_ZERO(&fds);
|
130
|
+
FD_SET(s->fd, &fds);
|
131
|
+
int ret = select(s->fd + 1, NULL, &fds, NULL, &tmout);
|
132
|
+
if (ret == -1) {
|
133
|
+
if (errno == EINTR || errno == EAGAIN) {
|
134
|
+
/* get current time */
|
135
|
+
struct timeval curr;
|
136
|
+
if (gettimeofday(&curr, NULL) == -1) {
|
137
|
+
s->errno_ = errno;
|
138
|
+
return TNT_ESYSTEM;
|
139
|
+
}
|
140
|
+
/* calculate timeout last time */
|
141
|
+
int64_t passd_usec = (curr.tv_sec - start_connect.tv_sec) * micro +
|
142
|
+
(curr.tv_usec - start_connect.tv_usec);
|
143
|
+
int64_t curr_tmeout = passd_usec - tmout_usec;
|
144
|
+
if (curr_tmeout <= 0) {
|
145
|
+
/* timeout */
|
146
|
+
return TNT_ETMOUT;
|
147
|
+
}
|
148
|
+
tmout.tv_sec = curr_tmeout / micro;
|
149
|
+
tmout.tv_usec = curr_tmeout % micro;
|
150
|
+
} else {
|
151
|
+
s->errno_ = errno;
|
152
|
+
return TNT_ESYSTEM;
|
153
|
+
}
|
154
|
+
} else if (ret == 0) {
|
155
|
+
/* timeout */
|
156
|
+
return TNT_ETMOUT;
|
157
|
+
} else {
|
158
|
+
/* we have a event on socket */
|
159
|
+
break;
|
160
|
+
}
|
161
|
+
}
|
162
|
+
/* checking error status */
|
163
|
+
int opt = 0;
|
164
|
+
socklen_t len = sizeof(opt);
|
165
|
+
if ((getsockopt(s->fd, SOL_SOCKET, SO_ERROR,
|
166
|
+
&opt, &len) == -1) || opt) {
|
167
|
+
s->errno_ = (opt) ? opt : errno;
|
168
|
+
return TNT_ESYSTEM;
|
169
|
+
}
|
170
|
+
} else {
|
171
|
+
s->errno_ = errno;
|
172
|
+
return TNT_ESYSTEM;
|
173
|
+
}
|
174
|
+
|
175
|
+
/* setting block */
|
176
|
+
result = tnt_io_nonblock(s, 0);
|
177
|
+
if (result != TNT_EOK)
|
178
|
+
return result;
|
179
|
+
return TNT_EOK;
|
180
|
+
}
|
181
|
+
|
182
|
+
static enum tnt_error
|
183
|
+
tnt_io_connect_tcp(struct tnt_stream_net *s, const char *host, int port)
|
184
|
+
{
|
185
|
+
/* resolving address */
|
186
|
+
struct sockaddr_in addr;
|
187
|
+
enum tnt_error result = tnt_io_resolve(&addr, host, port);
|
188
|
+
if (result != TNT_EOK)
|
189
|
+
return result;
|
190
|
+
|
191
|
+
return tnt_io_connect_do(s, (struct sockaddr *)&addr, sizeof(addr));
|
192
|
+
}
|
193
|
+
|
194
|
+
static enum tnt_error
|
195
|
+
tnt_io_connect_unix(struct tnt_stream_net *s, const char *path)
|
196
|
+
{
|
197
|
+
struct sockaddr_un addr;
|
198
|
+
memset(&addr, 0, sizeof(struct sockaddr_un));
|
199
|
+
addr.sun_family = AF_UNIX;
|
200
|
+
strcpy(addr.sun_path, path);
|
201
|
+
if (connect(s->fd, (struct sockaddr*)&addr, sizeof(addr)) != -1)
|
202
|
+
return TNT_EOK;
|
203
|
+
s->errno_ = errno;
|
204
|
+
return TNT_ESYSTEM;
|
205
|
+
}
|
206
|
+
|
207
|
+
static enum tnt_error tnt_io_xbufmax(struct tnt_stream_net *s, int opt, int min) {
|
208
|
+
int max = 128 * 1024 * 1024;
|
209
|
+
if (min == 0)
|
210
|
+
min = 16384;
|
211
|
+
unsigned int avg = 0;
|
212
|
+
while (min <= max) {
|
213
|
+
avg = ((unsigned int)(min + max)) / 2;
|
214
|
+
if (setsockopt(s->fd, SOL_SOCKET, opt, &avg, sizeof(avg)) == 0)
|
215
|
+
min = avg + 1;
|
216
|
+
else
|
217
|
+
max = avg - 1;
|
218
|
+
}
|
219
|
+
return TNT_EOK;
|
220
|
+
}
|
221
|
+
|
222
|
+
static enum tnt_error tnt_io_setopts(struct tnt_stream_net *s) {
|
223
|
+
int opt = 1;
|
224
|
+
if (s->opt.uri->host_hint != URI_UNIX) {
|
225
|
+
if (setsockopt(s->fd, IPPROTO_TCP, TCP_NODELAY, &opt, sizeof(opt)) == -1)
|
226
|
+
goto error;
|
227
|
+
}
|
228
|
+
|
229
|
+
tnt_io_xbufmax(s, SO_SNDBUF, s->opt.send_buf);
|
230
|
+
tnt_io_xbufmax(s, SO_RCVBUF, s->opt.recv_buf);
|
231
|
+
|
232
|
+
if (setsockopt(s->fd, SOL_SOCKET, SO_SNDTIMEO,
|
233
|
+
&s->opt.tmout_send, sizeof(s->opt.tmout_send)) == -1)
|
234
|
+
goto error;
|
235
|
+
if (setsockopt(s->fd, SOL_SOCKET, SO_RCVTIMEO,
|
236
|
+
&s->opt.tmout_recv, sizeof(s->opt.tmout_recv)) == -1)
|
237
|
+
goto error;
|
238
|
+
return TNT_EOK;
|
239
|
+
error:
|
240
|
+
s->errno_ = errno;
|
241
|
+
return TNT_ESYSTEM;
|
242
|
+
}
|
243
|
+
|
244
|
+
static int tnt_io_htopf(int host_hint) {
|
245
|
+
switch(host_hint) {
|
246
|
+
case URI_NAME:
|
247
|
+
case URI_IPV4:
|
248
|
+
return PF_INET;
|
249
|
+
case URI_IPV6:
|
250
|
+
return PF_INET6;
|
251
|
+
case URI_UNIX:
|
252
|
+
return PF_UNIX;
|
253
|
+
default:
|
254
|
+
return -1;
|
255
|
+
}
|
256
|
+
}
|
257
|
+
|
258
|
+
enum tnt_error
|
259
|
+
tnt_io_connect(struct tnt_stream_net *s)
|
260
|
+
{
|
261
|
+
struct uri *uri = s->opt.uri;
|
262
|
+
s->fd = socket(tnt_io_htopf(uri->host_hint), SOCK_STREAM, 0);
|
263
|
+
if (s->fd < 0) {
|
264
|
+
s->errno_ = errno;
|
265
|
+
return TNT_ESYSTEM;
|
266
|
+
}
|
267
|
+
enum tnt_error result = tnt_io_setopts(s);
|
268
|
+
if (result != TNT_EOK)
|
269
|
+
goto out;
|
270
|
+
switch (uri->host_hint) {
|
271
|
+
case URI_NAME:
|
272
|
+
case URI_IPV4:
|
273
|
+
case URI_IPV6: {
|
274
|
+
char host[128];
|
275
|
+
memcpy(host, uri->host, uri->host_len);
|
276
|
+
host[uri->host_len] = '\0';
|
277
|
+
uint32_t port = 3301;
|
278
|
+
if (uri->service)
|
279
|
+
port = strtol(uri->service, NULL, 10);
|
280
|
+
result = tnt_io_connect_tcp(s, host, port);
|
281
|
+
break;
|
282
|
+
}
|
283
|
+
case URI_UNIX: {
|
284
|
+
char service[128];
|
285
|
+
memcpy(service, uri->service, uri->service_len);
|
286
|
+
service[uri->service_len] = '\0';
|
287
|
+
result = tnt_io_connect_unix(s, service);
|
288
|
+
break;
|
289
|
+
}
|
290
|
+
default:
|
291
|
+
result = TNT_EFAIL;
|
292
|
+
}
|
293
|
+
if (result != TNT_EOK)
|
294
|
+
goto out;
|
295
|
+
s->connected = 1;
|
296
|
+
return TNT_EOK;
|
297
|
+
out:
|
298
|
+
tnt_io_close(s);
|
299
|
+
return result;
|
300
|
+
}
|
301
|
+
|
302
|
+
void tnt_io_close(struct tnt_stream_net *s)
|
303
|
+
{
|
304
|
+
if (s->fd > 0) {
|
305
|
+
close(s->fd);
|
306
|
+
s->fd = -1;
|
307
|
+
}
|
308
|
+
s->connected = 0;
|
309
|
+
}
|
310
|
+
|
311
|
+
ssize_t tnt_io_flush(struct tnt_stream_net *s) {
|
312
|
+
if (s->sbuf.off == 0)
|
313
|
+
return 0;
|
314
|
+
ssize_t rc = tnt_io_send_raw(s, s->sbuf.buf, s->sbuf.off, 1);
|
315
|
+
if (rc == -1)
|
316
|
+
return -1;
|
317
|
+
s->sbuf.off = 0;
|
318
|
+
return rc;
|
319
|
+
}
|
320
|
+
|
321
|
+
ssize_t
|
322
|
+
tnt_io_send_raw(struct tnt_stream_net *s, const char *buf, size_t size, int all)
|
323
|
+
{
|
324
|
+
size_t off = 0;
|
325
|
+
do {
|
326
|
+
ssize_t r;
|
327
|
+
if (s->sbuf.tx) {
|
328
|
+
r = s->sbuf.tx(&s->sbuf, buf + off, size - off);
|
329
|
+
} else {
|
330
|
+
do {
|
331
|
+
r = send(s->fd, buf + off, size - off, 0);
|
332
|
+
} while (r == -1 && (errno == EINTR));
|
333
|
+
}
|
334
|
+
if (r <= 0) {
|
335
|
+
s->error = TNT_ESYSTEM;
|
336
|
+
s->errno_ = errno;
|
337
|
+
return -1;
|
338
|
+
}
|
339
|
+
off += r;
|
340
|
+
} while (off != size && all);
|
341
|
+
return off;
|
342
|
+
}
|
343
|
+
|
344
|
+
ssize_t
|
345
|
+
tnt_io_sendv_raw(struct tnt_stream_net *s, struct iovec *iov, int count, int all)
|
346
|
+
{
|
347
|
+
size_t total = 0;
|
348
|
+
while (count > 0) {
|
349
|
+
ssize_t r;
|
350
|
+
if (s->sbuf.txv) {
|
351
|
+
r = s->sbuf.txv(&s->sbuf, iov, MIN(count, getiovmax()));
|
352
|
+
} else {
|
353
|
+
do {
|
354
|
+
r = writev(s->fd, iov, count);
|
355
|
+
} while (r == -1 && (errno == EINTR));
|
356
|
+
}
|
357
|
+
if (r <= 0) {
|
358
|
+
s->error = TNT_ESYSTEM;
|
359
|
+
s->errno_ = errno;
|
360
|
+
return -1;
|
361
|
+
}
|
362
|
+
total += r;
|
363
|
+
if (!all)
|
364
|
+
break;
|
365
|
+
while (count > 0) {
|
366
|
+
if (iov->iov_len > (size_t)r) {
|
367
|
+
iov->iov_base += r;
|
368
|
+
iov->iov_len -= r;
|
369
|
+
break;
|
370
|
+
} else {
|
371
|
+
r -= iov->iov_len;
|
372
|
+
iov++;
|
373
|
+
count--;
|
374
|
+
}
|
375
|
+
}
|
376
|
+
}
|
377
|
+
return total;
|
378
|
+
}
|
379
|
+
|
380
|
+
ssize_t
|
381
|
+
tnt_io_send(struct tnt_stream_net *s, const char *buf, size_t size)
|
382
|
+
{
|
383
|
+
if (s->sbuf.buf == NULL)
|
384
|
+
return tnt_io_send_raw(s, buf, size, 1);
|
385
|
+
if (size > s->sbuf.size) {
|
386
|
+
s->error = TNT_EBIG;
|
387
|
+
return -1;
|
388
|
+
}
|
389
|
+
if ((s->sbuf.off + size) <= s->sbuf.size) {
|
390
|
+
memcpy(s->sbuf.buf + s->sbuf.off, buf, size);
|
391
|
+
s->sbuf.off += size;
|
392
|
+
return size;
|
393
|
+
}
|
394
|
+
ssize_t r = tnt_io_send_raw(s, s->sbuf.buf, s->sbuf.off, 1);
|
395
|
+
if (r == -1)
|
396
|
+
return -1;
|
397
|
+
s->sbuf.off = size;
|
398
|
+
memcpy(s->sbuf.buf, buf, size);
|
399
|
+
return size;
|
400
|
+
}
|
401
|
+
|
402
|
+
inline static void
|
403
|
+
tnt_io_sendv_put(struct tnt_stream_net *s, struct iovec *iov, int count) {
|
404
|
+
int i;
|
405
|
+
for (i = 0 ; i < count ; i++) {
|
406
|
+
memcpy(s->sbuf.buf + s->sbuf.off,
|
407
|
+
iov[i].iov_base,
|
408
|
+
iov[i].iov_len);
|
409
|
+
s->sbuf.off += iov[i].iov_len;
|
410
|
+
}
|
411
|
+
}
|
412
|
+
|
413
|
+
ssize_t
|
414
|
+
tnt_io_sendv(struct tnt_stream_net *s, struct iovec *iov, int count)
|
415
|
+
{
|
416
|
+
if (s->sbuf.buf == NULL)
|
417
|
+
return tnt_io_sendv_raw(s, iov, count, 1);
|
418
|
+
size_t size = 0;
|
419
|
+
int i;
|
420
|
+
for (i = 0 ; i < count ; i++)
|
421
|
+
size += iov[i].iov_len;
|
422
|
+
if (size > s->sbuf.size) {
|
423
|
+
s->error = TNT_EBIG;
|
424
|
+
return -1;
|
425
|
+
}
|
426
|
+
if ((s->sbuf.off + size) <= s->sbuf.size) {
|
427
|
+
tnt_io_sendv_put(s, iov, count);
|
428
|
+
return size;
|
429
|
+
}
|
430
|
+
ssize_t r = tnt_io_send_raw(s, s->sbuf.buf, s->sbuf.off, 1);
|
431
|
+
if (r == -1)
|
432
|
+
return -1;
|
433
|
+
s->sbuf.off = 0;
|
434
|
+
tnt_io_sendv_put(s, iov, count);
|
435
|
+
return size;
|
436
|
+
}
|
437
|
+
|
438
|
+
ssize_t
|
439
|
+
tnt_io_recv_raw(struct tnt_stream_net *s, char *buf, size_t size, int all)
|
440
|
+
{
|
441
|
+
size_t off = 0;
|
442
|
+
do {
|
443
|
+
ssize_t r;
|
444
|
+
if (s->rbuf.tx) {
|
445
|
+
r = s->rbuf.tx(&s->rbuf, buf + off, size - off);
|
446
|
+
} else {
|
447
|
+
do {
|
448
|
+
r = recv(s->fd, buf + off, size - off, 0);
|
449
|
+
} while (r == -1 && (errno == EINTR));
|
450
|
+
}
|
451
|
+
if (r <= 0) {
|
452
|
+
s->error = TNT_ESYSTEM;
|
453
|
+
s->errno_ = errno;
|
454
|
+
return -1;
|
455
|
+
}
|
456
|
+
off += r;
|
457
|
+
} while (off != size && all);
|
458
|
+
return off;
|
459
|
+
}
|
460
|
+
|
461
|
+
ssize_t
|
462
|
+
tnt_io_recv(struct tnt_stream_net *s, char *buf, size_t size)
|
463
|
+
{
|
464
|
+
if (s->rbuf.buf == NULL)
|
465
|
+
return tnt_io_recv_raw(s, buf, size, 1);
|
466
|
+
size_t lv, rv, off = 0, left = size;
|
467
|
+
while (1) {
|
468
|
+
if ((s->rbuf.off + left) <= s->rbuf.top) {
|
469
|
+
memcpy(buf + off, s->rbuf.buf + s->rbuf.off, left);
|
470
|
+
s->rbuf.off += left;
|
471
|
+
return size;
|
472
|
+
}
|
473
|
+
|
474
|
+
lv = s->rbuf.top - s->rbuf.off;
|
475
|
+
rv = left - lv;
|
476
|
+
if (lv) {
|
477
|
+
memcpy(buf + off, s->rbuf.buf + s->rbuf.off, lv);
|
478
|
+
off += lv;
|
479
|
+
}
|
480
|
+
|
481
|
+
s->rbuf.off = 0;
|
482
|
+
ssize_t top = tnt_io_recv_raw(s, s->rbuf.buf, s->rbuf.size, 0);
|
483
|
+
if (top <= 0) {
|
484
|
+
s->errno_ = errno;
|
485
|
+
s->error = TNT_ESYSTEM;
|
486
|
+
return -1;
|
487
|
+
}
|
488
|
+
|
489
|
+
s->rbuf.top = top;
|
490
|
+
if (rv <= s->rbuf.top) {
|
491
|
+
memcpy(buf + off, s->rbuf.buf, rv);
|
492
|
+
s->rbuf.off = rv;
|
493
|
+
return size;
|
494
|
+
}
|
495
|
+
left -= lv;
|
496
|
+
}
|
497
|
+
return -1;
|
498
|
+
}
|
499
|
+
|
500
|
+
int getiovmax()
|
501
|
+
{
|
502
|
+
#if defined(IOV_MAX)
|
503
|
+
return IOV_MAX;
|
504
|
+
#elif defined(_SC_IOV_MAX)
|
505
|
+
static int iovmax = -1;
|
506
|
+
if (iovmax == -1) {
|
507
|
+
iovmax = sysconf(_SC_IOV_MAX);
|
508
|
+
/* On some embedded devices (arm-linux-uclibc based ip camera),
|
509
|
+
* sysconf(_SC_IOV_MAX) can not get the correct value. The return
|
510
|
+
* value is -1 and the errno is EINPROGRESS. Degrade the value to 1.
|
511
|
+
*/
|
512
|
+
if (iovmax == -1) iovmax = 1;
|
513
|
+
}
|
514
|
+
return iovmax;
|
515
|
+
#elif defined(UIO_MAXIOV)
|
516
|
+
return UIO_MAXIOV;
|
517
|
+
#else
|
518
|
+
return 1024;
|
519
|
+
#endif
|
520
|
+
}
|
@@ -0,0 +1,75 @@
|
|
1
|
+
|
2
|
+
/*
|
3
|
+
* Redistribution and use in source and binary forms, with or
|
4
|
+
* without modification, are permitted provided that the following
|
5
|
+
* conditions are met:
|
6
|
+
*
|
7
|
+
* 1. Redistributions of source code must retain the above
|
8
|
+
* copyright notice, this list of conditions and the
|
9
|
+
* following disclaimer.
|
10
|
+
*
|
11
|
+
* 2. Redistributions in binary form must reproduce the above
|
12
|
+
* copyright notice, this list of conditions and the following
|
13
|
+
* disclaimer in the documentation and/or other materials
|
14
|
+
* provided with the distribution.
|
15
|
+
*
|
16
|
+
* THIS SOFTWARE IS PROVIDED BY <COPYRIGHT HOLDER> ``AS IS'' AND
|
17
|
+
* ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
|
18
|
+
* TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
19
|
+
* A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL
|
20
|
+
* <COPYRIGHT HOLDER> OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT,
|
21
|
+
* INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
22
|
+
* DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
|
23
|
+
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR
|
24
|
+
* BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
|
25
|
+
* LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
26
|
+
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF
|
27
|
+
* THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
|
28
|
+
* SUCH DAMAGE.
|
29
|
+
*/
|
30
|
+
|
31
|
+
#include <stdlib.h>
|
32
|
+
#include <stdio.h>
|
33
|
+
#include <string.h>
|
34
|
+
#include <stdbool.h>
|
35
|
+
|
36
|
+
#include <unistd.h>
|
37
|
+
#include <sys/uio.h>
|
38
|
+
|
39
|
+
#include <tarantool/tnt_mem.h>
|
40
|
+
#include <tarantool/tnt_iob.h>
|
41
|
+
|
42
|
+
int
|
43
|
+
tnt_iob_init(struct tnt_iob *iob, size_t size,
|
44
|
+
tnt_iob_tx_t tx,
|
45
|
+
tnt_iob_txv_t txv, void *ptr)
|
46
|
+
{
|
47
|
+
iob->tx = tx;
|
48
|
+
iob->txv = txv;
|
49
|
+
iob->ptr = ptr;
|
50
|
+
iob->size = size;
|
51
|
+
iob->off = 0;
|
52
|
+
iob->top = 0;
|
53
|
+
iob->buf = NULL;
|
54
|
+
if (size > 0) {
|
55
|
+
iob->buf = tnt_mem_alloc(size);
|
56
|
+
if (iob->buf == NULL)
|
57
|
+
return -1;
|
58
|
+
memset(iob->buf, 0, size);
|
59
|
+
}
|
60
|
+
return 0;
|
61
|
+
}
|
62
|
+
|
63
|
+
void
|
64
|
+
tnt_iob_clear(struct tnt_iob *iob)
|
65
|
+
{
|
66
|
+
iob->top = 0;
|
67
|
+
iob->off = 0;
|
68
|
+
}
|
69
|
+
|
70
|
+
void
|
71
|
+
tnt_iob_free(struct tnt_iob *iob)
|
72
|
+
{
|
73
|
+
if (iob->buf)
|
74
|
+
tnt_mem_free(iob->buf);
|
75
|
+
}
|