-
-
Save kubo/e456246df93b4cc95c9e47e9c3db8835 to your computer and use it in GitHub Desktop.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
#include "libaddone.h" | |
#if defined __pic__ || defined __PIC__ | |
#error Compiled as PIC | |
#endif | |
int addone_no_pic(int i) | |
{ | |
return addone(i); | |
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
#include "libaddone.h" | |
#if !(defined __pic__ || defined __PIC__) | |
#error Compiled not as PIC | |
#endif | |
int addone_pic(int i) | |
{ | |
return addone(i); | |
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
#include "libaddone.h" | |
int addone(int i) | |
{ | |
return i + 1; | |
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
#ifndef LIBADDONE_H | |
#define LIBADDONE_H 1 | |
int addone(int); | |
#endif |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
#include <stdio.h> | |
#include <string.h> | |
#include "plthook.h" | |
extern int addone_pic(int i); | |
extern int addone_no_pic(int i); | |
int main() | |
{ | |
plthook_t *plthook; | |
unsigned int pos = 0; | |
const char *name; | |
void **addr; | |
printf("call from pic: %d\n", addone_pic(10)); | |
printf("call from no_pic: %d\n", addone_no_pic(10)); | |
plthook_open(&plthook, NULL); | |
while (plthook_enum(plthook, &pos, &name, &addr) == 0) { | |
if (strstr(name, "addone") != NULL) { | |
printf("%p:%s\n", *addr, name); | |
} | |
} | |
plthook_close(plthook); | |
return 0; | |
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
OBJS = main.o addone_no_pic.o addone_pic.o plthook_elf.o | |
all: prog | |
prog: $(OBJS) libaddone.so | |
$(CC) $(CFLAGS) $(LDFLAGS) -o $@ $(OBJS) -L. -laddone -Wl,-rpath,'$$ORIGIN' -ldl | |
libaddone.so: libaddone.c | |
$(CC) -shared -o $@ $< | |
addone_pic.o: addone_pic.c | |
$(CC) $(CFLAGS) $(CFLAGS_PIC) -c $< -fPIC | |
addone_no_pic.o: addone_no_pic.c | |
$(CC) $(CFLAGS) $(CFLAGS_NO_PIC) -c $< -fno-pic | |
check: prog | |
@echo ==================================== | |
./prog | |
@echo ==================================== | |
objdump -d prog | grep '<addone_.*:' -A10 | |
@echo ==================================== | |
clean: | |
$(RM) prog libaddone.so $(OBJS) |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
/* -*- indent-tabs-mode: nil -*- | |
* | |
* plthook.h -- the header file of plthook | |
* | |
* URL: https://github.com/kubo/plthook | |
* | |
* ------------------------------------------------------ | |
* | |
* Copyright 2013-2014 Kubo Takehiro <kubo@jiubao.org> | |
* | |
* Redistribution and use in source and binary forms, with or without modification, are | |
* permitted provided that the following conditions are met: | |
* | |
* 1. Redistributions of source code must retain the above copyright notice, this list of | |
* conditions and the following disclaimer. | |
* | |
* 2. Redistributions in binary form must reproduce the above copyright notice, this list | |
* of conditions and the following disclaimer in the documentation and/or other materials | |
* provided with the distribution. | |
* | |
* THIS SOFTWARE IS PROVIDED BY THE AUTHORS ''AS IS'' AND ANY EXPRESS OR IMPLIED | |
* WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND | |
* FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL <COPYRIGHT HOLDER> OR | |
* CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR | |
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR | |
* SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON | |
* ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING | |
* NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF | |
* ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | |
* | |
* The views and conclusions contained in the software and documentation are those of the | |
* authors and should not be interpreted as representing official policies, either expressed | |
* or implied, of the authors. | |
* | |
*/ | |
#ifndef PLTHOOK_H | |
#define PLTHOOK_H 1 | |
#define PLTHOOK_SUCCESS 0 | |
#define PLTHOOK_FILE_NOT_FOUND 1 | |
#define PLTHOOK_INVALID_FILE_FORMAT 2 | |
#define PLTHOOK_FUNCTION_NOT_FOUND 3 | |
#define PLTHOOK_INVALID_ARGUMENT 4 | |
#define PLTHOOK_OUT_OF_MEMORY 5 | |
#define PLTHOOK_INTERNAL_ERROR 6 | |
#define PLTHOOK_NOT_IMPLEMENTED 7 | |
typedef struct plthook plthook_t; | |
#ifdef __cplusplus | |
extern "C" { | |
#endif | |
int plthook_open(plthook_t **plthook_out, const char *filename); | |
int plthook_open_by_handle(plthook_t **plthook_out, void *handle); | |
int plthook_open_by_address(plthook_t **plthook_out, void *address); | |
int plthook_enum(plthook_t *plthook, unsigned int *pos, const char **name_out, void ***addr_out); | |
int plthook_replace(plthook_t *plthook, const char *funcname, void *funcaddr, void **oldfunc); | |
void plthook_close(plthook_t *plthook); | |
const char *plthook_error(void); | |
#ifdef __cplusplus | |
}; /* extern "C" */ | |
#endif | |
#endif |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
/* -*- indent-tabs-mode: nil -*- | |
* | |
* plthook_elf.c -- implementation of plthook for ELF format | |
* | |
* URL: https://github.com/kubo/plthook | |
* | |
* ------------------------------------------------------ | |
* | |
* Copyright 2013-2019 Kubo Takehiro <kubo@jiubao.org> | |
* | |
* Redistribution and use in source and binary forms, with or without modification, are | |
* permitted provided that the following conditions are met: | |
* | |
* 1. Redistributions of source code must retain the above copyright notice, this list of | |
* conditions and the following disclaimer. | |
* | |
* 2. Redistributions in binary form must reproduce the above copyright notice, this list | |
* of conditions and the following disclaimer in the documentation and/or other materials | |
* provided with the distribution. | |
* | |
* THIS SOFTWARE IS PROVIDED BY THE AUTHORS ''AS IS'' AND ANY EXPRESS OR IMPLIED | |
* WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND | |
* FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL <COPYRIGHT HOLDER> OR | |
* CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR | |
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR | |
* SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON | |
* ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING | |
* NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF | |
* ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | |
* | |
* The views and conclusions contained in the software and documentation are those of the | |
* authors and should not be interpreted as representing official policies, either expressed | |
* or implied, of the authors. | |
* | |
*/ | |
#if defined(__sun) && defined(_XOPEN_SOURCE) && !defined(__EXTENSIONS__) | |
#define __EXTENSIONS__ | |
#endif | |
#if defined(__linux__) && !defined(_GNU_SOURCE) | |
#define _GNU_SOURCE | |
#endif | |
#include <stdio.h> | |
#include <stdarg.h> | |
#include <stdlib.h> | |
#include <unistd.h> | |
#include <string.h> | |
#include <limits.h> | |
#include <sys/mman.h> | |
#include <errno.h> | |
#include <dlfcn.h> | |
#ifdef __sun | |
#include <sys/auxv.h> | |
#include <procfs.h> | |
#define ELF_TARGET_ALL | |
#endif /* __sun */ | |
#ifdef __FreeBSD__ | |
#include <sys/types.h> | |
#include <sys/user.h> | |
#include <libutil.h> | |
#endif | |
#include <elf.h> | |
#include <link.h> | |
#include "plthook.h" | |
#ifndef __GNUC__ | |
#define __attribute__(arg) | |
#endif | |
#if defined __FreeBSD__ && defined __i386__ && __ELF_WORD_SIZE == 64 | |
#error 32-bit application on 64-bit OS is not supported. | |
#endif | |
#if !defined(R_X86_64_JUMP_SLOT) && defined(R_X86_64_JMP_SLOT) | |
#define R_X86_64_JUMP_SLOT R_X86_64_JMP_SLOT | |
#endif | |
#if defined __x86_64__ || defined __x86_64 | |
#define R_JUMP_SLOT R_X86_64_JUMP_SLOT | |
#define R_GLOBAL_DATA R_X86_64_GLOB_DAT | |
#elif defined __i386__ || defined __i386 | |
#define R_JUMP_SLOT R_386_JMP_SLOT | |
#define R_GLOBAL_DATA R_386_GLOB_DAT | |
#define USE_REL | |
#elif defined __arm__ || defined __arm | |
#define R_JUMP_SLOT R_ARM_JUMP_SLOT | |
#define R_GLOBAL_DATA R_ARM_GLOB_DAT | |
#define USE_REL | |
#elif defined __aarch64__ || defined __aarch64 /* ARM64 */ | |
#define R_JUMP_SLOT R_AARCH64_JUMP_SLOT | |
#define R_GLOBAL_DATA R_AARCH64_GLOB_DAT | |
#elif defined __powerpc64__ | |
#define R_JUMP_SLOT R_PPC64_JMP_SLOT | |
#define R_GLOBAL_DATA R_PPC64_GLOB_DAT | |
#elif defined __powerpc__ | |
#define R_JUMP_SLOT R_PPC_JMP_SLOT | |
#define R_GLOBAL_DATA R_PPC_GLOB_DAT | |
#elif 0 /* disabled because not tested */ && (defined __sparcv9 || defined __sparc_v9__) | |
#define R_JUMP_SLOT R_SPARC_JMP_SLOT | |
#elif 0 /* disabled because not tested */ && (defined __sparc || defined __sparc__) | |
#define R_JUMP_SLOT R_SPARC_JMP_SLOT | |
#elif 0 /* disabled because not tested */ && (defined __ia64 || defined __ia64__) | |
#define R_JUMP_SLOT R_IA64_IPLTMSB | |
#else | |
#error unsupported OS | |
#endif | |
#ifdef USE_REL | |
#define Elf_Plt_Rel Elf_Rel | |
#define PLT_DT_REL DT_REL | |
#define PLT_DT_RELSZ DT_RELSZ | |
#define PLT_DT_RELENT DT_RELENT | |
#else | |
#define Elf_Plt_Rel Elf_Rela | |
#define PLT_DT_REL DT_RELA | |
#define PLT_DT_RELSZ DT_RELASZ | |
#define PLT_DT_RELENT DT_RELAENT | |
#endif | |
#if defined __LP64__ | |
#ifndef ELF_CLASS | |
#define ELF_CLASS ELFCLASS64 | |
#endif | |
#define SIZE_T_FMT "lu" | |
#define ELF_WORD_FMT "u" | |
#ifdef __ANDROID__ | |
#define ELF_XWORD_FMT "llu" | |
#else | |
#define ELF_XWORD_FMT "lu" | |
#endif | |
#define ELF_SXWORD_FMT "ld" | |
#define Elf_Half Elf64_Half | |
#define Elf_Xword Elf64_Xword | |
#define Elf_Sxword Elf64_Sxword | |
#define Elf_Ehdr Elf64_Ehdr | |
#define Elf_Phdr Elf64_Phdr | |
#define Elf_Sym Elf64_Sym | |
#define Elf_Dyn Elf64_Dyn | |
#define Elf_Rel Elf64_Rel | |
#define Elf_Rela Elf64_Rela | |
#ifndef ELF_R_SYM | |
#define ELF_R_SYM ELF64_R_SYM | |
#endif | |
#ifndef ELF_R_TYPE | |
#define ELF_R_TYPE ELF64_R_TYPE | |
#endif | |
#else /* __LP64__ */ | |
#ifndef ELF_CLASS | |
#define ELF_CLASS ELFCLASS32 | |
#endif | |
#define SIZE_T_FMT "u" | |
#ifdef __sun | |
#define ELF_WORD_FMT "lu" | |
#define ELF_XWORD_FMT "lu" | |
#define ELF_SXWORD_FMT "ld" | |
#else | |
#define ELF_WORD_FMT "u" | |
#define ELF_XWORD_FMT "u" | |
#define ELF_SXWORD_FMT "d" | |
#endif | |
#define Elf_Half Elf32_Half | |
#define Elf_Xword Elf32_Word | |
#define Elf_Sxword Elf32_Sword | |
#define Elf_Ehdr Elf32_Ehdr | |
#define Elf_Phdr Elf32_Phdr | |
#define Elf_Sym Elf32_Sym | |
#define Elf_Dyn Elf32_Dyn | |
#define Elf_Rel Elf32_Rel | |
#define Elf_Rela Elf32_Rela | |
#ifndef ELF_R_SYM | |
#define ELF_R_SYM ELF32_R_SYM | |
#endif | |
#ifndef ELF_R_TYPE | |
#define ELF_R_TYPE ELF32_R_TYPE | |
#endif | |
#endif /* __LP64__ */ | |
struct plthook { | |
const Elf_Sym *dynsym; | |
const char *dynstr; | |
size_t dynstr_size; | |
const char *plt_addr_base; | |
const Elf_Plt_Rel *rela_plt; | |
size_t rela_plt_cnt; | |
#ifdef R_GLOBAL_DATA | |
const Elf_Plt_Rel *rela_dyn; | |
size_t rela_dyn_cnt; | |
#endif | |
}; | |
static char errmsg[512]; | |
static size_t page_size; | |
#define ALIGN_ADDR(addr) ((void*)((size_t)(addr) & ~(page_size - 1))) | |
static int plthook_open_executable(plthook_t **plthook_out); | |
static int plthook_open_shared_library(plthook_t **plthook_out, const char *filename); | |
static const Elf_Dyn *find_dyn_by_tag(const Elf_Dyn *dyn, Elf_Sxword tag); | |
static int plthook_open_real(plthook_t **plthook_out, struct link_map *lmap); | |
#if defined __FreeBSD__ || defined __sun | |
static int check_elf_header(const Elf_Ehdr *ehdr); | |
#endif | |
static void set_errmsg(const char *fmt, ...) __attribute__((__format__ (__printf__, 1, 2))); | |
#if defined __ANDROID__ | |
struct dl_iterate_data { | |
char* addr; | |
struct link_map lmap; | |
}; | |
static int dl_iterate_cb(struct dl_phdr_info *info, size_t size, void *cb_data) | |
{ | |
struct dl_iterate_data *data = (struct dl_iterate_data*)cb_data; | |
Elf_Half idx = 0; | |
for (idx = 0; idx < info->dlpi_phnum; ++idx) { | |
const Elf_Phdr *phdr = &info->dlpi_phdr[idx]; | |
char* base = (char*)info->dlpi_addr + phdr->p_vaddr; | |
if (base <= data->addr && data->addr < base + phdr->p_memsz) { | |
break; | |
} | |
} | |
if (idx == info->dlpi_phnum) { | |
return 0; | |
} | |
for (idx = 0; idx < info->dlpi_phnum; ++idx) { | |
const Elf_Phdr *phdr = &info->dlpi_phdr[idx]; | |
if (phdr->p_type == PT_DYNAMIC) { | |
data->lmap.l_addr = info->dlpi_addr; | |
data->lmap.l_ld = (Elf_Dyn*)(info->dlpi_addr + phdr->p_vaddr); | |
return 1; | |
} | |
} | |
return 0; | |
} | |
#endif | |
int plthook_open(plthook_t **plthook_out, const char *filename) | |
{ | |
*plthook_out = NULL; | |
if (filename == NULL) { | |
return plthook_open_executable(plthook_out); | |
} else { | |
return plthook_open_shared_library(plthook_out, filename); | |
} | |
} | |
int plthook_open_by_handle(plthook_t **plthook_out, void *hndl) | |
{ | |
#if defined __ANDROID__ | |
char *addr; | |
if (hndl == NULL) { | |
set_errmsg("NULL handle"); | |
return PLTHOOK_FILE_NOT_FOUND; | |
} | |
addr = dlsym(hndl, "__INIT_ARRAY__"); | |
if (addr == NULL) { | |
addr = dlsym(hndl, "_end"); | |
} | |
return plthook_open_by_address(plthook_out, addr - 1); | |
#else | |
struct link_map *lmap = NULL; | |
if (hndl == NULL) { | |
set_errmsg("NULL handle"); | |
return PLTHOOK_FILE_NOT_FOUND; | |
} | |
if (dlinfo(hndl, RTLD_DI_LINKMAP, &lmap) != 0) { | |
set_errmsg("dlinfo error"); | |
return PLTHOOK_FILE_NOT_FOUND; | |
} | |
return plthook_open_real(plthook_out, lmap); | |
#endif | |
} | |
int plthook_open_by_address(plthook_t **plthook_out, void *address) | |
{ | |
#if defined __FreeBSD__ | |
return PLTHOOK_NOT_IMPLEMENTED; | |
#elif defined __ANDROID__ | |
struct dl_iterate_data data = {0,}; | |
data.addr = address; | |
dl_iterate_phdr(dl_iterate_cb, &data); | |
if (data.lmap.l_ld == NULL) { | |
set_errmsg("Could not find memory region containing address %p", address); | |
return PLTHOOK_INTERNAL_ERROR; | |
} | |
return plthook_open_real(plthook_out, &data.lmap); | |
#else | |
Dl_info info; | |
struct link_map *lmap = NULL; | |
*plthook_out = NULL; | |
if (dladdr1(address, &info, (void**)&lmap, RTLD_DL_LINKMAP) == 0) { | |
set_errmsg("dladdr error"); | |
return PLTHOOK_FILE_NOT_FOUND; | |
} | |
return plthook_open_real(plthook_out, lmap); | |
#endif | |
} | |
static int plthook_open_executable(plthook_t **plthook_out) | |
{ | |
#if defined __ANDROID__ | |
return plthook_open_shared_library(plthook_out, NULL); | |
#elif defined __linux__ | |
return plthook_open_real(plthook_out, _r_debug.r_map); | |
#elif defined __sun | |
const char *auxv_file = "/proc/self/auxv"; | |
#define NUM_AUXV_CNT 10 | |
FILE *fp = fopen(auxv_file, "r"); | |
auxv_t auxv; | |
struct r_debug *r_debug = NULL; | |
if (fp == NULL) { | |
set_errmsg("Could not open %s: %s", auxv_file, | |
strerror(errno)); | |
return PLTHOOK_INTERNAL_ERROR; | |
} | |
while (fread(&auxv, sizeof(auxv_t), 1, fp) == 1) { | |
if (auxv.a_type == AT_SUN_LDDATA) { | |
r_debug = (struct r_debug *)auxv.a_un.a_ptr; | |
break; | |
} | |
} | |
fclose(fp); | |
if (r_debug == NULL) { | |
set_errmsg("Could not find r_debug"); | |
return PLTHOOK_INTERNAL_ERROR; | |
} | |
return plthook_open_real(plthook_out, r_debug->r_map); | |
#elif defined __FreeBSD__ | |
return plthook_open_shared_library(plthook_out, NULL); | |
#else | |
set_errmsg("Opening the main program is not supported on this platform."); | |
return PLTHOOK_NOT_IMPLEMENTED; | |
#endif | |
} | |
static int plthook_open_shared_library(plthook_t **plthook_out, const char *filename) | |
{ | |
void *hndl = dlopen(filename, RTLD_LAZY | RTLD_NOLOAD); | |
#ifdef __ANDROID__ | |
int rv; | |
#else | |
struct link_map *lmap = NULL; | |
#endif | |
if (hndl == NULL) { | |
set_errmsg("dlopen error: %s", dlerror()); | |
return PLTHOOK_FILE_NOT_FOUND; | |
} | |
#ifdef __ANDROID__ | |
rv = plthook_open_by_handle(plthook_out, hndl); | |
dlclose(hndl); | |
return rv; | |
#else | |
if (dlinfo(hndl, RTLD_DI_LINKMAP, &lmap) != 0) { | |
set_errmsg("dlinfo error"); | |
dlclose(hndl); | |
return PLTHOOK_FILE_NOT_FOUND; | |
} | |
dlclose(hndl); | |
return plthook_open_real(plthook_out, lmap); | |
#endif | |
} | |
static const Elf_Dyn *find_dyn_by_tag(const Elf_Dyn *dyn, Elf_Sxword tag) | |
{ | |
while (dyn->d_tag != DT_NULL) { | |
if (dyn->d_tag == tag) { | |
return dyn; | |
} | |
dyn++; | |
} | |
return NULL; | |
} | |
#ifdef __linux__ | |
static int get_memory_permission(void *address) | |
{ | |
unsigned long addr = (unsigned long)address; | |
FILE *fp; | |
char buf[PATH_MAX]; | |
char perms[5]; | |
fp = fopen("/proc/self/maps", "r"); | |
if (fp == NULL) { | |
set_errmsg("failed to open /proc/self/maps"); | |
return 0; | |
} | |
while (fgets(buf, PATH_MAX, fp) != NULL) { | |
unsigned long start, end; | |
if (sscanf(buf, "%lx-%lx %4s", &start, &end, perms) != 3) { | |
continue; | |
} | |
if (start <= addr && addr < end) { | |
int prot = 0; | |
fclose(fp); | |
if (perms[0] == 'r') { | |
prot |= PROT_READ; | |
} else if (perms[0] != '-') { | |
goto unknown_perms; | |
} | |
if (perms[1] == 'w') { | |
prot |= PROT_WRITE; | |
} else if (perms[1] != '-') { | |
goto unknown_perms; | |
} | |
if (perms[2] == 'x') { | |
prot |= PROT_EXEC; | |
} else if (perms[2] != '-') { | |
goto unknown_perms; | |
} | |
if (perms[3] != 'p') { | |
goto unknown_perms; | |
} | |
if (perms[4] != '\0') { | |
perms[4] = '\0'; | |
goto unknown_perms; | |
} | |
return prot; | |
} | |
} | |
fclose(fp); | |
set_errmsg("Could not find memory region containing %p", (void*)addr); | |
return 0; | |
unknown_perms: | |
fclose(fp); | |
set_errmsg("Unexcepted memory permission %s at %p", perms, (void*)addr); | |
return 0; | |
} | |
#elif defined __FreeBSD__ | |
static int get_memory_permission(void *address) | |
{ | |
uint64_t addr = (uint64_t)address; | |
struct kinfo_vmentry *top; | |
int i, cnt; | |
top = kinfo_getvmmap(getpid(), &cnt); | |
if (top == NULL) { | |
set_errmsg("failed to call kinfo_getvmmap()\n"); | |
return 0; | |
} | |
for (i = 0; i < cnt; i++) { | |
struct kinfo_vmentry *kve = top + i; | |
if (kve->kve_start <= addr && addr < kve->kve_end) { | |
int prot = 0; | |
if (kve->kve_protection & KVME_PROT_READ) { | |
prot |= PROT_READ; | |
} | |
if (kve->kve_protection & KVME_PROT_WRITE) { | |
prot |= PROT_WRITE; | |
} | |
if (kve->kve_protection & KVME_PROT_EXEC) { | |
prot |= PROT_EXEC; | |
} | |
if (prot == 0) { | |
set_errmsg("Unknown kve_protection 0x%x at %p", kve->kve_protection, (void*)addr); | |
} | |
free(top); | |
return prot; | |
} | |
} | |
free(top); | |
set_errmsg("Could not find memory region containing %p", (void*)addr); | |
return 0; | |
} | |
#elif defined(__sun) | |
#define NUM_MAPS 20 | |
static int get_memory_permission(void *address) | |
{ | |
unsigned long addr = (unsigned long)address; | |
FILE *fp; | |
prmap_t maps[NUM_MAPS]; | |
size_t num; | |
fp = fopen("/proc/self/map", "r"); | |
if (fp == NULL) { | |
set_errmsg("failed to open /proc/self/map"); | |
return 0; | |
} | |
while ((num = fread(maps, sizeof(prmap_t), NUM_MAPS, fp)) > 0) { | |
size_t i; | |
for (i = 0; i < num; i++) { | |
prmap_t *map = &maps[i]; | |
if (map->pr_vaddr <= addr && addr < map->pr_vaddr + map->pr_size) { | |
int prot = 0; | |
if (map->pr_mflags & MA_READ) { | |
prot |= PROT_READ; | |
} | |
if (map->pr_mflags & MA_WRITE) { | |
prot |= PROT_WRITE; | |
} | |
if (map->pr_mflags & MA_EXEC) { | |
prot |= PROT_EXEC; | |
} | |
if (prot == 0) { | |
set_errmsg("Unknown pr_mflags 0x%x at %p", map->pr_mflags, (void*)addr); | |
} | |
fclose(fp); | |
return prot; | |
} | |
} | |
} | |
fclose(fp); | |
set_errmsg("Could not find memory region containing %p", (void*)addr); | |
return 0; | |
} | |
#else | |
#error Unsupported platform | |
#endif | |
static int plthook_open_real(plthook_t **plthook_out, struct link_map *lmap) | |
{ | |
plthook_t plthook = {NULL,}; | |
const Elf_Dyn *dyn; | |
const char *dyn_addr_base = NULL; | |
if (page_size == 0) { | |
page_size = sysconf(_SC_PAGESIZE); | |
} | |
#if defined __linux__ | |
plthook.plt_addr_base = (char*)lmap->l_addr; | |
#if defined __ANDROID__ | |
dyn_addr_base = (const char*)lmap->l_addr; | |
#endif | |
#elif defined __FreeBSD__ || defined __sun | |
const Elf_Ehdr *ehdr = (const Elf_Ehdr*)lmap->l_addr; | |
int rv_ = check_elf_header(ehdr); | |
if (rv_ != 0) { | |
return rv_; | |
} | |
if (ehdr->e_type == ET_DYN) { | |
dyn_addr_base = (const char*)lmap->l_addr; | |
plthook.plt_addr_base = (const char*)lmap->l_addr; | |
} | |
#else | |
#error unsupported OS | |
#endif | |
/* get .dynsym section */ | |
dyn = find_dyn_by_tag(lmap->l_ld, DT_SYMTAB); | |
if (dyn == NULL) { | |
set_errmsg("failed to find DT_SYMTAB"); | |
return PLTHOOK_INTERNAL_ERROR; | |
} | |
plthook.dynsym = (const Elf_Sym*)(dyn_addr_base + dyn->d_un.d_ptr); | |
/* Check sizeof(Elf_Sym) */ | |
dyn = find_dyn_by_tag(lmap->l_ld, DT_SYMENT); | |
if (dyn == NULL) { | |
set_errmsg("failed to find DT_SYMTAB"); | |
return PLTHOOK_INTERNAL_ERROR; | |
} | |
if (dyn->d_un.d_val != sizeof(Elf_Sym)) { | |
set_errmsg("DT_SYMENT size %" ELF_XWORD_FMT " != %" SIZE_T_FMT, dyn->d_un.d_val, sizeof(Elf_Sym)); | |
return PLTHOOK_INTERNAL_ERROR; | |
} | |
/* get .dynstr section */ | |
dyn = find_dyn_by_tag(lmap->l_ld, DT_STRTAB); | |
if (dyn == NULL) { | |
set_errmsg("failed to find DT_STRTAB"); | |
return PLTHOOK_INTERNAL_ERROR; | |
} | |
plthook.dynstr = dyn_addr_base + dyn->d_un.d_ptr; | |
/* get .dynstr size */ | |
dyn = find_dyn_by_tag(lmap->l_ld, DT_STRSZ); | |
if (dyn == NULL) { | |
set_errmsg("failed to find DT_STRSZ"); | |
return PLTHOOK_INTERNAL_ERROR; | |
} | |
plthook.dynstr_size = dyn->d_un.d_val; | |
/* get .rela.plt or .rel.plt section */ | |
dyn = find_dyn_by_tag(lmap->l_ld, DT_JMPREL); | |
if (dyn != NULL) { | |
plthook.rela_plt = (const Elf_Plt_Rel *)(dyn_addr_base + dyn->d_un.d_ptr); | |
dyn = find_dyn_by_tag(lmap->l_ld, DT_PLTRELSZ); | |
if (dyn == NULL) { | |
set_errmsg("failed to find DT_PLTRELSZ"); | |
return PLTHOOK_INTERNAL_ERROR; | |
} | |
plthook.rela_plt_cnt = dyn->d_un.d_val / sizeof(Elf_Plt_Rel); | |
} | |
#ifdef R_GLOBAL_DATA | |
/* get .rela.dyn or .rel.dyn section */ | |
dyn = find_dyn_by_tag(lmap->l_ld, PLT_DT_REL); | |
if (dyn != NULL) { | |
size_t total_size, elem_size; | |
plthook.rela_dyn = (const Elf_Plt_Rel *)(dyn_addr_base + dyn->d_un.d_ptr); | |
dyn = find_dyn_by_tag(lmap->l_ld, PLT_DT_RELSZ); | |
if (dyn == NULL) { | |
set_errmsg("failed to find PLT_DT_RELSZ"); | |
return PLTHOOK_INTERNAL_ERROR; | |
} | |
total_size = dyn->d_un.d_ptr; | |
dyn = find_dyn_by_tag(lmap->l_ld, PLT_DT_RELENT); | |
if (dyn == NULL) { | |
set_errmsg("failed to find PLT_DT_RELENT"); | |
return PLTHOOK_INTERNAL_ERROR; | |
} | |
elem_size = dyn->d_un.d_ptr; | |
plthook.rela_dyn_cnt = total_size / elem_size; | |
} | |
#endif | |
#ifdef R_GLOBAL_DATA | |
if (plthook.rela_plt == NULL && plthook.rela_dyn == NULL) { | |
set_errmsg("failed to find either of DT_JMPREL and DT_REL"); | |
return PLTHOOK_INTERNAL_ERROR; | |
} | |
#else | |
if (plthook.rela_plt == NULL) { | |
set_errmsg("failed to find DT_JMPREL"); | |
return PLTHOOK_INTERNAL_ERROR; | |
} | |
#endif | |
*plthook_out = malloc(sizeof(plthook_t)); | |
if (*plthook_out == NULL) { | |
set_errmsg("failed to allocate memory: %" SIZE_T_FMT " bytes", sizeof(plthook_t)); | |
return PLTHOOK_OUT_OF_MEMORY; | |
} | |
**plthook_out = plthook; | |
return 0; | |
} | |
#if defined __FreeBSD__ || defined __sun | |
static int check_elf_header(const Elf_Ehdr *ehdr) | |
{ | |
static const unsigned short s = 1; | |
/* Check endianness at runtime. */ | |
unsigned char elfdata = (*(const char*)&s) ? ELFDATA2LSB : ELFDATA2MSB; | |
if (ehdr == NULL) { | |
set_errmsg("invalid elf header address: NULL"); | |
return PLTHOOK_INTERNAL_ERROR; | |
} | |
if (memcmp(ehdr->e_ident, ELFMAG, SELFMAG) != 0) { | |
set_errmsg("invalid file signature: 0x%02x,0x%02x,0x%02x,0x%02x", | |
ehdr->e_ident[0], ehdr->e_ident[1], ehdr->e_ident[2], ehdr->e_ident[3]); | |
return PLTHOOK_INVALID_FILE_FORMAT; | |
} | |
if (ehdr->e_ident[EI_CLASS] != ELF_CLASS) { | |
set_errmsg("invalid elf class: 0x%02x", ehdr->e_ident[EI_CLASS]); | |
return PLTHOOK_INVALID_FILE_FORMAT; | |
} | |
if (ehdr->e_ident[EI_DATA] != elfdata) { | |
set_errmsg("invalid elf data: 0x%02x", ehdr->e_ident[EI_DATA]); | |
return PLTHOOK_INVALID_FILE_FORMAT; | |
} | |
if (ehdr->e_ident[EI_VERSION] != EV_CURRENT) { | |
set_errmsg("invalid elf version: 0x%02x", ehdr->e_ident[EI_VERSION]); | |
return PLTHOOK_INVALID_FILE_FORMAT; | |
} | |
if (ehdr->e_type != ET_EXEC && ehdr->e_type != ET_DYN) { | |
set_errmsg("invalid file type: 0x%04x", ehdr->e_type); | |
return PLTHOOK_INVALID_FILE_FORMAT; | |
} | |
if (ehdr->e_version != EV_CURRENT) { | |
set_errmsg("invalid object file version: %" ELF_WORD_FMT, ehdr->e_version); | |
return PLTHOOK_INVALID_FILE_FORMAT; | |
} | |
if (ehdr->e_ehsize != sizeof(Elf_Ehdr)) { | |
set_errmsg("invalid elf header size: %u", ehdr->e_ehsize); | |
return PLTHOOK_INVALID_FILE_FORMAT; | |
} | |
if (ehdr->e_phentsize != sizeof(Elf_Phdr)) { | |
set_errmsg("invalid program header table entry size: %u", ehdr->e_phentsize); | |
return PLTHOOK_INVALID_FILE_FORMAT; | |
} | |
return 0; | |
} | |
#endif | |
static int check_rel(const plthook_t *plthook, const Elf_Plt_Rel *plt, Elf_Xword r_type, const char **name_out, void ***addr_out) | |
{ | |
if (ELF_R_TYPE(plt->r_info) == r_type) { | |
size_t idx = ELF_R_SYM(plt->r_info); | |
idx = plthook->dynsym[idx].st_name; | |
if (idx + 1 > plthook->dynstr_size) { | |
set_errmsg("too big section header string table index: %" SIZE_T_FMT, idx); | |
return PLTHOOK_INVALID_FILE_FORMAT; | |
} | |
*name_out = plthook->dynstr + idx; | |
*addr_out = (void**)(plthook->plt_addr_base + plt->r_offset); | |
return 0; | |
} | |
return -1; | |
} | |
int plthook_enum(plthook_t *plthook, unsigned int *pos, const char **name_out, void ***addr_out) | |
{ | |
while (*pos < plthook->rela_plt_cnt) { | |
const Elf_Plt_Rel *plt = plthook->rela_plt + *pos; | |
int rv = check_rel(plthook, plt, R_JUMP_SLOT, name_out, addr_out); | |
(*pos)++; | |
if (rv >= 0) { | |
return rv; | |
} | |
} | |
#ifdef R_GLOBAL_DATA | |
while (*pos < plthook->rela_plt_cnt + plthook->rela_dyn_cnt) { | |
const Elf_Plt_Rel *plt = plthook->rela_dyn + (*pos - plthook->rela_plt_cnt); | |
int rv = check_rel(plthook, plt, R_GLOBAL_DATA, name_out, addr_out); | |
(*pos)++; | |
if (rv >= 0) { | |
return rv; | |
} | |
} | |
#endif | |
*name_out = NULL; | |
*addr_out = NULL; | |
return EOF; | |
} | |
int plthook_replace(plthook_t *plthook, const char *funcname, void *funcaddr, void **oldfunc) | |
{ | |
size_t funcnamelen = strlen(funcname); | |
unsigned int pos = 0; | |
const char *name; | |
void **addr; | |
int rv; | |
if (plthook == NULL) { | |
set_errmsg("invalid argument: The first argument is null."); | |
return PLTHOOK_INVALID_ARGUMENT; | |
} | |
while ((rv = plthook_enum(plthook, &pos, &name, &addr)) == 0) { | |
if (strncmp(name, funcname, funcnamelen) == 0) { | |
if (name[funcnamelen] == '\0' || name[funcnamelen] == '@') { | |
int prot = get_memory_permission(addr); | |
if (prot == 0) { | |
return PLTHOOK_INTERNAL_ERROR; | |
} | |
if (!(prot & PROT_WRITE)) { | |
if (mprotect(ALIGN_ADDR(addr), page_size, PROT_READ | PROT_WRITE) != 0) { | |
set_errmsg("Could not change the process memory permission at %p: %s", | |
ALIGN_ADDR(addr), strerror(errno)); | |
return PLTHOOK_INTERNAL_ERROR; | |
} | |
} | |
if (oldfunc) { | |
*oldfunc = *addr; | |
} | |
*addr = funcaddr; | |
if (!(prot & PROT_WRITE)) { | |
mprotect(ALIGN_ADDR(addr), page_size, prot); | |
} | |
return 0; | |
} | |
} | |
} | |
if (rv == EOF) { | |
set_errmsg("no such function: %s", funcname); | |
rv = PLTHOOK_FUNCTION_NOT_FOUND; | |
} | |
return rv; | |
} | |
void plthook_close(plthook_t *plthook) | |
{ | |
if (plthook != NULL) { | |
free(plthook); | |
} | |
} | |
const char *plthook_error(void) | |
{ | |
return errmsg; | |
} | |
static void set_errmsg(const char *fmt, ...) | |
{ | |
va_list ap; | |
va_start(ap, fmt); | |
vsnprintf(errmsg, sizeof(errmsg) - 1, fmt, ap); | |
va_end(ap); | |
} |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment