mirror of
https://github.com/jart/cosmopolitan.git
synced 2025-03-14 04:46:26 +00:00
Some checks failed
build / matrix_on_mode () (push) Has been cancelled
build / matrix_on_mode (optlinux) (push) Has been cancelled
build / matrix_on_mode (rel) (push) Has been cancelled
build / matrix_on_mode (tiny) (push) Has been cancelled
build / matrix_on_mode (tinylinux) (push) Has been cancelled
283 lines
9 KiB
C
283 lines
9 KiB
C
/*-*- mode:c;indent-tabs-mode:nil;c-basic-offset:2;tab-width:8;coding:utf-8 -*-│
|
|
│ vi: set et ft=c ts=2 sts=2 sw=2 fenc=utf-8 :vi │
|
|
╞══════════════════════════════════════════════════════════════════════════════╡
|
|
│ Copyright 2024 Justine Alexandra Roberts Tunney │
|
|
│ │
|
|
│ Permission to use, copy, modify, and/or distribute this software for │
|
|
│ any purpose with or without fee is hereby granted, provided that the │
|
|
│ above copyright notice and this permission notice appear in all copies. │
|
|
│ │
|
|
│ THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL │
|
|
│ WARRANTIES WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED │
|
|
│ WARRANTIES OF MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE │
|
|
│ AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL │
|
|
│ DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR │
|
|
│ PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER │
|
|
│ TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR │
|
|
│ PERFORMANCE OF THIS SOFTWARE. │
|
|
╚─────────────────────────────────────────────────────────────────────────────*/
|
|
#include "libc/calls/calls.h"
|
|
#include "libc/elf/def.h"
|
|
#include "libc/elf/elf.h"
|
|
#include "libc/elf/scalar.h"
|
|
#include "libc/elf/struct/ehdr.h"
|
|
#include "libc/elf/struct/phdr.h"
|
|
#include "libc/intrin/kprintf.h"
|
|
#include "libc/intrin/likely.h"
|
|
#include "libc/macros.h"
|
|
#include "libc/mem/mem.h"
|
|
#include "libc/runtime/runtime.h"
|
|
#include "libc/runtime/symbols.internal.h"
|
|
#include "libc/stdio/stdio.h"
|
|
#include "libc/str/str.h"
|
|
#include "libc/sysv/consts/map.h"
|
|
#include "libc/sysv/consts/o.h"
|
|
#include "libc/sysv/consts/prot.h"
|
|
#include "third_party/getopt/getopt.internal.h"
|
|
|
|
#define VERSION \
|
|
"renamestr v0.1\n" \
|
|
"https://github.com/jart/cosmopolitan\n"
|
|
|
|
#define MANUAL \
|
|
" -f FROM -t TO INPUT \n" \
|
|
"\n" \
|
|
"DESCRIPTION\n" \
|
|
"\n" \
|
|
" in-place string replacement in ELF binary .rodata\n" \
|
|
"\n" \
|
|
" this program may be used to replace strings in the\n" \
|
|
" .rodata sections of ELF binaries, in-place.\n" \
|
|
"\n" \
|
|
"FLAGS\n" \
|
|
"\n" \
|
|
" -h show usage\n" \
|
|
"\n" \
|
|
" -v show version\n" \
|
|
"\n" \
|
|
" -f FROM source string to replace\n" \
|
|
"\n" \
|
|
" -t TO target string replacement. must be shorter\n" \
|
|
" than FROM string for replacement to work\n" \
|
|
"\n" \
|
|
" INPUT ELF binary containing strings to replace\n" \
|
|
"\n"
|
|
|
|
static const char *prog;
|
|
static const char *exepath;
|
|
static Elf64_Shdr *rodata;
|
|
static char *rostart;
|
|
static char *roend;
|
|
static int exefd;
|
|
|
|
static wontreturn void Die(const char *thing, const char *reason) {
|
|
tinyprint(2, thing, ": ", reason, "\n", NULL);
|
|
exit(1);
|
|
}
|
|
|
|
static wontreturn void DieSys(const char *thing) {
|
|
perror(thing);
|
|
exit(1);
|
|
}
|
|
|
|
static wontreturn void ShowUsage(int rc, int fd) {
|
|
tinyprint(fd, "USAGE\n\n ", prog, MANUAL, NULL);
|
|
exit(rc);
|
|
}
|
|
|
|
static void Pwrite(const void *data, size_t size, uint64_t offset) {
|
|
ssize_t rc;
|
|
const char *p, *e;
|
|
for (p = data, e = p + size; p < e; p += (size_t)rc, offset += (size_t)rc) {
|
|
if ((rc = pwrite(exefd, p, e - p, offset)) == -1) {
|
|
DieSys(exepath);
|
|
}
|
|
}
|
|
}
|
|
|
|
struct String {
|
|
const char *str;
|
|
size_t len;
|
|
};
|
|
|
|
struct Param {
|
|
struct String from;
|
|
struct String to;
|
|
int count;
|
|
char *roloc;
|
|
};
|
|
|
|
struct Params {
|
|
int n;
|
|
struct Param p[4];
|
|
};
|
|
|
|
static struct Params params;
|
|
|
|
static void GetOpts(int argc, char *argv[]) {
|
|
int opt;
|
|
bool partial = false;
|
|
params.n = 0;
|
|
struct Param *param;
|
|
while ((opt = getopt(argc, argv, "hvf:t:")) != -1) {
|
|
if (params.n >= ARRAYLEN(params.p)) {
|
|
param = NULL;
|
|
} else {
|
|
param = &(params.p[params.n]);
|
|
}
|
|
switch (opt) {
|
|
case 'f':
|
|
if (!param) {
|
|
Die(prog, "too many replacements provided");
|
|
}
|
|
if (param->from.str) {
|
|
Die(prog, "from string already provided");
|
|
}
|
|
param->from.str = optarg;
|
|
param->from.len = strlen(optarg);
|
|
partial = !partial;
|
|
break;
|
|
case 't':
|
|
if (!param) {
|
|
Die(prog, "too many replacements provided");
|
|
}
|
|
if (param->to.str) {
|
|
Die(prog, "to string already provided");
|
|
}
|
|
param->to.str = optarg;
|
|
param->to.len = strlen(optarg);
|
|
partial = !partial;
|
|
break;
|
|
case 'v':
|
|
tinyprint(0, VERSION, NULL);
|
|
exit(0);
|
|
case 'h':
|
|
ShowUsage(0, 1);
|
|
default:
|
|
ShowUsage(1, 2);
|
|
}
|
|
if (param->from.str && param->to.str) {
|
|
if (param->from.len < param->to.len) {
|
|
Die(prog, "to.str longer than from.str, cannot replace");
|
|
}
|
|
params.n++;
|
|
}
|
|
}
|
|
if (params.n == 0) {
|
|
Die(prog, "no replacements provided");
|
|
}
|
|
if (partial) {
|
|
Die(prog, "partial replacement provided");
|
|
}
|
|
if (optind == argc) {
|
|
Die(prog, "missing input argument");
|
|
}
|
|
if (optind != argc - 1) {
|
|
Die(prog, "too many args");
|
|
}
|
|
exepath = argv[optind];
|
|
}
|
|
|
|
struct Input {
|
|
union {
|
|
char *map;
|
|
Elf64_Ehdr *elf;
|
|
unsigned char *umap;
|
|
};
|
|
size_t size;
|
|
const char *path;
|
|
};
|
|
|
|
static struct Input input;
|
|
|
|
static void OpenInput(const char *path) {
|
|
int fd;
|
|
if ((fd = open(path, O_RDWR)) == -1)
|
|
DieSys(path);
|
|
if ((input.size = lseek(fd, 0, SEEK_END)) == -1)
|
|
DieSys(path);
|
|
input.path = path;
|
|
input.map = mmap(0, input.size, PROT_READ | PROT_WRITE, MAP_PRIVATE, fd, 0);
|
|
if (input.map == MAP_FAILED)
|
|
DieSys(path);
|
|
if (!IsElf64Binary(input.elf, input.size))
|
|
Die(path, "not an elf64 binary");
|
|
exefd = fd;
|
|
}
|
|
|
|
static void ReplaceString(struct Param *param) {
|
|
size_t len;
|
|
char *x = (char *)memchr(param->roloc, 0, roend - param->roloc);
|
|
memmove(param->roloc, param->to.str, param->to.len);
|
|
if (UNLIKELY(x == NULL)) {
|
|
len = roend - param->roloc;
|
|
memmove(param->roloc + param->to.len, param->roloc + param->from.len,
|
|
len - param->from.len);
|
|
} else {
|
|
len = x - param->roloc;
|
|
memmove(param->roloc + param->to.len, param->roloc + param->from.len,
|
|
len + 1 - param->from.len);
|
|
}
|
|
param->roloc += param->to.len;
|
|
}
|
|
|
|
int main(int argc, char *argv[]) {
|
|
#ifdef MODE_DBG
|
|
ShowCrashReports();
|
|
#endif
|
|
|
|
prog = argv[0];
|
|
|
|
if (!prog)
|
|
prog = "renamestr";
|
|
|
|
GetOpts(argc, argv);
|
|
OpenInput(exepath);
|
|
rodata = FindElfSectionByName(
|
|
input.elf, input.size,
|
|
GetElfSectionNameStringTable(input.elf, input.size), ".rodata");
|
|
if (!rodata)
|
|
Die(exepath, "doesn't have .rodata");
|
|
|
|
rostart = GetElfSectionAddress(input.elf, input.size, rodata);
|
|
if (!rostart)
|
|
Die(prog, "could not get to start of .rodata");
|
|
roend = rostart + rodata->sh_size;
|
|
|
|
#ifdef MODE_DBG
|
|
kprintf("elf file to process: %s\n", exepath);
|
|
kprintf("file size is %ld\n", input.size);
|
|
#endif
|
|
for (int i = 0; i < params.n; ++i) {
|
|
struct Param *param = &(params.p[i]);
|
|
param->roloc = rostart;
|
|
param->count = 0;
|
|
#ifdef MODE_DBG
|
|
kprintf("need to replace '%s' with '%s'\n", param->from.str, param->to.str);
|
|
#endif
|
|
}
|
|
|
|
#define NEXT_ROLOC(z) \
|
|
memmem((z)->roloc, roend - (z)->roloc, (z)->from.str, (z)->from.len)
|
|
for (int i = 0; i < params.n; ++i) {
|
|
struct Param *param = &(params.p[i]);
|
|
for (param->roloc = NEXT_ROLOC(param); param->roloc != NULL;
|
|
param->roloc = NEXT_ROLOC(param)) {
|
|
ReplaceString(param);
|
|
param->count++;
|
|
}
|
|
}
|
|
#undef NEXT_ROLOC
|
|
|
|
Pwrite(input.map, input.size, 0);
|
|
if (close(exefd)) {
|
|
Die(prog, "unable to close file after writing");
|
|
}
|
|
|
|
for (int i = 0; i < params.n; ++i) {
|
|
struct Param *param = &(params.p[i]);
|
|
printf("'%s' -> '%s': %d replacements\n", param->from.str, param->to.str,
|
|
param->count);
|
|
}
|
|
return 0;
|
|
}
|