如何告诉/强制GNU ld将一个节/符号放在输出ELF文件的特定部分?

ers*_*zcz 13 linker elf ld multiboot

这将是一个很长的,所以抓住一些咖啡/茶/ yerba.

摘要

如何告诉/强制GNU ld将一个节/符号放在输出ELF文件的特定部分?

具体来说,我不是询问符号的物理/加载地址,而是询问文件中的偏移量.

背景

我正在尝试使用真实世界的BSD内核并使其与Multiboot兼容并由GRUB引导.为了使ELF图像与Multiboot兼容并且可以通过GRUB识别,需要将幻数(0x1BADB002)嵌入到文件的第一个8KiB中.

我指的是Multiboot 0.6.96.

由于原始内核是相当大的代码,我将使用基于OSDev wikiBare Bones内核的示例.由于该内核已经符合Multiboot标准,因此我将使用 extra_symbolwith值0xCAFEBABE作为我的问题的示例.

boot.s 包含:

.set CAFEBABE, 0xCAFEBABE
; ... snip ...
.section .extras
extra_symbol:
    .long CAFEBABE
Run Code Online (Sandbox Code Playgroud)

额外的符号 .text

最简单的选择是将具有该值的符号放在.text 其他任何内容之前:

.text BLOCK(4K) : ALIGN(4K)
{
    *(.extras)
    *(.multiboot)
    *(.text)
}
Run Code Online (Sandbox Code Playgroud)

这对于示例来说很好,但是如果真正的BSD内核.text 在文件中的0x2000(8KiB)之后启动.这种方法不是一种选择.

之前的额外部分.text

另一种选择是将整个.extras部分放在前面.text.在我的天真,我希望.text在链接器脚本之前放置这样的部分也会使它出现在输出ELF的前面:

SECTIONS
{
    // ... some stuff ...

    .extras : { *(.extras) }

    .text BLOCK(4K) : ALIGN(4K)
    {
        *(.multiboot)
        *(.text)
    }

    // ... more stuff ...

}
Run Code Online (Sandbox Code Playgroud)

但事实并非如此:

$ i586-elf-readelf -S myos.bin 
There are 10 section headers, starting at offset 0x6078:

Section Headers:
  [Nr] Name              Type            Addr     Off    Size   ES Flg Lk Inf Al
  [ 0]                   NULL            00000000 000000 000000 00      0   0  0
  [ 1] .extras           PROGBITS        00100000 006010 000004 00      0   0  1
  [ 2] .comment          PROGBITS        00000000 006014 000011 01  MS  0   0  1
  [ 3] .text             PROGBITS        00001000 001000 0001e4 00  AX  0   0 4096
  [ 4] .rodata.str1.1    PROGBITS        000011e4 0011e4 000016 01 AMS  0   0  1
  [ 5] .eh_frame         PROGBITS        000011fc 0011fc 000104 00   A  0   0  4
  [ 6] .bss              PROGBITS        00002000 002000 004010 00  WA  0   0 4096
  [ 7] .shstrtab         STRTAB          00000000 006025 000050 00      0   0  1
  [ 8] .symtab           SYMTAB          00000000 006208 000210 10      9  19  4
  [ 9] .strtab           STRTAB          00000000 006418 000130 00      0   0  1
Key to Flags:
  W (write), A (alloc), X (execute), M (merge), S (strings)
  I (info), L (link order), G (group), T (TLS), E (exclude), x (unknown)
  O (extra OS processing required) o (OS specific), p (processor specific)
Run Code Online (Sandbox Code Playgroud)

事实上,该部分首先出现在节头表中,但它在文件(0x6010)中的偏移量就在之后.text.事实上,它甚至在之后.bss.

这个问题

ELF规范明确指出:

尽管该图显示紧接在ELF头之后的程序头表,以及在这些部分之后的节头表,但实际文件可能不同.此外,部分和细分没有指定的顺序.只有ELF标题在文件中具有固定位置.

我如何利用它并告诉GNU ld将我的extra_symbol (可能是整个.extras部分)放在文件中的任何其他部分之前,最好是在ELF标题之后?

Flo*_*mer 2

为了将该节放在前面.text,您是否已尝试将其分配(A)和可执行(X)?

如果.interp黑客成功了,那当然也很好。