Как использовать модуль LuaJIT ffi при встраивании?

Я пытаюсь встроить LuaJIT в приложение C. Код выглядит так:

#include <lua.h>
#include <lualib.h>
#include <lauxlib.h>
#include <stdlib.h>
#include <stdio.h>

int barfunc(int foo)
{
    /* a dummy function to test with FFI */ 
    return foo + 1;
}

int
main(void)
{
    int status, result;
    lua_State *L;
    L = luaL_newstate();

    luaL_openlibs(L);

    /* Load the file containing the script we are going to run */
    status = luaL_loadfile(L, "hello.lua");
    if (status) {
        fprintf(stderr, "Couldn't load file: %s\n", lua_tostring(L, -1));
        exit(1);
    }

    /* Ask Lua to run our little script */
    result = lua_pcall(L, 0, LUA_MULTRET, 0);
    if (result) {
        fprintf(stderr, "Failed to run script: %s\n", lua_tostring(L, -1));
        exit(1);
    }

    lua_close(L);   /* Cya, Lua */

    return 0;
}

код Lua выглядит следующим образом:

-- Test FFI
local ffi = require("ffi")
ffi.cdef[[
int barfunc(int foo);
]]
local barreturn = ffi.C.barfunc(253)
io.write(barreturn)
io.write('\n')

Он сообщает об ошибке следующим образом:

Failed to run script: hello.lua:6: cannot resolve symbol 'barfunc'.

Я обыскал и обнаружил, что в модуле ffi действительно мало документа. Большое спасибо.

Ответ 1

Библиотека ffi требует luajit, поэтому вы должны запустить lua-код с luajit. Из документа: "Библиотека FFI тесно интегрирована в LuaJIT (она недоступна как отдельный модуль)".

Как вставить luajit? Посмотрите здесь http://luajit.org/install.html в разделе "Вложение LuaJIT"

В разделе mingw ваш пример запускается, если я добавляю

__declspec(dllexport) int barfunc(int foo)

в функции barfunc.

Под Windows luajit связан как dll.

Ответ 2

Как указывал misianne, вам нужно экспортировать функцию, которую вы можете сделать, используя extern, если вы используете GCC:

extern "C" int barfunc(int foo)
{
    /* a dummy function to test with FFI */ 
    return foo + 1;
}

Если у вас возникли проблемы с символами undefined в Linux с помощью GCC, позаботьтесь, чтобы компоновщик добавил все символы в таблицу динамических символов, передав флаг -rdynamic в GCC:

g++ -o application soure.cpp -rdynamic -I... -L... -llua

Ответ 3

Для тех из вас, кто пытается выполнить эту работу в Windows с помощью VС++ (2012 или позже), используя компилятор С++:

  • убедитесь, что вы используете расширение .cpp, так как это сделает компиляцию С++
  • заставить функцию иметь внешнюю связь C, чтобы ffi мог ссылаться на нее, с extern "C" { ... }
  • экспортировать функцию из исполняемого файла, __declspec(dllexport)
  • опционально укажите соглашение о вызове __cdecl, не требуется, поскольку оно должно быть по умолчанию и не переносимо
  • оберните заголовки Lua в extern "C" { include headers } или лучше просто #include "lua.hpp"

    #include "lua.hpp"  
    
    extern "C" {
    __declspec(dllexport) int __cdecl barfunc(int foo) { 
     return foo + 1;
    }}