1

我为 Lua 写了一个简单的 C 插件:

#include "lua.h"
#include "lualib.h"
#include "lauxlib.h"

static int bar (lua_State *L) {
  double arg1 = luaL_checknumber(L, 1);
  double arg2 = luaL_checknumber(L, 2);
  lua_Number res = arg1 + arg2;
  lua_pushnumber(L, res);
  return 1;
}

int luaopen_foo (lua_State *L) {
  static const struct luaL_Reg foo [] = {
    {"bar", bar},
    {NULL, NULL}
  };
  lua_newtable(L);
  luaL_setfuncs(L, foo, 0);
  lua_setglobal(L, "foo");
  return 1;
}

使用此 GCC 命令成功编译代码:

gcc -W -Wall -g -fPIC -shared -I/usr/include/lua5.3 -o foobar.so foobar.c

在 Lua 5.3 REPL 中,我也能够成功找到并导入模块,但函数调用的返回值始终为nil

root@b1898c1cc270:/# lua5.3
Lua 5.3.3  Copyright (C) 1994-2016 Lua.org, PUC-Rio
> local foo = require "foo"
> local res = foo.bar(3.0, 6.0)
> res
nil

没有错误被抛出,因为我能够printf在返回值之前在 C 代码中得到结果,我知道代码被调用并且结果计算成功。

有任何想法吗?

Edit1:通过不使用局部变量,我得到了这个堆栈跟踪而不是一个 nil 值:

root@d7340c919be4:/# lua5.3
Lua 5.3.3  Copyright (C) 1994-2016 Lua.org, PUC-Rio
> foo = require "foo"
> res = foo.bar(3.0, 6.0)
stdin:1: attempt to call a nil value (field 'bar')
stack traceback:
    stdin:1: in main chunk
    [C]: in ?
4

2 回答 2

3

luaL_setfuncs 只是将你的函数注册到一个表中。

相反,使用luaL_newlib. 它创建一个新表并在那里注册您的函数。然后你需要把表推到 lua 堆栈上。

luaL_newlib (L, foo);
return 1;
于 2018-04-06T10:46:46.030 回答
1

至于导致错误的原因,lua_setglobal从堆栈中弹出库表,这样luaopen_foo就不会返回它。然后(虽然我不明白为什么require "foo")改为返回库的文件路径(一个字符串),并且该字段("foo_filepath").barnil.

lua_setglobal因此,即使您没有luaL_newlib按照 Ravi 的建议使用创建库的宏,也可以删除调用来解决问题。

要将库表设置为全局foo 从其返回require,您必须先将表的第二个副本推入堆栈,lua_pushvalue(L, -1)然后lua_setglobal(L, "foo")将表的原始副本由返回luaopen_foo

于 2018-04-06T21:02:26.457 回答