1 1.7 christos /* $NetBSD: binpatch.c,v 1.7 2016/09/22 17:08:16 christos Exp $ */ 2 1.6 tsutsui 3 1.6 tsutsui /*- 4 1.6 tsutsui * Copyright (c) 2009 Izumi Tsutsui. All rights reserved. 5 1.6 tsutsui * 6 1.6 tsutsui * Redistribution and use in source and binary forms, with or without 7 1.6 tsutsui * modification, are permitted provided that the following conditions 8 1.6 tsutsui * are met: 9 1.6 tsutsui * 1. Redistributions of source code must retain the above copyright 10 1.6 tsutsui * notice, this list of conditions and the following disclaimer. 11 1.6 tsutsui * 2. Redistributions in binary form must reproduce the above copyright 12 1.6 tsutsui * notice, this list of conditions and the following disclaimer in the 13 1.6 tsutsui * documentation and/or other materials provided with the distribution. 14 1.6 tsutsui * 15 1.6 tsutsui * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR 16 1.6 tsutsui * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES 17 1.6 tsutsui * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. 18 1.6 tsutsui * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, 19 1.6 tsutsui * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT 20 1.6 tsutsui * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, 21 1.6 tsutsui * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY 22 1.6 tsutsui * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT 23 1.6 tsutsui * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF 24 1.6 tsutsui * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. 25 1.6 tsutsui */ 26 1.1 leo 27 1.1 leo /* 28 1.6 tsutsui * Copyright (c) 1996 Christopher G. Demetriou 29 1.1 leo * All rights reserved. 30 1.6 tsutsui * 31 1.1 leo * Redistribution and use in source and binary forms, with or without 32 1.1 leo * modification, are permitted provided that the following conditions 33 1.1 leo * are met: 34 1.1 leo * 1. Redistributions of source code must retain the above copyright 35 1.1 leo * notice, this list of conditions and the following disclaimer. 36 1.1 leo * 2. Redistributions in binary form must reproduce the above copyright 37 1.1 leo * notice, this list of conditions and the following disclaimer in the 38 1.1 leo * documentation and/or other materials provided with the distribution. 39 1.6 tsutsui * 3. The name of the author may not be used to endorse or promote products 40 1.6 tsutsui * derived from this software without specific prior written permission. 41 1.6 tsutsui * 42 1.1 leo * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR 43 1.1 leo * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES 44 1.1 leo * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. 45 1.1 leo * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, 46 1.1 leo * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT 47 1.1 leo * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, 48 1.1 leo * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY 49 1.1 leo * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT 50 1.1 leo * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF 51 1.1 leo * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. 52 1.6 tsutsui * 53 1.6 tsutsui * <<Id: LICENSE_GC,v 1.1 2001/10/01 23:24:05 cgd Exp>> 54 1.1 leo */ 55 1.1 leo 56 1.6 tsutsui #include <sys/cdefs.h> 57 1.6 tsutsui #ifndef lint 58 1.6 tsutsui __COPYRIGHT("@(#) Copyright (c) 1996\ 59 1.6 tsutsui Christopher G. Demetriou. All rights reserved."); 60 1.6 tsutsui #endif /* not lint */ 61 1.6 tsutsui 62 1.6 tsutsui #ifndef lint 63 1.7 christos __RCSID("$NetBSD: binpatch.c,v 1.7 2016/09/22 17:08:16 christos Exp $"); 64 1.6 tsutsui #endif /* not lint */ 65 1.6 tsutsui 66 1.1 leo #include <sys/types.h> 67 1.6 tsutsui #include <sys/mman.h> 68 1.6 tsutsui #include <sys/stat.h> 69 1.6 tsutsui #include <sys/inttypes.h> 70 1.6 tsutsui 71 1.6 tsutsui #include <err.h> 72 1.6 tsutsui #include <fcntl.h> 73 1.6 tsutsui #include <limits.h> 74 1.6 tsutsui #include <nlist.h> 75 1.1 leo #include <stdio.h> 76 1.4 mhitch #include <stdlib.h> 77 1.6 tsutsui #include <stdbool.h> 78 1.6 tsutsui #include <unistd.h> 79 1.1 leo 80 1.6 tsutsui #include "extern.h" 81 1.1 leo 82 1.6 tsutsui static void usage(void) __dead; 83 1.1 leo 84 1.6 tsutsui bool replace, verbose; 85 1.6 tsutsui u_long addr, offset; 86 1.6 tsutsui char *symbol; 87 1.6 tsutsui size_t size; 88 1.6 tsutsui uint64_t val; 89 1.6 tsutsui 90 1.6 tsutsui static const struct { 91 1.6 tsutsui const char *name; 92 1.6 tsutsui int (*check)(const char *, size_t); 93 1.7 christos int (*findoff)(const char *, size_t, u_long, size_t *, u_long); 94 1.6 tsutsui } exec_formats[] = { 95 1.6 tsutsui #ifdef NLIST_AOUT 96 1.6 tsutsui { "a.out", check_aout, findoff_aout, }, 97 1.6 tsutsui #endif 98 1.6 tsutsui #ifdef NLIST_ECOFF 99 1.6 tsutsui { "ECOFF", check_ecoff, findoff_ecoff, }, 100 1.6 tsutsui #endif 101 1.6 tsutsui #ifdef NLIST_ELF32 102 1.6 tsutsui { "ELF32", check_elf32, findoff_elf32, }, 103 1.6 tsutsui #endif 104 1.6 tsutsui #ifdef NLIST_ELF64 105 1.6 tsutsui { "ELF64", check_elf64, findoff_elf64, }, 106 1.6 tsutsui #endif 107 1.6 tsutsui #ifdef NLIST_COFF 108 1.6 tsutsui { "COFF", check_coff, findoff_coff, }, 109 1.6 tsutsui #endif 110 1.6 tsutsui }; 111 1.1 leo 112 1.1 leo 113 1.1 leo int 114 1.5 dsl main(int argc, char *argv[]) 115 1.1 leo { 116 1.6 tsutsui const char *fname; 117 1.6 tsutsui struct stat sb; 118 1.6 tsutsui struct nlist nl[2]; 119 1.6 tsutsui char *mappedfile; 120 1.6 tsutsui size_t valoff; 121 1.6 tsutsui void *valp; 122 1.6 tsutsui uint8_t uval8; 123 1.6 tsutsui int8_t sval8; 124 1.6 tsutsui uint16_t uval16; 125 1.6 tsutsui int16_t sval16; 126 1.6 tsutsui uint32_t uval32; 127 1.6 tsutsui int32_t sval32; 128 1.6 tsutsui uint64_t uval64; 129 1.6 tsutsui int64_t sval64; 130 1.6 tsutsui int ch, fd, rv, i, n; 131 1.7 christos u_long text_start; /* Start of kernel text (a.out) */ 132 1.6 tsutsui 133 1.6 tsutsui setprogname(argv[0]); 134 1.7 christos text_start = (unsigned long)~0; 135 1.6 tsutsui 136 1.6 tsutsui while ((ch = getopt(argc, argv, "bwldT:a:s:o:r:v")) != -1) 137 1.6 tsutsui switch (ch) { 138 1.6 tsutsui case 'b': 139 1.6 tsutsui size = sizeof(uint8_t); 140 1.6 tsutsui break; 141 1.6 tsutsui case 'w': 142 1.6 tsutsui size = sizeof(uint16_t); 143 1.6 tsutsui break; 144 1.6 tsutsui case 'l': 145 1.6 tsutsui size = sizeof(uint32_t); 146 1.6 tsutsui break; 147 1.6 tsutsui case 'd': 148 1.6 tsutsui size = sizeof(uint64_t); 149 1.6 tsutsui break; 150 1.6 tsutsui case 'a': 151 1.6 tsutsui if (addr != 0 || symbol != NULL) 152 1.6 tsutsui errx(EXIT_FAILURE, 153 1.6 tsutsui "only one address/symbol allowed"); 154 1.6 tsutsui addr = strtoul(optarg, NULL, 0); 155 1.6 tsutsui break; 156 1.6 tsutsui case 's': 157 1.6 tsutsui if (addr != 0 || symbol != NULL) 158 1.6 tsutsui errx(EXIT_FAILURE, 159 1.6 tsutsui "only one address/symbol allowed"); 160 1.6 tsutsui symbol = optarg; 161 1.6 tsutsui break; 162 1.6 tsutsui case 'o': 163 1.6 tsutsui if (offset != 0) 164 1.6 tsutsui err(EXIT_FAILURE, 165 1.6 tsutsui "only one offset allowed"); 166 1.6 tsutsui offset = strtoul(optarg, NULL, 0); 167 1.6 tsutsui break; 168 1.6 tsutsui case 'r': 169 1.6 tsutsui replace = true; 170 1.6 tsutsui val = strtoull(optarg, NULL, 0); 171 1.6 tsutsui break; 172 1.6 tsutsui case 'v': 173 1.6 tsutsui verbose = true; 174 1.6 tsutsui break; 175 1.6 tsutsui case 'T': 176 1.6 tsutsui text_start = strtoul(optarg, NULL, 0); 177 1.6 tsutsui break; 178 1.6 tsutsui case '?': 179 1.6 tsutsui default: 180 1.6 tsutsui usage(); 181 1.6 tsutsui } 182 1.6 tsutsui argc -= optind; 183 1.6 tsutsui argv += optind; 184 1.6 tsutsui 185 1.6 tsutsui if (argc != 1) 186 1.6 tsutsui usage(); 187 1.6 tsutsui 188 1.6 tsutsui if (addr == 0 && symbol == NULL) { 189 1.6 tsutsui warnx("no address or symbol specified"); 190 1.6 tsutsui usage(); 191 1.6 tsutsui } 192 1.6 tsutsui 193 1.6 tsutsui if (size == 0) 194 1.6 tsutsui size = sizeof(uint32_t); /* default to int */ 195 1.6 tsutsui 196 1.6 tsutsui fname = argv[0]; 197 1.6 tsutsui 198 1.6 tsutsui if ((fd = open(fname, replace ? O_RDWR : O_RDONLY, 0)) == -1) 199 1.6 tsutsui err(EXIT_FAILURE, "open %s", fname); 200 1.6 tsutsui 201 1.6 tsutsui if (symbol != NULL) { 202 1.6 tsutsui nl[0].n_name = symbol; 203 1.6 tsutsui nl[1].n_name = NULL; 204 1.6 tsutsui if ((rv = __fdnlist(fd, nl)) != 0) 205 1.6 tsutsui errx(EXIT_FAILURE, "could not find symbol %s in %s", 206 1.6 tsutsui symbol, fname); 207 1.6 tsutsui addr = nl[0].n_value; 208 1.6 tsutsui if (verbose) 209 1.6 tsutsui fprintf(stderr, "got symbol address 0x%lx from %s\n", 210 1.6 tsutsui addr, fname); 211 1.6 tsutsui } 212 1.6 tsutsui 213 1.6 tsutsui addr += offset * size; 214 1.6 tsutsui 215 1.6 tsutsui if (fstat(fd, &sb) == -1) 216 1.6 tsutsui err(EXIT_FAILURE, "fstat %s", fname); 217 1.6 tsutsui if (sb.st_size != (ssize_t)sb.st_size) 218 1.6 tsutsui errx(EXIT_FAILURE, "%s too big to map", fname); 219 1.6 tsutsui 220 1.6 tsutsui if ((mappedfile = mmap(NULL, sb.st_size, 221 1.6 tsutsui replace ? PROT_READ | PROT_WRITE : PROT_READ, 222 1.6 tsutsui MAP_FILE | MAP_SHARED, fd, 0)) == (char *)-1) 223 1.6 tsutsui err(EXIT_FAILURE, "mmap %s", fname); 224 1.6 tsutsui if (verbose) 225 1.6 tsutsui fprintf(stderr, "mapped %s\n", fname); 226 1.6 tsutsui 227 1.6 tsutsui n = __arraycount(exec_formats); 228 1.6 tsutsui for (i = 0; i < n; i++) { 229 1.6 tsutsui if ((*exec_formats[i].check)(mappedfile, sb.st_size) == 0) 230 1.6 tsutsui break; 231 1.6 tsutsui } 232 1.6 tsutsui if (i == n) 233 1.6 tsutsui errx(EXIT_FAILURE, "%s: unknown executable format", fname); 234 1.6 tsutsui 235 1.6 tsutsui if (verbose) { 236 1.6 tsutsui fprintf(stderr, "%s is an %s binary\n", fname, 237 1.6 tsutsui exec_formats[i].name); 238 1.7 christos if (text_start != (u_long)~0) 239 1.6 tsutsui fprintf(stderr, "kernel text loads at 0x%lx\n", 240 1.6 tsutsui text_start); 241 1.6 tsutsui } 242 1.6 tsutsui 243 1.6 tsutsui if ((*exec_formats[i].findoff)(mappedfile, sb.st_size, 244 1.7 christos addr, &valoff, text_start) != 0) 245 1.6 tsutsui errx(EXIT_FAILURE, "couldn't find file offset for %s in %s", 246 1.6 tsutsui symbol != NULL ? nl[0].n_name : "address" , fname); 247 1.6 tsutsui 248 1.6 tsutsui valp = mappedfile + valoff; 249 1.6 tsutsui 250 1.6 tsutsui if (symbol) 251 1.6 tsutsui printf("%s(0x%lx): ", symbol, addr); 252 1.1 leo else 253 1.6 tsutsui printf("0x%lx: ", addr); 254 1.1 leo 255 1.6 tsutsui switch (size) { 256 1.6 tsutsui case sizeof(uint8_t): 257 1.6 tsutsui uval8 = *(uint8_t *)valp; 258 1.6 tsutsui sval8 = *(int8_t *)valp; 259 1.6 tsutsui printf("0x%02" PRIx8 " (%" PRIu8, uval8, uval8); 260 1.6 tsutsui if (sval8 < 0) 261 1.6 tsutsui printf("/%" PRId8, sval8); 262 1.6 tsutsui printf(")"); 263 1.6 tsutsui break; 264 1.6 tsutsui case sizeof(uint16_t): 265 1.6 tsutsui uval16 = *(uint16_t *)valp; 266 1.6 tsutsui sval16 = *(int16_t *)valp; 267 1.6 tsutsui printf("0x%04" PRIx16 " (%" PRIu16, uval16, uval16); 268 1.6 tsutsui if (sval16 < 0) 269 1.6 tsutsui printf("/%" PRId16, sval16); 270 1.6 tsutsui printf(")"); 271 1.6 tsutsui break; 272 1.6 tsutsui case sizeof(uint32_t): 273 1.6 tsutsui uval32 = *(uint32_t *)valp; 274 1.6 tsutsui sval32 = *(int32_t *)valp; 275 1.6 tsutsui printf("0x%08" PRIx32 " (%" PRIu32, uval32, uval32); 276 1.6 tsutsui if (sval32 < 0) 277 1.6 tsutsui printf("/%" PRId32, sval32); 278 1.6 tsutsui printf(")"); 279 1.6 tsutsui break; 280 1.6 tsutsui case sizeof(uint64_t): 281 1.6 tsutsui uval64 = *(uint64_t *)valp; 282 1.6 tsutsui sval64 = *(int64_t *)valp; 283 1.6 tsutsui printf("0x%016" PRIx64 " (%" PRIu64, uval64, uval64); 284 1.6 tsutsui if (sval64 < 0) 285 1.6 tsutsui printf("/%" PRId64, sval64); 286 1.6 tsutsui printf(")"); 287 1.6 tsutsui break; 288 1.1 leo } 289 1.6 tsutsui printf(", at offset %#lx in %s\n", (unsigned long)valoff, fname); 290 1.1 leo 291 1.6 tsutsui if (!replace) 292 1.6 tsutsui goto done; 293 1.1 leo 294 1.6 tsutsui printf("new value: "); 295 1.1 leo 296 1.6 tsutsui switch (size) { 297 1.6 tsutsui case sizeof(uint8_t): 298 1.6 tsutsui uval8 = (uint8_t)val; 299 1.6 tsutsui sval8 = (int8_t)val; 300 1.6 tsutsui printf("0x%02" PRIx8 " (%" PRIu8, uval8, uval8); 301 1.6 tsutsui if (sval8 < 0) 302 1.6 tsutsui printf("/%" PRId8, sval8); 303 1.6 tsutsui printf(")"); 304 1.6 tsutsui *(uint8_t *)valp = uval8; 305 1.6 tsutsui break; 306 1.6 tsutsui case sizeof(uint16_t): 307 1.6 tsutsui uval16 = (uint16_t)val; 308 1.6 tsutsui sval16 = (int16_t)val; 309 1.6 tsutsui printf("0x%04" PRIx16 " (%" PRIu16, uval16, uval16); 310 1.6 tsutsui if (sval16 < 0) 311 1.6 tsutsui printf("/%" PRId16, sval16); 312 1.6 tsutsui printf(")"); 313 1.6 tsutsui *(uint16_t *)valp = uval16; 314 1.6 tsutsui break; 315 1.6 tsutsui case sizeof(uint32_t): 316 1.6 tsutsui uval32 = (uint32_t)val; 317 1.6 tsutsui sval32 = (int32_t)val; 318 1.6 tsutsui printf("0x%08" PRIx32 " (%" PRIu32, uval32, uval32); 319 1.6 tsutsui if (sval32 < 0) 320 1.6 tsutsui printf("/%" PRId32, sval32); 321 1.6 tsutsui printf(")"); 322 1.6 tsutsui *(uint32_t *)valp = uval32; 323 1.6 tsutsui break; 324 1.6 tsutsui case sizeof(uint64_t): 325 1.6 tsutsui uval64 = (uint64_t)val; 326 1.6 tsutsui sval64 = (int64_t)val; 327 1.6 tsutsui printf("0x%016" PRIx64 " (%" PRIu64, uval64, uval64); 328 1.6 tsutsui if (sval64 < 0) 329 1.6 tsutsui printf("/%" PRId64, sval64); 330 1.6 tsutsui printf(")"); 331 1.6 tsutsui *(uint64_t *)valp = uval64; 332 1.6 tsutsui break; 333 1.6 tsutsui } 334 1.6 tsutsui printf("\n"); 335 1.6 tsutsui 336 1.6 tsutsui done: 337 1.6 tsutsui munmap(mappedfile, sb.st_size); 338 1.6 tsutsui close(fd); 339 1.6 tsutsui 340 1.6 tsutsui if (verbose) 341 1.6 tsutsui fprintf(stderr, "exiting\n"); 342 1.6 tsutsui exit(EXIT_SUCCESS); 343 1.6 tsutsui } 344 1.1 leo 345 1.6 tsutsui static void 346 1.6 tsutsui usage(void) 347 1.1 leo { 348 1.6 tsutsui 349 1.6 tsutsui fprintf(stderr, 350 1.7 christos "Usage: %s [-b|-w|-l|-d] [-a address | -s symbol] [-o offset]" 351 1.7 christos " [-r value] [-T text_start] [-v] binary\n", getprogname()); 352 1.6 tsutsui exit(EXIT_FAILURE); 353 1.1 leo } 354