2013-10-17 22 views
4

从文档:http://luajit.org/running.html运行luajit目标文件由C

luajit -b test.lua test.obj     # Generate object file 
# Link test.obj with your application and load it with require("test") 

但没有解释如何做这些事情。我想他们假设任何使用Lua的人也是C程序员,而不是我的情况!我可以得到一些帮助吗? GCC为例。

我也想做同样的事情,除了从C字节数组头。我也找不到这方面的文档。

luajit -bt h -n test test.lua test.h 

这将创建头文件,但我不知道如何从C运行它。谢谢。

+0

您应该能够使用'luaL_loadstring'加载该C数组。一旦将函数返回到堆栈中,您可以使用'call'或'pcall'来执行它,或者如果您稍后需要它,可以将其保存到'package.preload'表中,具体取决于您想要的内容去做。 – greatwolf

回答

7

的基本用法如下:

  • 使用luajit
  • #include那那将被引用其符号源文件(一个或多个)在报头生成头文件
  • 编译源成一个可运行的可执行文件或用于lua的共享二进制模块,具体取决于您的使用情况。

这里的一个最小的例子来说明:

test.lua

return 
{ 
    fooprint = function (s) return print("from foo: "..s) end, 
    barprint = function (s) return print("from bar: "..s) end 
} 

test.h

// luajit -b test.lua test.h 
#define luaJIT_BC_test_SIZE 155 
static const char luaJIT_BC_test[] = { 
27,76,74,1,2,44,0,1,4,0,2,0,5,52,1,0,0,37,2,1,0,16,3,0,0,36,2,3,2,64,1,2,0,15, 
102,114,111,109,32,102,111,111,58,32,10,112,114,105,110,116,44,0,1,4,0,2,0,5, 
52,1,0,0,37,2,1,0,16,3,0,0,36,2,3,2,64,1,2,0,15,102,114,111,109,32,98,97,114, 
58,32,10,112,114,105,110,116,58,3,0,2,0,5,0,7,51,0,1,0,49,1,0,0,58,1,2,0,49,1, 
3,0,58,1,4,0,48,0,0,128,72,0,2,0,13,98,97,114,112,114,105,110,116,0,13,102, 
111,111,112,114,105,110,116,1,0,0,0,0 
}; 

runtest.cpp

// g++ -Wall -pedantic -g runtest.cpp -o runtest.exe -llua51 
#include <stdio.h> 
#include <assert.h> 

#include "lua.hpp" 
#include "test.h" 

static const char *runtest = 
"test = require 'test'\n" 
"test.fooprint('it works!')\n" 
"test.barprint('it works!')\n"; 


int main() 
{ 
    lua_State *L = luaL_newstate(); 
    luaL_openlibs(L); 

    lua_getglobal(L, "package"); 
    lua_getfield(L, -1, "preload"); 
    // package, preload, luaJIT_BC_test 
    bool err = luaL_loadbuffer(L, luaJIT_BC_test, luaJIT_BC_test_SIZE, NULL); 
    assert(!err); 

    // package.preload.test = luaJIT_BC_test 
    lua_setfield(L, -2, "test"); 

    // check that 'test' lib is now available; run the embedded test script 
    lua_settop(L, 0); 
    err = luaL_dostring(L, runtest); 
    assert(!err); 

    lua_close(L); 
} 

这是非常直接的。本示例采用字节码并将其放入此程序的lua环境的package.preload表中。其他lua脚本可以通过执行require 'test'来使用它。在runtest嵌入式LUA源不正是这个和输出:

from foo: it works! 
from bar: it works! 
+0

太棒了,谢谢!这是我需要知道的,luaL_loadbuffer。你能告诉我,如果使用普通的luac可以做到同样的事情吗? – Matthew

14

main.lua

print("Hello from main.lua") 

app.c

#include <stdio.h> 

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

int main(int argc, char **argv) 
{ 
    int status; 
    lua_State *L = luaL_newstate(); 
    luaL_openlibs(L); 
    lua_getglobal(L, "require"); 
    lua_pushliteral(L, "main"); 
    status = lua_pcall(L, 1, 0, 0); 
    if (status) { 
    fprintf(stderr, "Error: %s\n", lua_tostring(L, -1)); 
    return 1; 
    } 
    return 0; 
} 

Shell命令:

luajit -b main.lua main.o 
gcc -O2 -Wall -Wl,-E -o app app.c main.o -Ixx -Lxx -lluajit-5.1 -lm -ldl 

用LuaJIT include和库目录替换-Ixx-Lxx。如果您已将它安装在/usr/local(默认值)中,那么大多数GCC安装会在没有这两个选项的情况下找到它。

第一个命令将Lua源代码编译为字节码并将其嵌入到目标文件main.o中。

第二个命令编译并链接最小的C应用程序代码。请注意,它也链接在嵌入式字节码中。 -Wl,-E是必需的(在Linux上)从可执行文件导出所有符号。

现在移动原main.lua离开(以确保它的真正运行嵌入式字节码,而不是Lua的源代码文件),然后运行你的应用程序:

mv main.lua main.lua.orig 
./app 
# Output: Hello from main.lua 
+0

太棒了!谢谢! luac也可以生成.o文件吗? – Matthew

+0

不,只有'luajit -b'可以一步完成。有像'bin2c'这样的第三方工具可以帮助简单的Lua。看看[其他答案](http://stackoverflow.com/a/19426724/1657919)为一般方法。 –

+0

有没有办法像'assert(loadfile(“main”))'而不是'require(“main”)''做些事情? 'loadfile'似乎没有找到像'require'这样的链接模块,而只是运行搜索文件。我试图将参数(从app.c使用argv)传递给链接模块,并且不能使用'require',因为它不会返回模块作为我可以像'loadfile'那样调用的函数。 – ckemper