1. 程式人生 > >鏈接器下——鏈接器實戰

鏈接器下——鏈接器實戰

而不是 分析 header 系統服務 png nag 服務 函數 code

鏈接器實戰

1目標

模擬嵌入式開發,編寫一個“體積受限”的可執行程序,通過makefile 完成編譯, 運行後打印“Hello BT”

2分析過程

技術分享圖片

3解決方案

1.通過內嵌匯編自定義打印函數和退出函數(INT 80H)
2.通過鏈接腳本自定入口函數(不依賴任何庫和gcc內置功能)
3.刪除可執行程序中的無用信息(無用段信息、調試信息、等)

4打印函數設計

void print(const char* s, int l)
{
    asm volatile (
        "movl $4, %%eax\n"
        "movl $1, %%ebx\n"
        "movl %0, %%ecx\n"
        "movl %1, %%edx\n"
        "int $0x80     \n"
        :
        : "r"(s), "r"(l)
        : "eax", "ebx", "ecx", "edx"
    );
}

5退出函數設計

void exit(int code)
{
    asm volatile (
        "movl $1, %%eax\n"
        "movl %0, %%ebx\n"
        "int $0x80     \n"
        :
        : "r"(code)
        : "eax", "ebx"
    );
}

6鏈接腳本設計

ENTRY(program)

SECTIONS
{
    .text 0x08048000 + SIZEOF_HEADERS :
    {
        *(.text)
        *(.rodata)
    }

    /DISCARD/ :
    {
        *(*)
    }
}

7 makefie

CC := gcc
LD := ld
RM := rm -fr

TARGET := program.out
SRC := $(TARGET:.out=.c)
OBJ := $(TARGET:.out=.o)
LDS := $(TARGET:.out=.lds)

.PHONY : rebuild clean all

$(TARGET) : $(OBJ) $(LDS)
    $(LD) -static -T $(LDS) -o $@ $<
    @echo "Target File ==> $@"

$(OBJ) : $(SRC)
    $(CC) -fno-builtin -o $@ -c $^

rebuild : clean all

all : $(TARGET)

clean :
    $(RM) $(TARGET) $(OBJ)

8指定鏈接選項

  • -ld 命令 // GNU的鏈接器,將目標文件鏈接為可執行程序,GCC編譯器集成中的一員,重要的幕後工作者
  • -ld -static // 指定使用靜態鏈接的方式來產生最終程序,而不是默認的動態鏈接方式。
  • -gcc -fno-builtin //用於關閉GCC 內置函數的功能(GCC提供了很多內置函數(Built-in Function),它會將一些常用的C庫函數替換為編譯器的內置函數,以達到優化的目的)。

    9總結

    對於資源受限的嵌入式設備,需要考慮可執行程序的大小,通過內嵌匯編直接使用系統服務能夠劈開相關庫的使用可以通過如下方法控制可執行程序的體積大小。
    1.最小化庫的使用(必要情況下考慮自己實現相關函數)
    2.自定義鏈接腳本,刪除無用段信息

鏈接器下——鏈接器實戰