163 lines
		
	
	
		
			6.2 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
		
		
			
		
	
	
			163 lines
		
	
	
		
			6.2 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
|  | /*=========================================================================*\
 | ||
|  | * Auxiliar routines for class hierarchy manipulation | ||
|  | * LuaSocket toolkit | ||
|  | \*=========================================================================*/ | ||
|  | #include <string.h>
 | ||
|  | #include <stdio.h>
 | ||
|  | 
 | ||
|  | #include "auxiliar.h"
 | ||
|  | 
 | ||
|  | namespace NS_SLUA { | ||
|  | 
 | ||
|  | /*=========================================================================*\
 | ||
|  | * Exported functions | ||
|  | \*=========================================================================*/ | ||
|  | /*-------------------------------------------------------------------------*\
 | ||
|  | * Initializes the module | ||
|  | \*-------------------------------------------------------------------------*/ | ||
|  | int auxiliar_open(lua_State *L) { | ||
|  |     (void) L; | ||
|  |     return 0; | ||
|  | } | ||
|  | 
 | ||
|  | /*-------------------------------------------------------------------------*\
 | ||
|  | * Creates a new class with given methods | ||
|  | * Methods whose names start with __ are passed directly to the metatable. | ||
|  | \*-------------------------------------------------------------------------*/ | ||
|  | void auxiliar_newclass(lua_State *L, const char *classname, luaL_Reg *func) { | ||
|  |     luaL_newmetatable(L, classname); /* mt */ | ||
|  |     /* create __index table to place methods */ | ||
|  |     lua_pushstring(L, "__index");    /* mt,"__index" */ | ||
|  |     lua_newtable(L);                 /* mt,"__index",it */  | ||
|  |     /* put class name into class metatable */ | ||
|  |     lua_pushstring(L, "class");      /* mt,"__index",it,"class" */ | ||
|  |     lua_pushstring(L, classname);    /* mt,"__index",it,"class",classname */ | ||
|  |     lua_rawset(L, -3);               /* mt,"__index",it */ | ||
|  |     /* pass all methods that start with _ to the metatable, and all others
 | ||
|  |      * to the index table */ | ||
|  |     for (; func->name; func++) {     /* mt,"__index",it */ | ||
|  |         lua_pushstring(L, func->name); | ||
|  |         lua_pushcfunction(L, func->func); | ||
|  |         lua_rawset(L, func->name[0] == '_' ? -5: -3); | ||
|  |     } | ||
|  |     lua_rawset(L, -3);               /* mt */ | ||
|  |     lua_pop(L, 1); | ||
|  | } | ||
|  | 
 | ||
|  | /*-------------------------------------------------------------------------*\
 | ||
|  | * Prints the value of a class in a nice way | ||
|  | \*-------------------------------------------------------------------------*/ | ||
|  | int auxiliar_tostring(lua_State *L) { | ||
|  |     char buf[32]; | ||
|  |     if (!lua_getmetatable(L, 1)) goto error; | ||
|  |     lua_pushstring(L, "__index"); | ||
|  |     lua_gettable(L, -2); | ||
|  |     if (!lua_istable(L, -1)) goto error; | ||
|  |     lua_pushstring(L, "class"); | ||
|  |     lua_gettable(L, -2); | ||
|  |     if (!lua_isstring(L, -1)) goto error; | ||
|  |     sprintf(buf, "%p", lua_touserdata(L, 1)); | ||
|  |     lua_pushfstring(L, "%s: %s", lua_tostring(L, -1), buf); | ||
|  |     return 1; | ||
|  | error: | ||
|  |     lua_pushstring(L, "invalid object passed to 'auxiliar.c:__tostring'"); | ||
|  |     lua_error(L); | ||
|  |     return 1; | ||
|  | } | ||
|  | 
 | ||
|  | /*-------------------------------------------------------------------------*\
 | ||
|  | * Insert class into group | ||
|  | \*-------------------------------------------------------------------------*/ | ||
|  | void auxiliar_add2group(lua_State *L, const char *classname, const char *groupname) { | ||
|  |     luaL_getmetatable(L, classname); | ||
|  |     lua_pushstring(L, groupname); | ||
|  |     lua_pushboolean(L, 1); | ||
|  |     lua_rawset(L, -3); | ||
|  |     lua_pop(L, 1); | ||
|  | } | ||
|  | 
 | ||
|  | /*-------------------------------------------------------------------------*\
 | ||
|  | * Make sure argument is a boolean | ||
|  | \*-------------------------------------------------------------------------*/ | ||
|  | int auxiliar_checkboolean(lua_State *L, int objidx) { | ||
|  |     if (!lua_isboolean(L, objidx)) | ||
|  |         auxiliar_typeerror(L, objidx, lua_typename(L, LUA_TBOOLEAN)); | ||
|  |     return lua_toboolean(L, objidx); | ||
|  | } | ||
|  | 
 | ||
|  | /*-------------------------------------------------------------------------*\
 | ||
|  | * Return userdata pointer if object belongs to a given class, abort with  | ||
|  | * error otherwise | ||
|  | \*-------------------------------------------------------------------------*/ | ||
|  | void *auxiliar_checkclass(lua_State *L, const char *classname, int objidx) { | ||
|  |     void *data = auxiliar_getclassudata(L, classname, objidx); | ||
|  |     if (!data) { | ||
|  |         char msg[45]; | ||
|  |         sprintf(msg, "%.35s expected", classname); | ||
|  |         luaL_argerror(L, objidx, msg); | ||
|  |     } | ||
|  |     return data; | ||
|  | } | ||
|  | 
 | ||
|  | /*-------------------------------------------------------------------------*\
 | ||
|  | * Return userdata pointer if object belongs to a given group, abort with  | ||
|  | * error otherwise | ||
|  | \*-------------------------------------------------------------------------*/ | ||
|  | void *auxiliar_checkgroup(lua_State *L, const char *groupname, int objidx) { | ||
|  |     void *data = auxiliar_getgroupudata(L, groupname, objidx); | ||
|  |     if (!data) { | ||
|  |         char msg[45]; | ||
|  |         sprintf(msg, "%.35s expected", groupname); | ||
|  |         luaL_argerror(L, objidx, msg); | ||
|  |     } | ||
|  |     return data; | ||
|  | } | ||
|  | 
 | ||
|  | /*-------------------------------------------------------------------------*\
 | ||
|  | * Set object class | ||
|  | \*-------------------------------------------------------------------------*/ | ||
|  | void auxiliar_setclass(lua_State *L, const char *classname, int objidx) { | ||
|  |     luaL_getmetatable(L, classname); | ||
|  |     if (objidx < 0) objidx--; | ||
|  |     lua_setmetatable(L, objidx); | ||
|  | } | ||
|  | 
 | ||
|  | /*-------------------------------------------------------------------------*\
 | ||
|  | * Get a userdata pointer if object belongs to a given group. Return NULL  | ||
|  | * otherwise | ||
|  | \*-------------------------------------------------------------------------*/ | ||
|  | void *auxiliar_getgroupudata(lua_State *L, const char *groupname, int objidx) { | ||
|  |     if (!lua_getmetatable(L, objidx)) | ||
|  |         return NULL; | ||
|  |     lua_pushstring(L, groupname); | ||
|  |     lua_rawget(L, -2); | ||
|  |     if (lua_isnil(L, -1)) { | ||
|  |         lua_pop(L, 2); | ||
|  |         return NULL; | ||
|  |     } else { | ||
|  |         lua_pop(L, 2); | ||
|  |         return lua_touserdata(L, objidx); | ||
|  |     } | ||
|  | } | ||
|  | 
 | ||
|  | /*-------------------------------------------------------------------------*\
 | ||
|  | * Get a userdata pointer if object belongs to a given class. Return NULL  | ||
|  | * otherwise | ||
|  | \*-------------------------------------------------------------------------*/ | ||
|  | void *auxiliar_getclassudata(lua_State *L, const char *classname, int objidx) { | ||
|  |     return luaL_checkudata(L, objidx, classname); | ||
|  | } | ||
|  | 
 | ||
|  | /*-------------------------------------------------------------------------*\
 | ||
|  | * Throws error when argument does not have correct type. | ||
|  | * Used to be part of lauxlib in Lua 5.1, was dropped from 5.2. | ||
|  | \*-------------------------------------------------------------------------*/ | ||
|  | int auxiliar_typeerror (lua_State *L, int narg, const char *tname) { | ||
|  |   const char *msg = lua_pushfstring(L, "%s expected, got %s", tname,  | ||
|  |       luaL_typename(L, narg)); | ||
|  |   return luaL_argerror(L, narg, msg); | ||
|  | } | ||
|  | 
 | ||
|  | 
 | ||
|  | } // end NS_SLUA
 |