我想使用 ffi gem 在 Ruby 2.0 中加载自定义 C++ 函数。在 irb session 中加载自定义库时,我在 Attach_function 调用期间收到错误:
FFI::NotFoundError: Function 'add' not found in [/usr/local/lib/libbridge.so]
这是我创建和检查 libbridge 共享对象的方法:
我的库有一个头文件:
文件:bridge.h
std::string fooBar(std::string bar);
int add(int a, int b);
库实现:
文件:bridge.cpp
#include <iostream>
std::string fooBar(std::string bar)
{
return "foo" + bar;
}
int add(int a, int b)
{
return a + b;
}
以及我链接到新库的二进制代码:
文件:bridger.cpp
#include <iostream>
#include "bridge.h"
int main(int argc, char* argv[])
{
std::string foobar = fooBar("barrrr");
std::cout << foobar << "\n";
std::cout << "Adding 4 and 2: " << add(4,2) << "\n";
}
首先,我将库编译为目标代码:
$ g++ -c -Wall -Werror -fpic bridge.cpp
然后我将其设为共享对象:
$ g++ -shared -o libbridge.so bridge.o
我去掉调试符号:
$ strip -o libbridge.so libbridge.so
并将其移动到我的系统中:
$ sudo mv libbridge.so /usr/local/lib/
然后我使其可执行并运行 ldconfig:
$ sudo chmod 0755 /usr/local/lib/libbridge.so
$ sudo ldconfig
简短检查:
$ ldd /usr/local/lib/libbridge.so
linux-gate.so.1 => (0xb7781000)
libstdc++.so.6 => /usr/lib/i386-linux-gnu/libstdc++.so.6 (0xb7689000)
libm.so.6 => /lib/i386-linux-gnu/i686/cmov/libm.so.6 (0xb7663000)
libgcc_s.so.1 => /lib/i386-linux-gnu/libgcc_s.so.1 (0xb7645000)
libc.so.6 => /lib/i386-linux-gnu/i686/cmov/libc.so.6 (0xb74e2000)
/lib/ld-linux.so.2 (0xb7782000)
我从工作目录中删除了语言环境版本:
$ rm -f bridge.so
$ rm -f bridge.o
然后我编译我的二进制文件:
$ g++ -Wall -o bridger bridger.cpp -lbridge
我成功执行了它并得到了我期望的输出:
$ ./bridger
foobarrrr
Adding 4 and 2: 6
现在我创建一个 ruby 模块 ffi_test.rb:
require "ffi"
module FfiTest
extend FFI::Library
ffi_lib_flags :now, :global
ffi_lib "/usr/local/lib/libbridge.so"
attach_function :add, [:int, :int], :int
attach_function :fooBar, [:string], :string
end
现在,当我在 irb session 中加载该代码时,我总是收到此错误:
$ irb
2.0.0p247 :001 > load "ffi_test.rb"
FFI::NotFoundError: Function 'add' not found in [/usr/local/lib/libbridge.so]
from /home/me/.rvm/gems/ruby-2.0.0-p247@global/gems/ffi-1.9.0/lib/ffi/library.rb:251:in `attach_function'
from ffi_test.rb:8:in `<module:EricInterface>'
from ffi_test.rb:3:in `<top (required)>'
from (irb):1:in `load'
from (irb):1
from /home/me/.rvm/rubies/ruby-2.0.0-p247/bin/irb:16:in `<main>'
我不明白,这里的问题是什么。我的库显然导出了 :add 和 :fooBar 函数,否则我的桥接器二进制文件将无法成功链接到该库。相反,Ruby ffi 无法查看和安装这些函数。
有人能指出我正确的方向吗?
问候 菲利克斯
最佳答案
ffi 需要一个 c 函数,因此它找不到您的 c++ 函数,因为名称损坏了 c++。
您需要告诉链接器使函数可以从 c 访问,例如通过(在您的 .h 中)文件执行操作
extern "C" {
int add(int a, int b);
}
我怀疑 FFI 也不知道如何处理需要 std::string 作为参数的函数
关于c++ - 为什么 Ruby FFI 不附加自定义共享对象库中的函数,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/17809363/