C ++ CodeBlocks反汇编;方式太多代码?

时间:2014-12-17 02:34:44

标签: c++ assembly x86

我在CodeBlocks上运行调试器并查看了反汇编窗口。

我调试的程序的完整源代码如下:

int main(){}

我在窗口中看到的汇编代码是:

00401020    push   %ebp
00401021    mov    %esp,%ebp
00401023    push   %ebx
00401024    sub    $0x34,%esp
00401027    movl   $0x401150,(%esp)
0040102E    call   0x401984 <SetUnhandledExceptionFilter@4>
00401033    sub    $0x4,%esp
00401036    call   0x401330 <__cpu_features_init>
0040103B    call   0x401740 <fpreset>
00401040    lea    -0x10(%ebp),%eax
00401043    movl   $0x0,-0x10(%ebp)
0040104A    mov    %eax,0x10(%esp)
0040104E    mov    0x402000,%eax
00401053    movl   $0x404004,0x4(%esp)
0040105B    movl   $0x404000,(%esp)
00401062    mov    %eax,0xc(%esp)
00401066    lea    -0xc(%ebp),%eax
00401069    mov    %eax,0x8(%esp)
0040106D    call   0x40192c <__getmainargs>
00401072    mov    0x404008,%eax
00401077    test   %eax,%eax
00401079    jne    0x4010c5 <__mingw_CRTStartup+165>
0040107B    call   0x401934 <__p__fmode>
00401080    mov    0x402004,%edx
00401086    mov    %edx,(%eax)
00401088    call   0x4014f0 <_pei386_runtime_relocator>
0040108D    and    $0xfffffff0,%esp
00401090    call   0x401720 <__main>
00401095    call   0x40193c <__p__environ>
0040109A    mov    (%eax),%eax
0040109C    mov    %eax,0x8(%esp)
004010A0    mov    0x404004,%eax
004010A5    mov    %eax,0x4(%esp)
004010A9    mov    0x404000,%eax
004010AE    mov    %eax,(%esp)
004010B1    call   0x401318 <main>
004010B6    mov    %eax,%ebx
004010B8    call   0x401944 <_cexit>
004010BD    mov    %ebx,(%esp)
004010C0    call   0x40198c <ExitProcess@4>
004010C5    mov    0x4050f4,%ebx
004010CB    mov    %eax,0x402004
004010D0    mov    %eax,0x4(%esp)
004010D4    mov    0x10(%ebx),%eax
004010D7    mov    %eax,(%esp)
004010DA    call   0x40194c <_setmode>
004010DF    mov    0x404008,%eax
004010E4    mov    %eax,0x4(%esp)
004010E8    mov    0x30(%ebx),%eax
004010EB    mov    %eax,(%esp)
004010EE    call   0x40194c <_setmode>
004010F3    mov    0x404008,%eax
004010F8    mov    %eax,0x4(%esp)
004010FC    mov    0x50(%ebx),%eax
004010FF    mov    %eax,(%esp)
00401102    call   0x40194c <_setmode>
00401107    jmp    0x40107b <__mingw_CRTStartup+91>
0040110C    lea    0x0(%esi,%eiz,1),%esi

从这么少的C ++代码中获取这么多汇编代码是否正常?

通常,我的意思是,这接近MinGw编译器生成的汇编代码的平均数量,相对于我上面提供的C ++源代码量?

2 个答案:

答案 0 :(得分:23)

是的,这是相当典型的启动/关闭代码。

main运行之前,需要做一些事情:

  1. stdin / stdout / stderr打开
  2. cin / cout / cerr / clog打开,参考stdin / stdout / stderr
  3. 您定义的任何静态对象都已初始化
  4. 解析
  5. 命令行以生成argc / argv
  6. 环境被检索(可能)
  7. 同样,在您main退出后,还需要做更多的事情:

    1. 使用atexit设置的任何内容都会运行
    2. 您的静态物品被销毁
    3. cin / cout / cerr / clog被销毁
    4. 所有打开的输出流都被刷新并关闭
    5. 所有打开的输入流都已关闭
    6. 根据平台的不同,可能还有一些其他内容,例如设置一些默认的异常处理程序(对于C ++异常,某些特定于平台的异常,或两者兼而有之)。

      请注意,大部分内容都是固定代码,无论其执行或不包含,都会链接到每个程序中。从理论上讲,他们可以使用一些技巧(例如,&#34;弱外部&#34;)来避免在不需要的情况下链接到某些代码,但是上面的大部分内容都是这样使用的,所以接近普遍(并且处理它的代码是非常微不足道的),即使它不会被使用(例如它)也很难去任何工作来消除这些代码。你的情况,根本没有使用任何东西)。

      请注意,您显示的启动/关闭代码。它链接到您的程序,传统上来自名为crt0的文件(可能还有一些其他文件)。

      如果您浏览文件以查找为main生成的代码,您可能会发现它缩短了很多 - 可能只是简短而简单ret 。它可能是如此之小,以至于你错过了它在那里的事实。

答案 1 :(得分:1)

call 0x401318 <main>

基本上,

是您解码的代码。 main()是一个函数,它周围有代码,通常称为__start__end

您看到的内容部分归功于__start中的CRT支持代码,以及之后__end中的清理。