1

我正在使用 C++ 的 SWI-Prolog外语界面,试图集成一些其他资源。

它主要是有效的,但是任何抛出异常的尝试都会导致 SIGSEGV。异常通常用于验证用户参数,因此是接口的基本部分。

我正在从源代码(通过提供的脚本)编译 SWI-Prolog,并且 CXX 标志是

-c -O2 -gdwarf-2 -g3 -Wall -pthread -fPIC

我正在使用相同的标志来编译我的 C++ 代码,该代码组装在一个 .so 中,在 SWI-Prolog 中动态加载(我认为是通过 dlopen)。

在 SEGV 在 __cxa_allocate_exception 内的 <+36> 处显示 IP 之后检查堆栈(通过 GDB)。可能无法访问 __cxa_get_globals@plt。

        Dump of assembler code for function __cxa_allocate_exception:
0x00007ffff1d80220  <+0>:               push   %r12
0x00007ffff1d80222  <+2>:               lea    0x80(%rdi),%r12
0x00007ffff1d80229  <+9>:               push   %rbp
0x00007ffff1d8022a  <+10>:              mov    %r12,%rdi
0x00007ffff1d8022d  <+13>:              push   %rbx
0x00007ffff1d8022e  <+14>:              callq  0x7ffff1d1de30 <malloc@plt>
0x00007ffff1d80233  <+19>:              test   %rax,%rax
0x00007ffff1d80236  <+22>:              mov    %rax,%rbx
0x00007ffff1d80239  <+25>:              je     0x7ffff1d802d8 <__cxa_allocate_exception+184>
0x00007ffff1d8023f  <+31>:              callq  0x7ffff1d1efc0 <__cxa_get_globals@plt>
0x00007ffff1d80244  <+36>:              addl   $0x1,0x8(%rax)
0x00007ffff1d80248  <+40>:              test   $0x1,%bl
0x00007ffff1d8024b  <+43>:              mov    %rbx,%rdi
0x00007ffff1d8024e  <+46>:              mov    $0x80,%edx
0x00007ffff1d80253  <+51>:              jne    0x7ffff1d803d0 <__cxa_allocate_exception+432>
0x00007ffff1d80259  <+57>:              test   $0x2,%dil

我能找到的唯一资源似乎是相关的主张

异常需要 typeinfo 查找

这可能对 SIGSEGV 有意义。

但我现在无法继续。当然,我希望在一些神奇的 CXX 或 LD 标志。或者我应该装饰我的库入口点(我熟悉 Windows declspec,我广泛使用它们来构建 MFC 扩展 DLL)或其他什么?

4

1 回答 1

0

从 Prolog 调用时,您不能向 Prolog 内核抛出任何异常。C++ 接口将捕获 PlException 及其子类并将它们转换为 Prolog 异常。绝不能允许所有其他例外逃离您的图书馆。

由于 SWI Prolog 是 LGPL,因此您可能正在动态链接到它。因此,您必须确保抛出的所有 C++ 在 ELF 系统上都具有默认可见性。

于 2015-10-29T14:25:07.257 回答