关于程序的入口函数(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 puts@GLIBC_2.2.5 (2)
     2: 0000000000000000     0 FUNC    GLOBAL DEFAULT  UND exit@GLIBC_2.2.5 (2)
...
//查看elf文件 参数只有一个,printf 被优化成了puts

$ readelf -al a.out | grep FUN
     1: 0000000000000000     0 FUNC    GLOBAL DEFAULT  UND puts@GLIBC_2.2.5 (2)
     2: 0000000000000000     0 FUNC    GLOBAL DEFAULT  UND exit@GLIBC_2.2.5 (2)
    22: 0000000000000000     0 FUNC    GLOBAL DEFAULT  UND puts@@GLIBC_2.2.5
    25: 0000000000400390    24 FUNC    GLOBAL DEFAULT   10 _start
    27: 0000000000000000     0 FUNC    GLOBAL DEFAULT  UND exit@@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 puts@GLIBC_2.2.5 (2)
     2: 0000000000000000     0 FUNC    GLOBAL DEFAULT  UND exit@GLIBC_2.2.5 (2)
    22: 0000000000000000     0 FUNC    GLOBAL DEFAULT  UND puts@@GLIBC_2.2.5
    24: 0000000000400390    24 FUNC    GLOBAL DEFAULT   10 mymain
    27: 0000000000000000     0 FUNC    GLOBAL DEFAULT  UND exit@@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

三 关于 uboot start.s
 

通过连接脚本,把 start.o  放在代码段的 最开始的地方,让他好最先执行
SECTIONS
{
...
.text:
{
start.o(.text)
misc.o(.text)
*.text
*.text.*
}
...

  • 10
    点赞
  • 35
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 你好!根据你提供的信息,我理解你的问题是在函数'_start'中,出现了错误提示:(.text+0x20): 对'main'未定义的引用。 这个错误提示意味着在程序中,有一处地方调用了'main'函数,但是编译器无法找到'main'函数的定义。这通常发生在以下几种情况下: 1. 你的代码中没有定义'main'函数,或者'main'函数定义存在问题,导致编译器无法识别它。 2. 在编译时,你没有将所有需要的源文件都包含在编译命令中,导致编译器无法找到'main'函数的定义。 3. 你的代码中调用了一个其他源文件中定义的'main'函数,但是在编译时没有将该源文件包含在编译命令中,导致编译器无法找到'main'函数的定义。 为了解决这个问题,你需要检查你的代码中是否定义了'main'函数,并确保在编译时将所有需要的源文件都包含在编译命令中。如果你确认代码中确实没有定义'main'函数,那么你需要创建一个'main'函数来作为程序入口点。 希望这个回答能够帮到你!如果你还有其他问题,请随时问我。 ### 回答2: 这个错误通常会在编译C语言程序时出现,它意味着编译器在连接程序时找不到main()函数的实现。在C语言程序中,main()函数程序入口点,因此编译器需要在连接程序时找到该函数的实现。如果找不到,编译器就会抛出这个错误。 在解决这个错误之前,我们需要明确一些概念。在C语言中,程序往往由多个源代码文件组成,每个文件都可以包含多个函数定义。在编译每个源文件时,编译器会将每个函数的实现编译成目标代码,但并不会将这些函数的实现连接到最终的目标文件中。相反,连接器会负责将所有的目标代码组合成一个可执行文件。 因此,在遇到对‘main’未定义的引用错误时,我们需要检查以下几个方面: 1. 确定程序中是否有main()函数的实现。如果没有定义,在程序的任何位置都无法访问该函数。因此,首先需要确保程序中包含main()函数并且已经正确实现。 2. 确定所有的源代码文件是否都被编译为目标代码。如果缺少任何一个源文件的目标代码,编译器将无法连接所有的函数定义,导致出现对‘main’未定义的引用错误。 3. 确定编译器和连接器的参数是否正确。编译器和连接器都需要正确的参数才能正常工作。可能需要检查编译器和连接器的选项,以确保它们被正确设置。 4. 检查库文件是否正常。库文件通常包含外部函数的实现,如果库文件损坏或不可用,编译器将无法找到相应的函数定义,从而导致错误。 综上所述,如果出现对‘main’未定义的引用错误,需要检查程序中是否包含main()函数的实现,所有的源文件是否都被编译为目标代码,编译器和连接器的参数是否正确,以及库文件是否正常。正确处理这些问题,即可成功解决这个错误。 ### 回答3: 这个错误是由于编译器无法找到定义为主函数的“main”的原因所致。在C程序中,主函数程序入口点,如果没有定义主函数程序无法正常运行。通常,当编写C程序时,都需要包含一个名为“main”的函数,并在其中编写程序入口点。 当编译器在链接时找不到主函数的定义时,就会返回这个错误。解决这种错误的方法是确保程序源代码中有正确定义的“main函数,并且该函数程序中作为入口点。 在实际编程中,有一些常见的错误可能导致这个问题。其中一些包括: 1. 在编译时未将源文件添加到编译器中,导致链接时找不到主函数的定义。 2. 没有正确地定义主函数的参数,即int main()或int main(int argc, char *argv[])。 3. 主函数使用了错误的命名方式或拼写不正确,例如写成“master”而不是“main”。 4. 主函数定义在错误的位置,例如定义在其他函数内部而不是整个程序的顶部。 解决这个问题的方法是检查程序源代码,确保有正确定义的主函数,并检查是否存在上述常见错误。另外,还可以尝试重新编译程序,并指定正确的源文件和编译选项,以便编译器可以正确地找到主函数的定义。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值