關於程式的入口函式(main _start...)
阿新 • • 發佈:2018-11-14
參照來源:https://blog.csdn.net/cherisegege/article/details/80297320
ld有多種方法設定程序入口地址, 按一下順序: (編號越前, 優先順序越高)
1, ld命令列的-e選項
2, 連線指令碼的ENTRY(SYMBOL)命令
eg. ENTRY(_start) /* Entry point of application*/
3, 如果定義了start符號, 使用start符號值
4, 如果存在.text section, 使用.text section的第一位元組的位置值
5, 使用值0
(一)通常例子
#include <stdio.h>
int main()
{
printf("helloworld! \n");
return 0;
}
$ gcc hello.c -o hello
$ ./hello
使用者的程式碼是從main函式開始執行的,還有其它很多函式,比如_start函式。實際上程式真正的入口並不是main函式,我們以下面命令對hello.c程式碼進行編譯:
$ gcc hello.c -nostdlib
/usr/bin/ld: warning: cannot find entrysymbol _start; defaulting to 0000000000400144
-nostdlib命令是指不連結標準庫,報錯說找不到entry symbol _start,這裡是說找不到入口符號_start,也就是說程式的真正入口是_start函式。
實際上main函式只是使用者程式碼的入口,它會由系統庫去呼叫,在main函式之前,系統庫會做一些初始化工作,比如分配全域性變數的記憶體,初始化堆、執行緒等,當main函式執行完後,會通過exit()函式做一些清理工作,使用者可以自己實現_start函式:
(二)通過 _start 來實現
#include <stdio.h>
#include <stdlib.h>
int _start(void)
{
printf("hello world!\n");
exit(0);
}
執行如下編譯命令並執行:
$ gcc hello_start.c -nostartfiles -o hello_start
$ ./hello_start
hello world!
$ readelf -al a.out
...
Symbol table '.dynsym' contains 3 entries:
Num: Value Size Type Bind Vis Ndx Name
0: 0000000000000000 0 NOTYPE LOCAL DEFAULT UND
1: 0000000000000000 0 FUNC GLOBAL DEFAULT UND [email protected]_2.2.5 (2)
2: 0000000000000000 0 FUNC GLOBAL DEFAULT UND [email protected]_2.2.5 (2)
...
//檢視elf檔案 引數只有一個,printf 被優化成了puts
$ readelf -al a.out | grep FUN
1: 0000000000000000 0 FUNC GLOBAL DEFAULT UND [email protected]_2.2.5 (2)
2: 0000000000000000 0 FUNC GLOBAL DEFAULT UND [email protected]_2.2.5 (2)
22: 0000000000000000 0 FUNC GLOBAL DEFAULT UND [email protected]@GLIBC_2.2.5
25: 0000000000400390 24 FUNC GLOBAL DEFAULT 10 _start
27: 0000000000000000 0 FUNC GLOBAL DEFAULT UND [email protected]@GLIBC_2.2.5
(三)通過連結的時候指定來實現
#include <stdio.h>
#include <stdlib.h>
int mymain()
{
printf("helloworld!\n");
exit(0);
}
$ gcc hello_nomain.c -nostartfiles -e mymain -o hello_mymain
其中-e選項可以指定程式入口符號,檢視符號表如下:
$ readelf -s hello_mymain | grep FUNC
1: 0000000000000000 0 FUNC GLOBAL DEFAULT UND [email protected]_2.2.5 (2)
2: 0000000000000000 0 FUNC GLOBAL DEFAULT UND [email protected]_2.2.5 (2)
22: 0000000000000000 0 FUNC GLOBAL DEFAULT UND [email protected]@GLIBC_2.2.5
24: 0000000000400390 24 FUNC GLOBAL DEFAULT 10 mymain
27: 0000000000000000 0 FUNC GLOBAL DEFAULT UND [email protected]@GLIBC_2.2.5
//elf header 中的 入口 Entry point address 就是 函式的開始地址
ELF Header:
Magic: 7f 45 4c 46 02 01 01 00 00 00 00 00 00 00 00 00
Class: ELF64
Data: 2's complement, little endian
Version: 1 (current)
OS/ABI: UNIX - System V
ABI Version: 0
Type: EXEC (Executable file)
Machine: Advanced Micro Devices X86-64
Version: 0x1
Entry point address: 0x400390
Start of program headers: 64 (bytes into file)
Start of section headers: 5184 (bytes into file)
Flags: 0x0
Size of this header: 64 (bytes)
Size of program headers: 56 (bytes)
Number of program headers: 9
Size of section headers: 64 (bytes)
Number of section headers: 20
Section header string table index: 17