diff options
Diffstat (limited to 'src/socket.c')
| -rw-r--r-- | src/socket.c | 906 |
1 files changed, 0 insertions, 906 deletions
diff --git a/src/socket.c b/src/socket.c deleted file mode 100644 index d0df1c2f8..000000000 --- a/src/socket.c +++ /dev/null @@ -1,906 +0,0 @@ -/* -** socket.c - Socket module -** -** See Copyright Notice in mruby.h -*/ - -#ifdef _WIN32 - #define _WIN32_WINNT 0x0501 - - #include <winsock2.h> - #include <ws2tcpip.h> - #include <windows.h> - - #define SHUT_RDWR SD_BOTH -#else - #include <sys/types.h> - #include <sys/socket.h> - #include <sys/un.h> - #include <netinet/in.h> - #include <netinet/tcp.h> - #include <arpa/inet.h> - #include <fcntl.h> - #include <netdb.h> - #include <unistd.h> -#endif - -#include <stddef.h> -#include <string.h> - -#include "mruby.h" -#include "mruby/array.h" -#include "mruby/class.h" -#include "mruby/data.h" -#include "mruby/string.h" -#include "mruby/variable.h" -#include "error.h" - -#define E_SOCKET_ERROR (mrb_class_get(mrb, "SocketError")) - -#if !defined(mrb_cptr) -#define mrb_cptr_value(m,p) mrb_voidp_value((m),(p)) -#define mrb_cptr(o) mrb_voidp(o) -#define mrb_cptr_p(o) mrb_voidp_p(o) -#endif - -#ifdef _WIN32 -const char *inet_ntop(int af, const void *src, char *dst, socklen_t cnt) -{ - if (af == AF_INET) - { - struct sockaddr_in in; - memset(&in, 0, sizeof(in)); - in.sin_family = AF_INET; - memcpy(&in.sin_addr, src, sizeof(struct in_addr)); - getnameinfo((struct sockaddr *)&in, sizeof(struct - sockaddr_in), dst, cnt, NULL, 0, NI_NUMERICHOST); - return dst; - } - else if (af == AF_INET6) - { - struct sockaddr_in6 in; - memset(&in, 0, sizeof(in)); - in.sin6_family = AF_INET6; - memcpy(&in.sin6_addr, src, sizeof(struct in_addr6)); - getnameinfo((struct sockaddr *)&in, sizeof(struct - sockaddr_in6), dst, cnt, NULL, 0, NI_NUMERICHOST); - return dst; - } - return NULL; -} - -int inet_pton(int af, const char *src, void *dst) -{ - struct addrinfo hints, *res, *ressave; - - memset(&hints, 0, sizeof(struct addrinfo)); - hints.ai_family = af; - - if (getaddrinfo(src, NULL, &hints, &res) != 0) - { - printf("Couldn't resolve host %s\n", src); - return -1; - } - - ressave = res; - - while (res) - { - memcpy(dst, res->ai_addr, res->ai_addrlen); - res = res->ai_next; - } - - freeaddrinfo(ressave); - return 0; -} - -#endif - -static mrb_value -mrb_addrinfo_getaddrinfo(mrb_state *mrb, mrb_value klass) -{ - struct addrinfo hints, *res0, *res; - mrb_value ai, ary, family, lastai, nodename, protocol, sa, service, socktype; - mrb_int flags; - int arena_idx, error; - const char *hostname = NULL, *servname = NULL; - - ary = mrb_ary_new(mrb); - arena_idx = mrb_gc_arena_save(mrb); /* ary must be on arena! */ - - family = socktype = protocol = mrb_nil_value(); - flags = 0; - mrb_get_args(mrb, "oo|oooi", &nodename, &service, &family, &socktype, &protocol, &flags); - - if (mrb_string_p(nodename)) { - hostname = mrb_str_to_cstr(mrb, nodename); - } else if (mrb_nil_p(nodename)) { - hostname = NULL; - } else { - mrb_raise(mrb, E_TYPE_ERROR, "nodename must be String or nil"); - } - - if (mrb_string_p(service)) { - servname = mrb_str_to_cstr(mrb, service); - } else if (mrb_fixnum_p(service)) { - servname = mrb_str_to_cstr(mrb, mrb_funcall(mrb, service, "to_s", 0)); - } else if (mrb_nil_p(service)) { - servname = NULL; - } else { - mrb_raise(mrb, E_TYPE_ERROR, "service must be String, Fixnum, or nil"); - } - - memset(&hints, 0, sizeof(hints)); - hints.ai_flags = flags; - - if (mrb_fixnum_p(family)) { - hints.ai_family = mrb_fixnum(family); - } - - if (mrb_fixnum_p(socktype)) { - hints.ai_socktype = mrb_fixnum(socktype); - } - - if (mrb_fixnum_p(protocol)) { - hints.ai_protocol = mrb_fixnum(protocol); - } - - lastai = mrb_cv_get(mrb, klass, mrb_intern_lit(mrb, "_lastai")); - if (mrb_cptr_p(lastai)) { - freeaddrinfo(mrb_cptr(lastai)); - mrb_cv_set(mrb, klass, mrb_intern_lit(mrb, "_lastai"), mrb_nil_value()); - } - - error = getaddrinfo(hostname, servname, &hints, &res0); - if (error) { - mrb_raisef(mrb, E_SOCKET_ERROR, "getaddrinfo: %S", mrb_str_new_cstr(mrb, gai_strerror(error))); - } - mrb_cv_set(mrb, klass, mrb_intern_lit(mrb, "_lastai"), mrb_cptr_value(mrb, res0)); - - for (res = res0; res != NULL; res = res->ai_next) { - sa = mrb_str_new(mrb, (void *)res->ai_addr, res->ai_addrlen); - 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)); - mrb_ary_push(mrb, ary, ai); - mrb_gc_arena_restore(mrb, arena_idx); - } - - freeaddrinfo(res0); - mrb_cv_set(mrb, klass, mrb_intern_lit(mrb, "_lastai"), mrb_nil_value()); - - return ary; -} - -static mrb_value -mrb_addrinfo_getnameinfo(mrb_state *mrb, mrb_value self) -{ - mrb_int flags; - mrb_value ary, host, sastr, serv; - int error; - - flags = 0; - mrb_get_args(mrb, "|i", &flags); - host = mrb_str_buf_new(mrb, NI_MAXHOST); - serv = mrb_str_buf_new(mrb, NI_MAXSERV); - - sastr = mrb_iv_get(mrb, self, mrb_intern_lit(mrb, "@sockaddr")); - if (!mrb_string_p(sastr)) { - mrb_raise(mrb, E_SOCKET_ERROR, "invalid sockaddr"); - } - error = getnameinfo((struct sockaddr *)RSTRING_PTR(sastr), (socklen_t)RSTRING_LEN(sastr), RSTRING_PTR(host), NI_MAXHOST, RSTRING_PTR(serv), NI_MAXSERV, flags); - if (error != 0) { - mrb_raisef(mrb, E_SOCKET_ERROR, "getnameinfo: %s", gai_strerror(error)); - } - ary = mrb_ary_new_capa(mrb, 2); - mrb_str_resize(mrb, host, strlen(RSTRING_PTR(host))); - mrb_ary_push(mrb, ary, host); - mrb_str_resize(mrb, serv, strlen(RSTRING_PTR(serv))); - mrb_ary_push(mrb, ary, serv); - return ary; -} - -#ifndef _WIN32 -static mrb_value -mrb_addrinfo_unix_path(mrb_state *mrb, mrb_value self) -{ - mrb_value sastr; - - sastr = mrb_iv_get(mrb, self, mrb_intern_lit(mrb, "@sockaddr")); - if (((struct sockaddr *)RSTRING_PTR(sastr))->sa_family != AF_UNIX) - mrb_raise(mrb, E_SOCKET_ERROR, "need AF_UNIX address"); - return mrb_str_new_cstr(mrb, ((struct sockaddr_un *)RSTRING_PTR(sastr))->sun_path); -} -#endif - -static mrb_value -sa2addrlist(mrb_state *mrb, const struct sockaddr *sa, socklen_t salen) -{ - mrb_value ary, host; - unsigned short port; - const char *afstr; - - switch (sa->sa_family) { - case AF_INET: - afstr = "AF_INET"; - port = ((struct sockaddr_in *)sa)->sin_port; - break; - case AF_INET6: - afstr = "AF_INET6"; - port = ((struct sockaddr_in6 *)sa)->sin6_port; - break; - default: - mrb_raise(mrb, E_ARGUMENT_ERROR, "bad af"); - return mrb_nil_value(); - } - port = ntohs(port); - host = mrb_str_buf_new(mrb, NI_MAXHOST); - if (getnameinfo(sa, salen, RSTRING_PTR(host), NI_MAXHOST, NULL, 0, NI_NUMERICHOST) == -1) - mrb_sys_fail(mrb, "getnameinfo"); - mrb_str_resize(mrb, host, strlen(RSTRING_PTR(host))); - ary = mrb_ary_new_capa(mrb, 4); - mrb_ary_push(mrb, ary, mrb_str_new_cstr(mrb, afstr)); - mrb_ary_push(mrb, ary, mrb_fixnum_value(port)); - mrb_ary_push(mrb, ary, host); - mrb_ary_push(mrb, ary, host); - return ary; -} - -static int -socket_fd(mrb_state *mrb, mrb_value sock) -{ - return mrb_fixnum(mrb_funcall(mrb, sock, "fileno", 0)); -} - -static int -socket_family(int s) -{ - struct sockaddr_storage ss; - socklen_t salen; - - salen = sizeof(ss); - if (getsockname(s, (struct sockaddr *)&ss, &salen) == -1) - return AF_UNSPEC; - return ss.ss_family; -} - -static mrb_value -mrb_basicsocket_getpeereid(mrb_state *mrb, mrb_value self) -{ -#ifdef HAVE_GETPEEREID - mrb_value ary; - gid_t egid; - uid_t euid; - int s; - - s = socket_fd(mrb, self); - if (getpeereid(s, &euid, &egid) != 0) - mrb_sys_fail(mrb, "getpeereid"); - - ary = mrb_ary_new_capa(mrb, 2); - mrb_ary_push(mrb, ary, mrb_fixnum_value((mrb_int)euid)); - mrb_ary_push(mrb, ary, mrb_fixnum_value((mrb_int)egid)); - return ary; -#else - mrb_raise(mrb, E_RUNTIME_ERROR, "getpeereid is not avaialble on this system"); - return mrb_nil_value(); -#endif -} - -static mrb_value -mrb_basicsocket_getpeername(mrb_state *mrb, mrb_value self) -{ - struct sockaddr_storage ss; - socklen_t salen; - - salen = sizeof(ss); - if (getpeername(socket_fd(mrb, self), (struct sockaddr *)&ss, &salen) != 0) - mrb_sys_fail(mrb, "getpeername"); - - return mrb_str_new(mrb, (void *)&ss, salen); -} - -static mrb_value -mrb_basicsocket_getsockname(mrb_state *mrb, mrb_value self) -{ - struct sockaddr_storage ss; - socklen_t salen; - - salen = sizeof(ss); - if (getsockname(socket_fd(mrb, self), (struct sockaddr *)&ss, &salen) != 0) - mrb_sys_fail(mrb, "getsockname"); - - return mrb_str_new(mrb, (void *)&ss, salen); -} - -static mrb_value -mrb_basicsocket_getsockopt(mrb_state *mrb, mrb_value self) -{ - char opt[8]; - int s; - mrb_int family, level, optname; - mrb_value c, data; - socklen_t optlen; - - mrb_get_args(mrb, "ii", &level, &optname); - s = socket_fd(mrb, self); - optlen = sizeof(opt); - if (getsockopt(s, level, optname, opt, &optlen) == -1) - mrb_sys_fail(mrb, "getsockopt"); - c = mrb_const_get(mrb, mrb_obj_value(mrb_class_get(mrb, "Socket")), mrb_intern_lit(mrb, "Option")); - family = socket_family(s); - data = mrb_str_new(mrb, opt, optlen); - return mrb_funcall(mrb, c, "new", 4, mrb_fixnum_value(family), mrb_fixnum_value(level), mrb_fixnum_value(optname), data); -} - -static mrb_value -mrb_basicsocket_recv(mrb_state *mrb, mrb_value self) -{ - int n; - mrb_int maxlen, flags = 0; - mrb_value buf; - - mrb_get_args(mrb, "i|i", &maxlen, &flags); - buf = mrb_str_buf_new(mrb, maxlen); - n = recv(socket_fd(mrb, self), RSTRING_PTR(buf), maxlen, flags); - if (n == -1) - mrb_sys_fail(mrb, "recv"); - mrb_str_resize(mrb, buf, n); - return buf; -} - -static mrb_value -mrb_basicsocket_recvfrom(mrb_state *mrb, mrb_value self) -{ - int n; - mrb_int maxlen, flags = 0; - mrb_value ary, buf, sa; - socklen_t socklen; - - mrb_get_args(mrb, "i|i", &maxlen, &flags); - buf = mrb_str_buf_new(mrb, maxlen); - socklen = sizeof(struct sockaddr_storage); - sa = mrb_str_buf_new(mrb, socklen); - n = recvfrom(socket_fd(mrb, self), RSTRING_PTR(buf), maxlen, flags, (struct sockaddr *)RSTRING_PTR(sa), &socklen); - if (n == -1) - mrb_sys_fail(mrb, "recvfrom"); - mrb_str_resize(mrb, buf, n); - mrb_str_resize(mrb, sa, socklen); - ary = mrb_ary_new_capa(mrb, 2); - mrb_ary_push(mrb, ary, buf); - mrb_ary_push(mrb, ary, sa); - return ary; -} - -static mrb_value -mrb_basicsocket_send(mrb_state *mrb, mrb_value self) -{ - int n; - mrb_int flags; - mrb_value dest, mesg; - - dest = mrb_nil_value(); - mrb_get_args(mrb, "Si|S", &mesg, &flags, &dest); - if (mrb_nil_p(dest)) { - n = send(socket_fd(mrb, self), RSTRING_PTR(mesg), RSTRING_LEN(mesg), flags); - } else { - n = sendto(socket_fd(mrb, self), RSTRING_PTR(mesg), RSTRING_LEN(mesg), flags, (const void *)RSTRING_PTR(dest), RSTRING_LEN(dest)); - } - if (n == -1) - mrb_sys_fail(mrb, "send"); - return mrb_fixnum_value(n); -} - -static mrb_value -mrb_basicsocket_setnonblock(mrb_state *mrb, mrb_value self) -{ - int fd, flags; - mrb_value bool; -#ifdef _WIN32 - u_long mode = 1; -#endif - - mrb_get_args(mrb, "o", &bool); - fd = socket_fd(mrb, self); -#ifdef _WIN32 - flags = ioctlsocket(fd, FIONBIO, &mode); - if (flags != NO_ERROR) - mrb_sys_fail(mrb, "ioctlsocket"); -#else - flags = fcntl(fd, F_GETFL, 0); - if (flags == 1) - mrb_sys_fail(mrb, "fcntl"); - if (mrb_test(bool)) - flags |= O_NONBLOCK; - else - flags &= ~O_NONBLOCK; - if (fcntl(fd, F_SETFL, flags) == -1) - mrb_sys_fail(mrb, "fcntl"); -#endif - return mrb_nil_value(); -} - -static mrb_value -mrb_basicsocket_setsockopt(mrb_state *mrb, mrb_value self) -{ - int argc, s; - mrb_int level = 0, optname; - mrb_value optval, so; - - argc = mrb_get_args(mrb, "o|io", &so, &optname, &optval); - if (argc == 3) { - if (!mrb_fixnum_p(so)) { - mrb_raise(mrb, E_ARGUMENT_ERROR, "level is not an integer"); - } - level = mrb_fixnum(so); - if (mrb_string_p(optval)) { - /* that's good */ - } else if (mrb_type(optval) == MRB_TT_TRUE || mrb_type(optval) == MRB_TT_FALSE) { - mrb_int i = mrb_test(optval) ? 1 : 0; - optval = mrb_str_new(mrb, (char *)&i, sizeof(i)); - } else if (mrb_fixnum_p(optval)) { - if (optname == IP_MULTICAST_TTL || optname == IP_MULTICAST_LOOP) { - char uc = mrb_fixnum(optval); - optval = mrb_str_new(mrb, &uc, sizeof(uc)); - } else { - mrb_int i = mrb_fixnum(optval); - optval = mrb_str_new(mrb, (char *)&i, sizeof(i)); - } - } else { - mrb_raise(mrb, E_ARGUMENT_ERROR, "optval should be true, false, an integer, or a string"); - } - } else if (argc == 1) { - if (strcmp(mrb_obj_classname(mrb, so), "Socket::Option") != 0) - mrb_raisef(mrb, E_ARGUMENT_ERROR, "not an instance of Socket::Option"); - level = mrb_fixnum(mrb_funcall(mrb, so, "level", 0)); - optname = mrb_fixnum(mrb_funcall(mrb, so, "optname", 0)); - optval = mrb_funcall(mrb, so, "data", 0); - } else { - mrb_raisef(mrb, E_ARGUMENT_ERROR, "wrong number of arguments (%d for 3)", argc); - } - - s = socket_fd(mrb, self); - if (setsockopt(s, level, optname, RSTRING_PTR(optval), RSTRING_LEN(optval)) == -1) - mrb_sys_fail(mrb, "setsockopt"); - return mrb_fixnum_value(0); -} - -static mrb_value -mrb_basicsocket_shutdown(mrb_state *mrb, mrb_value self) -{ - mrb_int how = SHUT_RDWR; - - mrb_get_args(mrb, "|i", &how); - if (shutdown(socket_fd(mrb, self), how) != 0) - mrb_sys_fail(mrb, "shutdown"); - return mrb_fixnum_value(0); -} - -static mrb_value -mrb_ipsocket_ntop(mrb_state *mrb, mrb_value klass) -{ - mrb_int af, n; - char *addr, buf[50]; - - mrb_get_args(mrb, "is", &af, &addr, &n); - if ((af == AF_INET && n != 4) || (af == AF_INET6 && n != 16)) - mrb_raise(mrb, E_ARGUMENT_ERROR, "invalid address"); - if (inet_ntop(af, addr, buf, sizeof(buf)) == NULL) - mrb_raise(mrb, E_ARGUMENT_ERROR, "invalid address"); - return mrb_str_new_cstr(mrb, buf); -} - -static mrb_value -mrb_ipsocket_pton(mrb_state *mrb, mrb_value klass) -{ - mrb_int af, n; - char *bp, buf[50]; - - mrb_get_args(mrb, "is", &af, &bp, &n); - if (n > sizeof(buf) - 1) - mrb_raise(mrb, E_ARGUMENT_ERROR, "invalid address"); - memcpy(buf, bp, n); - buf[n] = '\0'; - - if (af == AF_INET) { - struct in_addr in; - if (inet_pton(AF_INET, buf, (void *)&in.s_addr) != 1) - goto invalid; - return mrb_str_new(mrb, (char *)&in.s_addr, 4); - } else if (af == AF_INET6) { - struct in6_addr in6; - if (inet_pton(AF_INET6, buf, (void *)&in6.s6_addr) != 1) - goto invalid; - return mrb_str_new(mrb, (char *)&in6.s6_addr, 16); - } else - mrb_raise(mrb, E_ARGUMENT_ERROR, "unsupported address family"); - -invalid: - mrb_raise(mrb, E_ARGUMENT_ERROR, "invalid address"); - return mrb_nil_value(); /* dummy */ -} - -static mrb_value -mrb_ipsocket_recvfrom(mrb_state *mrb, mrb_value self) -{ - struct sockaddr_storage ss; - socklen_t socklen; - mrb_value a, buf, pair; - mrb_int flags, maxlen, n; - int fd; - - fd = socket_fd(mrb, self); - flags = 0; - mrb_get_args(mrb, "i|i", &maxlen, &flags); - buf = mrb_str_buf_new(mrb, maxlen); - socklen = sizeof(ss); - n = recvfrom(fd, RSTRING_PTR(buf), maxlen, flags, - (struct sockaddr *)&ss, &socklen); - if (n == -1) { - mrb_sys_fail(mrb, "recvfrom"); - } - mrb_str_resize(mrb, buf, n); - a = sa2addrlist(mrb, (struct sockaddr *)&ss, socklen); - pair = mrb_ary_new_capa(mrb, 2); - mrb_ary_push(mrb, pair, buf); - mrb_ary_push(mrb, pair, a); - return pair; -} - -static mrb_value -mrb_socket_gethostname(mrb_state *mrb, mrb_value cls) -{ - mrb_value buf; - size_t bufsize; - -#ifdef HOST_NAME_MAX - bufsize = HOST_NAME_MAX + 1; -#else - bufsize = 256; -#endif - buf = mrb_str_buf_new(mrb, bufsize); - if (gethostname(RSTRING_PTR(buf), bufsize) != 0) - mrb_sys_fail(mrb, "gethostname"); - mrb_str_resize(mrb, buf, strlen(RSTRING_PTR(buf))); - return buf; -} - -static mrb_value -mrb_socket_accept(mrb_state *mrb, mrb_value klass) -{ - mrb_value ary, sastr; - int s1; - mrb_int s0; - socklen_t socklen; - - mrb_get_args(mrb, "i", &s0); - socklen = sizeof(struct sockaddr_storage); - sastr = mrb_str_buf_new(mrb, socklen); - s1 = accept(s0, (struct sockaddr *)RSTRING_PTR(sastr), &socklen); - if (s1 == -1) { - mrb_sys_fail(mrb, "accept"); - } - // XXX: possible descriptor leakage here! - mrb_str_resize(mrb, sastr, socklen); - ary = mrb_ary_new_capa(mrb, 2); - mrb_ary_push(mrb, ary, mrb_fixnum_value(s1)); - mrb_ary_push(mrb, ary, sastr); - return ary; -} - -static mrb_value -mrb_socket_bind(mrb_state *mrb, mrb_value klass) -{ - mrb_value sastr; - mrb_int s; - - mrb_get_args(mrb, "iS", &s, &sastr); - if (bind((int)s, (struct sockaddr *)RSTRING_PTR(sastr), (socklen_t)RSTRING_LEN(sastr)) == -1) { - mrb_sys_fail(mrb, "bind"); - } - return mrb_nil_value(); -} - -static mrb_value -mrb_socket_connect(mrb_state *mrb, mrb_value klass) -{ - mrb_value sastr; - mrb_int s; - - mrb_get_args(mrb, "iS", &s, &sastr); - if (connect((int)s, (struct sockaddr *)RSTRING_PTR(sastr), (socklen_t)RSTRING_LEN(sastr)) == -1) { - mrb_sys_fail(mrb, "connect"); - } - return mrb_nil_value(); -} - -static mrb_value -mrb_socket_listen(mrb_state *mrb, mrb_value klass) -{ - mrb_int backlog, s; - - mrb_get_args(mrb, "ii", &s, &backlog); - if (listen((int)s, (int)backlog) == -1) { - mrb_sys_fail(mrb, "listen"); - } - return mrb_nil_value(); -} - -static mrb_value -mrb_socket_sockaddr_family(mrb_state *mrb, mrb_value klass) -{ - mrb_value sa; - - mrb_get_args(mrb, "S", &sa); -#ifdef __linux__ - if (RSTRING_LEN(sa) < offsetof(struct sockaddr, sa_family) + sizeof(sa_family_t)) { - mrb_raisef(mrb, E_SOCKET_ERROR, "invalid sockaddr (too short)"); - } -#else - if (RSTRING_LEN(sa) < sizeof(struct sockaddr)) { - mrb_raisef(mrb, E_SOCKET_ERROR, "invalid sockaddr (too short)"); - } -#endif - return mrb_fixnum_value(((struct sockaddr *)RSTRING_PTR(sa))->sa_family); -} - -static mrb_value -mrb_socket_sockaddr_un(mrb_state *mrb, mrb_value klass) -{ -#ifdef _WIN32 - mrb_raise(mrb, E_NOTIMP_ERROR, "sockaddr_un unsupported on Windows"); - return mrb_nil_value(); -#else - struct sockaddr_un *sunp; - mrb_value path, s; - - mrb_get_args(mrb, "S", &path); - if (RSTRING_LEN(path) > sizeof(sunp->sun_path) - 1) { - mrb_raisef(mrb, E_ARGUMENT_ERROR, "too long unix socket path (max: %ubytes)", (unsigned int)sizeof(sunp->sun_path) - 1); - } - s = mrb_str_buf_new(mrb, sizeof(struct sockaddr_un)); - sunp = (struct sockaddr_un *)RSTRING_PTR(s); - sunp->sun_family = AF_UNIX; - memcpy(sunp->sun_path, RSTRING_PTR(path), RSTRING_LEN(path)); - sunp->sun_path[RSTRING_LEN(path)] = '\0'; - mrb_str_resize(mrb, s, sizeof(struct sockaddr_un)); - return s; -#endif -} - -static mrb_value -mrb_socket_socketpair(mrb_state *mrb, mrb_value klass) -{ -#ifdef _WIN32 - mrb_raise(mrb, E_NOTIMP_ERROR, "socketpair unsupported on Windows"); - return mrb_nil_value(); -#else - mrb_value ary; - mrb_int domain, type, protocol; - int sv[2]; - - mrb_get_args(mrb, "iii", &domain, &type, &protocol); - if (socketpair(domain, type, protocol, sv) == -1) { - mrb_sys_fail(mrb, "socketpair"); - } - // XXX: possible descriptor leakage here! - ary = mrb_ary_new_capa(mrb, 2); - mrb_ary_push(mrb, ary, mrb_fixnum_value(sv[0])); - mrb_ary_push(mrb, ary, mrb_fixnum_value(sv[1])); - return ary; -#endif -} - -static mrb_value -mrb_socket_socket(mrb_state *mrb, mrb_value klass) -{ - mrb_int domain, type, protocol; - int s; - - mrb_get_args(mrb, "iii", &domain, &type, &protocol); - s = socket(domain, type, protocol); - if (s == -1) - mrb_sys_fail(mrb, "socket"); - return mrb_fixnum_value(s); -} - -static mrb_value -mrb_tcpsocket_allocate(mrb_state *mrb, mrb_value klass) -{ - struct RClass *c = mrb_class_ptr(klass); - enum mrb_vtype ttype = MRB_INSTANCE_TT(c); - - /* copied from mrb_instance_alloc() */ - if (ttype == 0) ttype = MRB_TT_OBJECT; - return mrb_obj_value((struct RObject*)mrb_obj_alloc(mrb, ttype, c)); -} - -/* Windows overrides for IO methods on BasicSocket objects. - * This is because sockets on Windows are not the same as file - * descriptors, and thus functions which operate on file descriptors - * will break on socket descriptors. - */ -#ifdef _WIN32 -static mrb_value -mrb_win32_basicsocket_close(mrb_state *mrb, mrb_value self) -{ - if (closesocket(socket_fd(mrb, self)) != NO_ERROR) - mrb_raise(mrb, E_SOCKET_ERROR, "closesocket unsuccessful"); - return mrb_nil_value(); -} - -#define E_EOF_ERROR (mrb_class_get(mrb, "EOFError")) -static mrb_value -mrb_win32_basicsocket_sysread(mrb_state *mrb, mrb_value self) -{ - int sd, ret; - mrb_value buf = mrb_nil_value(); - mrb_int maxlen; - - mrb_get_args(mrb, "i|S", &maxlen, &buf); - if (maxlen < 0) { - return mrb_nil_value(); - } - - if (mrb_nil_p(buf)) { - buf = mrb_str_new(mrb, NULL, maxlen); - } - if (RSTRING_LEN(buf) != maxlen) { - buf = mrb_str_resize(mrb, buf, maxlen); - } - - sd = socket_fd(mrb, self); - ret = recv(sd, RSTRING_PTR(buf), maxlen, 0); - - switch (ret) { - case 0: /* EOF */ - if (maxlen == 0) { - buf = mrb_str_new_cstr(mrb, ""); - } else { - mrb_raise(mrb, E_EOF_ERROR, "sysread failed: End of File"); - } - break; - case SOCKET_ERROR: /* Error */ - mrb_sys_fail(mrb, "recv"); - break; - default: - if (RSTRING_LEN(buf) != ret) { - buf = mrb_str_resize(mrb, buf, ret); - } - break; - } - - return buf; -} - -static mrb_value -mrb_win32_basicsocket_sysseek(mrb_state *mrb, mrb_value self) -{ - mrb_raise(mrb, E_NOTIMP_ERROR, "sysseek not implemented for windows sockets"); - return mrb_nil_value(); -} - -static mrb_value -mrb_win32_basicsocket_syswrite(mrb_state *mrb, mrb_value self) -{ - int n; - SOCKET sd; - mrb_value str; - - sd = socket_fd(mrb, self); - mrb_get_args(mrb, "S", &str); - n = send(sd, RSTRING_PTR(str), RSTRING_LEN(str), 0); - if (n == SOCKET_ERROR) - mrb_sys_fail(mrb, "send"); - return mrb_fixnum_value(n); -} - -#endif - -void -mrb_mruby_socket_gem_init(mrb_state* mrb) -{ - struct RClass *io, *ai, *sock, *bsock, *ipsock, *tcpsock; - struct RClass *constants; - -#ifdef _WIN32 - WSADATA wsaData; - int result; - result = WSAStartup(MAKEWORD(2,2), &wsaData); - if (result != NO_ERROR) - mrb_raise(mrb, E_RUNTIME_ERROR, "WSAStartup failed"); -#else - struct RClass *usock; -#endif - - ai = mrb_define_class(mrb, "Addrinfo", mrb->object_class); - mrb_mod_cv_set(mrb, ai, mrb_intern_lit(mrb, "_lastai"), mrb_nil_value()); - mrb_define_class_method(mrb, ai, "getaddrinfo", mrb_addrinfo_getaddrinfo, MRB_ARGS_REQ(2)|MRB_ARGS_OPT(4)); - mrb_define_method(mrb, ai, "getnameinfo", mrb_addrinfo_getnameinfo, MRB_ARGS_OPT(1)); -#ifndef _WIN32 - mrb_define_method(mrb, ai, "unix_path", mrb_addrinfo_unix_path, MRB_ARGS_NONE()); -#endif - - io = mrb_class_get(mrb, "IO"); - - bsock = mrb_define_class(mrb, "BasicSocket", io); - mrb_define_method(mrb, bsock, "_recvfrom", mrb_basicsocket_recvfrom, MRB_ARGS_REQ(1)|MRB_ARGS_OPT(1)); - mrb_define_method(mrb, bsock, "_setnonblock", mrb_basicsocket_setnonblock, MRB_ARGS_REQ(1)); - mrb_define_method(mrb, bsock, "getpeereid", mrb_basicsocket_getpeereid, MRB_ARGS_NONE()); - mrb_define_method(mrb, bsock, "getpeername", mrb_basicsocket_getpeername, MRB_ARGS_NONE()); - mrb_define_method(mrb, bsock, "getsockname", mrb_basicsocket_getsockname, MRB_ARGS_NONE()); - mrb_define_method(mrb, bsock, "getsockopt", mrb_basicsocket_getsockopt, MRB_ARGS_REQ(2)); - mrb_define_method(mrb, bsock, "recv", mrb_basicsocket_recv, MRB_ARGS_REQ(1)|MRB_ARGS_OPT(1)); - // #recvmsg(maxlen, flags=0) - mrb_define_method(mrb, bsock, "send", mrb_basicsocket_send, MRB_ARGS_REQ(2)|MRB_ARGS_OPT(1)); - // #sendmsg - // #sendmsg_nonblock - mrb_define_method(mrb, bsock, "setsockopt", mrb_basicsocket_setsockopt, MRB_ARGS_REQ(1)|MRB_ARGS_OPT(2)); - mrb_define_method(mrb, bsock, "shutdown", mrb_basicsocket_shutdown, MRB_ARGS_OPT(1)); - - ipsock = mrb_define_class(mrb, "IPSocket", bsock); - mrb_define_class_method(mrb, ipsock, "ntop", mrb_ipsocket_ntop, MRB_ARGS_REQ(1)); - mrb_define_class_method(mrb, ipsock, "pton", mrb_ipsocket_pton, MRB_ARGS_REQ(2)); - mrb_define_method(mrb, ipsock, "recvfrom", mrb_ipsocket_recvfrom, MRB_ARGS_REQ(1)|MRB_ARGS_OPT(1)); - - tcpsock = mrb_define_class(mrb, "TCPSocket", ipsock); - mrb_define_class_method(mrb, tcpsock, "_allocate", mrb_tcpsocket_allocate, MRB_ARGS_NONE()); - mrb_define_class(mrb, "TCPServer", tcpsock); - - mrb_define_class(mrb, "UDPSocket", ipsock); - //#recvfrom_nonblock - - sock = mrb_define_class(mrb, "Socket", bsock); - mrb_define_class_method(mrb, sock, "_accept", mrb_socket_accept, MRB_ARGS_REQ(1)); - mrb_define_class_method(mrb, sock, "_bind", mrb_socket_bind, MRB_ARGS_REQ(3)); - mrb_define_class_method(mrb, sock, "_connect", mrb_socket_connect, MRB_ARGS_REQ(3)); - mrb_define_class_method(mrb, sock, "_listen", mrb_socket_listen, MRB_ARGS_REQ(2)); - mrb_define_class_method(mrb, sock, "_sockaddr_family", mrb_socket_sockaddr_family, MRB_ARGS_REQ(1)); - mrb_define_class_method(mrb, sock, "_socket", mrb_socket_socket, MRB_ARGS_REQ(3)); - //mrb_define_class_method(mrb, sock, "gethostbyaddr", mrb_socket_gethostbyaddr, MRB_ARGS_REQ(1)|MRB_ARGS_OPT(1)); - //mrb_define_class_method(mrb, sock, "gethostbyname", mrb_socket_gethostbyname, MRB_ARGS_REQ(1)|MRB_ARGS_OPT(1)); - mrb_define_class_method(mrb, sock, "gethostname", mrb_socket_gethostname, MRB_ARGS_NONE()); - //mrb_define_class_method(mrb, sock, "getservbyname", mrb_socket_getservbyname, MRB_ARGS_REQ(1)|MRB_ARGS_OPT(1)); - //mrb_define_class_method(mrb, sock, "getservbyport", mrb_socket_getservbyport, MRB_ARGS_REQ(1)|MRB_ARGS_OPT(1)); - mrb_define_class_method(mrb, sock, "sockaddr_un", mrb_socket_sockaddr_un, MRB_ARGS_REQ(1)); - mrb_define_class_method(mrb, sock, "socketpair", mrb_socket_socketpair, MRB_ARGS_REQ(3)); - //mrb_define_method(mrb, sock, "sysaccept", mrb_socket_accept, MRB_ARGS_NONE()); - -#ifndef _WIN32 - usock = mrb_define_class(mrb, "UNIXSocket", bsock); -#endif - //mrb_define_class_method(mrb, usock, "pair", mrb_unixsocket_open, MRB_ARGS_OPT(2)); - //mrb_define_class_method(mrb, usock, "socketpair", mrb_unixsocket_open, MRB_ARGS_OPT(2)); - - //mrb_define_method(mrb, usock, "recv_io", mrb_unixsocket_peeraddr, MRB_ARGS_NONE()); - //mrb_define_method(mrb, usock, "recvfrom", mrb_unixsocket_peeraddr, MRB_ARGS_NONE()); - //mrb_define_method(mrb, usock, "send_io", mrb_unixsocket_peeraddr, MRB_ARGS_NONE()); - - /* Windows IO Method Overrides on BasicSocket */ -#ifdef _WIN32 - mrb_define_method(mrb, bsock, "close", mrb_win32_basicsocket_close, MRB_ARGS_NONE()); - mrb_define_method(mrb, bsock, "sysread", mrb_win32_basicsocket_sysread, MRB_ARGS_REQ(1)|MRB_ARGS_OPT(1)); - mrb_define_method(mrb, bsock, "sysseek", mrb_win32_basicsocket_sysseek, MRB_ARGS_REQ(1)); - mrb_define_method(mrb, bsock, "syswrite", mrb_win32_basicsocket_syswrite, MRB_ARGS_REQ(1)); -#endif - - constants = mrb_define_module_under(mrb, sock, "Constants"); - -#define define_const(SYM) \ - do { \ - mrb_define_const(mrb, constants, #SYM, mrb_fixnum_value(SYM)); \ - } while (0) - -#include "const.cstub" -} - -void -mrb_mruby_socket_gem_final(mrb_state* mrb) -{ - mrb_value ai; - ai = mrb_mod_cv_get(mrb, mrb_class_get(mrb, "Addrinfo"), mrb_intern_lit(mrb, "_lastai")); - if (mrb_cptr_p(ai)) { - freeaddrinfo(mrb_cptr(ai)); - } -#ifdef _WIN32 - WSACleanup(); -#endif -} |
