静态精灵图像设置的段 %fs 在哪里?

Where is segment %fs for static elf images setup?

我想知道 %fs 寄存器是如何初始化的 手工创建精灵图像时。

我想要 运行 的简单片段是:

        .text
        nop
        movq %fs:0x28, %rax;
1:      jmp 1b

应该在 %fs 段中的偏移量 0x28 处读取。通常这是存储 stack canary 的地方。因为我手动创建了 elf 图像,所以我的代码根本没有设置 %fs 段,这预期会失败(?)。

以下是我创建小精灵图像的方法:

0000000000000000 <.text>:
   0:   90                      nop
   1:   64 48 8b 04 25 28 00    mov    %fs:0x28,%rax
   8:   00 00 
   a:   eb fe                   jmp    0xa

我通过

创建 .text 段
echo 9064488b042528000000ebfe | xxd -r -p > r2.bin

然后我转换成精灵:

ld -b binary -r -o raw.elf r2.bin
objcopy  --rename-section .data=.text --set-section-flags .data=alloc,code,load raw.elf

那时 raw.elf 包含我的说明。然后我 link 与 ld -T raw.ld -o out.elf -M --verbose 其中 raw.ld 是:

OUTPUT_FORMAT("elf64-x86-64", "elf64-x86-64", "elf64-x86-64")
OUTPUT_ARCH(i386:x86-64)
ENTRY(_entry)
PHDRS {
phdr4000000 PT_LOAD;
}
SECTIONS
{
  _entry = 0x4000000;
  .text 0x4000000 : { raw.elf (.text) } :phdr4000000
}

我现在可以使用 gdb 启动 out.elf:

gdb --args out.elf

并在 0x4000000 设置断点:

(gdb)break *0x4000000
(gdb)run

第一个 nop 可以通过 stepi 步进,但是堆栈金丝雀读取 mov %fs:0x28,%rax 段错误。

我想这是预料之中的,因为 OS 可能没有设置 %fs。 对于使用 gcc --static m.c -o m 编译的简单 m.c: int main() { return 0; } 程序,我可以从 %fs 读取。添加:

long can()
{
        long v = 0;
        __asm__("movq %%fs:0x28, %0;"
                : "=r"(val)::);
        return v;
}

让我从 %fs 读取 - 尽管我怀疑 %fs:28 是否已设置,因为 ld.so 不是 运行(它是静态图像)。

问题:

谁能指出在 static 图像的 c 运行 时间设置 %fs 的位置?

您需要先使用 ARCH_SET_FS 参数调用 arch_prctl,然后才能使用 %fs 段前缀。您将必须在某处分配后备存储(brkmmap 或堆栈的其他未使用部分)。

glibc 在 __libc_setup_tls in csu/libc-tls.c 中为静态链接的二进制文件执行此操作,隐藏在 TLS_INIT_TP 宏之后。