一个名为test.c 的文件可以作为cc 的输入,它是一个C 源文件。您可以在文本编辑器中打开它并阅读它。
在这种情况下,听起来 C 源代码中嵌入了一个 Lua 块。 C 代码可以与 Lua 库链接,并使用它来加载和执行 Lua 块。
这样的 C 文件可能看起来像这样:
// Disclaimer: tested with Lua 5.3 (64-bit)
#include "lua.h"
#include "lualib.h"
#include "lauxlib.h"
#define CODE "print(\"hello world\")\n";
// Or, if the chunk is compiled:
// #define CODE "\x1b\x4c\x75\x61\x53\x00\x19\x93\x0d\x0a\x1a\x0a\x04\x08\x04\x08\x08\x78\x56\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x28\x77\x40\x01\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x01\x02\x04\x00\x00\x00\x06\x00\x40\x00\x41\x40\x00\x00\x24\x40\x00\x01\x26\x00\x80\x00\x02\x00\x00\x00\x04\x06\x70\x72\x69\x6e\x74\x04\x0c\x68\x65\x6c\x6c\x6f\x20\x77\x6f\x72\x6c\x64\x01\x00\x00\x00\x01\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00"
int main(int argc, char **argv) {
lua_State *L = luaL_newstate();
luaL_openlibs(L);
luaL_loadbuffer(L, CODE, sizeof(CODE) - 1, NULL);
lua_call(L, 0, 0);
lua_close(L);
return 0;
}
C 源文件中的 Lua 块可能会也可能不会被编译。无论哪种情况,将其转换回文本或二进制 Lua 文件都不难。
例如,在 Lua 中(如果二进制编码如我的示例中所有字节为两位十六进制转义):
-- note: replaces file "out.lua" without asking
local binary = [[\x1b\x4c\x75\x61\x53\x00\x19\x93\x0d\x0a\x1a\x0a\x04\x08\x04\x08\x08\x78\x56\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x28\x77\x40\x01\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x01\x02\x04\x00\x00\x00\x06\x00\x40\x00\x41\x40\x00\x00\x24\x40\x00\x01\x26\x00\x80\x00\x02\x00\x00\x00\x04\x06\x70\x72\x69\x6e\x74\x04\x0c\x68\x65\x6c\x6c\x6f\x20\x77\x6f\x72\x6c\x64\x01\x00\x00\x00\x01\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00]]
binary = binary:gsub("..(..)", function(n)
return string.char(tonumber(n, 16))
end)
local f = assert(io.open("out.lua", "wb"))
assert(f:write(binary))
assert(f:close())
如果 Lua 块在开始时没有编译,则无需反编译。如果已编译,二进制 Lua 文件可以被反汇编或反编译,甚至可以正常加载到兼容的 Lua 解释器中。
C 文件中可能会出现其他复杂情况。可能有多个 Lua 块。在这种情况下,您可能需要单独处理每一个。也可能有一些额外的混淆(例如压缩)。但是,C 程序最终必须将数据转换为真正的 Lua 块,然后再将其交给 Lua 库,因此应该可以在切换时获取所需的数据。 (可以修改 C 或 Lua 源以将数据写出而不是加载它,或者数据可以直接从内存中转储,可能在调试器中。最坏的情况是数据流式传输到 @987654325 @ 不会同时将整个块加载到内存中,但这仍然可以轻松处理。)