GCC-プログラムのコンパイルプロセス

本論文では、コンパイルプロセスのためのgccコンパイラツールは、プログラムを記録します


コンパイル(合計)

私たちは、単純なプログラムhelloword、実行可能ファイルにプログラムの必要性を記述するときは、次のステップを通過する必要がありますコンパイルすることです。

hello.c
-----------------------------------------------
#include <stdio.h>

#define X 1

int main()
{
        int i=0;
        printf("hello world X = %d i=%d !\n",X,i);
        return 0;
}

gccをコンパイラツールをされて使用している場合は、GCCのhello.cでは、唯一の出力がファイルa.outをコンパイルすることができなければなりません。ヘルプは、プロセスの理解深めるためにこれらのステップを理解すること、事前コンパイル、コンパイル、アセンブルし、リンク(リンクは厳密に直接一緒に話すためにここに、ビルドプロセスの一部を話していない) - しかし、単純なコンパイルはいくつかのステップが含まれています。

プリコンパイル済み

#ifdefを含む前命令の先頭に「#」を持つC言語で、#はの#defineを含めると、プリコンパイラ・ディレクティブを属します。プリコンパイルされた段階では、GCCコンパイラディレクティブ正確に、前処理は、実際の値が、メディアファイルへの割り当ての#includeヘッダファイルに展開マクロ定義に#defineを配置するために使用されますが、コピーした後、正しくチェックしないんだろうプロセスのその後の流れ。

gcc -E -o hello.i hello.c

以上のように、これはgccの唯一の実装では、プロセスをプリコンパイルすることを追加する必要が-E一般的なプリコンパイルされた出力ファイル.iの接尾辞ながら、オプションを選択します。この抜粋ファイルパーツの結果は、の#includeが膨張した後の#define Xを使用する主な機能は、実際のところ1により置換されている間、関連する文とヘッダファイルは、ファイルにコピーされる参照します。

# 1 "hello.c"
# 1 "<built-in>"
# 1 "<command-line>"
# 1 "/usr/include/stdc-predef.h" 1 3 4
# 1 "<command-line>" 2
# 1 "hello.c"
# 1 "/usr/include/stdio.h" 1 3 4
# 27 "/usr/include/stdio.h" 3 4
# 1 "/usr/include/features.h" 1 3 4
# 367 "/usr/include/features.h" 3 4
# 1 "/usr/include/x86_64-linux-gnu/sys/cdefs.h" 1 3 4
# 410 "/usr/include/x86_64-linux-gnu/sys/cdefs.h" 3 4
# 1 "/usr/include/x86_64-linux-gnu/bits/wordsize.h" 1 3 4
# 411 "/usr/include/x86_64-linux-gnu/sys/cdefs.h" 2 3 4
# 368 "/usr/include/features.h" 2 3 4
# 391 "/usr/include/features.h" 3 4
# 1 "/usr/include/x86_64-linux-gnu/gnu/stubs.h" 1 3 4
# 10 "/usr/include/x86_64-linux-gnu/gnu/stubs.h" 3 4
# 1 "/usr/include/x86_64-linux-gnu/gnu/stubs-64.h" 1 3 4
# 11 "/usr/include/x86_64-linux-gnu/gnu/stubs.h" 2 3 4
# 392 "/usr/include/features.h" 2 3 4
# 28 "/usr/include/stdio.h" 2 3 4

...
...


extern int fprintf (FILE *__restrict __stream,
      const char *__restrict __format, ...);


extern int printf (const char *__restrict __format, ...);

...
...

# 5 "hello.c"
int main()
{
 int i=0;
 printf("hello world X = %d i=%d !\n",1,i);
 return 0;
}

コンパイル

次のプロセスに基づいて文書では、ここで、この資料の冒頭をコンパイルし、コンパイル、実行されるべきである。この論文の冒頭でコンパイルされ、同じものではありません短いため、いくつかのプロセスを説明することです。この文書にコンパイルは、唯一のプロセスをコンパイルするプリコンパイラを指します。コンパイラは、マシンに対応するアセンブラコードを生成し、同様のhello.cはhello.i例完了プリコンパイル、使用が-S コンパイルプロセスパラメータを。

gcc -S -o hello.s hello.i
        .file   "hello.c"
        .section        .rodata
.LC0:
        .string "hello world X = %d i=%d !\n"
        .text
        .globl  main
        .type   main, @function
main:
.LFB0:
        .cfi_startproc
        pushq   %rbp
        .cfi_def_cfa_offset 16
        .cfi_offset 6, -16
        movq    %rsp, %rbp
        .cfi_def_cfa_register 6
        subq    $16, %rsp
        movl    $0, -4(%rbp)
        movl    -4(%rbp), %eax
        movl    %eax, %edx
        movl    $1, %esi
        movl    $.LC0, %edi
        movl    $0, %eax
        call    printf
        movl    $0, %eax
        leave
        .cfi_def_cfa 7, 8
        ret
        .cfi_endproc
.LFE0:
        .size   main, .-main
        .ident  "GCC: (Ubuntu 5.4.0-6ubuntu1~16.04.12) 5.4.0 20160609"
        .section        .note.GNU-stack,"",@progbits

組み立て、リンク

行うにはコンパイル段階は、バイナリマシンコードに変換され、アセンブリコードの出力処理をコンパイルすることですが、それは最終的な実行可能ファイルを取得するために、ライブラリやファイルをリンクする必要性に直接依存して実行されません。国会使用して処理  -cパラメータを、リンクは直接.oファイルをリンク行くことができます直接のgccを実行することができます。

gcc -c -o hello.o hello.s
gcc -o hello hello.o

システムは、このプロセスの依存ライブラリであるため、リンクリスト私たち自身hello.oの例のみ、私はここでは、デフォルトでロードされ、他の依存ライブラリを参照してください、まだ成功しリンクすることができ、通常実行することができませんでした直接のgccへのリンクで-vパラメータを観察することができ、

 gcc -v -o hello  hello.o
Configured with: ../src/configure -v --with-pkgversion='Ubuntu 5.4.0-6ubuntu1~16.04.12' --with-bugurl=file:///usr/share/doc/gcc-5/README.Bugs --enable-languages=c,ada,c++,java,go,d,fortran,objc,obj-c++ --prefix=/usr --program-suffix=-5 --enable-shared --enable-linker-build-id --libexecdir=/usr/lib --without-included-gettext --enable-threads=posix --libdir=/usr/lib --enable-nls --with-sysroot=/ --enable-clocale=gnu --enable-libstdcxx-debug --enable-libstdcxx-time=yes --with-default-libstdcxx-abi=new --enable-gnu-unique-object --disable-vtable-verify --enable-libmpx --enable-plugin --with-system-zlib --disable-browser-plugin --enable-java-awt=gtk --enable-gtk-cairo --with-java-home=/usr/lib/jvm/java-1.5.0-gcj-5-amd64/jre --enable-java-home --with-jvm-root-dir=/usr/lib/jvm/java-1.5.0-gcj-5-amd64 --with-jvm-jar-dir=/usr/lib/jvm-exports/java-1.5.0-gcj-5-amd64 --with-arch-directory=amd64 --with-ecj-jar=/usr/share/java/eclipse-ecj.jar --enable-objc-gc --enable-multiarch --disable-werror --with-arch-32=i686 --with-abi=m64 --with-multilib-list=m32,m64,mx32 --enable-multilib --with-tune=generic --enable-checking=release --build=x86_64-linux-gnu --host=x86_64-linux-gnu --target=x86_64-linux-gnu
Thread model: posix
gcc version 5.4.0 20160609 (Ubuntu 5.4.0-6ubuntu1~16.04.12)
COMPILER_PATH=/usr/lib/gcc/x86_64-linux-gnu/5/:/usr/lib/gcc/x86_64-linux-gnu/5/:/usr/lib/gcc/x86_64-linux-gnu/:/usr/lib/gcc/x86_64-linux-gnu/5/:/usr/lib/gcc/x86_64-linux-gnu/
LIBRARY_PATH=/usr/lib/gcc/x86_64-linux-gnu/5/:/usr/lib/gcc/x86_64-linux-gnu/5/../../../x86_64-linux-gnu/:/usr/lib/gcc/x86_64-linux-gnu/5/../../../../lib/:/lib/x86_64-linux-gnu/:/lib/../lib/:/usr/lib/x86_64-linux-gnu/:/usr/lib/../lib/:/usr/lib/gcc/x86_64-linux-gnu/5/../../../:/lib/:/usr/lib/
COLLECT_GCC_OPTIONS='-v' '-o' 'hello' '-mtune=generic' '-march=x86-64'
 /usr/lib/gcc/x86_64-linux-gnu/5/collect2 -plugin /usr/lib/gcc/x86_64-linux-gnu/5/liblto_plugin.so -plugin-opt=/usr/lib/gcc/x86_64-linux-gnu/5/lto-wrapper -plugin-opt=-fresolution=/tmp/ccLdFZul.res -plugin-opt=-pass-through=-lgcc -plugin-opt=-pass-through=-lgcc_s -plugin-opt=-pass-through=-lc -plugin-opt=-pass-through=-lgcc -plugin-opt=-pass-through=-lgcc_s --sysroot=/ --build-id --eh-frame-hdr -m elf_x86_64 --hash-style=gnu --as-needed -dynamic-linker /lib64/ld-linux-x86-64.so.2 -z relro -o hello /usr/lib/gcc/x86_64-linux-gnu/5/../../../x86_64-linux-gnu/crt1.o /usr/lib/gcc/x86_64-linux-gnu/5/../../../x86_64-linux-gnu/crti.o /usr/lib/gcc/x86_64-linux-gnu/5/crtbegin.o -L/usr/lib/gcc/x86_64-linux-gnu/5 -L/usr/lib/gcc/x86_64-linux-gnu/5/../../../x86_64-linux-gnu -L/usr/lib/gcc/x86_64-linux-gnu/5/../../../../lib -L/lib/x86_64-linux-gnu -L/lib/../lib -L/usr/lib/x86_64-linux-gnu -L/usr/lib/../lib -L/usr/lib/gcc/x86_64-linux-gnu/5/../../.. hello.o -lgcc --as-needed -lgcc_s --no-as-needed -lc -lgcc --as-needed -lgcc_s --no-as-needed /usr/lib/gcc/x86_64-linux-gnu/5/crtend.o /usr/lib/gcc/x86_64-linux-gnu/5/../../../x86_64-linux-gnu/crtn.o

上記-v出力パラメータは-lcパラメータは、ライブラリシステムのlibcリンクされ、表示され、直接使用lddコマンドはまた、結果の実行可能ファイルのハローが依存見ることができる見ることができます検索用Ctrl + Fキー、使用しますライブラリ。

CryptonymAMS$: ldd hello
        linux-vdso.so.1 =>  (0x00007ffc397a6000)
        libc.so.6 => /lib/x86_64-linux-gnu/libc.so.6 (0x00007f4a8ed33000)
        /lib64/ld-linux-x86-64.so.2 (0x00007f4a8f0fd000)

こんにちはは、動的リンクであること、あなたが動的にリンクを表示するには、fileコマンドを使用することができ、システムにリンクされているダイナミックリンクライブラリを使用しています。

hello: ELF 64-bit LSB executable, x86-64, version 1 (SYSV), dynamically linked, interpreter /lib64/l, for GNU/Linux 2.6.32, BuildID[sha1]=ac17e411d6f0b61087de124ee029056f09f4c990, not stripped

これまでのところ、すべてのGCCコンパイラプロセスは、実行可能ファイルの出力を完了しています。

 

サプリメント

gccの一般的に使用されるパラメータ

-E :执行到预编译流程就停止,不执行后续编译、汇编等流程
-S :只执行到编译流程,不执行后续汇编等流程
-c :只执行到汇编流程,不执行链接流程

-o :指定输出文件的名字

-I :指定头文件路径
-L :指定库文件路径
-l :指定需要链接的库(如libc.so,只需要 -lc 即可)

-static :静态链接方式(可执行文件包含执行需要的所有信息,但是文件大小会比较大)
-nostdlib :不链接标准库

-v :查看gcc版本信息,编译过程使用时用于打印编译过程的输出

その他のパラメータは、直接またはman GCCのgcc --helpビューすることができます。

公開された19元の記事 ウォン称賛7 ビュー6927

おすすめ

転載: blog.csdn.net/G_METHOD/article/details/104163168