最佳答案
您可以通过将构造函数放在.init_array部分中来完成此操作.使用相同的参数调用.init_array中的函数(与.init不同)将调用main:argc,argv和env.
这是一个简单的例子.我使用LD_PRELOAD只是为了避免使用实际链接和使用共享库的代码使示例复杂化,但它也可以在更正常的场景中工作.
file:printargs.c
#include dio.h>
static int printargs(int argc,char** argv,char** env) {
puts("In printargs:");
for (int i = 0; i < argc; ++i)
printf(" Arg %d (%p) '%s'\n",i,(void*)argv[i],argv[i]);
return 0;
}
/* Put the function into the init_array */
__attribute__((section(".init_array"))) static void *ctr = &printargs;
构建和使用共享库
(如果使用-Wall,您将看到警告,因为未使用ctr.)
$gcc -o printargs.so -std=c11 -shared -fpic printargs.c
$LD_PRELOAD=./printargs.so /bin/echo Hello,world.
In printargs:
Arg 0 (0x7ffc7617102f) '/bin/echo'
Arg 1 (0x7ffc76171039) 'Hello,'
Arg 2 (0x7ffc76171040) 'world.'
Hello,world.
这个解决方案来自Mike Frysinger in the libc-help mailing list的建议,这个答案here on SO还有一个更简洁的版本.