以luasocket.c为例,第三方socket库
static const luaL_Reg mod[] = {
{"auxiliar", auxiliar_open},
{"except", except_open},
{"timeout", timeout_open},
{"buffer", buffer_open},
{"inet", inet_open},
{"tcp", tcp_open},
{"udp", udp_open},
{"select", select_open},
{NULL, NULL}
};
这里是关于所有的这个库中需要使用的函数的数组。将luaL_Reg这个结构体展开便可得知这个数组中每个元素的含义了。(在lauxlib.h中定义的)
typedef struct luaL_Reg {
const char *name;
lua_CFunction func;
} luaL_Reg;
再将lua_CFunction展开看下(在lua.h中定义)
typedef int (*lua_CFunction) (lua_State *L);
这个就定义了所有的用这种方式的函数的定义规则。
在lua中传入的参数是如何传给c代码?都是通过类似的函数LuaL_checkXXXX来从栈中取得。将返回值Push到栈中,函数返回值是返回值的个数。
而大量的luaL_XXX这种命名规则,是代表此函数不是LUA的核心代码。
再来看这个luasocket.c文件中的另一段代码
static int base_open(lua_State *L) {
if (socket_open()) {
/* export functions (and leave namespace table on top of stack) */
lua_newtable(L);
luaL_setfuncs(L, func, 0);
#ifdef LUASOCKET_DEBUG
lua_pushstring(L, "_DEBUG");
lua_pushboolean(L, 1);
lua_rawset(L, -3);
#endif
/* make version string available to scripts */
lua_pushstring(L, "_VERSION");
lua_pushstring(L, LUASOCKET_VERSION);
lua_rawset(L, -3);
return 1;
} else {
lua_pushstring(L, "unable to initialize library");
lua_error(L);
return 0;
}
}
/*-------------------------------------------------------------------------*\
* Initializes all library modules.
\*-------------------------------------------------------------------------*/
LUASOCKET_API int luaopen_socket_core(lua_State *L) {
int i;
base_open(L);
for (i = 0; mod[i].name; i++) mod[i].func(L);
return 1;
}
这个是将该函数作为一个模块放在一个table中。里面有函数地址等等。LUASOCKET_API就是extern 供其他文件直接引用.
linit.c这个文件非常重要
/*
** these libs are loaded by lua.c and are readily available to any Lua
** program
*/
static const luaL_Reg loadedlibs[] = {
{"_G", luaopen_base},
{LUA_LOADLIBNAME, luaopen_package},
{LUA_COLIBNAME, luaopen_coroutine},
{LUA_TABLIBNAME, luaopen_table},
{LUA_IOLIBNAME, luaopen_io},
{LUA_OSLIBNAME, luaopen_os},
{LUA_STRLIBNAME, luaopen_string},
{LUA_MATHLIBNAME, luaopen_math},
{LUA_UTF8LIBNAME, luaopen_utf8},
{LUA_DBLIBNAME, luaopen_debug},
#if defined(LUA_COMPAT_BITLIB)
{LUA_BITLIBNAME, luaopen_bit32},
#endif
{LUA_SOCKETLIBNAME, luaopen_socket_core},
{LUA_MIMELIBNAME, luaopen_mime_core},
{NULL, NULL}
};
loadLibs LuaL_Reg是之前看到的结构体。以luasocket库为例:(在lualib.h中定义)
#define LUA_SOCKETLIBNAME "socket"
LUAMOD_API int (luaopen_socket_core) (lua_State *L);
#define LUA_MIMELIBNAME "mime"
LUAMOD_API int (luaopen_mime_core) (lua_State *L);
而这里的名字就是我们要在lua中使用的模块名。 下面是用来外链函数的。以下是luaL_openlibs
LUALIB_API void luaL_openlibs (lua_State *L) {
const luaL_Reg *lib;
/* "require" functions from 'loadedlibs' and set results to global table */
for (lib = loadedlibs; lib->func; lib++) {
luaL_requiref(L, lib->name, lib->func, 1);
lua_pop(L, 1); /* remove lib */
}
}
将类库都加载到虚拟机中。