gcc 指定入口函数

本文讨论了在C++项目中遇到的测试代码与主函数冲突的问题,详细解释了如何通过更改主函数名称、使用GCC命令行选项以及调整退出方式来解决这一问题。同时提供了错误排查步骤和解决方案。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

近来写个测试代码

 

int main(int argc,char *argv[])
{
 int test_ret=0 ;

 test_ret += test_xxx1();

 test_ret +=test_xxxx1();


 printf("Total: %d functions, %d tests, %d errors\n",function_test, call_test, test_ret);

 return(test_ret);

}


结果影响到正常编译了,因为测试代码 的 TEST.C 文件会被 make自动找到,两个 main 函数冲突了

 

想起ARM 下可以各种方式指定 ENTRY 函数的

 

网上找了下,gcc 有两种方式 gcc -e entryfunc   gcc --Entry=entryfunc 事实上后一种试验时说是过时了

 

把main改成testmain

 

int testmain(int argc,char *argv[])
{
 int test_ret=0 ;

 test_ret += test_xxx1();

 test_ret +=test_xxxx1();


 printf("Total: %d functions, %d tests, %d errors\n",function_test, call_test, test_ret);

 return(test_ret);

}


 

gcc -o testapp -e testmain *.c

报以下错 

(.text+0x18): undefined reference to `main'
collect2: ld returned 1 exit status

 

原因是gcc 自动链ctrl.o 文件,这个ctrl代码中引用了 main

另加参数 -nostartfiles 忽略之

gcc -o testapp -e  -nostartfiles  testmain *.c

 

但执行时testmain返回时报错,是因为返回时少了必要清理代码

 

修改 testmain  把 return 改成 exit

 

int testmain(int argc,char *argv[])
{
 int test_ret=0 ;

 test_ret += test_xxx1();

 test_ret +=test_xxxx1();


 printf("Total: %d functions, %d tests, %d errors\n",function_test, call_test, test_ret);

 exit(test_ret);

}




 


 

 


 

 

<think>好的,我现在需要帮助用户了解GCC编译生成的镜像文件的入口地址是如何确定的,特别是ELF文件中的entry point。首先,我需要回忆一下ELF文件的结构和相关知识。根据用户提供的引用内容,尤其是引用[2]和引用[3],ARM编译器生成ELF格式的镜像文件,其中ELF头部包含e_entry字段,这个字段指定了程序的入口地址。 用户的问题集中在GCC编译生成的镜像文件的入口地址位置,也就是ELF文件的entry point如何确定。首先,我应该解释ELF文件的结构,特别是ELF头部中的e_entry字段的作用。引用[3]提到,e_entry字段是程序的进入点,病毒可能会修改这个值,这说明这个字段的重要性。 接下来,我需要说明入口地址是如何计算的。根据引用[1],入口地址是文件中的偏移加上装载地址。例如,如果入口在文件中的偏移是0x4f4,装载地址是0x4200000,那么入口地址就是0x42004f4。这说明入口地址的计算涉及链接脚本中的内存布局设置,特别是.text段的起始地址加上入口点的偏移量。 然后,我需要提到链接脚本的作用。在GCC编译过程中,链接脚本(通常是默认的或自定义的)决定了各个段(如.text、.data)在内存中的位置。入口点通常指向.text段的起始位置,这是由链接脚本中的ENTRY命令指定的,默认情况下是_start符号的地址。 另外,用户可能想知道如何查看ELF文件的入口地址,这里需要介绍readelf工具的使用,比如使用readelf -h命令来查看ELF头部的信息,其中会明确显示Entry point address。 可能还需要考虑不同架构或平台的情况,比如ARM映像文件的例子中,入口地址的计算方式是否与x86有所不同,但根据引用[2],ARM的镜像文件结构类似,因此基本逻辑应该相同。 最后,要确保回答符合用户的要求,结构清晰,分步骤解释,并引用提供的参考资料。同时,生成的相关问题需要围绕ELF文件结构、入口地址修改、链接脚本的作用以及查看入口地址的方法展开。</think>### GCC编译生成的镜像文件入口地址确定方法 ELF(Executable and Linkable Format)文件的入口地址由**ELF头部中的`e_entry`字段**直接定义,其值取决于**链接脚本配置**和**目标平台的内存映射规则**。具体步骤如下: --- #### 1. **ELF头部与入口字段** ELF文件头部(ELF Header)包含程序入口地址字段`e_entry`,该字段值为程序加载到内存后第一条执行的指令地址[^3]。通过`readelf -h <file>`命令可查看: ```bash $ readelf -h hello ELF Header: Entry point address: 0x400540 ``` --- #### 2. **入口地址的生成逻辑** - **链接脚本控制**:入口地址由链接脚本(Linker Script)中的`ENTRY()`命令指定。若未显式定义,默认使用符号`_start`的地址[^2]。 - **内存映射叠加**:实际入口地址 = **代码段(.text)的装载地址(VMA)** + **入口符号在文件中的偏移量**。例如: - 代码段装载地址:`0x400000` - `_start`符号偏移:`0x540` - 最终入口地址:`0x400540`[^1] --- #### 3. **自定义入口地址** 在链接脚本中可强制指定入口点: ```ld ENTRY(my_custom_entry) SECTIONS { . = 0x8000; /* 代码段装载地址 */ .text : { *(.text) } } ``` 此时`e_entry`将被设为`my_custom_entry`符号的绝对地址。 --- #### 4. **动态链接程序的特殊行为** 动态可执行文件(如Linux程序)的入口地址通常指向**动态链接器的初始化代码**,而非用户定义的`main()`函数。例如: ``` Entry point address: 0x7f7c4d4e0100 ``` ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值