368 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
		
		
			
		
	
	
			368 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
|  | /*=========================================================================*\
 | ||
|  | * Common option interface  | ||
|  | * LuaSocket toolkit | ||
|  | \*=========================================================================*/ | ||
|  | #include <string.h> 
 | ||
|  | 
 | ||
|  | #include "lauxlib.h"
 | ||
|  | 
 | ||
|  | #include "auxiliar.h"
 | ||
|  | #include "options.h"
 | ||
|  | #include "inet.h"
 | ||
|  | 
 | ||
|  | namespace NS_SLUA {     | ||
|  | 
 | ||
|  | /*=========================================================================*\
 | ||
|  | * 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[45]; | ||
|  |         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[45]; | ||
|  |         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_set_broadcast(lua_State *L, p_socket ps) | ||
|  | { | ||
|  |     return opt_setboolean(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)); | ||
|  | } | ||
|  | 
 | ||
|  | } // end NS_SLUA
 |