Linux(链接器的意义)

简介: Linux(链接器的意义)

前言

本篇文章我们来讲解链接器的意义。


一、链接器概念介绍

链接器(Linker)是计算机编译器系统中的一个重要组成部分,它用于将编译后生成的目标模块(Object Module)链接在一起,生成可执行文件或动态链接库(Dynamic Linking Library)。

链接器的主要任务是将符号(Symbol)引用解析到符号定义上,将多个目标模块合并为一个可执行文件或动态链接库,生成符号表(Symbol Table),并对程序代码做最后的检查和优化。

在编译过程中,C、C++等程序源代码先经过编译器的处理,生成目标代码文件,然后由链接器将多个目标代码合并成单个可执行文件或动态链接库,以便于程序执行和使用。


二、目标文件

首先我们先编写两个文件一个是main.c一个是func.c。

main.c:

extern void func(void);
int g_a = 0;
int main(void)
{
    func();
    return 0;
}

func.c:

#include <stdio.h>
void func(void)
{
    printf("Hello World\n");
}

将这两个文件编译成目标文件.o

使用nm命令查看两个目标文件的信息:


通过nm命令我们可以发现这两个目标文件里面的各个段都是没有具体地址的。但是我们可以查看到具体的段大小。

目录文件需要注意的地方:

1.各个段没有具体的起始地址,只有段大小信息。

2.各个标识符没有实际地址,只有段中的相对地址。

3.段和标识符的实际地址需要链接器具体确定。

链接器的几个主要作用:

符号解析:将目标文件中的符号引用和定义进行匹配,并生成符号表信息,方便程序执行时进行符号地址绑定。

文件合并:将多个目标文件合并为单个可执行文件或动态链接库。

地址空间分配:将符号或变量分配到内存中的具体地址。

重定位:合并多个目标模块时,需要对地址进行重定位,以便在程序的正确地址空间中运行。

跳转补丁:对目标代码进行修补,以确保跳转地址的正确性和有效性。

使用链接器将他们链接起来:

链接完成后我们就可以查看到具体的段地址等信息了。


三、main函数是第一个被执行的函数吗?

在我们编写代码的时候第一个执行的函数就是main函数,那么很多人就可能会认为第一个执行的程序就是main函数,其实这是不正确的。下面我就来证明一下main函数不是第一个被执行的程序。

首先使用objdump -d 命令来生成一个汇编文件:

使用cat 命令来查看这个文件:

在_start这里我们可以查看到main函数:

这里也可以看到_start函数的地址:0000000000000530

使用objdump - h命令显示文件的段表,包括段的名称、大小、虚拟地址、文件偏移等信息。

这里可以看到text段和_start的段地址是一样的。所以我们可以证明执行代码的时候首先执行的是_start函数,而不是main函数。

我们查看_start里面的信息就可以知道程序开始时到底做了什么操作:

四、链接脚本的意义和作用

链接脚本(Linker Script)是用于指导链接器如何将目标文件链接生成最终的可执行文件或者动态链接库的脚本。链接脚本描述了代码的内存分配、数据的布局、初始化、以及其他一些链接时需要的信息。

链接脚本通常使用一种简单的编程语言,例如 GNU Linker 的链接脚本是使用类似 C 的语言写成的脚本。通常它包含了以下信息:

地址空间布局:链接脚本可以指定代码段、数据段、BSS段等在内存中的位置和大小,并决定它们的起始地址和结束地址。

符号表:链接脚本可以定义符号表,这些符号可以是全局变量、函数、常量等,它们将在链接过程中被绑定。

初始化和清除:链接器会按照链接脚本指定的顺序对段进行初始化,清除,或者进行其他的一些工作。

程序入口:链接脚本中可以指定程序入口,例如 main() 函数所在的地址。

链接脚本的主要作用是描述生成可执行文件或者动态链接库所需的各种相关信息,同时还可以完成其他一些链接时的必要处理,并附加一些必要的元数据,例如程序入口点和动态链接库函数符号等。通过使用链接脚本,程序员可以更加精细地控制生成的可执行文件或者动态链接库的各种属性,同时也可以有效地解决某些链接时需要的问题。

下面我们来编写一个链接脚本:

ENTRY(main)     /* 设置程序入口为 main 函数 */
SECTIONS {
    . = 0x08000000;  /* 指定起始地址 */
    /* 定义代码段,包含 .text 和 .rodata 段 */
    .text   ALIGN(4) :
    {
        *(.text)        /* 提取目标文件中的所有代码段 */
        *(.rodata)      /* 提取目标文件中的只读数据段 */
    } >FLASH
    /* 定义数据段,包含 .data 和 .bss 段 */
    .data   ALIGN(4) :
    {
        *(.data)        /* 提取目标文件中的所有数据段 */
    } >SRAM
    /* 定义bss段 */
    .bss (NOLOAD) :
    {
        _sbss = .;      /* 设置 bss 段的起始地址 */
        *(.bss)         /* 提取目标文件中的 bss 数据段 */
        *(COMMON)       /* 提取目标文件中的 common 数据段 */
        _ebss = .;      /* 设置 bss 段的结束地址 */
    } >SRAM
    /* 设置栈和堆 */
    _stack_end = ORIGIN(SRAM) + LENGTH(SRAM);  /* 设置栈的结束地址 */
    _heap_start = _stack_end;      /* 设置堆的起始地址 */
    _heap_end = ORIGIN(SRAM) + LENGTH(SRAM);    /* 设置堆的结束地址 */
}

此链接脚本是用于将代码段和只读数据段从FLASH存储器分配到0x08000000和数据段和BSS段从SRAM存储器分配到0x20000000。它还定义了堆和栈的位置,以及一些变量。

该脚本首先定义程序入口为 main 函数,然后使用 SECTIONS 区块来定义不同段的位置和大小。代码段和只读数据段使用 ALIGN 对齐到4字节,并通过 >FLASH 分配到 FLASH 存储器中。数据段使用 ALIGN 对齐到4字节,并通过 >SRAM 分配到 SRAM 存储器中。BSS段没有实际数据,仅分配段空间,通过 NOLOAD 让链接器不会从可执行文件中复制数据到RAM中从而节省存储空间,使用 _sbss 和 _ebss 定义BSS段起始和结束地址。最后,通过自定义变量来设置栈和堆的位置。


总结

本篇文章就讲解到这里了,希望大家好好掌握这些知识点。


相关文章
|
2月前
|
Linux 数据处理 开发者
Linux命令ld.bfd:二进制文件的强大链接器
`ld.bfd`是GNU链接器的变体,利用BFD库处理多种目标文件格式(如ELF, COFF)。它收集文件,解析符号,执行重定位,生成可执行文件。特点包括多格式支持,高效符号管理和诊断信息。常用命令如`ld.bfd -o output file1.o file2.o -lc`。注意文件路径、链接顺序,利用诊断信息和文档,保持工具更新以优化使用。
|
3月前
|
缓存 Linux Windows
【Linux】开始认识软硬链接
上篇文章我们学习了文件系统,了解未打开的文件在磁盘中是如何储存的。
30 4
|
2月前
|
Linux 编译器 数据处理
探索Linux中的`ld`命令:链接器的奥秘
`ld`是Linux中的链接器,将编译后的目标文件链接成可执行或共享库。作为GNU Binutils的一部分,它处理符号引用、重定位,支持多种文件格式和库类型。常用参数包括`-o`指定输出文件,`-l`链接库,`-L`指定搜索路径。编译器驱动程序常用于简化链接过程。理解符号解析、选择静态/动态链接及管理库版本是使用`ld`的关键。使用`--verbose`调试链接问题,并考虑优化选项。
|
2月前
|
Linux 编译器 数据处理
深入了解Linux命令ld.gold:快速链接器的奥秘
`ld.gold`是GNU的快速链接器,设计用于加速大型项目的链接,尤其擅长并行处理和增量链接。它与标准的`ld`高度兼容,可通过`-fuse-ld=gold`选项启用。例如,`gcc -o my_program file1.c file2.c file3.c -Wl,--ld-as-needed -fuse-ld=gold`命令使用`ld.gold`链接多个源文件。最佳实践包括确保环境支持、利用多线程和启用增量链接。
|
3月前
|
Linux Windows
【Linux】详解软硬链接
【Linux】详解软硬链接
|
1月前
|
网络协议 Ubuntu Linux
|
1月前
|
存储 Linux Windows
【Linux】文件系统软硬链接的那些事儿
本文介绍了Linux文件系统的磁盘结构、逻辑结构以及文件和inode的关系,重点讲解了软链接和硬链接的区别,强调了它们在文件管理中的作用。
43 7
|
2月前
|
Shell Linux 网络安全
Linux怎样在使用ssh 链接时就指定gcc 的版本
Linux怎样在使用ssh 链接时就指定gcc 的版本
32 7
|
2月前
|
Linux vr&ar C语言
Linux怎样更新Centos下Gcc版本支持C17?Centos7快速安装gcc8.3.1 可支持C++17(附gcc相关链接整理)
Linux怎样更新Centos下Gcc版本支持C17?Centos7快速安装gcc8.3.1 可支持C++17(附gcc相关链接整理)
151 2
|
2月前
|
Ubuntu JavaScript Linux
linux 命令行下载BT种子和磁力链接 ubuntu linux 命令行下载种子
linux 命令行下载BT种子和磁力链接 ubuntu linux 命令行下载种子
439 3