libbb: get rid of statics in dump.c; code shrinks a lot too
function old new delta alloc_dumper - 26 +26 hexdump_main 600 601 +1 static.done 1 - -1 static.ateof 1 - -1 bb_dump_vflag 1 - -1 static.savp 4 - -4 static.nextfs 4 - -4 static.curp 4 - -4 exitval 4 - -4 endfu 4 - -4 bb_dump_length 4 - -4 bb_dump_fshead 4 - -4 bb_dump_blocksize 4 - -4 _argv 4 - -4 bb_dump_add 365 358 -7 savaddress 8 - -8 eaddress 8 - -8 bb_dump_skip 8 - -8 address 8 - -8 bb_dump_dump 2748 2672 -76 next 538 445 -93 ------------------------------------------------------------------------------ (add/remove: 1/16 grow/shrink: 1/3 up/down: 27/-247) Total: -220 bytes text data bss dec hex filename 789458 607 6764 796829 c289d busybox_old 789309 601 6696 796606 c27be busybox_unstripped
This commit is contained in:
parent
8ddb6410ed
commit
55f7912dda
@ -24,7 +24,7 @@
|
|||||||
#define ishexdigit(c) (isxdigit)(c)
|
#define ishexdigit(c) (isxdigit)(c)
|
||||||
|
|
||||||
static void
|
static void
|
||||||
odoffset(int argc, char ***argvp)
|
odoffset(dumper_t *dumper, int argc, char ***argvp)
|
||||||
{
|
{
|
||||||
char *num, *p;
|
char *num, *p;
|
||||||
int base;
|
int base;
|
||||||
@ -57,7 +57,7 @@ odoffset(int argc, char ***argvp)
|
|||||||
|
|
||||||
base = 0;
|
base = 0;
|
||||||
/*
|
/*
|
||||||
* bb_dump_skip over leading '+', 'x[0-9a-fA-f]' or '0x', and
|
* skip over leading '+', 'x[0-9a-fA-f]' or '0x', and
|
||||||
* set base.
|
* set base.
|
||||||
*/
|
*/
|
||||||
if (p[0] == '+')
|
if (p[0] == '+')
|
||||||
@ -70,11 +70,13 @@ odoffset(int argc, char ***argvp)
|
|||||||
base = 16;
|
base = 16;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* bb_dump_skip over the number */
|
/* skip over the number */
|
||||||
if (base == 16)
|
if (base == 16)
|
||||||
for (num = p; ishexdigit(*p); ++p);
|
for (num = p; ishexdigit(*p); ++p)
|
||||||
|
continue;
|
||||||
else
|
else
|
||||||
for (num = p; isdecdigit(*p); ++p);
|
for (num = p; isdecdigit(*p); ++p)
|
||||||
|
continue;
|
||||||
|
|
||||||
/* check for no number */
|
/* check for no number */
|
||||||
if (num == p)
|
if (num == p)
|
||||||
@ -87,23 +89,23 @@ odoffset(int argc, char ***argvp)
|
|||||||
base = 10;
|
base = 10;
|
||||||
}
|
}
|
||||||
|
|
||||||
bb_dump_skip = strtol(num, &end, base ? base : 8);
|
dumper->dump_skip = strtol(num, &end, base ? base : 8);
|
||||||
|
|
||||||
/* if end isn't the same as p, we got a non-octal digit */
|
/* if end isn't the same as p, we got a non-octal digit */
|
||||||
if (end != p)
|
if (end != p)
|
||||||
bb_dump_skip = 0;
|
dumper->dump_skip = 0;
|
||||||
else {
|
else {
|
||||||
if (*p) {
|
if (*p) {
|
||||||
if (*p == 'b') {
|
if (*p == 'b') {
|
||||||
bb_dump_skip *= 512;
|
dumper->dump_skip *= 512;
|
||||||
++p;
|
++p;
|
||||||
} else if (*p == 'B') {
|
} else if (*p == 'B') {
|
||||||
bb_dump_skip *= 1024;
|
dumper->dump_skip *= 1024;
|
||||||
++p;
|
++p;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (*p)
|
if (*p)
|
||||||
bb_dump_skip = 0;
|
dumper->dump_skip = 0;
|
||||||
else {
|
else {
|
||||||
++*argvp;
|
++*argvp;
|
||||||
/*
|
/*
|
||||||
@ -121,8 +123,8 @@ odoffset(int argc, char ***argvp)
|
|||||||
if (base == 10) {
|
if (base == 10) {
|
||||||
x_or_d = 'd';
|
x_or_d = 'd';
|
||||||
DO_X_OR_D:
|
DO_X_OR_D:
|
||||||
bb_dump_fshead->nextfu->fmt[TYPE_OFFSET]
|
dumper->fshead->nextfu->fmt[TYPE_OFFSET]
|
||||||
= bb_dump_fshead->nextfs->nextfu->fmt[TYPE_OFFSET]
|
= dumper->fshead->nextfs->nextfu->fmt[TYPE_OFFSET]
|
||||||
= x_or_d;
|
= x_or_d;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -161,36 +163,35 @@ int od_main(int argc, char **argv)
|
|||||||
int ch;
|
int ch;
|
||||||
int first = 1;
|
int first = 1;
|
||||||
char *p;
|
char *p;
|
||||||
bb_dump_vflag = FIRST;
|
dumper_t *dumper = alloc_dumper();
|
||||||
bb_dump_length = -1;
|
|
||||||
|
|
||||||
while ((ch = getopt(argc, argv, od_opts)) > 0) {
|
while ((ch = getopt(argc, argv, od_opts)) > 0) {
|
||||||
if (ch == 'v') {
|
if (ch == 'v') {
|
||||||
bb_dump_vflag = ALL;
|
dumper->dump_vflag = ALL;
|
||||||
} else if (((p = strchr(od_opts, ch)) != NULL) && (*p != '\0')) {
|
} else if (((p = strchr(od_opts, ch)) != NULL) && (*p != '\0')) {
|
||||||
if (first) {
|
if (first) {
|
||||||
first = 0;
|
first = 0;
|
||||||
bb_dump_add("\"%07.7_Ao\n\"");
|
bb_dump_add(dumper, "\"%07.7_Ao\n\"");
|
||||||
bb_dump_add("\"%07.7_ao \"");
|
bb_dump_add(dumper, "\"%07.7_ao \"");
|
||||||
} else {
|
} else {
|
||||||
bb_dump_add("\" \"");
|
bb_dump_add(dumper, "\" \"");
|
||||||
}
|
}
|
||||||
bb_dump_add(add_strings[(int)od_o2si[(p-od_opts)]]);
|
bb_dump_add(dumper, add_strings[(int)od_o2si[(p - od_opts)]]);
|
||||||
} else { /* P, p, s, w, or other unhandled */
|
} else { /* P, p, s, w, or other unhandled */
|
||||||
bb_show_usage();
|
bb_show_usage();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (!bb_dump_fshead) {
|
if (!dumper->fshead) {
|
||||||
bb_dump_add("\"%07.7_Ao\n\"");
|
bb_dump_add(dumper, "\"%07.7_Ao\n\"");
|
||||||
bb_dump_add("\"%07.7_ao \" 8/2 \"%06o \" \"\\n\"");
|
bb_dump_add(dumper, "\"%07.7_ao \" 8/2 \"%06o \" \"\\n\"");
|
||||||
}
|
}
|
||||||
|
|
||||||
argc -= optind;
|
argc -= optind;
|
||||||
argv += optind;
|
argv += optind;
|
||||||
|
|
||||||
odoffset(argc, &argv);
|
odoffset(dumper, argc, &argv);
|
||||||
|
|
||||||
return bb_dump_dump(argv);
|
return bb_dump_dump(dumper, argv);
|
||||||
}
|
}
|
||||||
#endif /* ENABLE_DESKTOP */
|
#endif /* ENABLE_DESKTOP */
|
||||||
|
|
||||||
|
@ -18,10 +18,10 @@
|
|||||||
#define F_UINT 0x200 /* %[ouXx] */
|
#define F_UINT 0x200 /* %[ouXx] */
|
||||||
#define F_TEXT 0x400 /* no conversions */
|
#define F_TEXT 0x400 /* no conversions */
|
||||||
|
|
||||||
enum _vflag { ALL, DUP, FIRST, WAIT }; /* -v values */
|
enum dump_vflag_t { ALL, DUP, FIRST, WAIT }; /* -v values */
|
||||||
|
|
||||||
typedef struct _pr {
|
typedef struct PR {
|
||||||
struct _pr *nextpr; /* next print unit */
|
struct PR *nextpr; /* next print unit */
|
||||||
unsigned flags; /* flag values */
|
unsigned flags; /* flag values */
|
||||||
int bcnt; /* byte count */
|
int bcnt; /* byte count */
|
||||||
char *cchar; /* conversion character */
|
char *cchar; /* conversion character */
|
||||||
@ -29,30 +29,31 @@ typedef struct _pr {
|
|||||||
char *nospace; /* no whitespace version */
|
char *nospace; /* no whitespace version */
|
||||||
} PR;
|
} PR;
|
||||||
|
|
||||||
typedef struct _fu {
|
typedef struct FU {
|
||||||
struct _fu *nextfu; /* next format unit */
|
struct FU *nextfu; /* next format unit */
|
||||||
struct _pr *nextpr; /* next print unit */
|
struct PR *nextpr; /* next print unit */
|
||||||
unsigned flags; /* flag values */
|
unsigned flags; /* flag values */
|
||||||
int reps; /* repetition count */
|
int reps; /* repetition count */
|
||||||
int bcnt; /* byte count */
|
int bcnt; /* byte count */
|
||||||
char *fmt; /* format string */
|
char *fmt; /* format string */
|
||||||
} FU;
|
} FU;
|
||||||
|
|
||||||
typedef struct _fs { /* format strings */
|
typedef struct FS { /* format strings */
|
||||||
struct _fs *nextfs; /* linked list of format strings */
|
struct FS *nextfs; /* linked list of format strings */
|
||||||
struct _fu *nextfu; /* linked list of format units */
|
struct FU *nextfu; /* linked list of format units */
|
||||||
int bcnt;
|
int bcnt;
|
||||||
} FS;
|
} FS;
|
||||||
|
|
||||||
extern void bb_dump_add(const char *fmt) FAST_FUNC;
|
typedef struct dumper_t {
|
||||||
extern int bb_dump_dump(char **argv) FAST_FUNC;
|
off_t dump_skip; /* bytes to skip */
|
||||||
extern int bb_dump_size(FS * fs) FAST_FUNC;
|
int dump_length; /* max bytes to read */
|
||||||
|
smallint dump_vflag; /*enum dump_vflag_t*/
|
||||||
|
FS *fshead;
|
||||||
|
} dumper_t;
|
||||||
|
|
||||||
extern FS *bb_dump_fshead; /* head of format strings */
|
dumper_t* alloc_dumper(void) FAST_FUNC;
|
||||||
extern int bb_dump_blocksize; /* data block size */
|
extern void bb_dump_add(dumper_t *dumper, const char *fmt) FAST_FUNC;
|
||||||
extern int bb_dump_length; /* max bytes to read */
|
extern int bb_dump_dump(dumper_t *dumper, char **argv) FAST_FUNC;
|
||||||
extern smallint /*enum _vflag*/ bb_dump_vflag;
|
|
||||||
extern off_t bb_dump_skip; /* bytes to skip */
|
|
||||||
|
|
||||||
#if __GNUC_PREREQ(4,1)
|
#if __GNUC_PREREQ(4,1)
|
||||||
# pragma GCC visibility pop
|
# pragma GCC visibility pop
|
||||||
|
288
libbb/dump.c
288
libbb/dump.c
@ -14,19 +14,6 @@
|
|||||||
#include "libbb.h"
|
#include "libbb.h"
|
||||||
#include "dump.h"
|
#include "dump.h"
|
||||||
|
|
||||||
FS *bb_dump_fshead; /* head of format strings */
|
|
||||||
off_t bb_dump_skip; /* bytes to skip */
|
|
||||||
int bb_dump_blocksize; /* data block size */
|
|
||||||
int bb_dump_length = -1; /* max bytes to read */
|
|
||||||
smallint /*enum _vflag*/ bb_dump_vflag = FIRST;
|
|
||||||
|
|
||||||
static FU *endfu;
|
|
||||||
static char **_argv;
|
|
||||||
static off_t savaddress; /* saved address/offset in stream */
|
|
||||||
static off_t eaddress; /* end address */
|
|
||||||
static off_t address; /* address/offset in stream */
|
|
||||||
static int exitval; /* final exit value */
|
|
||||||
|
|
||||||
static const char index_str[] ALIGN1 = ".#-+ 0123456789";
|
static const char index_str[] ALIGN1 = ".#-+ 0123456789";
|
||||||
|
|
||||||
static const char size_conv_str[] ALIGN1 =
|
static const char size_conv_str[] ALIGN1 =
|
||||||
@ -34,7 +21,36 @@ static const char size_conv_str[] ALIGN1 =
|
|||||||
|
|
||||||
static const char lcc[] ALIGN1 = "diouxX";
|
static const char lcc[] ALIGN1 = "diouxX";
|
||||||
|
|
||||||
int FAST_FUNC bb_dump_size(FS *fs)
|
|
||||||
|
typedef struct priv_dumper_t {
|
||||||
|
dumper_t pub;
|
||||||
|
|
||||||
|
char **argv;
|
||||||
|
FU *endfu;
|
||||||
|
off_t savaddress; /* saved address/offset in stream */
|
||||||
|
off_t eaddress; /* end address */
|
||||||
|
off_t address; /* address/offset in stream */
|
||||||
|
int blocksize;
|
||||||
|
smallint exitval; /* final exit value */
|
||||||
|
|
||||||
|
/* former statics */
|
||||||
|
smallint next__done;
|
||||||
|
smallint get__ateof; // = 1;
|
||||||
|
unsigned char *get__curp;
|
||||||
|
unsigned char *get__savp;
|
||||||
|
} priv_dumper_t;
|
||||||
|
|
||||||
|
dumper_t* FAST_FUNC alloc_dumper(void)
|
||||||
|
{
|
||||||
|
priv_dumper_t *dumper = xzalloc(sizeof(*dumper));
|
||||||
|
dumper->pub.dump_length = -1;
|
||||||
|
dumper->pub.dump_vflag = FIRST;
|
||||||
|
dumper->get__ateof = 1;
|
||||||
|
return &dumper->pub;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static NOINLINE int bb_dump_size(FS *fs)
|
||||||
{
|
{
|
||||||
FU *fu;
|
FU *fu;
|
||||||
int bcnt, cur_size;
|
int bcnt, cur_size;
|
||||||
@ -52,13 +68,14 @@ int FAST_FUNC bb_dump_size(FS *fs)
|
|||||||
if (*fmt != '%')
|
if (*fmt != '%')
|
||||||
continue;
|
continue;
|
||||||
/*
|
/*
|
||||||
* bb_dump_skip any special chars -- save precision in
|
* skip any special chars -- save precision in
|
||||||
* case it's a %s format.
|
* case it's a %s format.
|
||||||
*/
|
*/
|
||||||
while (strchr(index_str + 1, *++fmt));
|
while (strchr(index_str + 1, *++fmt));
|
||||||
if (*fmt == '.' && isdigit(*++fmt)) {
|
if (*fmt == '.' && isdigit(*++fmt)) {
|
||||||
prec = atoi(fmt);
|
prec = atoi(fmt);
|
||||||
while (isdigit(*++fmt));
|
while (isdigit(*++fmt))
|
||||||
|
continue;
|
||||||
}
|
}
|
||||||
p = strchr(size_conv_str + 12, *fmt);
|
p = strchr(size_conv_str + 12, *fmt);
|
||||||
if (!p) {
|
if (!p) {
|
||||||
@ -79,7 +96,7 @@ int FAST_FUNC bb_dump_size(FS *fs)
|
|||||||
return cur_size;
|
return cur_size;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void rewrite(FS *fs)
|
static void rewrite(priv_dumper_t *dumper, FS *fs)
|
||||||
{
|
{
|
||||||
enum { NOTOKAY, USEBCNT, USEPREC } sokay;
|
enum { NOTOKAY, USEBCNT, USEPREC } sokay;
|
||||||
PR *pr, **nextpr = NULL;
|
PR *pr, **nextpr = NULL;
|
||||||
@ -104,7 +121,7 @@ static void rewrite(FS *fs)
|
|||||||
* uninitialized 1st time through.
|
* uninitialized 1st time through.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/* bb_dump_skip preceding text and up to the next % sign */
|
/* skip preceding text and up to the next % sign */
|
||||||
for (p1 = fmtp; *p1 && *p1 != '%'; ++p1)
|
for (p1 = fmtp; *p1 && *p1 != '%'; ++p1)
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
@ -121,11 +138,11 @@ static void rewrite(FS *fs)
|
|||||||
*/
|
*/
|
||||||
if (fu->bcnt) {
|
if (fu->bcnt) {
|
||||||
sokay = USEBCNT;
|
sokay = USEBCNT;
|
||||||
/* bb_dump_skip to conversion character */
|
/* skip to conversion character */
|
||||||
for (++p1; strchr(index_str, *p1); ++p1)
|
for (++p1; strchr(index_str, *p1); ++p1)
|
||||||
continue;
|
continue;
|
||||||
} else {
|
} else {
|
||||||
/* bb_dump_skip any special chars, field width */
|
/* skip any special chars, field width */
|
||||||
while (strchr(index_str + 1, *++p1))
|
while (strchr(index_str + 1, *++p1))
|
||||||
continue;
|
continue;
|
||||||
if (*p1 == '.' && isdigit(*++p1)) {
|
if (*p1 == '.' && isdigit(*++p1)) {
|
||||||
@ -198,7 +215,7 @@ static void rewrite(FS *fs)
|
|||||||
++p2;
|
++p2;
|
||||||
switch (p1[1]) {
|
switch (p1[1]) {
|
||||||
case 'A':
|
case 'A':
|
||||||
endfu = fu;
|
dumper->endfu = fu;
|
||||||
fu->flags |= F_IGNORE;
|
fu->flags |= F_IGNORE;
|
||||||
/* FALLTHROUGH */
|
/* FALLTHROUGH */
|
||||||
case 'a':
|
case 'a':
|
||||||
@ -274,7 +291,7 @@ static void rewrite(FS *fs)
|
|||||||
}
|
}
|
||||||
/*
|
/*
|
||||||
* if the format string interprets any data at all, and it's
|
* if the format string interprets any data at all, and it's
|
||||||
* not the same as the bb_dump_blocksize, and its last format unit
|
* not the same as the blocksize, and its last format unit
|
||||||
* interprets any data at all, and has no iteration count,
|
* interprets any data at all, and has no iteration count,
|
||||||
* repeat it as necessary.
|
* repeat it as necessary.
|
||||||
*
|
*
|
||||||
@ -282,10 +299,10 @@ static void rewrite(FS *fs)
|
|||||||
* gets output from the last iteration of the format unit.
|
* gets output from the last iteration of the format unit.
|
||||||
*/
|
*/
|
||||||
for (fu = fs->nextfu;; fu = fu->nextfu) {
|
for (fu = fs->nextfu;; fu = fu->nextfu) {
|
||||||
if (!fu->nextfu && fs->bcnt < bb_dump_blocksize
|
if (!fu->nextfu && fs->bcnt < dumper->blocksize
|
||||||
&& !(fu->flags & F_SETREP) && fu->bcnt
|
&& !(fu->flags & F_SETREP) && fu->bcnt
|
||||||
) {
|
) {
|
||||||
fu->reps += (bb_dump_blocksize - fs->bcnt) / fu->bcnt;
|
fu->reps += (dumper->blocksize - fs->bcnt) / fu->bcnt;
|
||||||
}
|
}
|
||||||
if (fu->reps > 1) {
|
if (fu->reps > 1) {
|
||||||
for (pr = fu->nextpr;; pr = pr->nextpr)
|
for (pr = fu->nextpr;; pr = pr->nextpr)
|
||||||
@ -301,7 +318,7 @@ static void rewrite(FS *fs)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static void do_skip(const char *fname, int statok)
|
static void do_skip(priv_dumper_t *dumper, const char *fname, int statok)
|
||||||
{
|
{
|
||||||
struct stat sbuf;
|
struct stat sbuf;
|
||||||
|
|
||||||
@ -309,125 +326,122 @@ static void do_skip(const char *fname, int statok)
|
|||||||
if (fstat(STDIN_FILENO, &sbuf)) {
|
if (fstat(STDIN_FILENO, &sbuf)) {
|
||||||
bb_simple_perror_msg_and_die(fname);
|
bb_simple_perror_msg_and_die(fname);
|
||||||
}
|
}
|
||||||
if ((!(S_ISCHR(sbuf.st_mode) ||
|
if (!(S_ISCHR(sbuf.st_mode) || S_ISBLK(sbuf.st_mode) || S_ISFIFO(sbuf.st_mode))
|
||||||
S_ISBLK(sbuf.st_mode) ||
|
&& dumper->pub.dump_skip >= sbuf.st_size
|
||||||
S_ISFIFO(sbuf.st_mode))) && bb_dump_skip >= sbuf.st_size) {
|
) {
|
||||||
/* If bb_dump_size valid and bb_dump_skip >= size */
|
/* If bb_dump_size valid and pub.dump_skip >= size */
|
||||||
bb_dump_skip -= sbuf.st_size;
|
dumper->pub.dump_skip -= sbuf.st_size;
|
||||||
address += sbuf.st_size;
|
dumper->address += sbuf.st_size;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (fseek(stdin, bb_dump_skip, SEEK_SET)) {
|
if (fseek(stdin, dumper->pub.dump_skip, SEEK_SET)) {
|
||||||
bb_simple_perror_msg_and_die(fname);
|
bb_simple_perror_msg_and_die(fname);
|
||||||
}
|
}
|
||||||
savaddress = address += bb_dump_skip;
|
dumper->address += dumper->pub.dump_skip;
|
||||||
bb_dump_skip = 0;
|
dumper->savaddress = dumper->address;
|
||||||
|
dumper->pub.dump_skip = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int next(char **argv)
|
static NOINLINE int next(priv_dumper_t *dumper)
|
||||||
{
|
{
|
||||||
static smallint done;
|
|
||||||
|
|
||||||
int statok;
|
int statok;
|
||||||
|
|
||||||
if (argv) {
|
|
||||||
_argv = argv;
|
|
||||||
return 1;
|
|
||||||
}
|
|
||||||
for (;;) {
|
for (;;) {
|
||||||
if (*_argv) {
|
if (*dumper->argv) {
|
||||||
if (!(freopen(*_argv, "r", stdin))) {
|
if (!(freopen(*dumper->argv, "r", stdin))) {
|
||||||
bb_simple_perror_msg(*_argv);
|
bb_simple_perror_msg(*dumper->argv);
|
||||||
exitval = 1;
|
dumper->exitval = 1;
|
||||||
++_argv;
|
++dumper->argv;
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
done = statok = 1;
|
dumper->next__done = statok = 1;
|
||||||
} else {
|
} else {
|
||||||
if (done)
|
if (dumper->next__done)
|
||||||
return 0;
|
return 0;
|
||||||
done = 1;
|
dumper->next__done = 1;
|
||||||
statok = 0;
|
statok = 0;
|
||||||
}
|
}
|
||||||
if (bb_dump_skip)
|
if (dumper->pub.dump_skip)
|
||||||
do_skip(statok ? *_argv : "stdin", statok);
|
do_skip(dumper, statok ? *dumper->argv : "stdin", statok);
|
||||||
if (*_argv)
|
if (*dumper->argv)
|
||||||
++_argv;
|
++dumper->argv;
|
||||||
if (!bb_dump_skip)
|
if (!dumper->pub.dump_skip)
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
/* NOTREACHED */
|
/* NOTREACHED */
|
||||||
}
|
}
|
||||||
|
|
||||||
static unsigned char *get(void)
|
static unsigned char *get(priv_dumper_t *dumper)
|
||||||
{
|
{
|
||||||
static smallint ateof = 1;
|
|
||||||
static unsigned char *curp = NULL, *savp; /*DBU:[dave@cray.com]initialize curp */
|
|
||||||
|
|
||||||
int n;
|
int n;
|
||||||
int need, nread;
|
int need, nread;
|
||||||
unsigned char *tmpp;
|
unsigned char *tmpp;
|
||||||
|
int blocksize = dumper->blocksize;
|
||||||
|
|
||||||
if (!curp) {
|
if (!dumper->get__curp) {
|
||||||
address = (off_t)0; /*DBU:[dave@cray.com] initialize,initialize..*/
|
dumper->address = (off_t)0; /*DBU:[dave@cray.com] initialize,initialize..*/
|
||||||
curp = xmalloc(bb_dump_blocksize);
|
dumper->get__curp = xmalloc(blocksize);
|
||||||
savp = xmalloc(bb_dump_blocksize);
|
dumper->get__savp = xmalloc(blocksize);
|
||||||
} else {
|
} else {
|
||||||
tmpp = curp;
|
tmpp = dumper->get__curp;
|
||||||
curp = savp;
|
dumper->get__curp = dumper->get__savp;
|
||||||
savp = tmpp;
|
dumper->get__savp = tmpp;
|
||||||
address = savaddress += bb_dump_blocksize;
|
dumper->savaddress += blocksize;
|
||||||
|
dumper->address = dumper->savaddress;
|
||||||
}
|
}
|
||||||
for (need = bb_dump_blocksize, nread = 0;;) {
|
need = blocksize;
|
||||||
|
nread = 0;
|
||||||
|
while (1) {
|
||||||
/*
|
/*
|
||||||
* if read the right number of bytes, or at EOF for one file,
|
* if read the right number of bytes, or at EOF for one file,
|
||||||
* and no other files are available, zero-pad the rest of the
|
* and no other files are available, zero-pad the rest of the
|
||||||
* block and set the end flag.
|
* block and set the end flag.
|
||||||
*/
|
*/
|
||||||
if (!bb_dump_length || (ateof && !next(NULL))) {
|
if (!dumper->pub.dump_length || (dumper->get__ateof && !next(dumper))) {
|
||||||
if (need == bb_dump_blocksize) {
|
if (need == blocksize) {
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
if (bb_dump_vflag != ALL && !memcmp(curp, savp, nread)) {
|
if (dumper->pub.dump_vflag != ALL && !memcmp(dumper->get__curp, dumper->get__savp, nread)) {
|
||||||
if (bb_dump_vflag != DUP) {
|
if (dumper->pub.dump_vflag != DUP) {
|
||||||
puts("*");
|
puts("*");
|
||||||
}
|
}
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
memset(curp + nread, 0, need);
|
memset(dumper->get__curp + nread, 0, need);
|
||||||
eaddress = address + nread;
|
dumper->eaddress = dumper->address + nread;
|
||||||
return curp;
|
return dumper->get__curp;
|
||||||
}
|
}
|
||||||
n = fread(curp + nread, sizeof(unsigned char),
|
n = fread(dumper->get__curp + nread, sizeof(unsigned char),
|
||||||
bb_dump_length == -1 ? need : MIN(bb_dump_length, need), stdin);
|
dumper->pub.dump_length == -1 ? need : MIN(dumper->pub.dump_length, need), stdin);
|
||||||
if (!n) {
|
if (!n) {
|
||||||
if (ferror(stdin)) {
|
if (ferror(stdin)) {
|
||||||
bb_simple_perror_msg(_argv[-1]);
|
bb_simple_perror_msg(dumper->argv[-1]);
|
||||||
}
|
}
|
||||||
ateof = 1;
|
dumper->get__ateof = 1;
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
ateof = 0;
|
dumper->get__ateof = 0;
|
||||||
if (bb_dump_length != -1) {
|
if (dumper->pub.dump_length != -1) {
|
||||||
bb_dump_length -= n;
|
dumper->pub.dump_length -= n;
|
||||||
}
|
}
|
||||||
need -= n;
|
need -= n;
|
||||||
if (!need) {
|
if (!need) {
|
||||||
if (bb_dump_vflag == ALL || bb_dump_vflag == FIRST
|
if (dumper->pub.dump_vflag == ALL || dumper->pub.dump_vflag == FIRST
|
||||||
|| memcmp(curp, savp, bb_dump_blocksize)
|
|| memcmp(dumper->get__curp, dumper->get__savp, blocksize)
|
||||||
) {
|
) {
|
||||||
if (bb_dump_vflag == DUP || bb_dump_vflag == FIRST) {
|
if (dumper->pub.dump_vflag == DUP || dumper->pub.dump_vflag == FIRST) {
|
||||||
bb_dump_vflag = WAIT;
|
dumper->pub.dump_vflag = WAIT;
|
||||||
}
|
}
|
||||||
return curp;
|
return dumper->get__curp;
|
||||||
}
|
}
|
||||||
if (bb_dump_vflag == WAIT) {
|
if (dumper->pub.dump_vflag == WAIT) {
|
||||||
puts("*");
|
puts("*");
|
||||||
}
|
}
|
||||||
bb_dump_vflag = DUP;
|
dumper->pub.dump_vflag = DUP;
|
||||||
address = savaddress += bb_dump_blocksize;
|
dumper->savaddress += blocksize;
|
||||||
need = bb_dump_blocksize;
|
dumper->address = dumper->savaddress;
|
||||||
|
need = blocksize;
|
||||||
nread = 0;
|
nread = 0;
|
||||||
} else {
|
} else {
|
||||||
nread += n;
|
nread += n;
|
||||||
@ -515,29 +529,31 @@ static void conv_u(PR *pr, unsigned char *p)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static void display(void)
|
static void display(priv_dumper_t* dumper)
|
||||||
{
|
{
|
||||||
/* extern FU *endfu; */
|
|
||||||
FS *fs;
|
FS *fs;
|
||||||
FU *fu;
|
FU *fu;
|
||||||
PR *pr;
|
PR *pr;
|
||||||
int cnt;
|
int cnt;
|
||||||
unsigned char *bp;
|
unsigned char *bp, *savebp;
|
||||||
off_t saveaddress;
|
off_t saveaddress;
|
||||||
unsigned char savech = 0, *savebp;
|
unsigned char savech = '\0';
|
||||||
|
|
||||||
while ((bp = get()) != NULL) {
|
while ((bp = get(dumper)) != NULL) {
|
||||||
for (fs = bb_dump_fshead, savebp = bp, saveaddress = address; fs;
|
fs = dumper->pub.fshead;
|
||||||
fs = fs->nextfs, bp = savebp, address = saveaddress) {
|
savebp = bp;
|
||||||
|
saveaddress = dumper->address;
|
||||||
|
for (; fs; fs = fs->nextfs, bp = savebp, dumper->address = saveaddress) {
|
||||||
for (fu = fs->nextfu; fu; fu = fu->nextfu) {
|
for (fu = fs->nextfu; fu; fu = fu->nextfu) {
|
||||||
if (fu->flags & F_IGNORE) {
|
if (fu->flags & F_IGNORE) {
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
for (cnt = fu->reps; cnt; --cnt) {
|
for (cnt = fu->reps; cnt; --cnt) {
|
||||||
for (pr = fu->nextpr; pr; address += pr->bcnt,
|
for (pr = fu->nextpr; pr; dumper->address += pr->bcnt,
|
||||||
bp += pr->bcnt, pr = pr->nextpr) {
|
bp += pr->bcnt, pr = pr->nextpr) {
|
||||||
if (eaddress && address >= eaddress &&
|
if (dumper->eaddress && dumper->address >= dumper->eaddress
|
||||||
!(pr->flags & (F_TEXT | F_BPAD))) {
|
&& !(pr->flags & (F_TEXT | F_BPAD))
|
||||||
|
) {
|
||||||
bpad(pr);
|
bpad(pr);
|
||||||
}
|
}
|
||||||
if (cnt == 1 && pr->nospace) {
|
if (cnt == 1 && pr->nospace) {
|
||||||
@ -547,7 +563,7 @@ static void display(void)
|
|||||||
/* PRINT; */
|
/* PRINT; */
|
||||||
switch (pr->flags) {
|
switch (pr->flags) {
|
||||||
case F_ADDRESS:
|
case F_ADDRESS:
|
||||||
printf(pr->fmt, (unsigned) address);
|
printf(pr->fmt, (unsigned) dumper->address);
|
||||||
break;
|
break;
|
||||||
case F_BPAD:
|
case F_BPAD:
|
||||||
printf(pr->fmt, "");
|
printf(pr->fmt, "");
|
||||||
@ -633,21 +649,21 @@ static void display(void)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (endfu) {
|
if (dumper->endfu) {
|
||||||
/*
|
/*
|
||||||
* if eaddress not set, error or file bb_dump_size was multiple of
|
* if eaddress not set, error or file size was multiple
|
||||||
* bb_dump_blocksize, and no partial block ever found.
|
* of blocksize, and no partial block ever found.
|
||||||
*/
|
*/
|
||||||
if (!eaddress) {
|
if (!dumper->eaddress) {
|
||||||
if (!address) {
|
if (!dumper->address) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
eaddress = address;
|
dumper->eaddress = dumper->address;
|
||||||
}
|
}
|
||||||
for (pr = endfu->nextpr; pr; pr = pr->nextpr) {
|
for (pr = dumper->endfu->nextpr; pr; pr = pr->nextpr) {
|
||||||
switch (pr->flags) {
|
switch (pr->flags) {
|
||||||
case F_ADDRESS:
|
case F_ADDRESS:
|
||||||
printf(pr->fmt, (unsigned) eaddress);
|
printf(pr->fmt, (unsigned) dumper->eaddress);
|
||||||
break;
|
break;
|
||||||
case F_TEXT:
|
case F_TEXT:
|
||||||
printf(pr->fmt);
|
printf(pr->fmt);
|
||||||
@ -657,52 +673,58 @@ static void display(void)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
int FAST_FUNC bb_dump_dump(char **argv)
|
#define dumper ((priv_dumper_t*)pub_dumper)
|
||||||
|
int FAST_FUNC bb_dump_dump(dumper_t *pub_dumper, char **argv)
|
||||||
{
|
{
|
||||||
FS *tfs;
|
FS *tfs;
|
||||||
|
int blocksize;
|
||||||
|
|
||||||
/* figure out the data block bb_dump_size */
|
/* figure out the data block bb_dump_size */
|
||||||
for (bb_dump_blocksize = 0, tfs = bb_dump_fshead; tfs; tfs = tfs->nextfs) {
|
blocksize = 0;
|
||||||
|
tfs = dumper->pub.fshead;
|
||||||
|
while (tfs) {
|
||||||
tfs->bcnt = bb_dump_size(tfs);
|
tfs->bcnt = bb_dump_size(tfs);
|
||||||
if (bb_dump_blocksize < tfs->bcnt) {
|
if (blocksize < tfs->bcnt) {
|
||||||
bb_dump_blocksize = tfs->bcnt;
|
blocksize = tfs->bcnt;
|
||||||
}
|
}
|
||||||
|
tfs = tfs->nextfs;
|
||||||
}
|
}
|
||||||
|
dumper->blocksize = blocksize;
|
||||||
|
|
||||||
/* rewrite the rules, do syntax checking */
|
/* rewrite the rules, do syntax checking */
|
||||||
for (tfs = bb_dump_fshead; tfs; tfs = tfs->nextfs) {
|
for (tfs = dumper->pub.fshead; tfs; tfs = tfs->nextfs) {
|
||||||
rewrite(tfs);
|
rewrite(dumper, tfs);
|
||||||
}
|
}
|
||||||
|
|
||||||
next(argv);
|
dumper->argv = argv;
|
||||||
display();
|
display(dumper);
|
||||||
|
|
||||||
return exitval;
|
return dumper->exitval;
|
||||||
}
|
}
|
||||||
|
|
||||||
void FAST_FUNC bb_dump_add(const char *fmt)
|
void FAST_FUNC bb_dump_add(dumper_t* pub_dumper, const char *fmt)
|
||||||
{
|
{
|
||||||
static FS **nextfs;
|
|
||||||
|
|
||||||
const char *p;
|
const char *p;
|
||||||
char *p1;
|
char *p1;
|
||||||
char *p2;
|
char *p2;
|
||||||
FS *tfs;
|
FS *tfs;
|
||||||
FU *tfu, **nextfu;
|
FU *tfu, **nextfupp;
|
||||||
const char *savep;
|
const char *savep;
|
||||||
|
|
||||||
/* start new linked list of format units */
|
/* start new linked list of format units */
|
||||||
tfs = xzalloc(sizeof(FS)); /*DBU:[dave@cray.com] start out NULL */
|
tfs = xzalloc(sizeof(FS)); /*DBU:[dave@cray.com] start out NULL */
|
||||||
if (!bb_dump_fshead) {
|
if (!dumper->pub.fshead) {
|
||||||
bb_dump_fshead = tfs;
|
dumper->pub.fshead = tfs;
|
||||||
} else {
|
} else {
|
||||||
*nextfs = tfs;
|
FS *fslast = dumper->pub.fshead;
|
||||||
|
while (fslast->nextfs)
|
||||||
|
fslast = fslast->nextfs;
|
||||||
|
fslast->nextfs = tfs;
|
||||||
}
|
}
|
||||||
nextfs = &tfs->nextfs;
|
nextfupp = &tfs->nextfu;
|
||||||
nextfu = &tfs->nextfu;
|
|
||||||
|
|
||||||
/* take the format string and break it up into format units */
|
/* take the format string and break it up into format units */
|
||||||
for (p = fmt;;) {
|
for (p = fmt;;) {
|
||||||
/* bb_dump_skip leading white space */
|
|
||||||
p = skip_whitespace(p);
|
p = skip_whitespace(p);
|
||||||
if (!*p) {
|
if (!*p) {
|
||||||
break;
|
break;
|
||||||
@ -712,8 +734,8 @@ void FAST_FUNC bb_dump_add(const char *fmt)
|
|||||||
/* NOSTRICT */
|
/* NOSTRICT */
|
||||||
/* DBU:[dave@cray.com] zalloc so that forward pointers start out NULL */
|
/* DBU:[dave@cray.com] zalloc so that forward pointers start out NULL */
|
||||||
tfu = xzalloc(sizeof(FU));
|
tfu = xzalloc(sizeof(FU));
|
||||||
*nextfu = tfu;
|
*nextfupp = tfu;
|
||||||
nextfu = &tfu->nextfu;
|
nextfupp = &tfu->nextfu;
|
||||||
tfu->reps = 1;
|
tfu->reps = 1;
|
||||||
|
|
||||||
/* if leading digit, repetition count */
|
/* if leading digit, repetition count */
|
||||||
@ -726,11 +748,11 @@ void FAST_FUNC bb_dump_add(const char *fmt)
|
|||||||
/* may overwrite either white space or slash */
|
/* may overwrite either white space or slash */
|
||||||
tfu->reps = atoi(savep);
|
tfu->reps = atoi(savep);
|
||||||
tfu->flags = F_SETREP;
|
tfu->flags = F_SETREP;
|
||||||
/* bb_dump_skip trailing white space */
|
/* skip trailing white space */
|
||||||
p = skip_whitespace(++p);
|
p = skip_whitespace(++p);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* bb_dump_skip slash and trailing white space */
|
/* skip slash and trailing white space */
|
||||||
if (*p == '/') {
|
if (*p == '/') {
|
||||||
p = skip_whitespace(++p);
|
p = skip_whitespace(++p);
|
||||||
}
|
}
|
||||||
@ -745,7 +767,7 @@ void FAST_FUNC bb_dump_add(const char *fmt)
|
|||||||
bb_error_msg_and_die("bad format {%s}", fmt);
|
bb_error_msg_and_die("bad format {%s}", fmt);
|
||||||
}
|
}
|
||||||
tfu->bcnt = atoi(savep);
|
tfu->bcnt = atoi(savep);
|
||||||
/* bb_dump_skip trailing white space */
|
/* skip trailing white space */
|
||||||
p = skip_whitespace(++p);
|
p = skip_whitespace(++p);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -15,7 +15,7 @@
|
|||||||
/* This is a NOEXEC applet. Be very careful! */
|
/* This is a NOEXEC applet. Be very careful! */
|
||||||
|
|
||||||
|
|
||||||
static void bb_dump_addfile(char *name)
|
static void bb_dump_addfile(dumper_t *dumper, char *name)
|
||||||
{
|
{
|
||||||
char *p;
|
char *p;
|
||||||
FILE *fp;
|
FILE *fp;
|
||||||
@ -27,7 +27,7 @@ static void bb_dump_addfile(char *name)
|
|||||||
p = skip_whitespace(buf);
|
p = skip_whitespace(buf);
|
||||||
|
|
||||||
if (*p && (*p != '#')) {
|
if (*p && (*p != '#')) {
|
||||||
bb_dump_add(p);
|
bb_dump_add(dumper, p);
|
||||||
}
|
}
|
||||||
free(buf);
|
free(buf);
|
||||||
}
|
}
|
||||||
@ -56,6 +56,7 @@ static const struct suffix_mult suffixes[] = {
|
|||||||
int hexdump_main(int argc, char **argv) MAIN_EXTERNALLY_VISIBLE;
|
int hexdump_main(int argc, char **argv) MAIN_EXTERNALLY_VISIBLE;
|
||||||
int hexdump_main(int argc, char **argv)
|
int hexdump_main(int argc, char **argv)
|
||||||
{
|
{
|
||||||
|
dumper_t *dumper = alloc_dumper();
|
||||||
const char *p;
|
const char *p;
|
||||||
int ch;
|
int ch;
|
||||||
#if ENABLE_FEATURE_HEXDUMP_REVERSE
|
#if ENABLE_FEATURE_HEXDUMP_REVERSE
|
||||||
@ -63,9 +64,6 @@ int hexdump_main(int argc, char **argv)
|
|||||||
smallint rdump = 0;
|
smallint rdump = 0;
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
bb_dump_vflag = FIRST;
|
|
||||||
bb_dump_length = -1;
|
|
||||||
|
|
||||||
if (ENABLE_HD && !applet_name[2]) { /* we are "hd" */
|
if (ENABLE_HD && !applet_name[2]) { /* we are "hd" */
|
||||||
ch = 'C';
|
ch = 'C';
|
||||||
goto hd_applet;
|
goto hd_applet;
|
||||||
@ -78,30 +76,30 @@ int hexdump_main(int argc, char **argv)
|
|||||||
if (!p)
|
if (!p)
|
||||||
bb_show_usage();
|
bb_show_usage();
|
||||||
if ((p - hexdump_opts) < 5) {
|
if ((p - hexdump_opts) < 5) {
|
||||||
bb_dump_add(add_first);
|
bb_dump_add(dumper, add_first);
|
||||||
bb_dump_add(add_strings[(int)(p - hexdump_opts)]);
|
bb_dump_add(dumper, add_strings[(int)(p - hexdump_opts)]);
|
||||||
}
|
}
|
||||||
/* Save a little bit of space below by omitting the 'else's. */
|
/* Save a little bit of space below by omitting the 'else's. */
|
||||||
if (ch == 'C') {
|
if (ch == 'C') {
|
||||||
hd_applet:
|
hd_applet:
|
||||||
bb_dump_add("\"%08.8_Ax\n\"");
|
bb_dump_add(dumper, "\"%08.8_Ax\n\"");
|
||||||
bb_dump_add("\"%08.8_ax \" 8/1 \"%02x \" \" \" 8/1 \"%02x \" ");
|
bb_dump_add(dumper, "\"%08.8_ax \" 8/1 \"%02x \" \" \" 8/1 \"%02x \" ");
|
||||||
bb_dump_add("\" |\" 16/1 \"%_p\" \"|\\n\"");
|
bb_dump_add(dumper, "\" |\" 16/1 \"%_p\" \"|\\n\"");
|
||||||
}
|
}
|
||||||
if (ch == 'e') {
|
if (ch == 'e') {
|
||||||
bb_dump_add(optarg);
|
bb_dump_add(dumper, optarg);
|
||||||
} /* else */
|
} /* else */
|
||||||
if (ch == 'f') {
|
if (ch == 'f') {
|
||||||
bb_dump_addfile(optarg);
|
bb_dump_addfile(dumper, optarg);
|
||||||
} /* else */
|
} /* else */
|
||||||
if (ch == 'n') {
|
if (ch == 'n') {
|
||||||
bb_dump_length = xatoi_u(optarg);
|
dumper->dump_length = xatoi_u(optarg);
|
||||||
} /* else */
|
} /* else */
|
||||||
if (ch == 's') {
|
if (ch == 's') {
|
||||||
bb_dump_skip = xatoul_range_sfx(optarg, 0, LONG_MAX, suffixes);
|
dumper->dump_skip = xatoul_range_sfx(optarg, 0, LONG_MAX, suffixes);
|
||||||
} /* else */
|
} /* else */
|
||||||
if (ch == 'v') {
|
if (ch == 'v') {
|
||||||
bb_dump_vflag = ALL;
|
dumper->dump_vflag = ALL;
|
||||||
}
|
}
|
||||||
#if ENABLE_FEATURE_HEXDUMP_REVERSE
|
#if ENABLE_FEATURE_HEXDUMP_REVERSE
|
||||||
if (ch == 'R') {
|
if (ch == 'R') {
|
||||||
@ -110,18 +108,18 @@ int hexdump_main(int argc, char **argv)
|
|||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!bb_dump_fshead) {
|
if (!dumper->fshead) {
|
||||||
bb_dump_add(add_first);
|
bb_dump_add(dumper, add_first);
|
||||||
bb_dump_add("\"%07.7_ax \" 8/2 \"%04x \" \"\\n\"");
|
bb_dump_add(dumper, "\"%07.7_ax \" 8/2 \"%04x \" \"\\n\"");
|
||||||
}
|
}
|
||||||
|
|
||||||
argv += optind;
|
argv += optind;
|
||||||
|
|
||||||
#if !ENABLE_FEATURE_HEXDUMP_REVERSE
|
#if !ENABLE_FEATURE_HEXDUMP_REVERSE
|
||||||
return bb_dump_dump(argv);
|
return bb_dump_dump(dumper, argv);
|
||||||
#else
|
#else
|
||||||
if (!rdump) {
|
if (!rdump) {
|
||||||
return bb_dump_dump(argv);
|
return bb_dump_dump(dumper, argv);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* -R: reverse of 'hexdump -Cv' */
|
/* -R: reverse of 'hexdump -Cv' */
|
||||||
|
Loading…
x
Reference in New Issue
Block a user