1. 程式人生 > >關於程式的入口函式(main _start...)

關於程式的入口函式(main _start...)

參照來源: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