使用SWIG从Perl调用C库时发生崩溃(AIX 5.1)

我试图在AIX 5.1机器上从Perl调用C库.我创建了一个非常简单的测试项目来尝试练习这个.

我的C共享库(test.cpp):

#include <stdio.h>
#include <iostream>

void myfunc()
{
    printf("in myfunc()\n");
    std::cout << "in myfunc() also" << std::endl;
}

我的SWIG接口文件(test.i):

%module test
%{
void myfunc();
%}
void myfunc();

然后我像这样构建共享对象:

swig -c++ -perl test.i
g++ -c test_wrap.cxx -I/usr/opt/perl5/lib/5.6.0/aix/CORE -o test_wrap.o
g++ -c test.cpp -o test.o
ld -G -bI:/usr/opt/perl5/lib/5.6.0/aix/CORE/perl.exp -bnoentry -bexpall -lc_r test.o test_wrap.o -o test.so

此时,我有一个test.so共享对象,应该可以在perl中加载(通过SWIG生成的test.pm).我有一个非常简单的perl脚本来尝试加载共享对象并调用我正在导出的一个函数(test.pl):

#!/usr/bin/perl
use test;
test::myfunc();

当我运行test.pl时,我得到以下输出:

in myfunc()
Illegal instruction (core dumped)

如果我在myfunc中注释掉std :: cout用法,它的工作没有问题.似乎在C STL中使用任何东西都会导致核心转储(我试图声明一个std :: vector和std :: stringstream,两者都会导致核心转储).我可以创建一个独立的C可执行文件,它使用STL而没有任何问题,只有在从perl加载时我的共享对象中调用才会遇到麻烦.

我也试过使用xlc而不是gcc,但我得到了相同的结果.我想我需要传递一些时髦的链接器标志,以确保所有链接都正确发生?欢迎任何想法……

编辑:如果我使用gcc / xlc链接而不是直接调用链接器(ld),我立即得到分段错误.当perl试图简单地加载共享库时,它看起来崩溃了.像我上面那样调用ld是我最接近它的工作,但我想我可能缺少一些库或C库的特殊AIX链接器标志.

Edit2:好的,我已经开始工作了.在链接方面,AIX非常脆弱.我最终提出了以下似乎正常工作的链接命令:

ld -G -bI:/usr/opt/perl5/lib/5.6.0/aix/CORE/perl.exp -bnoentry -bexpall -lC -lc -ldl test.o test_wrap.o -o test.so

我链接的图书馆是最相关的.事实证明,提到图书馆的顺序也很重要(呃).另请注意,这是针对AIX 5.1附带的Perl 5.6.0构建的.我已经尝试针对Perl 5.8.8构建这个相同的简单应用程序,但它不起作用.但是,我很确定更直接的链接方法(使用直接gcc / xlc而不是直接调用ld)似乎效果更好.所以这个问题似乎是Perl发行版或链接器中的一个错误.

希望这将有助于一些可怜的灵魂被诅咒与AIX合作……

最佳答案
你不是只是将你的libstdc添加到ld命令中吗?例如,-lstdc?

复制问题后,我在Linux上做了什么:

gcc -g -lstdc++ -shared test*.o -o test.so

然后问题就消失了.

(试图为ld获取正确的库列表是太多的工作,所以我只是告诉gcc为我做这件事.)

转载注明原文:使用SWIG从Perl调用C库时发生崩溃(AIX 5.1) - 代码日志