blob: f1d5df2d6537a003e89d8ede2038b4c1d3802cfa [file] [log] [blame]
"Robert P. J. Day"63fc1a92006-07-02 19:47:05 +00001/* vi: set sw=4 ts=4: */
Glenn L McGrath60281112001-11-02 11:39:46 +00002/*
3 * Support code for the hexdump and od applets,
4 * based on code from util-linux v 2.11l
5 *
6 * Copyright (c) 1989
7 * The Regents of the University of California. All rights reserved.
8 *
Bernhard Reutner-Fischer30385572006-01-31 17:57:48 +00009 * Licensed under GPLv2 or later, see file LICENSE in this tarball for details.
Glenn L McGrath60281112001-11-02 11:39:46 +000010 *
11 * Original copyright notice is retained at the end of this file.
12 */
13
Rob Landleyea224be2006-06-18 20:20:07 +000014#include "libbb.h"
Glenn L McGrath60281112001-11-02 11:39:46 +000015#include <string.h>
Eric Andersen8ee2b272004-03-27 11:26:32 +000016#include <unistd.h>
Glenn L McGrath60281112001-11-02 11:39:46 +000017#include <ctype.h> /* for isdigit() */
Manuel Novoa III cad53642003-03-19 09:13:01 +000018#include "dump.h"
Glenn L McGrath60281112001-11-02 11:39:46 +000019
Manuel Novoa III cad53642003-03-19 09:13:01 +000020enum _vflag bb_dump_vflag = FIRST;
21FS *bb_dump_fshead; /* head of format strings */
Glenn L McGrath60281112001-11-02 11:39:46 +000022static FU *endfu;
23static char **_argv;
Glenn L McGrath9fef17d2002-08-22 18:41:20 +000024static off_t savaddress; /* saved address/offset in stream */
25static off_t eaddress; /* end address */
26static off_t address; /* address/offset in stream */
Manuel Novoa III cad53642003-03-19 09:13:01 +000027off_t bb_dump_skip; /* bytes to skip */
28static int exitval; /* final exit value */
29int bb_dump_blocksize; /* data block size */
30int bb_dump_length = -1; /* max bytes to read */
Glenn L McGrath60281112001-11-02 11:39:46 +000031
Manuel Novoa III cad53642003-03-19 09:13:01 +000032static const char index_str[] = ".#-+ 0123456789";
Glenn L McGrath60281112001-11-02 11:39:46 +000033
Manuel Novoa III cad53642003-03-19 09:13:01 +000034static const char size_conv_str[] =
35"\x1\x4\x4\x4\x4\x4\x4\x8\x8\x8\x8\010cdiouxXeEfgG";
36
37static const char lcc[] = "diouxX";
38
39int bb_dump_size(FS * fs)
Glenn L McGrath60281112001-11-02 11:39:46 +000040{
"Robert P. J. Day"68229832006-07-01 13:08:46 +000041 FU *fu;
42 int bcnt, cur_size;
43 char *fmt;
Manuel Novoa III cad53642003-03-19 09:13:01 +000044 const char *p;
Glenn L McGrath60281112001-11-02 11:39:46 +000045 int prec;
46
Manuel Novoa III cad53642003-03-19 09:13:01 +000047 /* figure out the data block bb_dump_size needed for each format unit */
48 for (cur_size = 0, fu = fs->nextfu; fu; fu = fu->nextfu) {
Glenn L McGrath60281112001-11-02 11:39:46 +000049 if (fu->bcnt) {
Manuel Novoa III cad53642003-03-19 09:13:01 +000050 cur_size += fu->bcnt * fu->reps;
Glenn L McGrath60281112001-11-02 11:39:46 +000051 continue;
52 }
53 for (bcnt = prec = 0, fmt = fu->fmt; *fmt; ++fmt) {
54 if (*fmt != '%')
55 continue;
56 /*
Manuel Novoa III cad53642003-03-19 09:13:01 +000057 * bb_dump_skip any special chars -- save precision in
Glenn L McGrath60281112001-11-02 11:39:46 +000058 * case it's a %s format.
59 */
Manuel Novoa III cad53642003-03-19 09:13:01 +000060 while (strchr(index_str + 1, *++fmt));
Glenn L McGrath60281112001-11-02 11:39:46 +000061 if (*fmt == '.' && isdigit(*++fmt)) {
62 prec = atoi(fmt);
63 while (isdigit(*++fmt));
64 }
Manuel Novoa III cad53642003-03-19 09:13:01 +000065 if (!(p = strchr(size_conv_str + 12, *fmt))) {
66 if (*fmt == 's') {
67 bcnt += prec;
68 } else if (*fmt == '_') {
69 ++fmt;
70 if ((*fmt == 'c') || (*fmt == 'p') || (*fmt == 'u')) {
71 bcnt += 1;
72 }
Glenn L McGrath60281112001-11-02 11:39:46 +000073 }
Manuel Novoa III cad53642003-03-19 09:13:01 +000074 } else {
75 bcnt += size_conv_str[p - (size_conv_str + 12)];
Glenn L McGrath60281112001-11-02 11:39:46 +000076 }
77 }
Manuel Novoa III cad53642003-03-19 09:13:01 +000078 cur_size += bcnt * fu->reps;
Glenn L McGrath60281112001-11-02 11:39:46 +000079 }
Manuel Novoa III cad53642003-03-19 09:13:01 +000080 return (cur_size);
Glenn L McGrath60281112001-11-02 11:39:46 +000081}
82
Manuel Novoa III cad53642003-03-19 09:13:01 +000083static void rewrite(FS * fs)
Glenn L McGrath60281112001-11-02 11:39:46 +000084{
85 enum { NOTOKAY, USEBCNT, USEPREC } sokay;
Rob Landleyea224be2006-06-18 20:20:07 +000086 PR *pr, **nextpr = NULL;
87 FU *fu;
88 char *p1, *p2, *p3;
Glenn L McGrath60281112001-11-02 11:39:46 +000089 char savech, *fmtp;
Manuel Novoa III cad53642003-03-19 09:13:01 +000090 const char *byte_count_str;
Glenn L McGrath60281112001-11-02 11:39:46 +000091 int nconv, prec = 0;
92
93 for (fu = fs->nextfu; fu; fu = fu->nextfu) {
94 /*
95 * break each format unit into print units; each
96 * conversion character gets its own.
97 */
98 for (nconv = 0, fmtp = fu->fmt; *fmtp; nextpr = &pr->nextpr) {
99 /* NOSTRICT */
Eric Andersen4a11e0f2003-04-19 23:18:35 +0000100 /* DBU:[dvae@cray.com] calloc so that forward ptrs start out NULL*/
Rob Landleyea224be2006-06-18 20:20:07 +0000101 pr = xzalloc(sizeof(PR));
Glenn L McGrath60281112001-11-02 11:39:46 +0000102 if (!fu->nextpr)
103 fu->nextpr = pr;
Eric Andersen4a11e0f2003-04-19 23:18:35 +0000104 /* ignore nextpr -- its unused inside the loop and is
105 * uninitialized 1st time thru.
106 */
Glenn L McGrath60281112001-11-02 11:39:46 +0000107
Manuel Novoa III cad53642003-03-19 09:13:01 +0000108 /* bb_dump_skip preceding text and up to the next % sign */
Glenn L McGrath60281112001-11-02 11:39:46 +0000109 for (p1 = fmtp; *p1 && *p1 != '%'; ++p1);
110
111 /* only text in the string */
112 if (!*p1) {
113 pr->fmt = fmtp;
114 pr->flags = F_TEXT;
115 break;
116 }
117
118 /*
119 * get precision for %s -- if have a byte count, don't
120 * need it.
121 */
122 if (fu->bcnt) {
123 sokay = USEBCNT;
Manuel Novoa III cad53642003-03-19 09:13:01 +0000124 /* bb_dump_skip to conversion character */
125 for (++p1; strchr(index_str, *p1); ++p1);
Glenn L McGrath60281112001-11-02 11:39:46 +0000126 } else {
Manuel Novoa III cad53642003-03-19 09:13:01 +0000127 /* bb_dump_skip any special chars, field width */
128 while (strchr(index_str + 1, *++p1));
Glenn L McGrath60281112001-11-02 11:39:46 +0000129 if (*p1 == '.' && isdigit(*++p1)) {
130 sokay = USEPREC;
131 prec = atoi(p1);
132 while (isdigit(*++p1));
Glenn L McGrath9fef17d2002-08-22 18:41:20 +0000133 } else
Glenn L McGrath60281112001-11-02 11:39:46 +0000134 sokay = NOTOKAY;
135 }
136
Glenn L McGrath9fef17d2002-08-22 18:41:20 +0000137 p2 = p1 + 1; /* set end pointer */
Glenn L McGrath60281112001-11-02 11:39:46 +0000138
139 /*
140 * figure out the byte count for each conversion;
141 * rewrite the format as necessary, set up blank-
Manuel Novoa III cad53642003-03-19 09:13:01 +0000142 * pbb_dump_adding for end of data.
Glenn L McGrath60281112001-11-02 11:39:46 +0000143 */
Manuel Novoa III cad53642003-03-19 09:13:01 +0000144
145 if (*p1 == 'c') {
Glenn L McGrath60281112001-11-02 11:39:46 +0000146 pr->flags = F_CHAR;
Manuel Novoa III cad53642003-03-19 09:13:01 +0000147 DO_BYTE_COUNT_1:
148 byte_count_str = "\001";
149 DO_BYTE_COUNT:
150 if (fu->bcnt) {
151 do {
152 if (fu->bcnt == *byte_count_str) {
153 break;
154 }
155 } while (*++byte_count_str);
Glenn L McGrath60281112001-11-02 11:39:46 +0000156 }
Manuel Novoa III cad53642003-03-19 09:13:01 +0000157 /* Unlike the original, output the remainder of the format string. */
158 if (!*byte_count_str) {
159 bb_error_msg_and_die("bad byte count for conversion character %s.", p1);
160 }
161 pr->bcnt = *byte_count_str;
162 } else if (*p1 == 'l') {
Glenn L McGrath60281112001-11-02 11:39:46 +0000163 ++p2;
Manuel Novoa III cad53642003-03-19 09:13:01 +0000164 ++p1;
165 DO_INT_CONV:
166 {
167 const char *e;
168 if (!(e = strchr(lcc, *p1))) {
169 goto DO_BAD_CONV_CHAR;
170 }
Glenn L McGrath60281112001-11-02 11:39:46 +0000171 pr->flags = F_INT;
Manuel Novoa III cad53642003-03-19 09:13:01 +0000172 if (e > lcc + 1) {
173 pr->flags = F_UINT;
174 }
175 byte_count_str = "\004\002\001";
176 goto DO_BYTE_COUNT;
Glenn L McGrath60281112001-11-02 11:39:46 +0000177 }
178 /* NOTREACHED */
Manuel Novoa III cad53642003-03-19 09:13:01 +0000179 } else if (strchr(lcc, *p1)) {
180 goto DO_INT_CONV;
181 } else if (strchr("eEfgG", *p1)) {
Glenn L McGrath60281112001-11-02 11:39:46 +0000182 pr->flags = F_DBL;
Manuel Novoa III cad53642003-03-19 09:13:01 +0000183 byte_count_str = "\010\004";
184 goto DO_BYTE_COUNT;
185 } else if (*p1 == 's') {
Glenn L McGrath60281112001-11-02 11:39:46 +0000186 pr->flags = F_STR;
Manuel Novoa III cad53642003-03-19 09:13:01 +0000187 if (sokay == USEBCNT) {
Glenn L McGrath60281112001-11-02 11:39:46 +0000188 pr->bcnt = fu->bcnt;
Manuel Novoa III cad53642003-03-19 09:13:01 +0000189 } else if (sokay == USEPREC) {
Glenn L McGrath60281112001-11-02 11:39:46 +0000190 pr->bcnt = prec;
Manuel Novoa III cad53642003-03-19 09:13:01 +0000191 } else { /* NOTOKAY */
192 bb_error_msg_and_die("%%s requires a precision or a byte count.");
Glenn L McGrath60281112001-11-02 11:39:46 +0000193 }
Manuel Novoa III cad53642003-03-19 09:13:01 +0000194 } else if (*p1 == '_') {
Glenn L McGrath60281112001-11-02 11:39:46 +0000195 ++p2;
Glenn L McGrath9fef17d2002-08-22 18:41:20 +0000196 switch (p1[1]) {
Glenn L McGrath60281112001-11-02 11:39:46 +0000197 case 'A':
198 endfu = fu;
199 fu->flags |= F_IGNORE;
200 /* FALLTHROUGH */
201 case 'a':
202 pr->flags = F_ADDRESS;
203 ++p2;
Manuel Novoa III cad53642003-03-19 09:13:01 +0000204 if ((p1[2] != 'd') && (p1[2] != 'o') && (p1[2] != 'x')) {
205 goto DO_BAD_CONV_CHAR;
Glenn L McGrath60281112001-11-02 11:39:46 +0000206 }
Manuel Novoa III cad53642003-03-19 09:13:01 +0000207 *p1 = p1[2];
Glenn L McGrath60281112001-11-02 11:39:46 +0000208 break;
209 case 'c':
210 pr->flags = F_C;
Glenn L McGrath9fef17d2002-08-22 18:41:20 +0000211 /* *p1 = 'c'; set in conv_c */
Manuel Novoa III cad53642003-03-19 09:13:01 +0000212 goto DO_BYTE_COUNT_1;
Glenn L McGrath60281112001-11-02 11:39:46 +0000213 case 'p':
214 pr->flags = F_P;
215 *p1 = 'c';
Manuel Novoa III cad53642003-03-19 09:13:01 +0000216 goto DO_BYTE_COUNT_1;
Glenn L McGrath60281112001-11-02 11:39:46 +0000217 case 'u':
218 pr->flags = F_U;
Glenn L McGrath9fef17d2002-08-22 18:41:20 +0000219 /* *p1 = 'c'; set in conv_u */
Manuel Novoa III cad53642003-03-19 09:13:01 +0000220 goto DO_BYTE_COUNT_1;
Glenn L McGrath60281112001-11-02 11:39:46 +0000221 default:
Manuel Novoa III cad53642003-03-19 09:13:01 +0000222 goto DO_BAD_CONV_CHAR;
Glenn L McGrath60281112001-11-02 11:39:46 +0000223 }
Manuel Novoa III cad53642003-03-19 09:13:01 +0000224 } else {
225 DO_BAD_CONV_CHAR:
226 bb_error_msg_and_die("bad conversion character %%%s.\n", p1);
Glenn L McGrath60281112001-11-02 11:39:46 +0000227 }
228
229 /*
230 * copy to PR format string, set conversion character
231 * pointer, update original.
232 */
233 savech = *p2;
234 p1[1] = '\0';
Manuel Novoa III cad53642003-03-19 09:13:01 +0000235 pr->fmt = bb_xstrdup(fmtp);
Glenn L McGrath60281112001-11-02 11:39:46 +0000236 *p2 = savech;
237 pr->cchar = pr->fmt + (p1 - fmtp);
Eric Andersen4a11e0f2003-04-19 23:18:35 +0000238
239 /* DBU:[dave@cray.com] w/o this, trailing fmt text, space is lost.
Eric Andersenc7bda1c2004-03-15 08:29:22 +0000240 * Skip subsequent text and up to the next % sign and tack the
241 * additional text onto fmt: eg. if fmt is "%x is a HEX number",
Eric Andersen4a11e0f2003-04-19 23:18:35 +0000242 * we lose the " is a HEX number" part of fmt.
243 */
244 for (p3 = p2; *p3 && *p3 != '%'; p3++);
245 if (p3 > p2)
246 {
247 savech = *p3;
248 *p3 = '\0';
Rob Landleyea224be2006-06-18 20:20:07 +0000249 pr->fmt = xrealloc(pr->fmt, strlen(pr->fmt)+(p3-p2)+1);
Eric Andersen4a11e0f2003-04-19 23:18:35 +0000250 strcat(pr->fmt, p2);
251 *p3 = savech;
252 p2 = p3;
253 }
254
Glenn L McGrath60281112001-11-02 11:39:46 +0000255 fmtp = p2;
256
257 /* only one conversion character if byte count */
Glenn L McGrath9fef17d2002-08-22 18:41:20 +0000258 if (!(pr->flags & F_ADDRESS) && fu->bcnt && nconv++) {
Manuel Novoa III cad53642003-03-19 09:13:01 +0000259 bb_error_msg_and_die("byte count with multiple conversion characters.\n");
Glenn L McGrath60281112001-11-02 11:39:46 +0000260 }
261 }
262 /*
263 * if format unit byte count not specified, figure it out
264 * so can adjust rep count later.
265 */
266 if (!fu->bcnt)
267 for (pr = fu->nextpr; pr; pr = pr->nextpr)
268 fu->bcnt += pr->bcnt;
269 }
270 /*
271 * if the format string interprets any data at all, and it's
Manuel Novoa III cad53642003-03-19 09:13:01 +0000272 * not the same as the bb_dump_blocksize, and its last format unit
Glenn L McGrath60281112001-11-02 11:39:46 +0000273 * interprets any data at all, and has no iteration count,
274 * repeat it as necessary.
275 *
276 * if, rep count is greater than 1, no trailing whitespace
277 * gets output from the last iteration of the format unit.
278 */
279 for (fu = fs->nextfu;; fu = fu->nextfu) {
Manuel Novoa III cad53642003-03-19 09:13:01 +0000280 if (!fu->nextfu && fs->bcnt < bb_dump_blocksize &&
Glenn L McGrath9fef17d2002-08-22 18:41:20 +0000281 !(fu->flags & F_SETREP) && fu->bcnt)
Manuel Novoa III cad53642003-03-19 09:13:01 +0000282 fu->reps += (bb_dump_blocksize - fs->bcnt) / fu->bcnt;
Glenn L McGrath60281112001-11-02 11:39:46 +0000283 if (fu->reps > 1) {
284 for (pr = fu->nextpr;; pr = pr->nextpr)
285 if (!pr->nextpr)
286 break;
287 for (p1 = pr->fmt, p2 = NULL; *p1; ++p1)
288 p2 = isspace(*p1) ? p1 : NULL;
289 if (p2)
290 pr->nospace = p2;
291 }
292 if (!fu->nextfu)
293 break;
294 }
295}
296
Manuel Novoa III cad53642003-03-19 09:13:01 +0000297static void do_skip(char *fname, int statok)
Glenn L McGrath60281112001-11-02 11:39:46 +0000298{
299 struct stat sbuf;
300
301 if (statok) {
Eric Andersen70060d22004-03-27 10:02:48 +0000302 if (fstat(STDIN_FILENO, &sbuf)) {
Manuel Novoa III cad53642003-03-19 09:13:01 +0000303 bb_perror_msg_and_die("%s", fname);
Glenn L McGrath60281112001-11-02 11:39:46 +0000304 }
Glenn L McGrath9fef17d2002-08-22 18:41:20 +0000305 if ((!(S_ISCHR(sbuf.st_mode) ||
306 S_ISBLK(sbuf.st_mode) ||
Manuel Novoa III cad53642003-03-19 09:13:01 +0000307 S_ISFIFO(sbuf.st_mode))) && bb_dump_skip >= sbuf.st_size) {
308 /* If bb_dump_size valid and bb_dump_skip >= size */
309 bb_dump_skip -= sbuf.st_size;
Glenn L McGrath60281112001-11-02 11:39:46 +0000310 address += sbuf.st_size;
311 return;
312 }
313 }
Manuel Novoa III cad53642003-03-19 09:13:01 +0000314 if (fseek(stdin, bb_dump_skip, SEEK_SET)) {
315 bb_perror_msg_and_die("%s", fname);
Glenn L McGrath60281112001-11-02 11:39:46 +0000316 }
Manuel Novoa III cad53642003-03-19 09:13:01 +0000317 savaddress = address += bb_dump_skip;
318 bb_dump_skip = 0;
Glenn L McGrath60281112001-11-02 11:39:46 +0000319}
320
Manuel Novoa III cad53642003-03-19 09:13:01 +0000321static int next(char **argv)
Glenn L McGrath60281112001-11-02 11:39:46 +0000322{
323 static int done;
324 int statok;
325
326 if (argv) {
327 _argv = argv;
Glenn L McGrath9fef17d2002-08-22 18:41:20 +0000328 return (1);
Glenn L McGrath60281112001-11-02 11:39:46 +0000329 }
330 for (;;) {
331 if (*_argv) {
332 if (!(freopen(*_argv, "r", stdin))) {
Manuel Novoa III cad53642003-03-19 09:13:01 +0000333 bb_perror_msg("%s", *_argv);
Glenn L McGrath60281112001-11-02 11:39:46 +0000334 exitval = 1;
335 ++_argv;
336 continue;
337 }
338 statok = done = 1;
339 } else {
340 if (done++)
Glenn L McGrath9fef17d2002-08-22 18:41:20 +0000341 return (0);
Glenn L McGrath60281112001-11-02 11:39:46 +0000342 statok = 0;
343 }
Manuel Novoa III cad53642003-03-19 09:13:01 +0000344 if (bb_dump_skip)
345 do_skip(statok ? *_argv : "stdin", statok);
Glenn L McGrath60281112001-11-02 11:39:46 +0000346 if (*_argv)
347 ++_argv;
Manuel Novoa III cad53642003-03-19 09:13:01 +0000348 if (!bb_dump_skip)
Glenn L McGrath9fef17d2002-08-22 18:41:20 +0000349 return (1);
Glenn L McGrath60281112001-11-02 11:39:46 +0000350 }
351 /* NOTREACHED */
352}
353
Mike Frysinger294254c2006-02-19 22:59:12 +0000354static unsigned char *get(void)
Glenn L McGrath60281112001-11-02 11:39:46 +0000355{
356 static int ateof = 1;
Mike Frysinger294254c2006-02-19 22:59:12 +0000357 static unsigned char *curp=NULL, *savp; /*DBU:[dave@cray.com]initialize curp */
"Robert P. J. Day"68229832006-07-01 13:08:46 +0000358 int n;
Glenn L McGrath60281112001-11-02 11:39:46 +0000359 int need, nread;
Mike Frysinger294254c2006-02-19 22:59:12 +0000360 unsigned char *tmpp;
Glenn L McGrath60281112001-11-02 11:39:46 +0000361
362 if (!curp) {
Eric Andersen4a11e0f2003-04-19 23:18:35 +0000363 address = (off_t)0; /*DBU:[dave@cray.com] initialize,initialize..*/
Mike Frysinger294254c2006-02-19 22:59:12 +0000364 curp = (unsigned char *) xmalloc(bb_dump_blocksize);
365 savp = (unsigned char *) xmalloc(bb_dump_blocksize);
Glenn L McGrath60281112001-11-02 11:39:46 +0000366 } else {
367 tmpp = curp;
368 curp = savp;
369 savp = tmpp;
Manuel Novoa III cad53642003-03-19 09:13:01 +0000370 address = savaddress += bb_dump_blocksize;
Glenn L McGrath60281112001-11-02 11:39:46 +0000371 }
Manuel Novoa III cad53642003-03-19 09:13:01 +0000372 for (need = bb_dump_blocksize, nread = 0;;) {
Glenn L McGrath60281112001-11-02 11:39:46 +0000373 /*
374 * if read the right number of bytes, or at EOF for one file,
375 * and no other files are available, zero-pad the rest of the
376 * block and set the end flag.
377 */
Manuel Novoa III cad53642003-03-19 09:13:01 +0000378 if (!bb_dump_length || (ateof && !next((char **) NULL))) {
379 if (need == bb_dump_blocksize) {
Mike Frysinger294254c2006-02-19 22:59:12 +0000380 return ((unsigned char *) NULL);
Glenn L McGrath60281112001-11-02 11:39:46 +0000381 }
Mike Frysinger78bd5042006-04-16 05:51:47 +0000382 if (bb_dump_vflag != ALL && !memcmp(curp, savp, nread)) {
Manuel Novoa III cad53642003-03-19 09:13:01 +0000383 if (bb_dump_vflag != DUP) {
Glenn L McGrath60281112001-11-02 11:39:46 +0000384 printf("*\n");
385 }
Mike Frysinger294254c2006-02-19 22:59:12 +0000386 return ((unsigned char *) NULL);
Glenn L McGrath60281112001-11-02 11:39:46 +0000387 }
Bernhard Reutner-Fischer30385572006-01-31 17:57:48 +0000388 memset((char *) curp + nread, 0, need);
Glenn L McGrath60281112001-11-02 11:39:46 +0000389 eaddress = address + nread;
Glenn L McGrath9fef17d2002-08-22 18:41:20 +0000390 return (curp);
Glenn L McGrath60281112001-11-02 11:39:46 +0000391 }
Mike Frysinger294254c2006-02-19 22:59:12 +0000392 n = fread((char *) curp + nread, sizeof(unsigned char),
Manuel Novoa III cad53642003-03-19 09:13:01 +0000393 bb_dump_length == -1 ? need : MIN(bb_dump_length, need), stdin);
Glenn L McGrath60281112001-11-02 11:39:46 +0000394 if (!n) {
395 if (ferror(stdin)) {
Manuel Novoa III cad53642003-03-19 09:13:01 +0000396 bb_perror_msg("%s", _argv[-1]);
Glenn L McGrath60281112001-11-02 11:39:46 +0000397 }
398 ateof = 1;
399 continue;
400 }
401 ateof = 0;
Manuel Novoa III cad53642003-03-19 09:13:01 +0000402 if (bb_dump_length != -1) {
403 bb_dump_length -= n;
Glenn L McGrath60281112001-11-02 11:39:46 +0000404 }
405 if (!(need -= n)) {
Manuel Novoa III cad53642003-03-19 09:13:01 +0000406 if (bb_dump_vflag == ALL || bb_dump_vflag == FIRST
Mike Frysinger78bd5042006-04-16 05:51:47 +0000407 || memcmp(curp, savp, bb_dump_blocksize)) {
Manuel Novoa III cad53642003-03-19 09:13:01 +0000408 if (bb_dump_vflag == DUP || bb_dump_vflag == FIRST) {
409 bb_dump_vflag = WAIT;
Glenn L McGrath60281112001-11-02 11:39:46 +0000410 }
Glenn L McGrath9fef17d2002-08-22 18:41:20 +0000411 return (curp);
Glenn L McGrath60281112001-11-02 11:39:46 +0000412 }
Manuel Novoa III cad53642003-03-19 09:13:01 +0000413 if (bb_dump_vflag == WAIT) {
Glenn L McGrath60281112001-11-02 11:39:46 +0000414 printf("*\n");
415 }
Manuel Novoa III cad53642003-03-19 09:13:01 +0000416 bb_dump_vflag = DUP;
417 address = savaddress += bb_dump_blocksize;
418 need = bb_dump_blocksize;
Glenn L McGrath60281112001-11-02 11:39:46 +0000419 nread = 0;
420 } else {
421 nread += n;
422 }
423 }
424}
425
Glenn L McGrath9fef17d2002-08-22 18:41:20 +0000426static void bpad(PR * pr)
Glenn L McGrath60281112001-11-02 11:39:46 +0000427{
Rob Landleya6e60372006-06-28 14:36:50 +0000428 char *p1, *p2;
Glenn L McGrath60281112001-11-02 11:39:46 +0000429
430 /*
431 * remove all conversion flags; '-' is the only one valid
432 * with %s, and it's not useful here.
433 */
434 pr->flags = F_BPAD;
435 *pr->cchar = 's';
436 for (p1 = pr->fmt; *p1 != '%'; ++p1);
Rob Landleya6e60372006-06-28 14:36:50 +0000437 for (p2 = ++p1; *p1 && strchr(" -0+#", *p1); ++p1)
438 if (pr->nospace) pr->nospace--;
Glenn L McGrath9fef17d2002-08-22 18:41:20 +0000439 while ((*p2++ = *p1++) != 0);
Glenn L McGrath60281112001-11-02 11:39:46 +0000440}
441
Manuel Novoa III cad53642003-03-19 09:13:01 +0000442static const char conv_str[] =
443 "\0\\0\0"
444 "\007\\a\0" /* \a */
445 "\b\\b\0"
446 "\f\\b\0"
447 "\n\\n\0"
448 "\r\\r\0"
449 "\t\\t\0"
450 "\v\\v\0"
451 "\0";
Glenn L McGrath60281112001-11-02 11:39:46 +0000452
Manuel Novoa III cad53642003-03-19 09:13:01 +0000453
Mike Frysinger294254c2006-02-19 22:59:12 +0000454static void conv_c(PR * pr, unsigned char * p)
Manuel Novoa III cad53642003-03-19 09:13:01 +0000455{
456 const char *str = conv_str;
457 char buf[10];
458
459 do {
460 if (*p == *str) {
461 ++str;
462 goto strpr;
463 }
464 str += 4;
465 } while (*str);
466
Glenn L McGrath60281112001-11-02 11:39:46 +0000467 if (isprint(*p)) {
468 *pr->cchar = 'c';
Glenn L McGrath9fef17d2002-08-22 18:41:20 +0000469 (void) printf(pr->fmt, *p);
Glenn L McGrath60281112001-11-02 11:39:46 +0000470 } else {
Manuel Novoa III cad53642003-03-19 09:13:01 +0000471 sprintf(buf, "%03o", (int) *p);
472 str = buf;
Glenn L McGrath9fef17d2002-08-22 18:41:20 +0000473 strpr:
Glenn L McGrath60281112001-11-02 11:39:46 +0000474 *pr->cchar = 's';
475 printf(pr->fmt, str);
476 }
477}
478
Mike Frysinger294254c2006-02-19 22:59:12 +0000479static void conv_u(PR * pr, unsigned char * p)
Glenn L McGrath60281112001-11-02 11:39:46 +0000480{
Manuel Novoa III cad53642003-03-19 09:13:01 +0000481 static const char list[] =
482 "nul\0soh\0stx\0etx\0eot\0enq\0ack\0bel\0"
483 "bs\0_ht\0_lf\0_vt\0_ff\0_cr\0_so\0_si\0_"
484 "dle\0dcl\0dc2\0dc3\0dc4\0nak\0syn\0etb\0"
485 "can\0em\0_sub\0esc\0fs\0_gs\0_rs\0_us";
Glenn L McGrath60281112001-11-02 11:39:46 +0000486
487 /* od used nl, not lf */
488 if (*p <= 0x1f) {
489 *pr->cchar = 's';
Glenn L McGratheeb06bf2004-07-23 01:35:41 +0000490 printf(pr->fmt, list + (4 * (int)*p));
Glenn L McGrath60281112001-11-02 11:39:46 +0000491 } else if (*p == 0x7f) {
492 *pr->cchar = 's';
493 printf(pr->fmt, "del");
494 } else if (isprint(*p)) {
495 *pr->cchar = 'c';
496 printf(pr->fmt, *p);
497 } else {
498 *pr->cchar = 'x';
Glenn L McGrath9fef17d2002-08-22 18:41:20 +0000499 printf(pr->fmt, (int) *p);
Glenn L McGrath60281112001-11-02 11:39:46 +0000500 }
501}
502
Manuel Novoa III cad53642003-03-19 09:13:01 +0000503static void display(void)
Glenn L McGrath60281112001-11-02 11:39:46 +0000504{
Glenn L McGrath9fef17d2002-08-22 18:41:20 +0000505/* extern FU *endfu; */
"Robert P. J. Day"68229832006-07-01 13:08:46 +0000506 FS *fs;
507 FU *fu;
508 PR *pr;
509 int cnt;
510 unsigned char *bp;
Glenn L McGrath9fef17d2002-08-22 18:41:20 +0000511
Manuel Novoa III cad53642003-03-19 09:13:01 +0000512 off_t saveaddress;
Mike Frysinger294254c2006-02-19 22:59:12 +0000513 unsigned char savech = 0, *savebp;
Glenn L McGrath60281112001-11-02 11:39:46 +0000514
515 while ((bp = get()) != NULL) {
Manuel Novoa III cad53642003-03-19 09:13:01 +0000516 for (fs = bb_dump_fshead, savebp = bp, saveaddress = address; fs;
Glenn L McGrath9fef17d2002-08-22 18:41:20 +0000517 fs = fs->nextfs, bp = savebp, address = saveaddress) {
518 for (fu = fs->nextfu; fu; fu = fu->nextfu) {
Glenn L McGrath60281112001-11-02 11:39:46 +0000519 if (fu->flags & F_IGNORE) {
520 break;
521 }
522 for (cnt = fu->reps; cnt; --cnt) {
Glenn L McGrath9fef17d2002-08-22 18:41:20 +0000523 for (pr = fu->nextpr; pr; address += pr->bcnt,
524 bp += pr->bcnt, pr = pr->nextpr) {
525 if (eaddress && address >= eaddress &&
526 !(pr->flags & (F_TEXT | F_BPAD))) {
Glenn L McGrath60281112001-11-02 11:39:46 +0000527 bpad(pr);
528 }
Glenn L McGrath9fef17d2002-08-22 18:41:20 +0000529 if (cnt == 1 && pr->nospace) {
Glenn L McGrath60281112001-11-02 11:39:46 +0000530 savech = *pr->nospace;
531 *pr->nospace = '\0';
Glenn L McGrath60281112001-11-02 11:39:46 +0000532 }
Glenn L McGrath9fef17d2002-08-22 18:41:20 +0000533/* PRINT; */
534 switch (pr->flags) {
535 case F_ADDRESS:
Glenn L McGrathff5309a2004-05-02 08:38:53 +0000536 printf(pr->fmt, (unsigned int) address);
Glenn L McGrath9fef17d2002-08-22 18:41:20 +0000537 break;
538 case F_BPAD:
539 printf(pr->fmt, "");
540 break;
541 case F_C:
542 conv_c(pr, bp);
543 break;
544 case F_CHAR:
545 printf(pr->fmt, *bp);
546 break;
547 case F_DBL:{
548 double dval;
549 float fval;
550
551 switch (pr->bcnt) {
552 case 4:
Mike Frysinger1a540302006-04-16 05:58:21 +0000553 memmove((char *) &fval, (char *) bp,
Glenn L McGrath9fef17d2002-08-22 18:41:20 +0000554 sizeof(fval));
555 printf(pr->fmt, fval);
556 break;
557 case 8:
Mike Frysinger1a540302006-04-16 05:58:21 +0000558 memmove((char *) &dval, (char *) bp,
Glenn L McGrath9fef17d2002-08-22 18:41:20 +0000559 sizeof(dval));
560 printf(pr->fmt, dval);
561 break;
562 }
563 break;
564 }
565 case F_INT:{
566 int ival;
567 short sval;
568
569 switch (pr->bcnt) {
570 case 1:
571 printf(pr->fmt, (int) *bp);
572 break;
573 case 2:
Mike Frysinger1a540302006-04-16 05:58:21 +0000574 memmove((char *) &sval, (char *) bp,
Glenn L McGrath9fef17d2002-08-22 18:41:20 +0000575 sizeof(sval));
576 printf(pr->fmt, (int) sval);
577 break;
578 case 4:
Mike Frysinger1a540302006-04-16 05:58:21 +0000579 memmove((char *) &ival, (char *) bp,
Glenn L McGrath9fef17d2002-08-22 18:41:20 +0000580 sizeof(ival));
581 printf(pr->fmt, ival);
582 break;
583 }
584 break;
585 }
586 case F_P:
587 printf(pr->fmt, isprint(*bp) ? *bp : '.');
588 break;
589 case F_STR:
590 printf(pr->fmt, (char *) bp);
591 break;
592 case F_TEXT:
593 printf(pr->fmt);
594 break;
595 case F_U:
596 conv_u(pr, bp);
597 break;
598 case F_UINT:{
Eric Andersen0f56de62004-01-30 22:52:27 +0000599 unsigned int ival;
600 unsigned short sval;
Glenn L McGrath9fef17d2002-08-22 18:41:20 +0000601
602 switch (pr->bcnt) {
603 case 1:
Eric Andersen0f56de62004-01-30 22:52:27 +0000604 printf(pr->fmt, (unsigned int) * bp);
Glenn L McGrath9fef17d2002-08-22 18:41:20 +0000605 break;
606 case 2:
Mike Frysinger1a540302006-04-16 05:58:21 +0000607 memmove((char *) &sval, (char *) bp,
Glenn L McGrath9fef17d2002-08-22 18:41:20 +0000608 sizeof(sval));
Eric Andersen0f56de62004-01-30 22:52:27 +0000609 printf(pr->fmt, (unsigned int) sval);
Glenn L McGrath9fef17d2002-08-22 18:41:20 +0000610 break;
611 case 4:
Mike Frysinger1a540302006-04-16 05:58:21 +0000612 memmove((char *) &ival, (char *) bp,
Glenn L McGrath9fef17d2002-08-22 18:41:20 +0000613 sizeof(ival));
614 printf(pr->fmt, ival);
615 break;
616 }
617 break;
618 }
619 }
620 if (cnt == 1 && pr->nospace) {
Glenn L McGrath60281112001-11-02 11:39:46 +0000621 *pr->nospace = savech;
622 }
623 }
624 }
625 }
626 }
627 }
628 if (endfu) {
629 /*
Manuel Novoa III cad53642003-03-19 09:13:01 +0000630 * if eaddress not set, error or file bb_dump_size was multiple of
631 * bb_dump_blocksize, and no partial block ever found.
Glenn L McGrath60281112001-11-02 11:39:46 +0000632 */
633 if (!eaddress) {
634 if (!address) {
635 return;
636 }
637 eaddress = address;
638 }
639 for (pr = endfu->nextpr; pr; pr = pr->nextpr) {
Glenn L McGrath9fef17d2002-08-22 18:41:20 +0000640 switch (pr->flags) {
Glenn L McGrath60281112001-11-02 11:39:46 +0000641 case F_ADDRESS:
Glenn L McGrathd2b860f2004-03-05 05:47:19 +0000642 (void) printf(pr->fmt, (unsigned int) eaddress);
Glenn L McGrath60281112001-11-02 11:39:46 +0000643 break;
644 case F_TEXT:
Glenn L McGrath9fef17d2002-08-22 18:41:20 +0000645 (void) printf(pr->fmt);
Glenn L McGrath60281112001-11-02 11:39:46 +0000646 break;
647 }
648 }
649 }
650}
651
Manuel Novoa III cad53642003-03-19 09:13:01 +0000652int bb_dump_dump(char **argv)
Glenn L McGrath60281112001-11-02 11:39:46 +0000653{
"Robert P. J. Day"68229832006-07-01 13:08:46 +0000654 FS *tfs;
Glenn L McGrath60281112001-11-02 11:39:46 +0000655
Manuel Novoa III cad53642003-03-19 09:13:01 +0000656 /* figure out the data block bb_dump_size */
657 for (bb_dump_blocksize = 0, tfs = bb_dump_fshead; tfs; tfs = tfs->nextfs) {
658 tfs->bcnt = bb_dump_size(tfs);
659 if (bb_dump_blocksize < tfs->bcnt) {
660 bb_dump_blocksize = tfs->bcnt;
Glenn L McGrath60281112001-11-02 11:39:46 +0000661 }
662 }
663 /* rewrite the rules, do syntax checking */
Manuel Novoa III cad53642003-03-19 09:13:01 +0000664 for (tfs = bb_dump_fshead; tfs; tfs = tfs->nextfs) {
Glenn L McGrath60281112001-11-02 11:39:46 +0000665 rewrite(tfs);
666 }
667
668 next(argv);
669 display();
670
Glenn L McGrath9fef17d2002-08-22 18:41:20 +0000671 return (exitval);
Glenn L McGrath60281112001-11-02 11:39:46 +0000672}
673
Manuel Novoa III cad53642003-03-19 09:13:01 +0000674void bb_dump_add(const char *fmt)
Glenn L McGrath60281112001-11-02 11:39:46 +0000675{
Rob Landleyea224be2006-06-18 20:20:07 +0000676 const char *p;
677 char *p1;
678 char *p2;
Glenn L McGrath60281112001-11-02 11:39:46 +0000679 static FS **nextfs;
680 FS *tfs;
681 FU *tfu, **nextfu;
Manuel Novoa III cad53642003-03-19 09:13:01 +0000682 const char *savep;
Glenn L McGrath60281112001-11-02 11:39:46 +0000683
684 /* start new linked list of format units */
Rob Landleyea224be2006-06-18 20:20:07 +0000685 tfs = xzalloc(sizeof(FS)); /*DBU:[dave@cray.com] start out NULL */
Manuel Novoa III cad53642003-03-19 09:13:01 +0000686 if (!bb_dump_fshead) {
687 bb_dump_fshead = tfs;
Glenn L McGrath60281112001-11-02 11:39:46 +0000688 } else {
689 *nextfs = tfs;
690 }
691 nextfs = &tfs->nextfs;
692 nextfu = &tfs->nextfu;
693
694 /* take the format string and break it up into format units */
695 for (p = fmt;;) {
Manuel Novoa III cad53642003-03-19 09:13:01 +0000696 /* bb_dump_skip leading white space */
Rob Landleyea224be2006-06-18 20:20:07 +0000697 p = skip_whitespace(p);
Glenn L McGrath60281112001-11-02 11:39:46 +0000698 if (!*p) {
699 break;
700 }
701
702 /* allocate a new format unit and link it in */
703 /* NOSTRICT */
Eric Andersen4a11e0f2003-04-19 23:18:35 +0000704 /* DBU:[dave@cray.com] calloc so that forward pointers start out NULL */
Rob Landleyea224be2006-06-18 20:20:07 +0000705 tfu = xzalloc(sizeof(FU));
Glenn L McGrath60281112001-11-02 11:39:46 +0000706 *nextfu = tfu;
707 nextfu = &tfu->nextfu;
708 tfu->reps = 1;
709
710 /* if leading digit, repetition count */
711 if (isdigit(*p)) {
712 for (savep = p; isdigit(*p); ++p);
713 if (!isspace(*p) && *p != '/') {
Manuel Novoa III cad53642003-03-19 09:13:01 +0000714 bb_error_msg_and_die("bad format {%s}", fmt);
Glenn L McGrath60281112001-11-02 11:39:46 +0000715 }
716 /* may overwrite either white space or slash */
717 tfu->reps = atoi(savep);
718 tfu->flags = F_SETREP;
Manuel Novoa III cad53642003-03-19 09:13:01 +0000719 /* bb_dump_skip trailing white space */
Rob Landleyea224be2006-06-18 20:20:07 +0000720 p = skip_whitespace(++p);
Glenn L McGrath60281112001-11-02 11:39:46 +0000721 }
722
Manuel Novoa III cad53642003-03-19 09:13:01 +0000723 /* bb_dump_skip slash and trailing white space */
Glenn L McGrath60281112001-11-02 11:39:46 +0000724 if (*p == '/') {
Rob Landleyea224be2006-06-18 20:20:07 +0000725 p = skip_whitespace(++p);
Glenn L McGrath60281112001-11-02 11:39:46 +0000726 }
727
728 /* byte count */
729 if (isdigit(*p)) {
730 for (savep = p; isdigit(*p); ++p);
731 if (!isspace(*p)) {
Manuel Novoa III cad53642003-03-19 09:13:01 +0000732 bb_error_msg_and_die("bad format {%s}", fmt);
Glenn L McGrath60281112001-11-02 11:39:46 +0000733 }
734 tfu->bcnt = atoi(savep);
Manuel Novoa III cad53642003-03-19 09:13:01 +0000735 /* bb_dump_skip trailing white space */
Rob Landleyea224be2006-06-18 20:20:07 +0000736 p = skip_whitespace(++p);
Glenn L McGrath60281112001-11-02 11:39:46 +0000737 }
738
739 /* format */
740 if (*p != '"') {
Manuel Novoa III cad53642003-03-19 09:13:01 +0000741 bb_error_msg_and_die("bad format {%s}", fmt);
Glenn L McGrath60281112001-11-02 11:39:46 +0000742 }
743 for (savep = ++p; *p != '"';) {
744 if (*p++ == 0) {
Manuel Novoa III cad53642003-03-19 09:13:01 +0000745 bb_error_msg_and_die("bad format {%s}", fmt);
Glenn L McGrath60281112001-11-02 11:39:46 +0000746 }
747 }
Manuel Novoa III cad53642003-03-19 09:13:01 +0000748 tfu->fmt = xmalloc(p - savep + 1);
Glenn L McGrath60281112001-11-02 11:39:46 +0000749 strncpy(tfu->fmt, savep, p - savep);
750 tfu->fmt[p - savep] = '\0';
Glenn L McGrath9fef17d2002-08-22 18:41:20 +0000751/* escape(tfu->fmt); */
Glenn L McGrath60281112001-11-02 11:39:46 +0000752
753 p1 = tfu->fmt;
754
755 /* alphabetic escape sequences have to be done in place */
756 for (p2 = p1;; ++p1, ++p2) {
757 if (!*p1) {
758 *p2 = *p1;
759 break;
760 }
761 if (*p1 == '\\') {
Manuel Novoa III cad53642003-03-19 09:13:01 +0000762 const char *cs = conv_str + 4;
763 ++p1;
764 *p2 = *p1;
765 do {
766 if (*p1 == cs[2]) {
767 *p2 = cs[0];
768 break;
769 }
770 cs += 4;
771 } while (*cs);
Glenn L McGrath60281112001-11-02 11:39:46 +0000772 }
773 }
774
775 p++;
776 }
777}
Glenn L McGrath9fef17d2002-08-22 18:41:20 +0000778
Glenn L McGrath60281112001-11-02 11:39:46 +0000779/*
780 * Copyright (c) 1989 The Regents of the University of California.
781 * All rights reserved.
782 *
783 * Redistribution and use in source and binary forms, with or without
784 * modification, are permitted provided that the following conditions
785 * are met:
786 * 1. Redistributions of source code must retain the above copyright
787 * notice, this list of conditions and the following disclaimer.
788 * 2. Redistributions in binary form must reproduce the above copyright
789 * notice, this list of conditions and the following disclaimer in the
790 * documentation and/or other materials provided with the distribution.
Aaron Lehmann69d41782002-06-23 22:25:24 +0000791 * 3. Neither the name of the University nor the names of its contributors
Glenn L McGrath60281112001-11-02 11:39:46 +0000792 * may be used to endorse or promote products derived from this software
793 * without specific prior written permission.
794 *
795 * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
796 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
797 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
798 * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
799 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
800 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
801 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
802 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
803 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
804 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
805 * SUCH DAMAGE.
806 */