From 5018f9e7673f533bbe3240c2ecce703b47d76a73 Mon Sep 17 00:00:00 2001 From: David Walter Seikel Date: Fri, 28 Mar 2014 15:44:43 +1000 Subject: Make test_c a real skang module, and tweaks to the rest of the system to support C skang modules. --- ClientHamr/GuiLua/skang.lua | 19 +++-- ClientHamr/GuiLua/test.lua | 5 +- ClientHamr/GuiLua/test_c.c | 177 ++++++++++++++++++++++++++++++++------------ 3 files changed, 147 insertions(+), 54 deletions(-) diff --git a/ClientHamr/GuiLua/skang.lua b/ClientHamr/GuiLua/skang.lua index c5c2096..6f06131 100644 --- a/ClientHamr/GuiLua/skang.lua +++ b/ClientHamr/GuiLua/skang.lua @@ -77,10 +77,12 @@ local versions = { -- Trying to capture best practices here for creating modules, especially since module() is broken and deprecated. -- TODO - Should parse in license type to. -moduleBegin = function (name, author, copyright, version, timestamp, skin) +moduleBegin = function (name, author, copyright, version, timestamp, skin, isLua) local _M = {} -- This is what we return to require(). local level = 2 + if 'nil' == type(isLua) then isLua = true end + package.loaded[name] = _M -- Stuff the result into where require() can find it, instead of returning it at the end. -- Returning it at the end does the same thing. -- This is so that we can have all the module stuff at the top, in this function. @@ -102,6 +104,7 @@ moduleBegin = function (name, author, copyright, version, timestamp, skin) _M._M = _M -- So that references to _M below the setfenv() actually go to the real _M. _M._NAME = name _M._PACKAGE = string.gsub(_M._NAME, "[^.]*$", "") -- Strip the name down to the package name. + _M.isLua = isLua -- Parse in an entire copyright message, and strip that down into bits, to put back together. local date, owner = string.match(copyright, '[Cc]opyright (%d%d%d%d) (.*)') @@ -129,12 +132,15 @@ moduleBegin = function (name, author, copyright, version, timestamp, skin) setmetatable(_M, Thing) _M.savedEnvironment = savedEnvironment - -- setfenv() sets the environment for the FUNCTION, stack level deep. - -- The number is the stack level - - -- 0 running thread, 1 current function, 2 function that called this function, etc - setfenv(level, _M) -- Use the result for the modules internal global environment, so they don't need to qualify internal names. + -- NOTE - setfenv() wont work if the environment it refers to is a C function. Worse, getfenv() returns the global environment, so we can't tell. + if isLua then + -- setfenv() sets the environment for the FUNCTION, stack level deep. + -- The number is the stack level - + -- 0 running thread, 1 current function, 2 function that called this function, etc + setfenv(level, _M) -- Use the result for the modules internal global environment, so they don't need to qualify internal names. -- Dunno if this causes problems with the do ... end style of joining modules. It does. So we need to restore in moduleEnd(). -- Next question, does this screw with the environment of the skang module? No it doesn't, coz that's set up at require 'skang' time. + end print('Loaded module ' .. _M._NAME .. ' version ' .. _M.VERSION .. ', ' .. _M.COPYRIGHT .. '.\n ' .. _M.VERSION_DESC) @@ -146,7 +152,7 @@ moduleEnd = function (module) -- TODO - Look for _NAME.properties, and load it into the modules Things. -- TODO - Parse command line parameters at some point. -- http://stackoverflow.com/questions/3745047/help-locate-c-sample-code-to-read-lua-command-line-arguments - setfenv(2, module.savedEnvironment) + if module.isLua then setfenv(2, module.savedEnvironment) end end -- Call this now so that from now on, this is like any other module. @@ -424,6 +430,7 @@ thing = function (names, ...) thing.required = params[5] or thing.required thing.acl = params[6] or thing.acl thing.boss = params[7] or thing.boss + thing.module = params[8] or thing.module -- Mostly for things like C functions, where get/setfenv() wont do what we need. -- PUll out named arguments. for k, v in pairs(params) do diff --git a/ClientHamr/GuiLua/test.lua b/ClientHamr/GuiLua/test.lua index 3c2e9ab..d566354 100644 --- a/ClientHamr/GuiLua/test.lua +++ b/ClientHamr/GuiLua/test.lua @@ -49,10 +49,11 @@ local skang = require 'skang' local test = require 'test' local test_c = require 'test_c' -print('foo = ' .. test.foo .. ' ->> ' .. skang.things.foo.help) print('End ' .. test.bar .. ' ' .. test.VERSION .. ' ' .. skang.things.ffunc.help .. ' ->> ' .. skang.things.f.action) +print('foo = ' .. test.foo .. ' ->> ' .. skang.things.foo.help) +print('cfunc ->> ' .. skang.things.cfunc.help) test.ffunc('one', 2) -test_c.ffunc(0, 'zero') +test_c.cfunc(0, 'zero') --skang.things.ffunc('seven', 'aight') print('') diff --git a/ClientHamr/GuiLua/test_c.c b/ClientHamr/GuiLua/test_c.c index 8d3ab81..e3fd801 100644 --- a/ClientHamr/GuiLua/test_c.c +++ b/ClientHamr/GuiLua/test_c.c @@ -8,9 +8,6 @@ http://lua.2524044.n2.nabble.com/C-Lua-modules-not-compatible-with-every-Lua-int http://lua-users.org/wiki/LuaProxyDllFour http://stackoverflow.com/questions/11492194/how-do-you-create-a-lua-plugin-that-calls-the-c-lua-api?rq=1 http://lua-users.org/lists/lua-l/2008-01/msg00671.html - - - */ @@ -19,72 +16,160 @@ http://lua-users.org/lists/lua-l/2008-01/msg00671.html //#include -static int ffunc (lua_State *L) +static const char *ourName = "test_c"; +int skang, _M; + +static int cfunc (lua_State *L) { double arg1 = luaL_checknumber(L, 1); const char *arg2 = luaL_checkstring(L, 2); - printf("Inside test_c.ffunc(%f, %s)\n", arg1, arg2); + printf("Inside %s.cfunc(%f, %s)\n", ourName, arg1, arg2); return 0; } -static const struct luaL_reg test_c [] = +static void dumpStack(lua_State *L, int i) { - {"ffunc", ffunc}, - {NULL, NULL} -}; - + int type = lua_type(L, i); + + switch (type) + { + case LUA_TNONE : printf("Stack %d is empty\n", i); break; + case LUA_TNIL : printf("Stack %d is a nil\n", i); break; + case LUA_TBOOLEAN : printf("Stack %d is a boolean\n", i); break; + case LUA_TNUMBER : printf("Stack %d is a number\n", i); break; + case LUA_TSTRING : printf("Stack %d is a string - %s\n", i, lua_tostring(L, i)); break; + case LUA_TFUNCTION : printf("Stack %d is a function\n", i); break; + case LUA_TTHREAD : printf("Stack %d is a thread\n", i); break; + case LUA_TTABLE : printf("Stack %d is a table\n", i); break; + case LUA_TUSERDATA : printf("Stack %d is a userdata\n", i); break; + case LUA_TLIGHTUSERDATA : printf("Stack %d is a light userdata\n", i); break; + default : printf("Stack %d is unknown\n", i); break; + } +} /* local test_c = require 'test_c' Lua's require() function will strip any stuff from the front of the name -separated by a hypen, so 'GuiLua-test_c' -> 'test_c'. Then it will +separated by a hyphen, so 'GuiLua-test_c' -> 'test_c'. Then it will search through a path, and eventually find this test_c.so (or test_c.dll -or whatever), then call luaopen_test_c(), which should return a table. +or whatever), then call luaopen_test_c(), which should return a table. +The argument (only thing on the stack) for this function will be +'test_c'. Normally luaL_register() creates a table of functions, that is the table returned, but we want to do something different with skang. - */ int luaopen_test_c(lua_State *L) { -// This is a moving target, old ways get deperecated, new ways get added, -// would have to check version before doing any of these. -// luaL_openlib(L, "test_c", test_c, 0); // Lua 5.0 way. -// luaL_register (L, "test_c", test_c); // Lua 5.1 way. -// luaL_newlib() or luaL_setfuncs() // Lua 5.2 way. - // Creates a global table "test_c", does the package.loaded[test_c] thing. - lua_newtable(L); - luaL_register (L, NULL, test_c); // Lua 5.1 way. - // Puts the funcions in a table on top of the stack. - -/* BUT REALLY ... - -We are in fact NOT putting any functions into the returned table. - -skang.moduleBegin() returns the table we need to send back to Lua. - it saves getfenv(2) as the old environment, which should in theory be L - and setfenv(_M, 2) to set the tbale to be it's own environment - it does the package.loaded[test_c] thing for us - it returns the table it created, so we should just leave that on the stack as our result - -skang.thing() also uses getfenv(2) to grab the module's table + // In theory, the only thing on the stack now is 'test_c' from the require() call. + +// pseudo-indices, special tables that can be accessed like the stack - +// LUA_GLOBALSINDEX - thread environment, where globals are +// LUA_ENVIRONINDEX - C function environment, in this case luaopen_test_c() is the C function +// LUA_REGISTRYINDEX - C registry, global, for unique keys use the module name as a string, or a lightuserdata address to a C object in our module. +// lua_upvalueindex(n) - C function upvalues + +// The only locals we care about are skang and _M. +// All modules go into package.loaded[name] as well. +// skang is essentially a global anyway. +// _M we pass back as the result, and our functions get added to it by skang.thing() +// Not entirely true, skang.things is a proxy table, skang.things.ffunc.func would be our function. +// skang.things.ffunc.module is our _M ('environment of our calling function', so that's the 'environment' right here, since we call skang.thing(). +// test_c.ffunc() -> test_c.__index(test_c, 'ffunc') -> skang.things[ffunc].value() -> test_c.c->ffunc() +// this is a C function, not a Lua function + +// local skang = require 'skang' + lua_getglobal(L, "require"); + lua_pushstring(L, "skang"); + lua_call(L, 1, 1); + skang = lua_gettop(L); +// dumpStack(L, skang); + +// local _M = skang.moduleBegin('test_c', nil, 'Copyright 2014 David Seikel', '0.1', '2014-03-27 03:57:00', nil, true) + lua_getfield(L, skang, "moduleBegin"); + lua_pushstring(L, ourName); + lua_pushnil(L); // Author comes from copyright. + lua_pushstring(L, "Copyright 2014 David Seikel"); + lua_pushstring(L, "0.1"); + lua_pushstring(L, "2014-03-27 03:57:00"); + lua_pushnil(L); // No skin. + lua_pushboolean(L, 0); // We are not a Lua module. + lua_call(L, 7, 1); // call 'skang.moduleBegin' with 7 arguments and 1 result. + _M = lua_gettop(L); +// dumpStack(L, _M); + + // At this point the stack should be - 'test_c', skang, _M. Let's test that. +/* + int top = 0, i; + + top = lua_gettop(L); + printf("MODULE test_c has %d stack items.\n", top); + for (i = 1; i <= top; i++) + dumpStack(L, i); */ -/* TODO - load skang, create things, etc. - -local skang = require "skang" -local _M = skang.moduleBegin("test_c", nil, "Copyright 2014 David Seikel", "0.1", "2014-03-27 03:57:00") + // Save this module in the C registry. + lua_setfield(L, LUA_REGISTRYINDEX, ourName); + +// skang.thing('cfooble,c', 'Help text goes here', 1, 'number', \"'edit', 'The fooble:', 1, 1, 10, 50\", true) + lua_getfield(L, skang, "thing"); + lua_pushstring(L, "cfooble,c"); + lua_pushstring(L, "Help text"); + lua_pushnumber(L, 1); + lua_pushstring(L, "number"); + lua_pushstring(L, "'edit', 'The cfooble:', 1, 1, 10, 50"); + lua_pushboolean(L, 1); // Is required. + lua_pushnil(L); // Default ACL. + lua_pushnil(L); // No boss. + lua_getfield(L, LUA_REGISTRYINDEX, ourName); // Coz getfenv() can't find C environment. + lua_call(L, 9, 0); + +// skang.thing('cbar', 'Help text', 'Default') + lua_getfield(L, skang, "thing"); + lua_pushstring(L, "cbar"); + lua_pushstring(L, "Help text"); + lua_pushstring(L, "Default"); + lua_pushnil(L); // No type. + lua_pushnil(L); // No widget. + lua_pushnil(L); // Not required. + lua_pushnil(L); // Default ACL. + lua_pushnil(L); // No boss. + lua_getfield(L, LUA_REGISTRYINDEX, ourName); // Coz getfenv() can't find C environment. + lua_call(L, 9, 0); + +// skang.thing('cfoo') + lua_getfield(L, skang, "thing"); + lua_pushstring(L, "cfoo"); + lua_pushnil(L); // No help. + lua_pushnil(L); // No default. + lua_pushnil(L); // No type. + lua_pushnil(L); // No widget. + lua_pushnil(L); // Not required. + lua_pushnil(L); // Default ACL. + lua_pushnil(L); // No boss. + lua_getfield(L, LUA_REGISTRYINDEX, ourName); // Coz getfenv() can't find C environment. + lua_call(L, 9, 0); + +// skang.thing('cfunc', 'Help Text', ffunc, 'number,string') + lua_getfield(L, skang, "thing"); + lua_pushstring(L, "cfunc"); + lua_pushstring(L, "cfunc does nothing really"); + lua_pushcfunction(L, cfunc); + lua_pushstring(L, "number,string"); + lua_pushnil(L); // No widget. + lua_pushnil(L); // Not required. + lua_pushnil(L); // Default ACL. + lua_pushnil(L); // No boss. + lua_getfield(L, LUA_REGISTRYINDEX, ourName); // Coz getfenv() can't find C environment. + lua_call(L, 9, 0); + +// skang.moduleEnd(_M) + lua_getfield(L, skang, "moduleEnd"); + lua_getfield(L, LUA_REGISTRYINDEX, ourName); + lua_call(L, 1, 1); -skang.thing("fooble,f", "Help text goes here", 1, "number", "'edit', 'The fooble:', 1, 1, 10, 50", true) -skang.thing("bar", "Help text", "Default") -skang.thing("foo") -skang.thing("ffunc", "Help Text", ffunc, "number,string") - -skang.moduleEnd(_M) - -*/ return 1; } -- cgit v1.1