elimination of common code + better error message

This commit is contained in:
Roberto Ierusalimschy 2005-07-12 18:17:46 -03:00
parent a569099b70
commit a05190fa3b
1 changed files with 51 additions and 39 deletions

View File

@ -1,5 +1,5 @@
/* /*
** $Id: loadlib.c,v 1.31 2005/07/05 19:29:03 roberto Exp roberto $ ** $Id: loadlib.c,v 1.32 2005/07/11 16:41:57 roberto Exp roberto $
** Dynamic library loader for Lua ** Dynamic library loader for Lua
** See Copyright Notice in lua.h ** See Copyright Notice in lua.h
** **
@ -329,48 +329,42 @@ static int ll_loadlib (lua_State *L) {
*/ */
static int loader_Lua (lua_State *L) { static const char *findfile (lua_State *L, const char *pname) {
const char *name = luaL_checkstring(L, 1); const char *name = luaL_checkstring(L, 1);
const char *fname = luaL_gsub(L, name, ".", LUA_DIRSEP); const char *fname = luaL_gsub(L, name, ".", LUA_DIRSEP);
const char *path = NULL; const char *path;
#if defined(LUA_COMPAT_PATH) lua_getfield(L, LUA_ENVIRONINDEX, pname);
/* try first `LUA_PATH' for compatibility */
lua_pushstring(L, "LUA_PATH");
lua_rawget(L, LUA_GLOBALSINDEX);
path = lua_tostring(L, -1); path = lua_tostring(L, -1);
#endif
if (!path) {
lua_pop(L, 1);
lua_getfield(L, LUA_ENVIRONINDEX, "path");
path = lua_tostring(L, -1);
}
if (path == NULL) if (path == NULL)
luaL_error(L, LUA_QL("package.path") " must be a string"); luaL_error(L, LUA_QL("package.%s") " must be a string", pname);
fname = luaL_searchpath(L, fname, path); return luaL_searchpath(L, fname, path);
}
static void loaderror (lua_State *L, const char *msg) {
luaL_error(L, "error loading package " LUA_QS " (%s)",
lua_tostring(L, 1), msg);
}
static int loader_Lua (lua_State *L) {
const char *fname;
fname = findfile(L, "path");
if (fname == NULL) return 0; /* library not found in this path */ if (fname == NULL) return 0; /* library not found in this path */
if (luaL_loadfile(L, fname) != 0) if (luaL_loadfile(L, fname) != 0)
luaL_error(L, "error loading package " LUA_QS " (%s)", loaderror(L, lua_tostring(L, -1));
name, lua_tostring(L, -1));
return 1; /* library loaded successfully */ return 1; /* library loaded successfully */
} }
static int loader_C (lua_State *L) { static int loader_C (lua_State *L) {
const char *name = luaL_checkstring(L, 1);
const char *fname = luaL_gsub(L, name, ".", LUA_DIRSEP);
const char *path;
const char *funcname; const char *funcname;
lua_getfield(L, LUA_ENVIRONINDEX, "cpath"); const char *fname = findfile(L, "cpath");
path = lua_tostring(L, -1);
if (path == NULL)
luaL_error(L, LUA_QL("package.cpath") " must be a string");
fname = luaL_searchpath(L, fname, path);
if (fname == NULL) return 0; /* library not found in this path */ if (fname == NULL) return 0; /* library not found in this path */
funcname = luaL_gsub(L, name, ".", LUA_OFSEP); funcname = luaL_gsub(L, lua_tostring(L, 1), ".", LUA_OFSEP);
funcname = lua_pushfstring(L, "%s%s", POF, funcname); funcname = lua_pushfstring(L, POF"%s", funcname);
if (ll_loadfunc(L, fname, funcname) != 1) if (ll_loadfunc(L, fname, funcname) != 1)
luaL_error(L, "error loading package " LUA_QS " (%s)", loaderror(L, lua_tostring(L, -2));
name, lua_tostring(L, -2));
return 1; /* library loaded successfully */ return 1; /* library loaded successfully */
} }
@ -384,13 +378,13 @@ static int loader_preload (lua_State *L) {
} }
static void require_aux (lua_State *L, const char *name) { static int require_aux (lua_State *L, const char *name) {
int i; int i;
int loadedtable = lua_gettop(L) + 1; int loadedtable = lua_gettop(L) + 1;
lua_getfield(L, LUA_REGISTRYINDEX, "_LOADED"); lua_getfield(L, LUA_REGISTRYINDEX, "_LOADED");
lua_getfield(L, loadedtable, name); lua_getfield(L, loadedtable, name);
if (lua_toboolean(L, -1)) /* is it there? */ if (lua_toboolean(L, -1)) /* is it there? */
return; /* package is already loaded; return its result */ return 1; /* package is already loaded */
/* else must load it; iterate over available loaders */ /* else must load it; iterate over available loaders */
lua_getfield(L, LUA_ENVIRONINDEX, "loaders"); lua_getfield(L, LUA_ENVIRONINDEX, "loaders");
if (!lua_istable(L, -1)) if (!lua_istable(L, -1))
@ -398,7 +392,7 @@ static void require_aux (lua_State *L, const char *name) {
for (i=1; ; i++) { for (i=1; ; i++) {
lua_rawgeti(L, -1, i); /* get a loader */ lua_rawgeti(L, -1, i); /* get a loader */
if (lua_isnil(L, -1)) if (lua_isnil(L, -1))
luaL_error(L, "package " LUA_QS " not found", name); return 0; /* package not found */
lua_pushstring(L, name); lua_pushstring(L, name);
lua_call(L, 1, 1); /* call it */ lua_call(L, 1, 1); /* call it */
if (lua_isnil(L, -1)) lua_pop(L, 1); /* did not found module */ if (lua_isnil(L, -1)) lua_pop(L, 1); /* did not found module */
@ -416,7 +410,20 @@ static void require_aux (lua_State *L, const char *name) {
if (!lua_isnil(L, -1)) /* non-nil return? */ if (!lua_isnil(L, -1)) /* non-nil return? */
lua_setfield(L, loadedtable, name); /* _LOADED[name] = returned value */ lua_setfield(L, loadedtable, name); /* _LOADED[name] = returned value */
lua_getfield(L, loadedtable, name); /* return _LOADED[name] */ lua_getfield(L, loadedtable, name); /* return _LOADED[name] */
return; return 1;
}
static void ll_error (lua_State *L, const char *name) {
const char *msg;
lua_settop(L, 1);
lua_getfield(L, LUA_ENVIRONINDEX, "path");
lua_getfield(L, LUA_ENVIRONINDEX, "cpath");
msg = lua_pushfstring(L, "package " LUA_QS " not found in following paths:\n"
" Lua path: %s\n C path: %s\n", name,
lua_tostring(L, -2), lua_tostring(L, -1));
msg = luaL_gsub(L, msg, LUA_PATHSEP, "\n ");
luaL_error(L, msg);
} }
@ -429,7 +436,8 @@ static int ll_require (lua_State *L) {
lua_pushlstring(L, name, pt - name); lua_pushlstring(L, name, pt - name);
require_aux(L, lua_tostring(L, -1)); require_aux(L, lua_tostring(L, -1));
} }
require_aux(L, name); /* load module itself */ if (!require_aux(L, name)) /* load module itself */
ll_error(L, name);
return 1; return 1;
} }
@ -490,15 +498,19 @@ static int ll_module (lua_State *L) {
/* }====================================================== */ /* }====================================================== */
/* auxiliary mark (for internal use) */
#define AUXMARK "\1"
static void setpath (lua_State *L, const char *fname, const char *envname, static void setpath (lua_State *L, const char *fname, const char *envname,
const char *def) { const char *def) {
const char *path = getenv(envname); const char *path = getenv(envname);
if (path == NULL) lua_pushstring(L, def); if (path == NULL) /* no environment variable? */
lua_pushstring(L, def); /* use default */
else { else {
/* replace ";;" by default path */ /* replace ";;" by ";AUXMARK;" and then AUXMARK by default path */
path = luaL_gsub(L, path, LUA_PATHSEP LUA_PATHSEP, path = luaL_gsub(L, path, LUA_PATHSEP LUA_PATHSEP,
LUA_PATHSEP"\1"LUA_PATHSEP); LUA_PATHSEP AUXMARK LUA_PATHSEP);
luaL_gsub(L, path, "\1", def); luaL_gsub(L, path, AUXMARK, def);
lua_remove(L, -2); lua_remove(L, -2);
} }
setprogdir(L); setprogdir(L);
@ -507,8 +519,8 @@ static void setpath (lua_State *L, const char *fname, const char *envname,
static const luaL_reg ll_funcs[] = { static const luaL_reg ll_funcs[] = {
{"require", ll_require},
{"module", ll_module}, {"module", ll_module},
{"require", ll_require},
{NULL, NULL} {NULL, NULL}
}; };