binpatch.c revision 1.7 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