2

我正在学习ELF。我想在链接共享库时找到位置相关的可执行文件和位置无关的可执行文件之间的elf格式的区别。

但是在与共享库链接时,我无法为可执行文件生成与位置相关的代码。

/*Lib.c*/
static int a;
extern int b;
int c=1;
extern void exit();
void set_value()
{
    a=1;
    b=1;
    c=1;
}
void run()
{
    set_value();
    exit();
}

拳头,使用 gcc 生成共享动态库:

gcc -m32 -nostdlib -o Lib.so Lib.c

请注意,我不使用 -fpic 为 Lib.so 生成与位置无关的代码。

现在,我有另一个文件 main.c 需要与 Lib.so 链接:

/*main.c*/
extern void run();
int b=2;
void nomain()
{
    run();
}
void exit()
{
    asm("int $0x80 \n\t"
        ::"a"(1),"b"(42));

}

使用以下命令将 main.c 与 Lib.so 链接:

gcc  -m32 -e nomain  -nostartfiles  -fno-builtin  -o a.out main.c ./Lib.so

但是,当与共享库链接时,gcc 默认会将 main.c 编译为与位置无关的代码,即使此库不使用 pic。

我想知道 gcc 是否有一些选项可以为可执行文件生成位置相关?

我发布了 Lib.so 和 a.out 的部分信息。我们可以看到 a.out 中有 '.plt' 和 '.got.plt' 部分,这意味着 a.out 使用 PIC。

  /*Section for Lib.so*/
  Section Headers:
  [Nr] Name              Type            Addr     Off    Size   ES Flg Lk Inf Al
  [ 0]                   NULL            00000000 000000 000000 00      0   0  0
  [ 1] .gnu.hash         GNU_HASH        000000b4 0000b4 00003c 04   A  2   0  4
  [ 2] .dynsym           DYNSYM          000000f0 0000f0 000090 10   A  3   1  4
  [ 3] .dynstr           STRTAB          00000180 000180 000030 00   A  0   0  1
  [ 4] .rel.dyn          REL             000001b0 0001b0 000028 08   A  2   0  4
  [ 5] .text             PROGBITS        000001d8 0001d8 000033 00  AX  0   0  4
  [ 6] .dynamic          DYNAMIC         0000120c 00020c 000078 08  WA  3   0  4
  [ 7] .got.plt          PROGBITS        00001284 000284 00000c 04  WA  0   0  4
  [ 8] .data             PROGBITS        00001290 000290 000004 00  WA  0   0  4
  [ 9] .bss              NOBITS          00001294 000294 000004 00  WA  0   0  4
  [10] .comment          PROGBITS        00000000 000294 00002e 00      0   0  1
  [11] .shstrtab         STRTAB          00000000 0002c2 00006a 00      0   0  1
  [12] .symtab           SYMTAB          00000000 00055c 000170 10     13  15  4
  [13] .strtab           STRTAB          00000000 0006cc 000057 00      0   0  1
  /*Section for a.out*/
  Section Headers:
  [Nr] Name              Type            Addr     Off    Size   ES Flg Lk Inf Al
  [ 0]                   NULL            00000000 000000 000000 00      0   0  0
  [ 1] .interp           PROGBITS        080480f4 0000f4 000013 00   A  0   0  1
  [ 2] .gnu.hash         GNU_HASH        08048108 000108 000034 04   A  3   0  4
  [ 3] .dynsym           DYNSYM          0804813c 00013c 000070 10   A  4   1  4
  [ 4] .dynstr           STRTAB          080481ac 0001ac 000037 00   A  0   0  1
  [ 5] .rel.plt          REL             080481e4 0001e4 000008 08   A  3   6  4
  [ 6] .plt              PROGBITS        080481ec 0001ec 000020 04  AX  0   0  4
  [ 7] .text             PROGBITS        0804820c 00020c 000020 00  AX  0   0  4
  [ 8] .dynamic          DYNAMIC         0804922c 00022c 000090 08  WA  4   0  4
  [ 9] .got.plt          PROGBITS        080492bc 0002bc 000010 04  WA  0   0  4
  [10] .data             PROGBITS        080492cc 0002cc 000004 00  WA  0   0  4
  [11] .comment          PROGBITS        00000000 0002d0 00002e 00      0   0  1
  [12] .shstrtab         STRTAB          00000000 0002fe 00006d 00      0   0  1
  [13] .symtab           SYMTAB          00000000 0005c4 000160 10     14  15  4
  [14] .strtab           STRTAB          00000000 000724 000051 00      0   0  1
4

1 回答 1

0

你应该Lib.sogcc -Wall -Wextra -m32 -shared -Wl,-soname,Lib.so -o Lib.so Lib.c.

您在这里所做的只是生成一个没有重定位信息的正常程序。

查看本教程以了解有关动态库的更多信息。

注意:另外,不要忘记设置LD_LIBRARY_PATH环境变量以指向Lib.so.

于 2013-04-21T07:17:57.310 回答