奇怪的编译代码

时间:2012-05-09 19:46:30

标签: c++ qt assembly x86 google-nativeclient

我用google的nacl编译器编译了一些Qt代码,但是ncval验证器并没有理解它。许多人中的一个例子:

src/corelib/animation/qabstractanimation.cpp:165

以下是相关代码:

#define Q_GLOBAL_STATIC(TYPE, NAME)                                  \
    static TYPE *NAME()                                              \
    {                                                                \
        static TYPE thisVariable;                                    \
        static QGlobalStatic<TYPE > thisGlobalStatic(&thisVariable); \
        return thisGlobalStatic.pointer;                             \
    }

#ifndef QT_NO_THREAD
Q_GLOBAL_STATIC(QThreadStorage<QUnifiedTimer *>, unifiedTimer)
#endif

编译为:

00000480 <_ZL12unifiedTimerv>:
     480:       55                      push   %ebp
     481:       89 e5                   mov    %esp,%ebp
     483:       57                      push   %edi
     484:       56                      push   %esi
     485:       53                      push   %ebx
     486:       83 ec 2c                sub    $0x2c,%esp
     489:       c7 04 24 28 00 2e 10    movl   $0x102e0028,(%esp)
     490:       8d 74 26 00             lea    0x0(%esi,%eiz,1),%esi
     494:       8d bc 27 00 00 00 00    lea    0x0(%edi,%eiz,1),%edi
     49b:       e8 fc ff ff ff          call   49c <_ZL12unifiedTimerv+0x1c>
     4a0:       84 c0                   test   %al,%al
     4a2:       74 1c                   je     4c0 <_ZL12unifiedTimerv+0x40>
     4a4:       0f b6 05 2c 00 2e 10    movzbl 0x102e002c,%eax
     4ab:       83 f0 01                xor    $0x1,%eax
     4ae:       84 c0                   test   %al,%al
     4b0:       74 0e                   je     4c0 <_ZL12unifiedTimerv+0x40>
     4b2:       b8 01 00 00 00          mov    $0x1,%eax
     4b7:       eb 27                   jmp    4e0 <_ZL12unifiedTimerv+0x60>
     4b9:       8d b4 26 00 00 00 00    lea    0x0(%esi,%eiz,1),%esi
     4c0:       b8 00 00 00 00          mov    $0x0,%eax
     4c5:       eb 19                   jmp    4e0 <_ZL12unifiedTimerv+0x60>
     4c7:       90                      nop
     4c8:       90                      nop
     4c9:       90                      nop
     4ca:       90                      nop
     4cb:       90                      nop

检查 49b处的呼叫指令:这是验证器无法理解的内容。究竟是什么原因导致编译器发出调用自身中间的指令?有没有解决的办法?我用-g -O0 -fno-inline编译了。编译错误?

2 个答案:

答案 0 :(得分:3)

据推测,它实际上是对外部符号的调用,它将在链接时填充。实际上会被调用的是externalSymbol-4,这有点奇怪 - 也许这就是把ncval验证器从气味中抛出来的。

答案 1 :(得分:1)

这是一个动态库还是没有链接到可执行文件的静态对象?

在动态库中,这可能是因为代码构建为依赖于位置并链接到动态库中。尝试使用“objdump -d -r -R”,如果你看到TEXTREL,就是这种情况。 NaCl动态链接故事不支持TEXTREL。 (通过在编译代码期间使用-fPIC标志解决)

使用静态对象尝试在链接到静态可执行文件后进行验证。