123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374 |
- /*=========================================================================*\
- * Common option interface
- * LuaSocket toolkit
- \*=========================================================================*/
- #include <string.h>
- #include "lauxlib.h"
- #include "auxiliar.h"
- #include "options.h"
- #include "inet.h"
- /*=========================================================================*\
- * Internal functions prototypes
- \*=========================================================================*/
- static int opt_setmembership(lua_State *L, p_socket ps, int level, int name);
- static int opt_ip6_setmembership(lua_State *L, p_socket ps, int level, int name);
- static int opt_setboolean(lua_State *L, p_socket ps, int level, int name);
- static int opt_getboolean(lua_State *L, p_socket ps, int level, int name);
- static int opt_setint(lua_State *L, p_socket ps, int level, int name);
- static int opt_getint(lua_State *L, p_socket ps, int level, int name);
- static int opt_set(lua_State *L, p_socket ps, int level, int name,
- void *val, int len);
- static int opt_get(lua_State *L, p_socket ps, int level, int name,
- void *val, int* len);
- /*=========================================================================*\
- * Exported functions
- \*=========================================================================*/
- /*-------------------------------------------------------------------------*\
- * Calls appropriate option handler
- \*-------------------------------------------------------------------------*/
- int opt_meth_setoption(lua_State *L, p_opt opt, p_socket ps)
- {
- const char *name = luaL_checkstring(L, 2); /* obj, name, ... */
- while (opt->name && strcmp(name, opt->name))
- opt++;
- if (!opt->func) {
- char msg[57];
- sprintf(msg, "unsupported option `%.35s'", name);
- luaL_argerror(L, 2, msg);
- }
- return opt->func(L, ps);
- }
- int opt_meth_getoption(lua_State *L, p_opt opt, p_socket ps)
- {
- const char *name = luaL_checkstring(L, 2); /* obj, name, ... */
- while (opt->name && strcmp(name, opt->name))
- opt++;
- if (!opt->func) {
- char msg[57];
- sprintf(msg, "unsupported option `%.35s'", name);
- luaL_argerror(L, 2, msg);
- }
- return opt->func(L, ps);
- }
- /* enables reuse of local address */
- int opt_set_reuseaddr(lua_State *L, p_socket ps)
- {
- return opt_setboolean(L, ps, SOL_SOCKET, SO_REUSEADDR);
- }
- int opt_get_reuseaddr(lua_State *L, p_socket ps)
- {
- return opt_getboolean(L, ps, SOL_SOCKET, SO_REUSEADDR);
- }
- /* enables reuse of local port */
- int opt_set_reuseport(lua_State *L, p_socket ps)
- {
- return opt_setboolean(L, ps, SOL_SOCKET, SO_REUSEPORT);
- }
- int opt_get_reuseport(lua_State *L, p_socket ps)
- {
- return opt_getboolean(L, ps, SOL_SOCKET, SO_REUSEPORT);
- }
- /* disables the Naggle algorithm */
- int opt_set_tcp_nodelay(lua_State *L, p_socket ps)
- {
- return opt_setboolean(L, ps, IPPROTO_TCP, TCP_NODELAY);
- }
- int opt_get_tcp_nodelay(lua_State *L, p_socket ps)
- {
- return opt_getboolean(L, ps, IPPROTO_TCP, TCP_NODELAY);
- }
- int opt_set_keepalive(lua_State *L, p_socket ps)
- {
- return opt_setboolean(L, ps, SOL_SOCKET, SO_KEEPALIVE);
- }
- int opt_get_keepalive(lua_State *L, p_socket ps)
- {
- return opt_getboolean(L, ps, SOL_SOCKET, SO_KEEPALIVE);
- }
- int opt_set_dontroute(lua_State *L, p_socket ps)
- {
- return opt_setboolean(L, ps, SOL_SOCKET, SO_DONTROUTE);
- }
- int opt_get_dontroute(lua_State *L, p_socket ps)
- {
- return opt_getboolean(L, ps, SOL_SOCKET, SO_DONTROUTE);
- }
- int opt_set_broadcast(lua_State *L, p_socket ps)
- {
- return opt_setboolean(L, ps, SOL_SOCKET, SO_BROADCAST);
- }
- int opt_get_broadcast(lua_State *L, p_socket ps)
- {
- return opt_getboolean(L, ps, SOL_SOCKET, SO_BROADCAST);
- }
- int opt_set_ip6_unicast_hops(lua_State *L, p_socket ps)
- {
- return opt_setint(L, ps, IPPROTO_IPV6, IPV6_UNICAST_HOPS);
- }
- int opt_get_ip6_unicast_hops(lua_State *L, p_socket ps)
- {
- return opt_getint(L, ps, IPPROTO_IPV6, IPV6_UNICAST_HOPS);
- }
- int opt_set_ip6_multicast_hops(lua_State *L, p_socket ps)
- {
- return opt_setint(L, ps, IPPROTO_IPV6, IPV6_MULTICAST_HOPS);
- }
- int opt_get_ip6_multicast_hops(lua_State *L, p_socket ps)
- {
- return opt_getint(L, ps, IPPROTO_IPV6, IPV6_MULTICAST_HOPS);
- }
- int opt_set_ip_multicast_loop(lua_State *L, p_socket ps)
- {
- return opt_setboolean(L, ps, IPPROTO_IP, IP_MULTICAST_LOOP);
- }
- int opt_get_ip_multicast_loop(lua_State *L, p_socket ps)
- {
- return opt_getboolean(L, ps, IPPROTO_IP, IP_MULTICAST_LOOP);
- }
- int opt_set_ip6_multicast_loop(lua_State *L, p_socket ps)
- {
- return opt_setboolean(L, ps, IPPROTO_IPV6, IPV6_MULTICAST_LOOP);
- }
- int opt_get_ip6_multicast_loop(lua_State *L, p_socket ps)
- {
- return opt_getboolean(L, ps, IPPROTO_IPV6, IPV6_MULTICAST_LOOP);
- }
- int opt_set_linger(lua_State *L, p_socket ps)
- {
- struct linger li; /* obj, name, table */
- if (!lua_istable(L, 3)) auxiliar_typeerror(L,3,lua_typename(L, LUA_TTABLE));
- lua_pushstring(L, "on");
- lua_gettable(L, 3);
- if (!lua_isboolean(L, -1))
- luaL_argerror(L, 3, "boolean 'on' field expected");
- li.l_onoff = (u_short) lua_toboolean(L, -1);
- lua_pushstring(L, "timeout");
- lua_gettable(L, 3);
- if (!lua_isnumber(L, -1))
- luaL_argerror(L, 3, "number 'timeout' field expected");
- li.l_linger = (u_short) lua_tonumber(L, -1);
- return opt_set(L, ps, SOL_SOCKET, SO_LINGER, (char *) &li, sizeof(li));
- }
- int opt_get_linger(lua_State *L, p_socket ps)
- {
- struct linger li; /* obj, name */
- int len = sizeof(li);
- int err = opt_get(L, ps, SOL_SOCKET, SO_LINGER, (char *) &li, &len);
- if (err)
- return err;
- lua_newtable(L);
- lua_pushboolean(L, li.l_onoff);
- lua_setfield(L, -2, "on");
- lua_pushinteger(L, li.l_linger);
- lua_setfield(L, -2, "timeout");
- return 1;
- }
- int opt_set_ip_multicast_ttl(lua_State *L, p_socket ps)
- {
- return opt_setint(L, ps, IPPROTO_IP, IP_MULTICAST_TTL);
- }
- int opt_set_ip_multicast_if(lua_State *L, p_socket ps)
- {
- const char *address = luaL_checkstring(L, 3); /* obj, name, ip */
- struct in_addr val;
- val.s_addr = htonl(INADDR_ANY);
- if (strcmp(address, "*") && !inet_aton(address, &val))
- luaL_argerror(L, 3, "ip expected");
- return opt_set(L, ps, IPPROTO_IP, IP_MULTICAST_IF,
- (char *) &val, sizeof(val));
- }
- int opt_get_ip_multicast_if(lua_State *L, p_socket ps)
- {
- struct in_addr val;
- socklen_t len = sizeof(val);
- if (getsockopt(*ps, IPPROTO_IP, IP_MULTICAST_IF, (char *) &val, &len) < 0) {
- lua_pushnil(L);
- lua_pushstring(L, "getsockopt failed");
- return 2;
- }
- lua_pushstring(L, inet_ntoa(val));
- return 1;
- }
- int opt_set_ip_add_membership(lua_State *L, p_socket ps)
- {
- return opt_setmembership(L, ps, IPPROTO_IP, IP_ADD_MEMBERSHIP);
- }
- int opt_set_ip_drop_membersip(lua_State *L, p_socket ps)
- {
- return opt_setmembership(L, ps, IPPROTO_IP, IP_DROP_MEMBERSHIP);
- }
- int opt_set_ip6_add_membership(lua_State *L, p_socket ps)
- {
- return opt_ip6_setmembership(L, ps, IPPROTO_IPV6, IPV6_ADD_MEMBERSHIP);
- }
- int opt_set_ip6_drop_membersip(lua_State *L, p_socket ps)
- {
- return opt_ip6_setmembership(L, ps, IPPROTO_IPV6, IPV6_DROP_MEMBERSHIP);
- }
- int opt_get_ip6_v6only(lua_State *L, p_socket ps)
- {
- return opt_getboolean(L, ps, IPPROTO_IPV6, IPV6_V6ONLY);
- }
- int opt_set_ip6_v6only(lua_State *L, p_socket ps)
- {
- return opt_setboolean(L, ps, IPPROTO_IPV6, IPV6_V6ONLY);
- }
- /*=========================================================================*\
- * Auxiliar functions
- \*=========================================================================*/
- static int opt_setmembership(lua_State *L, p_socket ps, int level, int name)
- {
- struct ip_mreq val; /* obj, name, table */
- if (!lua_istable(L, 3)) auxiliar_typeerror(L,3,lua_typename(L, LUA_TTABLE));
- lua_pushstring(L, "multiaddr");
- lua_gettable(L, 3);
- if (!lua_isstring(L, -1))
- luaL_argerror(L, 3, "string 'multiaddr' field expected");
- if (!inet_aton(lua_tostring(L, -1), &val.imr_multiaddr))
- luaL_argerror(L, 3, "invalid 'multiaddr' ip address");
- lua_pushstring(L, "interface");
- lua_gettable(L, 3);
- if (!lua_isstring(L, -1))
- luaL_argerror(L, 3, "string 'interface' field expected");
- val.imr_interface.s_addr = htonl(INADDR_ANY);
- if (strcmp(lua_tostring(L, -1), "*") &&
- !inet_aton(lua_tostring(L, -1), &val.imr_interface))
- luaL_argerror(L, 3, "invalid 'interface' ip address");
- return opt_set(L, ps, level, name, (char *) &val, sizeof(val));
- }
- static int opt_ip6_setmembership(lua_State *L, p_socket ps, int level, int name)
- {
- struct ipv6_mreq val; /* obj, opt-name, table */
- memset(&val, 0, sizeof(val));
- if (!lua_istable(L, 3)) auxiliar_typeerror(L,3,lua_typename(L, LUA_TTABLE));
- lua_pushstring(L, "multiaddr");
- lua_gettable(L, 3);
- if (!lua_isstring(L, -1))
- luaL_argerror(L, 3, "string 'multiaddr' field expected");
- if (!inet_pton(AF_INET6, lua_tostring(L, -1), &val.ipv6mr_multiaddr))
- luaL_argerror(L, 3, "invalid 'multiaddr' ip address");
- lua_pushstring(L, "interface");
- lua_gettable(L, 3);
- /* By default we listen to interface on default route
- * (sigh). However, interface= can override it. We should
- * support either number, or name for it. Waiting for
- * windows port of if_nametoindex */
- if (!lua_isnil(L, -1)) {
- if (lua_isnumber(L, -1)) {
- val.ipv6mr_interface = (unsigned int) lua_tonumber(L, -1);
- } else
- luaL_argerror(L, -1, "number 'interface' field expected");
- }
- return opt_set(L, ps, level, name, (char *) &val, sizeof(val));
- }
- static
- int opt_get(lua_State *L, p_socket ps, int level, int name, void *val, int* len)
- {
- socklen_t socklen = *len;
- if (getsockopt(*ps, level, name, (char *) val, &socklen) < 0) {
- lua_pushnil(L);
- lua_pushstring(L, "getsockopt failed");
- return 2;
- }
- *len = socklen;
- return 0;
- }
- static
- int opt_set(lua_State *L, p_socket ps, int level, int name, void *val, int len)
- {
- if (setsockopt(*ps, level, name, (char *) val, len) < 0) {
- lua_pushnil(L);
- lua_pushstring(L, "setsockopt failed");
- return 2;
- }
- lua_pushnumber(L, 1);
- return 1;
- }
- static int opt_getboolean(lua_State *L, p_socket ps, int level, int name)
- {
- int val = 0;
- int len = sizeof(val);
- int err = opt_get(L, ps, level, name, (char *) &val, &len);
- if (err)
- return err;
- lua_pushboolean(L, val);
- return 1;
- }
- int opt_get_error(lua_State *L, p_socket ps)
- {
- int val = 0;
- socklen_t len = sizeof(val);
- if (getsockopt(*ps, SOL_SOCKET, SO_ERROR, (char *) &val, &len) < 0) {
- lua_pushnil(L);
- lua_pushstring(L, "getsockopt failed");
- return 2;
- }
- lua_pushstring(L, socket_strerror(val));
- return 1;
- }
- static int opt_setboolean(lua_State *L, p_socket ps, int level, int name)
- {
- int val = auxiliar_checkboolean(L, 3); /* obj, name, bool */
- return opt_set(L, ps, level, name, (char *) &val, sizeof(val));
- }
- static int opt_getint(lua_State *L, p_socket ps, int level, int name)
- {
- int val = 0;
- int len = sizeof(val);
- int err = opt_get(L, ps, level, name, (char *) &val, &len);
- if (err)
- return err;
- lua_pushnumber(L, val);
- return 1;
- }
- static int opt_setint(lua_State *L, p_socket ps, int level, int name)
- {
- int val = (int) lua_tonumber(L, 3); /* obj, name, int */
- return opt_set(L, ps, level, name, (char *) &val, sizeof(val));
- }
|