如何使用gcc更改C程序的入口点?

发布于 2024-12-05 15:31:39 字数 170 浏览 1 评论 0原文

如何更改使用 gcc 编译的 C 程序的入口点?
就像下面的代码一样

#include<stdio.h>
int entry()  //entry is the entry point instead of main
 {
   return 0;
 }

How to change the entry point of a C program compiled with gcc ?
Just like in the following code

#include<stdio.h>
int entry()  //entry is the entry point instead of main
 {
   return 0;
 }

如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。

扫码二维码加入Web技术交流群

发布评论

需要 登录 才能够评论, 你可以免费 注册 一个本站的账号。

评论(5

香橙ぽ 2024-12-12 15:31:39

这是一个链接器设置:

-Wl,-eentry

-Wl,... 事物将参数传递给链接器,链接器采用 -e 参数来设置入口函数

It's a linker setting:

-Wl,-eentry

the -Wl,... thing passes arguments to the linker, and the linker takes a -e argument to set the entry function

未蓝澄海的烟 2024-12-12 15:31:39

如果您使用的系统提供 GNU Binutils(例如 Linux),
您可以使用 objcopy 命令
使任意函数成为新的入口点。

假设一个名为 program.c 的文件包含 entry 函数:

$ cat > program.c
#include <stdio.h>
int entry()
{
    return 0;
}
^D
  1. 首先使用 -c 编译它以生成可重定位目标文件:

    <前><代码>$ gcc -c 程序.c -o 程序.o

  2. 然后将 entry 重新定义为 main< /代码>:

    $ objcopy --redefine-sym 条目=主程序.o
    

  3. 现在使用 gcc 编译新的目标文件:

    $ gcc 程序.o -o 程序
    

注意:如果您的程序已经有一个名为 main 的函数,在步骤 2 之前,您可以执行单独的 objcopy 调用:

objcopy --redefine-sym oldmain=main program.o

If you are on a system that provides GNU Binutils (like Linux),
you can use the objcopy command
to make an arbitrary function the new entry point.

Suppose a file called program.c containing the entry function:

$ cat > program.c
#include <stdio.h>
int entry()
{
    return 0;
}
^D
  1. You first compile it using -c to generate a relocatable object file:

    $ gcc -c program.c -o program.o
    
  2. Then you redefine entry to be main:

    $ objcopy --redefine-sym entry=main program.o
    
  3. Now use gcc to compile the new object file:

    $ gcc program.o -o program
    

NOTE: If your program already has a function called main, before step 2, you can perform a separate objcopy invocation:

objcopy --redefine-sym oldmain=main program.o
ぽ尐不点ル 2024-12-12 15:31:39

您可以将源代码修改为:

#include<stdio.h>

const char my_interp[] __attribute__((section(".interp"))) = "/lib/ld-linux.so.2";

int entry()  //entry is the entry point instead of main
{
   exit(0);
}

“.interp”部分将使您的程序能够调用外部共享库。
exit 调用将使您的入口函数退出程序而不是返回。

然后将程序构建为可执行的共享库:

$ gcc -shared -fPIC -e entry test_main.c -o test_main.so
$ ./test_main

You can modify your source code as:

#include<stdio.h>

const char my_interp[] __attribute__((section(".interp"))) = "/lib/ld-linux.so.2";

int entry()  //entry is the entry point instead of main
{
   exit(0);
}

The ".interp" section will let your program able to call external shared library.
The exit call will make your entry function to exit program instead of return.

Then build the program as a shared library which is executable:

$ gcc -shared -fPIC -e entry test_main.c -o test_main.so
$ ./test_main
乱世争霸 2024-12-12 15:31:39

最小可运行示例和其他答案的注释

main.c

#include <stdio.h>
#include <stdlib.h>

int mymain(void) {
    puts("hello");
}

编译并运行:

gcc -nostartfiles -Wl,--entry=mymain -o main.out main.c
# or -Wl,-emymain
./main.out 1 2 3

注释:

  • 没有 -nostartfiles,链接失败并显示:

    /usr/lib/gcc/x86_64-linux-gnu/7/../../../x86_64-linux-gnu/Scrt1.o:在函数 `_start' 中:
    (.text+0x20):对“main”的未定义引用
    collect2:错误:ld 返回 1 退出状态
    

    可能是因为 glibc 设置代码在 main 之前运行 _start 通常调用 main

  • 命令行参数不是为您设置的,大概是因为它们将由在 main 之前运行的 glibc 代码设置,因此尝试使用它们会打印未定义的值。我还没有找到适合他们的方法。

另一个选择是删除 -nostartfiles ,只定义一个虚拟 main:

#include <stdio.h>
#include <stdlib.h>

int main(void) {}

int mymain(int argc, char **argv) {
    (void)argv;
    printf("%d\n", argc);
    exit(0);
}

,然后编译并运行良好:

gcc -nostartfiles -Wl,--entry=mymain -o main.out main.c
# or -Wl,-emymain
./main.out 1 2 3

CLI args 仍然损坏,但也许更多的东西 stdlib 的东西会像这样工作。

在 Ubuntu 23.10 中测试。

Minimal runnable example and notes on other answers

main.c

#include <stdio.h>
#include <stdlib.h>

int mymain(void) {
    puts("hello");
}

compile and run:

gcc -nostartfiles -Wl,--entry=mymain -o main.out main.c
# or -Wl,-emymain
./main.out 1 2 3

The notes:

  • without -nostartfiles, the link fails with:

    /usr/lib/gcc/x86_64-linux-gnu/7/../../../x86_64-linux-gnu/Scrt1.o: In function `_start':
    (.text+0x20): undefined reference to `main'
    collect2: error: ld returned 1 exit status
    

    presumably because the glibc setup code that runs before main in _start normally calls main.

  • command line arguments are not setup for you, presumably because they would be setup by the glibc code that runs before main, so trying to use them prints undefined values. I haven't found a method that works for them.

Another option is to remove -nostartfiles is to just define a dummy main:

#include <stdio.h>
#include <stdlib.h>

int main(void) {}

int mymain(int argc, char **argv) {
    (void)argv;
    printf("%d\n", argc);
    exit(0);
}

which then compile and runs fine with:

gcc -nostartfiles -Wl,--entry=mymain -o main.out main.c
# or -Wl,-emymain
./main.out 1 2 3

CLI args are still broken, but perhaps more things stdlib things will work like this.

Tested in Ubuntu 23.10.

花开半夏魅人心 2024-12-12 15:31:39

如果您在托管环境(即 Linux、Windows 等)上执行此操作,那么使用链接器重新定义哪个函数将成为 main() 可能是一个更好的主意:这样

-Wl,--defsym=main=some_other_entry_point

,启动时调用的函数仍然是 main(),因此操作系统的 _start() 代码仍将正常工作。如果您尝试使用 --entry 实际更改入口点名称,您可能会遇到很多问题,因为操作系统的 _start不会被调用。

示例:

#include <stdio.h>

int main(void) {
    printf("wrong\n");
}

int some_other_entry_point(void) {
    printf("right\n");
}
gcc -Wl,--defsym=main=some_other_entry_point -o test test.c
$ ./test
right

https://godbolt.org/z/aWGYEMhaE

If you're doing this on a hosted environment (i.e. Linux, Windows etc.) it's probably a much better idea to use the linker to redefine which function becomes main() with this:

-Wl,--defsym=main=some_other_entry_point

This way, the function being called at startup is still main(), so your operating system's _start() code will still work as normal. If you try to actually change the entry point name with --entry, you'll probably run into a lot of issues because the OS's _start won't be called.

Example:

#include <stdio.h>

int main(void) {
    printf("wrong\n");
}

int some_other_entry_point(void) {
    printf("right\n");
}
gcc -Wl,--defsym=main=some_other_entry_point -o test test.c
$ ./test
right

https://godbolt.org/z/aWGYEMhaE

~没有更多了~
我们使用 Cookies 和其他技术来定制您的体验包括您的登录状态等。通过阅读我们的 隐私政策 了解更多相关信息。 单击 接受 或继续使用网站,即表示您同意使用 Cookies 和您的相关数据。
原文