source: EcnlProtoTool/trunk/mrbgems/mruby-socket/src/socket.c@ 331

Last change on this file since 331 was 331, checked in by coas-nagasima, 6 years ago

prototoolに関連するプロジェクトをnewlibからmuslを使うよう変更・更新
ntshellをnewlibの下位の実装から、muslのsyscallの実装に変更・更新
以下のOSSをアップデート
・mruby-1.3.0
・musl-1.1.18
・onigmo-6.1.3
・tcc-0.9.27
以下のOSSを追加
・openssl-1.1.0e
・curl-7.57.0
・zlib-1.2.11
以下のmrbgemsを追加
・iij/mruby-digest
・iij/mruby-env
・iij/mruby-errno
・iij/mruby-iijson
・iij/mruby-ipaddr
・iij/mruby-mock
・iij/mruby-require
・iij/mruby-tls-openssl

  • Property svn:eol-style set to native
  • Property svn:mime-type set to text/x-csrc;charset=UTF-8
File size: 25.9 KB
Line 
1/*
2** socket.c - Socket module
3**
4** See Copyright Notice in mruby.h
5*/
6
7#ifdef _WIN32
8 #define _WIN32_WINNT 0x0501
9
10 #include <winsock2.h>
11 #include <ws2tcpip.h>
12 #include <windows.h>
13
14 #define SHUT_RDWR SD_BOTH
15#else
16 #include <sys/types.h>
17 #include <sys/socket.h>
18 #include <sys/un.h>
19 #include <netinet/in.h>
20 #include <netinet/tcp.h>
21 #include <arpa/inet.h>
22 #include <fcntl.h>
23 #include <netdb.h>
24 #include <unistd.h>
25#endif
26
27#include <stddef.h>
28#include <string.h>
29
30#include "mruby.h"
31#include "mruby/array.h"
32#include "mruby/class.h"
33#include "mruby/data.h"
34#include "mruby/string.h"
35#include "mruby/variable.h"
36#include "error.h"
37
38#define E_SOCKET_ERROR (mrb_class_get(mrb, "SocketError"))
39
40#if !defined(mrb_cptr)
41#define mrb_cptr_value(m,p) mrb_voidp_value((m),(p))
42#define mrb_cptr(o) mrb_voidp(o)
43#define mrb_cptr_p(o) mrb_voidp_p(o)
44#endif
45
46#ifdef _WIN32
47const char *inet_ntop(int af, const void *src, char *dst, socklen_t cnt)
48{
49 if (af == AF_INET)
50 {
51 struct sockaddr_in in;
52 memset(&in, 0, sizeof(in));
53 in.sin_family = AF_INET;
54 memcpy(&in.sin_addr, src, sizeof(struct in_addr));
55 getnameinfo((struct sockaddr *)&in, sizeof(struct
56 sockaddr_in), dst, cnt, NULL, 0, NI_NUMERICHOST);
57 return dst;
58 }
59 else if (af == AF_INET6)
60 {
61 struct sockaddr_in6 in;
62 memset(&in, 0, sizeof(in));
63 in.sin6_family = AF_INET6;
64 memcpy(&in.sin6_addr, src, sizeof(struct in_addr6));
65 getnameinfo((struct sockaddr *)&in, sizeof(struct
66 sockaddr_in6), dst, cnt, NULL, 0, NI_NUMERICHOST);
67 return dst;
68 }
69 return NULL;
70}
71
72int inet_pton(int af, const char *src, void *dst)
73{
74 struct addrinfo hints, *res, *ressave;
75
76 memset(&hints, 0, sizeof(struct addrinfo));
77 hints.ai_family = af;
78
79 if (getaddrinfo(src, NULL, &hints, &res) != 0)
80 {
81 printf("Couldn't resolve host %s\n", src);
82 return -1;
83 }
84
85 ressave = res;
86
87 while (res)
88 {
89 memcpy(dst, res->ai_addr, res->ai_addrlen);
90 res = res->ai_next;
91 }
92
93 freeaddrinfo(ressave);
94 return 0;
95}
96
97#endif
98
99static mrb_value
100mrb_addrinfo_getaddrinfo(mrb_state *mrb, mrb_value klass)
101{
102 struct addrinfo hints, *res0, *res;
103 mrb_value ai, ary, family, lastai, nodename, protocol, sa, service, socktype;
104 mrb_int flags;
105 int arena_idx, error;
106 const char *hostname = NULL, *servname = NULL;
107
108 ary = mrb_ary_new(mrb);
109 arena_idx = mrb_gc_arena_save(mrb); /* ary must be on arena! */
110
111 family = socktype = protocol = mrb_nil_value();
112 flags = 0;
113 mrb_get_args(mrb, "oo|oooi", &nodename, &service, &family, &socktype, &protocol, &flags);
114
115 if (mrb_string_p(nodename)) {
116 hostname = mrb_str_to_cstr(mrb, nodename);
117 } else if (mrb_nil_p(nodename)) {
118 hostname = NULL;
119 } else {
120 mrb_raise(mrb, E_TYPE_ERROR, "nodename must be String or nil");
121 }
122
123 if (mrb_string_p(service)) {
124 servname = mrb_str_to_cstr(mrb, service);
125 } else if (mrb_fixnum_p(service)) {
126 servname = mrb_str_to_cstr(mrb, mrb_funcall(mrb, service, "to_s", 0));
127 } else if (mrb_nil_p(service)) {
128 servname = NULL;
129 } else {
130 mrb_raise(mrb, E_TYPE_ERROR, "service must be String, Fixnum, or nil");
131 }
132
133 memset(&hints, 0, sizeof(hints));
134 hints.ai_flags = flags;
135
136 if (mrb_fixnum_p(family)) {
137 hints.ai_family = mrb_fixnum(family);
138 }
139
140 if (mrb_fixnum_p(socktype)) {
141 hints.ai_socktype = mrb_fixnum(socktype);
142 }
143
144 lastai = mrb_cv_get(mrb, klass, mrb_intern_lit(mrb, "_lastai"));
145 if (mrb_cptr_p(lastai)) {
146 freeaddrinfo(mrb_cptr(lastai));
147 mrb_cv_set(mrb, klass, mrb_intern_lit(mrb, "_lastai"), mrb_nil_value());
148 }
149
150 error = getaddrinfo(hostname, servname, &hints, &res0);
151 if (error) {
152 mrb_raisef(mrb, E_SOCKET_ERROR, "getaddrinfo: %S", mrb_str_new_cstr(mrb, gai_strerror(error)));
153 }
154 mrb_cv_set(mrb, klass, mrb_intern_lit(mrb, "_lastai"), mrb_cptr_value(mrb, res0));
155
156 for (res = res0; res != NULL; res = res->ai_next) {
157 sa = mrb_str_new(mrb, (void *)res->ai_addr, res->ai_addrlen);
158 ai = mrb_funcall(mrb, klass, "new", 4, sa, mrb_fixnum_value(res->ai_family), mrb_fixnum_value(res->ai_socktype), mrb_fixnum_value(res->ai_protocol));
159 mrb_ary_push(mrb, ary, ai);
160 mrb_gc_arena_restore(mrb, arena_idx);
161 }
162
163 freeaddrinfo(res0);
164 mrb_cv_set(mrb, klass, mrb_intern_lit(mrb, "_lastai"), mrb_nil_value());
165
166 return ary;
167}
168
169static mrb_value
170mrb_addrinfo_getnameinfo(mrb_state *mrb, mrb_value self)
171{
172 mrb_int flags;
173 mrb_value ary, host, sastr, serv;
174 int error;
175
176 flags = 0;
177 mrb_get_args(mrb, "|i", &flags);
178 host = mrb_str_buf_new(mrb, NI_MAXHOST);
179 serv = mrb_str_buf_new(mrb, NI_MAXSERV);
180
181 sastr = mrb_iv_get(mrb, self, mrb_intern_lit(mrb, "@sockaddr"));
182 if (!mrb_string_p(sastr)) {
183 mrb_raise(mrb, E_SOCKET_ERROR, "invalid sockaddr");
184 }
185 error = getnameinfo((struct sockaddr *)RSTRING_PTR(sastr), (socklen_t)RSTRING_LEN(sastr), RSTRING_PTR(host), NI_MAXHOST, RSTRING_PTR(serv), NI_MAXSERV, flags);
186 if (error != 0) {
187 mrb_raisef(mrb, E_SOCKET_ERROR, "getnameinfo: %s", gai_strerror(error));
188 }
189 ary = mrb_ary_new_capa(mrb, 2);
190 mrb_str_resize(mrb, host, strlen(RSTRING_PTR(host)));
191 mrb_ary_push(mrb, ary, host);
192 mrb_str_resize(mrb, serv, strlen(RSTRING_PTR(serv)));
193 mrb_ary_push(mrb, ary, serv);
194 return ary;
195}
196
197#ifndef _WIN32
198static mrb_value
199mrb_addrinfo_unix_path(mrb_state *mrb, mrb_value self)
200{
201 mrb_value sastr;
202
203 sastr = mrb_iv_get(mrb, self, mrb_intern_lit(mrb, "@sockaddr"));
204 if (((struct sockaddr *)RSTRING_PTR(sastr))->sa_family != AF_UNIX)
205 mrb_raise(mrb, E_SOCKET_ERROR, "need AF_UNIX address");
206 return mrb_str_new_cstr(mrb, ((struct sockaddr_un *)RSTRING_PTR(sastr))->sun_path);
207}
208#endif
209
210static mrb_value
211sa2addrlist(mrb_state *mrb, const struct sockaddr *sa, socklen_t salen)
212{
213 mrb_value ary, host;
214 unsigned short port;
215 const char *afstr;
216
217 switch (sa->sa_family) {
218 case AF_INET:
219 afstr = "AF_INET";
220 port = ((struct sockaddr_in *)sa)->sin_port;
221 break;
222 case AF_INET6:
223 afstr = "AF_INET6";
224 port = ((struct sockaddr_in6 *)sa)->sin6_port;
225 break;
226 default:
227 mrb_raise(mrb, E_ARGUMENT_ERROR, "bad af");
228 return mrb_nil_value();
229 }
230 port = ntohs(port);
231 host = mrb_str_buf_new(mrb, NI_MAXHOST);
232 if (getnameinfo(sa, salen, RSTRING_PTR(host), NI_MAXHOST, NULL, 0, NI_NUMERICHOST) == -1)
233 mrb_sys_fail(mrb, "getnameinfo");
234 mrb_str_resize(mrb, host, strlen(RSTRING_PTR(host)));
235 ary = mrb_ary_new_capa(mrb, 4);
236 mrb_ary_push(mrb, ary, mrb_str_new_cstr(mrb, afstr));
237 mrb_ary_push(mrb, ary, mrb_fixnum_value(port));
238 mrb_ary_push(mrb, ary, host);
239 mrb_ary_push(mrb, ary, host);
240 return ary;
241}
242
243static int
244socket_fd(mrb_state *mrb, mrb_value sock)
245{
246 return mrb_fixnum(mrb_funcall(mrb, sock, "fileno", 0));
247}
248
249static int
250socket_family(int s)
251{
252 struct sockaddr_storage ss;
253 socklen_t salen;
254
255 salen = sizeof(ss);
256 if (getsockname(s, (struct sockaddr *)&ss, &salen) == -1)
257 return AF_UNSPEC;
258 return ss.ss_family;
259}
260
261static mrb_value
262mrb_basicsocket_getpeereid(mrb_state *mrb, mrb_value self)
263{
264#ifdef HAVE_GETPEEREID
265 mrb_value ary;
266 gid_t egid;
267 uid_t euid;
268 int s;
269
270 s = socket_fd(mrb, self);
271 if (getpeereid(s, &euid, &egid) != 0)
272 mrb_sys_fail(mrb, "getpeereid");
273
274 ary = mrb_ary_new_capa(mrb, 2);
275 mrb_ary_push(mrb, ary, mrb_fixnum_value((mrb_int)euid));
276 mrb_ary_push(mrb, ary, mrb_fixnum_value((mrb_int)egid));
277 return ary;
278#else
279 mrb_raise(mrb, E_RUNTIME_ERROR, "getpeereid is not avaialble on this system");
280 return mrb_nil_value();
281#endif
282}
283
284static mrb_value
285mrb_basicsocket_getpeername(mrb_state *mrb, mrb_value self)
286{
287 struct sockaddr_storage ss;
288 socklen_t salen;
289
290 salen = sizeof(ss);
291 if (getpeername(socket_fd(mrb, self), (struct sockaddr *)&ss, &salen) != 0)
292 mrb_sys_fail(mrb, "getpeername");
293
294 return mrb_str_new(mrb, (void *)&ss, salen);
295}
296
297static mrb_value
298mrb_basicsocket_getsockname(mrb_state *mrb, mrb_value self)
299{
300 struct sockaddr_storage ss;
301 socklen_t salen;
302
303 salen = sizeof(ss);
304 if (getsockname(socket_fd(mrb, self), (struct sockaddr *)&ss, &salen) != 0)
305 mrb_sys_fail(mrb, "getsockname");
306
307 return mrb_str_new(mrb, (void *)&ss, salen);
308}
309
310static mrb_value
311mrb_basicsocket_getsockopt(mrb_state *mrb, mrb_value self)
312{
313 char opt[8];
314 int s;
315 mrb_int family, level, optname;
316 mrb_value c, data;
317 socklen_t optlen;
318
319 mrb_get_args(mrb, "ii", &level, &optname);
320 s = socket_fd(mrb, self);
321 optlen = sizeof(opt);
322 if (getsockopt(s, level, optname, opt, &optlen) == -1)
323 mrb_sys_fail(mrb, "getsockopt");
324 c = mrb_const_get(mrb, mrb_obj_value(mrb_class_get(mrb, "Socket")), mrb_intern_lit(mrb, "Option"));
325 family = socket_family(s);
326 data = mrb_str_new(mrb, opt, optlen);
327 return mrb_funcall(mrb, c, "new", 4, mrb_fixnum_value(family), mrb_fixnum_value(level), mrb_fixnum_value(optname), data);
328}
329
330static mrb_value
331mrb_basicsocket_recv(mrb_state *mrb, mrb_value self)
332{
333 int n;
334 mrb_int maxlen, flags = 0;
335 mrb_value buf;
336
337 mrb_get_args(mrb, "i|i", &maxlen, &flags);
338 buf = mrb_str_buf_new(mrb, maxlen);
339 n = recv(socket_fd(mrb, self), RSTRING_PTR(buf), maxlen, flags);
340 if (n == -1)
341 mrb_sys_fail(mrb, "recv");
342 mrb_str_resize(mrb, buf, n);
343 return buf;
344}
345
346static mrb_value
347mrb_basicsocket_recvfrom(mrb_state *mrb, mrb_value self)
348{
349 int n;
350 mrb_int maxlen, flags = 0;
351 mrb_value ary, buf, sa;
352 socklen_t socklen;
353
354 mrb_get_args(mrb, "i|i", &maxlen, &flags);
355 buf = mrb_str_buf_new(mrb, maxlen);
356 socklen = sizeof(struct sockaddr_storage);
357 sa = mrb_str_buf_new(mrb, socklen);
358 n = recvfrom(socket_fd(mrb, self), RSTRING_PTR(buf), maxlen, flags, (struct sockaddr *)RSTRING_PTR(sa), &socklen);
359 if (n == -1)
360 mrb_sys_fail(mrb, "recvfrom");
361 mrb_str_resize(mrb, buf, n);
362 mrb_str_resize(mrb, sa, socklen);
363 ary = mrb_ary_new_capa(mrb, 2);
364 mrb_ary_push(mrb, ary, buf);
365 mrb_ary_push(mrb, ary, sa);
366 return ary;
367}
368
369static mrb_value
370mrb_basicsocket_send(mrb_state *mrb, mrb_value self)
371{
372 int n;
373 mrb_int flags;
374 mrb_value dest, mesg;
375
376 dest = mrb_nil_value();
377 mrb_get_args(mrb, "Si|S", &mesg, &flags, &dest);
378 if (mrb_nil_p(dest)) {
379 n = send(socket_fd(mrb, self), RSTRING_PTR(mesg), RSTRING_LEN(mesg), flags);
380 } else {
381 n = sendto(socket_fd(mrb, self), RSTRING_PTR(mesg), RSTRING_LEN(mesg), flags, (const void *)RSTRING_PTR(dest), RSTRING_LEN(dest));
382 }
383 if (n == -1)
384 mrb_sys_fail(mrb, "send");
385 return mrb_fixnum_value(n);
386}
387
388static mrb_value
389mrb_basicsocket_setnonblock(mrb_state *mrb, mrb_value self)
390{
391 int fd, flags;
392 mrb_value bool;
393#ifdef _WIN32
394 u_long mode = 1;
395#endif
396
397 mrb_get_args(mrb, "o", &bool);
398 fd = socket_fd(mrb, self);
399#ifdef _WIN32
400 flags = ioctlsocket(fd, FIONBIO, &mode);
401 if (flags != NO_ERROR)
402 mrb_sys_fail(mrb, "ioctlsocket");
403#else
404 flags = fcntl(fd, F_GETFL, 0);
405 if (flags == 1)
406 mrb_sys_fail(mrb, "fcntl");
407 if (mrb_test(bool))
408 flags |= O_NONBLOCK;
409 else
410 flags &= ~O_NONBLOCK;
411 if (fcntl(fd, F_SETFL, flags) == -1)
412 mrb_sys_fail(mrb, "fcntl");
413#endif
414 return mrb_nil_value();
415}
416
417static mrb_value
418mrb_basicsocket_setsockopt(mrb_state *mrb, mrb_value self)
419{
420 int argc, s;
421 mrb_int level = 0, optname;
422 mrb_value optval, so;
423
424 argc = mrb_get_args(mrb, "o|io", &so, &optname, &optval);
425 if (argc == 3) {
426 if (!mrb_fixnum_p(so)) {
427 mrb_raise(mrb, E_ARGUMENT_ERROR, "level is not an integer");
428 }
429 level = mrb_fixnum(so);
430 if (mrb_string_p(optval)) {
431 /* that's good */
432 } else if (mrb_type(optval) == MRB_TT_TRUE || mrb_type(optval) == MRB_TT_FALSE) {
433 mrb_int i = mrb_test(optval) ? 1 : 0;
434 optval = mrb_str_new(mrb, (char *)&i, sizeof(i));
435 } else if (mrb_fixnum_p(optval)) {
436 if (optname == IP_MULTICAST_TTL || optname == IP_MULTICAST_LOOP) {
437 char uc = mrb_fixnum(optval);
438 optval = mrb_str_new(mrb, &uc, sizeof(uc));
439 } else {
440 mrb_int i = mrb_fixnum(optval);
441 optval = mrb_str_new(mrb, (char *)&i, sizeof(i));
442 }
443 } else {
444 mrb_raise(mrb, E_ARGUMENT_ERROR, "optval should be true, false, an integer, or a string");
445 }
446 } else if (argc == 1) {
447 if (strcmp(mrb_obj_classname(mrb, so), "Socket::Option") != 0)
448 mrb_raisef(mrb, E_ARGUMENT_ERROR, "not an instance of Socket::Option");
449 level = mrb_fixnum(mrb_funcall(mrb, so, "level", 0));
450 optname = mrb_fixnum(mrb_funcall(mrb, so, "optname", 0));
451 optval = mrb_funcall(mrb, so, "data", 0);
452 } else {
453 mrb_raisef(mrb, E_ARGUMENT_ERROR, "wrong number of arguments (%d for 3)", argc);
454 }
455
456 s = socket_fd(mrb, self);
457 if (setsockopt(s, level, optname, RSTRING_PTR(optval), RSTRING_LEN(optval)) == -1)
458 mrb_sys_fail(mrb, "setsockopt");
459 return mrb_fixnum_value(0);
460}
461
462static mrb_value
463mrb_basicsocket_shutdown(mrb_state *mrb, mrb_value self)
464{
465 mrb_int how = SHUT_RDWR;
466
467 mrb_get_args(mrb, "|i", &how);
468 if (shutdown(socket_fd(mrb, self), how) != 0)
469 mrb_sys_fail(mrb, "shutdown");
470 return mrb_fixnum_value(0);
471}
472
473static mrb_value
474mrb_ipsocket_ntop(mrb_state *mrb, mrb_value klass)
475{
476 mrb_int af, n;
477 char *addr, buf[50];
478
479 mrb_get_args(mrb, "is", &af, &addr, &n);
480 if ((af == AF_INET && n != 4) || (af == AF_INET6 && n != 16))
481 mrb_raise(mrb, E_ARGUMENT_ERROR, "invalid address");
482 if (inet_ntop(af, addr, buf, sizeof(buf)) == NULL)
483 mrb_raise(mrb, E_ARGUMENT_ERROR, "invalid address");
484 return mrb_str_new_cstr(mrb, buf);
485}
486
487static mrb_value
488mrb_ipsocket_pton(mrb_state *mrb, mrb_value klass)
489{
490 mrb_int af, n;
491 char *bp, buf[50];
492
493 mrb_get_args(mrb, "is", &af, &bp, &n);
494 if (n > sizeof(buf) - 1)
495 mrb_raise(mrb, E_ARGUMENT_ERROR, "invalid address");
496 memcpy(buf, bp, n);
497 buf[n] = '\0';
498
499 if (af == AF_INET) {
500 struct in_addr in;
501 if (inet_pton(AF_INET, buf, (void *)&in.s_addr) != 1)
502 goto invalid;
503 return mrb_str_new(mrb, (char *)&in.s_addr, 4);
504 } else if (af == AF_INET6) {
505 struct in6_addr in6;
506 if (inet_pton(AF_INET6, buf, (void *)&in6.s6_addr) != 1)
507 goto invalid;
508 return mrb_str_new(mrb, (char *)&in6.s6_addr, 16);
509 } else
510 mrb_raise(mrb, E_ARGUMENT_ERROR, "unsupported address family");
511
512invalid:
513 mrb_raise(mrb, E_ARGUMENT_ERROR, "invalid address");
514 return mrb_nil_value(); /* dummy */
515}
516
517static mrb_value
518mrb_ipsocket_recvfrom(mrb_state *mrb, mrb_value self)
519{
520 struct sockaddr_storage ss;
521 socklen_t socklen;
522 mrb_value a, buf, pair;
523 mrb_int flags, maxlen, n;
524 int fd;
525
526 fd = socket_fd(mrb, self);
527 flags = 0;
528 mrb_get_args(mrb, "i|i", &maxlen, &flags);
529 buf = mrb_str_buf_new(mrb, maxlen);
530 socklen = sizeof(ss);
531 n = recvfrom(fd, RSTRING_PTR(buf), maxlen, flags,
532 (struct sockaddr *)&ss, &socklen);
533 if (n == -1) {
534 mrb_sys_fail(mrb, "recvfrom");
535 }
536 mrb_str_resize(mrb, buf, n);
537 a = sa2addrlist(mrb, (struct sockaddr *)&ss, socklen);
538 pair = mrb_ary_new_capa(mrb, 2);
539 mrb_ary_push(mrb, pair, buf);
540 mrb_ary_push(mrb, pair, a);
541 return pair;
542}
543
544static mrb_value
545mrb_socket_gethostname(mrb_state *mrb, mrb_value cls)
546{
547 mrb_value buf;
548 size_t bufsize;
549
550#ifdef HOST_NAME_MAX
551 bufsize = HOST_NAME_MAX + 1;
552#else
553 bufsize = 256;
554#endif
555 buf = mrb_str_buf_new(mrb, bufsize);
556 if (gethostname(RSTRING_PTR(buf), bufsize) != 0)
557 mrb_sys_fail(mrb, "gethostname");
558 mrb_str_resize(mrb, buf, strlen(RSTRING_PTR(buf)));
559 return buf;
560}
561
562static mrb_value
563mrb_socket_accept(mrb_state *mrb, mrb_value klass)
564{
565 mrb_value ary, sastr;
566 int s1;
567 mrb_int s0;
568 socklen_t socklen;
569
570 mrb_get_args(mrb, "i", &s0);
571 socklen = sizeof(struct sockaddr_storage);
572 sastr = mrb_str_buf_new(mrb, socklen);
573 s1 = accept(s0, (struct sockaddr *)RSTRING_PTR(sastr), &socklen);
574 if (s1 == -1) {
575 mrb_sys_fail(mrb, "accept");
576 }
577 // XXX: possible descriptor leakage here!
578 mrb_str_resize(mrb, sastr, socklen);
579 ary = mrb_ary_new_capa(mrb, 2);
580 mrb_ary_push(mrb, ary, mrb_fixnum_value(s1));
581 mrb_ary_push(mrb, ary, sastr);
582 return ary;
583}
584
585static mrb_value
586mrb_socket_bind(mrb_state *mrb, mrb_value klass)
587{
588 mrb_value sastr;
589 mrb_int s;
590
591 mrb_get_args(mrb, "iS", &s, &sastr);
592 if (bind((int)s, (struct sockaddr *)RSTRING_PTR(sastr), (socklen_t)RSTRING_LEN(sastr)) == -1) {
593 mrb_sys_fail(mrb, "bind");
594 }
595 return mrb_nil_value();
596}
597
598static mrb_value
599mrb_socket_connect(mrb_state *mrb, mrb_value klass)
600{
601 mrb_value sastr;
602 mrb_int s;
603
604 mrb_get_args(mrb, "iS", &s, &sastr);
605 if (connect((int)s, (struct sockaddr *)RSTRING_PTR(sastr), (socklen_t)RSTRING_LEN(sastr)) == -1) {
606 mrb_sys_fail(mrb, "connect");
607 }
608 return mrb_nil_value();
609}
610
611static mrb_value
612mrb_socket_listen(mrb_state *mrb, mrb_value klass)
613{
614 mrb_int backlog, s;
615
616 mrb_get_args(mrb, "ii", &s, &backlog);
617 if (listen((int)s, (int)backlog) == -1) {
618 mrb_sys_fail(mrb, "listen");
619 }
620 return mrb_nil_value();
621}
622
623static mrb_value
624mrb_socket_sockaddr_family(mrb_state *mrb, mrb_value klass)
625{
626 mrb_value sa;
627
628 mrb_get_args(mrb, "S", &sa);
629#ifdef __linux__
630 if (RSTRING_LEN(sa) < offsetof(struct sockaddr, sa_family) + sizeof(sa_family_t)) {
631 mrb_raisef(mrb, E_SOCKET_ERROR, "invalid sockaddr (too short)");
632 }
633#else
634 if (RSTRING_LEN(sa) < sizeof(struct sockaddr)) {
635 mrb_raisef(mrb, E_SOCKET_ERROR, "invalid sockaddr (too short)");
636 }
637#endif
638 return mrb_fixnum_value(((struct sockaddr *)RSTRING_PTR(sa))->sa_family);
639}
640
641static mrb_value
642mrb_socket_sockaddr_un(mrb_state *mrb, mrb_value klass)
643{
644#ifdef _WIN32
645 mrb_raise(mrb, E_NOTIMP_ERROR, "sockaddr_un unsupported on Windows");
646 return mrb_nil_value();
647#else
648 struct sockaddr_un *sunp;
649 mrb_value path, s;
650
651 mrb_get_args(mrb, "S", &path);
652 if (RSTRING_LEN(path) > sizeof(sunp->sun_path) - 1) {
653 mrb_raisef(mrb, E_ARGUMENT_ERROR, "too long unix socket path (max: %ubytes)", (unsigned int)sizeof(sunp->sun_path) - 1);
654 }
655 s = mrb_str_buf_new(mrb, sizeof(struct sockaddr_un));
656 sunp = (struct sockaddr_un *)RSTRING_PTR(s);
657 sunp->sun_family = AF_UNIX;
658 memcpy(sunp->sun_path, RSTRING_PTR(path), RSTRING_LEN(path));
659 sunp->sun_path[RSTRING_LEN(path)] = '\0';
660 mrb_str_resize(mrb, s, sizeof(struct sockaddr_un));
661 return s;
662#endif
663}
664
665static mrb_value
666mrb_socket_socketpair(mrb_state *mrb, mrb_value klass)
667{
668#ifdef _WIN32
669 mrb_raise(mrb, E_NOTIMP_ERROR, "socketpair unsupported on Windows");
670 return mrb_nil_value();
671#else
672 mrb_value ary;
673 mrb_int domain, type, protocol;
674 int sv[2];
675
676 mrb_get_args(mrb, "iii", &domain, &type, &protocol);
677 if (socketpair(domain, type, protocol, sv) == -1) {
678 mrb_sys_fail(mrb, "socketpair");
679 }
680 // XXX: possible descriptor leakage here!
681 ary = mrb_ary_new_capa(mrb, 2);
682 mrb_ary_push(mrb, ary, mrb_fixnum_value(sv[0]));
683 mrb_ary_push(mrb, ary, mrb_fixnum_value(sv[1]));
684 return ary;
685#endif
686}
687
688static mrb_value
689mrb_socket_socket(mrb_state *mrb, mrb_value klass)
690{
691 mrb_int domain, type, protocol;
692 int s;
693
694 mrb_get_args(mrb, "iii", &domain, &type, &protocol);
695 s = socket(domain, type, protocol);
696 if (s == -1)
697 mrb_sys_fail(mrb, "socket");
698 return mrb_fixnum_value(s);
699}
700
701static mrb_value
702mrb_tcpsocket_allocate(mrb_state *mrb, mrb_value klass)
703{
704 struct RClass *c = mrb_class_ptr(klass);
705 enum mrb_vtype ttype = MRB_INSTANCE_TT(c);
706
707 /* copied from mrb_instance_alloc() */
708 if (ttype == 0) ttype = MRB_TT_OBJECT;
709 return mrb_obj_value((struct RObject*)mrb_obj_alloc(mrb, ttype, c));
710}
711
712/* Windows overrides for IO methods on BasicSocket objects.
713 * This is because sockets on Windows are not the same as file
714 * descriptors, and thus functions which operate on file descriptors
715 * will break on socket descriptors.
716 */
717#ifdef _WIN32
718static mrb_value
719mrb_win32_basicsocket_close(mrb_state *mrb, mrb_value self)
720{
721 if (closesocket(socket_fd(mrb, self)) != NO_ERROR)
722 mrb_raise(mrb, E_SOCKET_ERROR, "closesocket unsuccessful");
723 return mrb_nil_value();
724}
725
726#define E_EOF_ERROR (mrb_class_get(mrb, "EOFError"))
727static mrb_value
728mrb_win32_basicsocket_sysread(mrb_state *mrb, mrb_value self)
729{
730 int sd, ret;
731 mrb_value buf = mrb_nil_value();
732 mrb_int maxlen;
733
734 mrb_get_args(mrb, "i|S", &maxlen, &buf);
735 if (maxlen < 0) {
736 return mrb_nil_value();
737 }
738
739 if (mrb_nil_p(buf)) {
740 buf = mrb_str_new(mrb, NULL, maxlen);
741 }
742 if (RSTRING_LEN(buf) != maxlen) {
743 buf = mrb_str_resize(mrb, buf, maxlen);
744 }
745
746 sd = socket_fd(mrb, self);
747 ret = recv(sd, RSTRING_PTR(buf), maxlen, 0);
748
749 switch (ret) {
750 case 0: /* EOF */
751 if (maxlen == 0) {
752 buf = mrb_str_new_cstr(mrb, "");
753 } else {
754 mrb_raise(mrb, E_EOF_ERROR, "sysread failed: End of File");
755 }
756 break;
757 case SOCKET_ERROR: /* Error */
758 mrb_sys_fail(mrb, "recv");
759 break;
760 default:
761 if (RSTRING_LEN(buf) != ret) {
762 buf = mrb_str_resize(mrb, buf, ret);
763 }
764 break;
765 }
766
767 return buf;
768}
769
770static mrb_value
771mrb_win32_basicsocket_sysseek(mrb_state *mrb, mrb_value self)
772{
773 mrb_raise(mrb, E_NOTIMP_ERROR, "sysseek not implemented for windows sockets");
774 return mrb_nil_value();
775}
776
777static mrb_value
778mrb_win32_basicsocket_syswrite(mrb_state *mrb, mrb_value self)
779{
780 int n;
781 SOCKET sd;
782 mrb_value str;
783
784 sd = socket_fd(mrb, self);
785 mrb_get_args(mrb, "S", &str);
786 n = send(sd, RSTRING_PTR(str), RSTRING_LEN(str), 0);
787 if (n == SOCKET_ERROR)
788 mrb_sys_fail(mrb, "send");
789 return mrb_fixnum_value(n);
790}
791
792#endif
793
794void
795mrb_mruby_socket_gem_init(mrb_state* mrb)
796{
797 struct RClass *io, *ai, *sock, *bsock, *ipsock, *tcpsock;
798 struct RClass *constants;
799
800#ifdef _WIN32
801 WSADATA wsaData;
802 int result;
803 result = WSAStartup(MAKEWORD(2,2), &wsaData);
804 if (result != NO_ERROR)
805 mrb_raise(mrb, E_RUNTIME_ERROR, "WSAStartup failed");
806#else
807 struct RClass *usock;
808#endif
809
810 ai = mrb_define_class(mrb, "Addrinfo", mrb->object_class);
811 mrb_mod_cv_set(mrb, ai, mrb_intern_lit(mrb, "_lastai"), mrb_nil_value());
812 mrb_define_class_method(mrb, ai, "getaddrinfo", mrb_addrinfo_getaddrinfo, MRB_ARGS_REQ(2)|MRB_ARGS_OPT(4));
813 mrb_define_method(mrb, ai, "getnameinfo", mrb_addrinfo_getnameinfo, MRB_ARGS_OPT(1));
814#ifndef _WIN32
815 mrb_define_method(mrb, ai, "unix_path", mrb_addrinfo_unix_path, MRB_ARGS_NONE());
816#endif
817
818 io = mrb_class_get(mrb, "IO");
819
820 bsock = mrb_define_class(mrb, "BasicSocket", io);
821 mrb_define_method(mrb, bsock, "_recvfrom", mrb_basicsocket_recvfrom, MRB_ARGS_REQ(1)|MRB_ARGS_OPT(1));
822 mrb_define_method(mrb, bsock, "_setnonblock", mrb_basicsocket_setnonblock, MRB_ARGS_REQ(1));
823 mrb_define_method(mrb, bsock, "getpeereid", mrb_basicsocket_getpeereid, MRB_ARGS_NONE());
824 mrb_define_method(mrb, bsock, "getpeername", mrb_basicsocket_getpeername, MRB_ARGS_NONE());
825 mrb_define_method(mrb, bsock, "getsockname", mrb_basicsocket_getsockname, MRB_ARGS_NONE());
826 mrb_define_method(mrb, bsock, "getsockopt", mrb_basicsocket_getsockopt, MRB_ARGS_REQ(2));
827 mrb_define_method(mrb, bsock, "recv", mrb_basicsocket_recv, MRB_ARGS_REQ(1)|MRB_ARGS_OPT(1));
828 // #recvmsg(maxlen, flags=0)
829 mrb_define_method(mrb, bsock, "send", mrb_basicsocket_send, MRB_ARGS_REQ(2)|MRB_ARGS_OPT(1));
830 // #sendmsg
831 // #sendmsg_nonblock
832 mrb_define_method(mrb, bsock, "setsockopt", mrb_basicsocket_setsockopt, MRB_ARGS_REQ(1)|MRB_ARGS_OPT(2));
833 mrb_define_method(mrb, bsock, "shutdown", mrb_basicsocket_shutdown, MRB_ARGS_OPT(1));
834
835 ipsock = mrb_define_class(mrb, "IPSocket", bsock);
836 mrb_define_class_method(mrb, ipsock, "ntop", mrb_ipsocket_ntop, MRB_ARGS_REQ(1));
837 mrb_define_class_method(mrb, ipsock, "pton", mrb_ipsocket_pton, MRB_ARGS_REQ(2));
838 mrb_define_method(mrb, ipsock, "recvfrom", mrb_ipsocket_recvfrom, MRB_ARGS_REQ(1)|MRB_ARGS_OPT(1));
839
840 tcpsock = mrb_define_class(mrb, "TCPSocket", ipsock);
841 mrb_define_class_method(mrb, tcpsock, "_allocate", mrb_tcpsocket_allocate, MRB_ARGS_NONE());
842 mrb_define_class(mrb, "TCPServer", tcpsock);
843
844 mrb_define_class(mrb, "UDPSocket", ipsock);
845 //#recvfrom_nonblock
846
847 sock = mrb_define_class(mrb, "Socket", bsock);
848 mrb_define_class_method(mrb, sock, "_accept", mrb_socket_accept, MRB_ARGS_REQ(1));
849 mrb_define_class_method(mrb, sock, "_bind", mrb_socket_bind, MRB_ARGS_REQ(3));
850 mrb_define_class_method(mrb, sock, "_connect", mrb_socket_connect, MRB_ARGS_REQ(3));
851 mrb_define_class_method(mrb, sock, "_listen", mrb_socket_listen, MRB_ARGS_REQ(2));
852 mrb_define_class_method(mrb, sock, "_sockaddr_family", mrb_socket_sockaddr_family, MRB_ARGS_REQ(1));
853 mrb_define_class_method(mrb, sock, "_socket", mrb_socket_socket, MRB_ARGS_REQ(3));
854 //mrb_define_class_method(mrb, sock, "gethostbyaddr", mrb_socket_gethostbyaddr, MRB_ARGS_REQ(1)|MRB_ARGS_OPT(1));
855 //mrb_define_class_method(mrb, sock, "gethostbyname", mrb_socket_gethostbyname, MRB_ARGS_REQ(1)|MRB_ARGS_OPT(1));
856 mrb_define_class_method(mrb, sock, "gethostname", mrb_socket_gethostname, MRB_ARGS_NONE());
857 //mrb_define_class_method(mrb, sock, "getservbyname", mrb_socket_getservbyname, MRB_ARGS_REQ(1)|MRB_ARGS_OPT(1));
858 //mrb_define_class_method(mrb, sock, "getservbyport", mrb_socket_getservbyport, MRB_ARGS_REQ(1)|MRB_ARGS_OPT(1));
859 mrb_define_class_method(mrb, sock, "sockaddr_un", mrb_socket_sockaddr_un, MRB_ARGS_REQ(1));
860 mrb_define_class_method(mrb, sock, "socketpair", mrb_socket_socketpair, MRB_ARGS_REQ(3));
861 //mrb_define_method(mrb, sock, "sysaccept", mrb_socket_accept, MRB_ARGS_NONE());
862
863#ifndef _WIN32
864 usock = mrb_define_class(mrb, "UNIXSocket", bsock);
865#endif
866 //mrb_define_class_method(mrb, usock, "pair", mrb_unixsocket_open, MRB_ARGS_OPT(2));
867 //mrb_define_class_method(mrb, usock, "socketpair", mrb_unixsocket_open, MRB_ARGS_OPT(2));
868
869 //mrb_define_method(mrb, usock, "recv_io", mrb_unixsocket_peeraddr, MRB_ARGS_NONE());
870 //mrb_define_method(mrb, usock, "recvfrom", mrb_unixsocket_peeraddr, MRB_ARGS_NONE());
871 //mrb_define_method(mrb, usock, "send_io", mrb_unixsocket_peeraddr, MRB_ARGS_NONE());
872
873 /* Windows IO Method Overrides on BasicSocket */
874#ifdef _WIN32
875 mrb_define_method(mrb, bsock, "close", mrb_win32_basicsocket_close, MRB_ARGS_NONE());
876 mrb_define_method(mrb, bsock, "sysread", mrb_win32_basicsocket_sysread, MRB_ARGS_REQ(1)|MRB_ARGS_OPT(1));
877 mrb_define_method(mrb, bsock, "sysseek", mrb_win32_basicsocket_sysseek, MRB_ARGS_REQ(1));
878 mrb_define_method(mrb, bsock, "syswrite", mrb_win32_basicsocket_syswrite, MRB_ARGS_REQ(1));
879#endif
880
881 constants = mrb_define_module_under(mrb, sock, "Constants");
882
883#define define_const(SYM) \
884 do { \
885 mrb_define_const(mrb, constants, #SYM, mrb_fixnum_value(SYM)); \
886 } while (0)
887
888#include "const.cstub"
889}
890
891void
892mrb_mruby_socket_gem_final(mrb_state* mrb)
893{
894 mrb_value ai;
895 ai = mrb_mod_cv_get(mrb, mrb_class_get(mrb, "Addrinfo"), mrb_intern_lit(mrb, "_lastai"));
896 if (mrb_cptr_p(ai)) {
897 freeaddrinfo(mrb_cptr(ai));
898 }
899#ifdef _WIN32
900 WSACleanup();
901#endif
902}
Note: See TracBrowser for help on using the repository browser.