mmap区域上的memset / memcpy失败

问题描述 投票:1回答:1

我正在尝试从另一个程序加载静态链接的程序并执行它。我的步骤是:

  • 解析ELF
  • 从程序头中解析段
  • 对于每个PT_LOAD
  • 加载它
  • 跳转到起始地址

如果elf_bytes是mmap的ELF文件,则加载PT_LOAD段是load(&p, elf_bytes + p.p_offset)

加载功能:

int load(const Elf64_Phdr *phdr, const void *elf_bytes_for_phdr) {
    fprintf(stderr, "loading phdr of type %x from 0x%x to +=%zu bytes\n", phdr->p_type, phdr->p_vaddr, phdr->p_memsz);
    const size_t pagesize = getpagesize();
    const size_t unaligned_bytes = phdr->p_vaddr % pagesize;

    void *base_addr = phdr->p_vaddr - unaligned_bytes;
    size_t total_bytes = phdr->p_memsz + unaligned_bytes;

    void *region = mmap(
            base_addr,
            total_bytes,
            phdr->p_flags,
            MAP_FIXED | MAP_PRIVATE | MAP_ANONYMOUS,
            0, 0
    );
    if (region != MAP_FAILED) {
        memset(region, 0, unaligned_bytes);
        // return memcpy(region + unaligned_bytes, elf_bytes_for_phdr, phdr->p_filesz) == region + unaligned_bytes;
        return memset(region + unaligned_bytes, /*elf_bytes_for_phdr*/0, 1) == region + unaligned_bytes;
    }
    return 1;
}

memset和memcpy都失败了;内核在地址0x400000发送一个SIGSEGV,恰好是region。那里什么都没有:

gdb$ shell pmap 10751
00007ff000000000      8K r-x-- ld_simple_loader
00007ff000201000      4K r---- ld_simple_loader
00007ff000202000      4K rw--- ld_simple_loader
00007ffff79e4000   1948K r-x-- libc-2.27.so
00007ffff7bcb000   2048K ----- libc-2.27.so
00007ffff7dcb000     16K r---- libc-2.27.so
00007ffff7dcf000      8K rw--- libc-2.27.so
00007ffff7dd1000     16K rw---   [ anon ]
00007ffff7dd5000    156K r-x-- ld-2.27.so
00007ffff7fdc000      8K rw---   [ anon ]
00007ffff7ff7000     12K r----   [ anon ]
00007ffff7ffa000      8K r-x--   [ anon ]
00007ffff7ffc000      4K r---- ld-2.27.so
00007ffff7ffd000      4K rw--- ld-2.27.so
00007ffff7ffe000      4K rw---   [ anon ]
00007ffffffde000    132K rw---   [ stack ]
ffffffffff600000      4K r-x--   [ anon ]
 total             4384K

因为加载器从一个非常高的地址开始(实际上是为了避免这个问题)。这通过与-Wl,-Ttext-segment=00007ff000000000连接。

(我也尝试过首先对地区进行大规模的拍摄。)

c linux elf mmap
1个回答
1
投票
void *region = mmap(
        base_addr,
        total_bytes,
        phdr->p_flags,
        MAP_FIXED | MAP_PRIVATE | MAP_ANONYMOUS,
        0, 0
);

mmap的第一段通常涵盖.text并具有R-X(但没有写入)权限。

没有.p_flags中的写入权限,尝试写入该内存(自然)会失败。

你可能想用phdr->p_flags | PROT_WRITE代替。

注意:某些安全策略(例如SELinux)不允许可写和可执行映射。在这样的系统上,您需要使用PROT_WRITE映射内存,根据需要复制数据,然后使用所需保护的mprotect

© www.soinside.com 2019 - 2024. All rights reserved.