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.
Files changed (107) hide show
  1. checksums.yaml +7 -0
  2. data/README.md +84 -0
  3. data/ext/lwtarantool/conn.c +314 -0
  4. data/ext/lwtarantool/depend +23 -0
  5. data/ext/lwtarantool/errors.c +24 -0
  6. data/ext/lwtarantool/extconf.rb +8 -0
  7. data/ext/lwtarantool/lwtarantool.c +12 -0
  8. data/ext/lwtarantool/lwtarantool.h +38 -0
  9. data/ext/lwtarantool/request.c +124 -0
  10. data/ext/lwtarantool/vendor/msgpuck/CMakeLists.txt +73 -0
  11. data/ext/lwtarantool/vendor/msgpuck/hints.c +674 -0
  12. data/ext/lwtarantool/vendor/msgpuck/msgpuck.c +375 -0
  13. data/ext/lwtarantool/vendor/msgpuck/msgpuck.h +2195 -0
  14. data/ext/lwtarantool/vendor/msgpuck/test/CMakeLists.txt +25 -0
  15. data/ext/lwtarantool/vendor/msgpuck/test/msgpuck.c +1126 -0
  16. data/ext/lwtarantool/vendor/msgpuck/test/test.c +105 -0
  17. data/ext/lwtarantool/vendor/msgpuck/test/test.h +123 -0
  18. data/ext/lwtarantool/vendor/tarantool-c/CMakeLists.txt +89 -0
  19. data/ext/lwtarantool/vendor/tarantool-c/cmake/FindMsgPuck.cmake +18 -0
  20. data/ext/lwtarantool/vendor/tarantool-c/cmake/FindSphinx.cmake +10 -0
  21. data/ext/lwtarantool/vendor/tarantool-c/doc/CMakeLists.txt +15 -0
  22. data/ext/lwtarantool/vendor/tarantool-c/doc/source/example.c +340 -0
  23. data/ext/lwtarantool/vendor/tarantool-c/include/CMakeLists.txt +6 -0
  24. data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tarantool.h +67 -0
  25. data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_auth.h +72 -0
  26. data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_buf.h +88 -0
  27. data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_call.h +83 -0
  28. data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_delete.h +52 -0
  29. data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_execute.h +24 -0
  30. data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_insert.h +62 -0
  31. data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_io.h +67 -0
  32. data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_iob.h +62 -0
  33. data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_iter.h +301 -0
  34. data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_mem.h +102 -0
  35. data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_net.h +242 -0
  36. data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_object.h +248 -0
  37. data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_opt.h +138 -0
  38. data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_ping.h +49 -0
  39. data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_proto.h +295 -0
  40. data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_reply.h +204 -0
  41. data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_request.h +377 -0
  42. data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_schema.h +165 -0
  43. data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_select.h +59 -0
  44. data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_stream.h +79 -0
  45. data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_update.h +226 -0
  46. data/ext/lwtarantool/vendor/tarantool-c/include/tp.h +1998 -0
  47. data/ext/lwtarantool/vendor/tarantool-c/test/CMakeLists.txt +36 -0
  48. data/ext/lwtarantool/vendor/tarantool-c/test/common.c +233 -0
  49. data/ext/lwtarantool/vendor/tarantool-c/test/common.h +28 -0
  50. data/ext/lwtarantool/vendor/tarantool-c/test/plain_test.c +152 -0
  51. data/ext/lwtarantool/vendor/tarantool-c/test/tarantool_call.c +57 -0
  52. data/ext/lwtarantool/vendor/tarantool-c/test/tarantool_disconnect.c +31 -0
  53. data/ext/lwtarantool/vendor/tarantool-c/test/tarantool_tcp.c +840 -0
  54. data/ext/lwtarantool/vendor/tarantool-c/test/tarantool_unix.c +384 -0
  55. data/ext/lwtarantool/vendor/tarantool-c/test/test.c +72 -0
  56. data/ext/lwtarantool/vendor/tarantool-c/test/test.h +94 -0
  57. data/ext/lwtarantool/vendor/tarantool-c/test/tnt_assoc.c +6 -0
  58. data/ext/lwtarantool/vendor/tarantool-c/test/tnt_assoc.h +67 -0
  59. data/ext/lwtarantool/vendor/tarantool-c/third_party/PMurHash.c +317 -0
  60. data/ext/lwtarantool/vendor/tarantool-c/third_party/PMurHash.h +68 -0
  61. data/ext/lwtarantool/vendor/tarantool-c/third_party/base64.c +300 -0
  62. data/ext/lwtarantool/vendor/tarantool-c/third_party/base64.h +91 -0
  63. data/ext/lwtarantool/vendor/tarantool-c/third_party/mhash.h +589 -0
  64. data/ext/lwtarantool/vendor/tarantool-c/third_party/sha1.c +219 -0
  65. data/ext/lwtarantool/vendor/tarantool-c/third_party/sha1.h +24 -0
  66. data/ext/lwtarantool/vendor/tarantool-c/third_party/uri.c +6528 -0
  67. data/ext/lwtarantool/vendor/tarantool-c/third_party/uri.h +81 -0
  68. data/ext/lwtarantool/vendor/tarantool-c/tnt/CMakeLists.txt +83 -0
  69. data/ext/lwtarantool/vendor/tarantool-c/tnt/pmatomic.h +494 -0
  70. data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_assoc.c +9 -0
  71. data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_assoc.h +172 -0
  72. data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_auth.c +118 -0
  73. data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_buf.c +171 -0
  74. data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_call.c +79 -0
  75. data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_delete.c +52 -0
  76. data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_execute.c +60 -0
  77. data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_insert.c +60 -0
  78. data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_io.c +520 -0
  79. data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_iob.c +75 -0
  80. data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_iter.c +293 -0
  81. data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_mem.c +82 -0
  82. data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_net.c +336 -0
  83. data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_object.c +476 -0
  84. data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_opt.c +117 -0
  85. data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_ping.c +38 -0
  86. data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_proto_internal.h +43 -0
  87. data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_reply.c +300 -0
  88. data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_request.c +336 -0
  89. data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_schema.c +334 -0
  90. data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_select.c +58 -0
  91. data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_stream.c +71 -0
  92. data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_update.c +247 -0
  93. data/ext/lwtarantool/vendor/tarantool-c/tntrpl/CMakeLists.txt +69 -0
  94. data/ext/lwtarantool/vendor/tarantool-c/tntrpl/tnt_dir.c +173 -0
  95. data/ext/lwtarantool/vendor/tarantool-c/tntrpl/tnt_dir.h +58 -0
  96. data/ext/lwtarantool/vendor/tarantool-c/tntrpl/tnt_log.c +329 -0
  97. data/ext/lwtarantool/vendor/tarantool-c/tntrpl/tnt_log.h +119 -0
  98. data/ext/lwtarantool/vendor/tarantool-c/tntrpl/tnt_rpl.c +189 -0
  99. data/ext/lwtarantool/vendor/tarantool-c/tntrpl/tnt_rpl.h +47 -0
  100. data/ext/lwtarantool/vendor/tarantool-c/tntrpl/tnt_snapshot.c +163 -0
  101. data/ext/lwtarantool/vendor/tarantool-c/tntrpl/tnt_snapshot.h +50 -0
  102. data/ext/lwtarantool/vendor/tarantool-c/tntrpl/tnt_xlog.c +163 -0
  103. data/ext/lwtarantool/vendor/tarantool-c/tntrpl/tnt_xlog.h +50 -0
  104. data/lib/lwtarantool/connection.rb +84 -0
  105. data/lib/lwtarantool/request.rb +58 -0
  106. data/lib/lwtarantool.rb +23 -0
  107. 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
+ }