101335b0dSGarrett D'Amore /*
201335b0dSGarrett D'Amore * This file and its contents are supplied under the terms of the
301335b0dSGarrett D'Amore * Common Development and Distribution License ("CDDL"), version 1.0.
45aec55ebSGarrett D'Amore * You may only use this file in accordance with the terms of version
501335b0dSGarrett D'Amore * 1.0 of the CDDL.
601335b0dSGarrett D'Amore *
701335b0dSGarrett D'Amore * A full copy of the text of the CDDL should have accompanied this
801335b0dSGarrett D'Amore * source. A copy of the CDDL is also available via the Internet
901335b0dSGarrett D'Amore * http://www.illumos.org/license/CDDL.
1001335b0dSGarrett D'Amore */
1101335b0dSGarrett D'Amore
1201335b0dSGarrett D'Amore /*
1301335b0dSGarrett D'Amore * Copyright 2010 Nexenta Systems, Inc. All rights reserved.
1401335b0dSGarrett D'Amore */
1501335b0dSGarrett D'Amore
1601335b0dSGarrett D'Amore /*
1701335b0dSGarrett D'Amore * od - octal dump. Not really just octal anymore; read the POSIX
1801335b0dSGarrett D'Amore * specification for it -- its more complex than you think!
1901335b0dSGarrett D'Amore *
2001335b0dSGarrett D'Amore * NB: We followed the POSIX semantics fairly strictly, where the
2101335b0dSGarrett D'Amore * legacy code's behavior was in conflict. In many cases the legacy
2201335b0dSGarrett D'Amore * Solaris code was so completely broken as to be completely unusable.
2301335b0dSGarrett D'Amore * (For example, the long double support was broken beyond
2401335b0dSGarrett D'Amore * imagination!) Note that GNU coreutils violates POSIX in a few
2501335b0dSGarrett D'Amore * interesting ways, such as changing the numbering of the addresses
2601335b0dSGarrett D'Amore * when skipping. (Address starts should always be at 0, according to
2701335b0dSGarrett D'Amore * the sample output in the Open Group man page.)
2801335b0dSGarrett D'Amore */
2901335b0dSGarrett D'Amore
3001335b0dSGarrett D'Amore #include <stdio.h>
3101335b0dSGarrett D'Amore #include <stdlib.h>
3201335b0dSGarrett D'Amore #include <sys/types.h>
3301335b0dSGarrett D'Amore #include <string.h>
3401335b0dSGarrett D'Amore #include <err.h>
3501335b0dSGarrett D'Amore #include <wchar.h>
3601335b0dSGarrett D'Amore #include <locale.h>
3701335b0dSGarrett D'Amore #include <unistd.h>
3801335b0dSGarrett D'Amore #include <sys/stat.h>
3901335b0dSGarrett D'Amore
4001335b0dSGarrett D'Amore #define _(x) gettext(x)
4101335b0dSGarrett D'Amore
42*84441f85SGarrett D'Amore
43*84441f85SGarrett D'Amore #ifndef TEXT_DOMAIN
44*84441f85SGarrett D'Amore #define TEXT_DOMAIN "SYS_TEST"
45*84441f85SGarrett D'Amore #endif
46*84441f85SGarrett D'Amore
4701335b0dSGarrett D'Amore /* address format */
4801335b0dSGarrett D'Amore static char *afmt = "%07llo";
4901335b0dSGarrett D'Amore static char *cfmt = " ";
5001335b0dSGarrett D'Amore
5101335b0dSGarrett D'Amore static FILE *input = NULL;
5201335b0dSGarrett D'Amore static size_t lcm = 1;
5301335b0dSGarrett D'Amore static size_t blocksize = 16;
5401335b0dSGarrett D'Amore static int numfiles = 0;
5501335b0dSGarrett D'Amore static int curfile = 0;
5601335b0dSGarrett D'Amore static char **files = NULL;
5701335b0dSGarrett D'Amore static off_t limit = -1;
5801335b0dSGarrett D'Amore
5901335b0dSGarrett D'Amore /*
6001335b0dSGarrett D'Amore * This structure describes our ring buffer. Its always a power of 2
6101335b0dSGarrett D'Amore * in size to make wrap around calculations fast using a mask instead
6201335b0dSGarrett D'Amore * of doing modulo.
6301335b0dSGarrett D'Amore *
6401335b0dSGarrett D'Amore * The size is calculated thusly: We need three "blocks" of data, as
6501335b0dSGarrett D'Amore * we process a block at a time (one block == one line of od output.)
6601335b0dSGarrett D'Amore *
6701335b0dSGarrett D'Amore * We need lookahead of an extra block to support multibyte chars. We
6801335b0dSGarrett D'Amore * also have a look behind so that we can avoid printing lines that
6901335b0dSGarrett D'Amore * are identical to what we've already printed. Finally, we need the
7001335b0dSGarrett D'Amore * current block.
7101335b0dSGarrett D'Amore *
7201335b0dSGarrett D'Amore * The block size is determined by the least common multiple of the
7301335b0dSGarrett D'Amore * data items being displayed. Usually it will be 16, but sometimes
7401335b0dSGarrett D'Amore * it is 24 (when 12-byte long doubles are presented.)
7501335b0dSGarrett D'Amore *
7601335b0dSGarrett D'Amore * The data buffer is allocaed via memalign to make sure it is
7701335b0dSGarrett D'Amore * properly aligned.
7801335b0dSGarrett D'Amore */
7901335b0dSGarrett D'Amore typedef struct buffer {
8001335b0dSGarrett D'Amore char *data; /* data buffer */
8101335b0dSGarrett D'Amore int prod; /* producer index */
8201335b0dSGarrett D'Amore int cons; /* consumer index */
8301335b0dSGarrett D'Amore int mask; /* buffer size - 1, wraparound index */
8401335b0dSGarrett D'Amore int navail; /* total bytes avail */
8501335b0dSGarrett D'Amore } buffer_t;
8601335b0dSGarrett D'Amore
8701335b0dSGarrett D'Amore /*
8801335b0dSGarrett D'Amore * This structure is used to provide information on a specific output
8901335b0dSGarrett D'Amore * format. We link them together in a list representing the output
9001335b0dSGarrett D'Amore * formats that the user has selected.
9101335b0dSGarrett D'Amore */
9201335b0dSGarrett D'Amore typedef struct output {
9301335b0dSGarrett D'Amore int width; /* bytes consumed per call */
9401335b0dSGarrett D'Amore void (*func)(buffer_t *, int); /* output function */
9501335b0dSGarrett D'Amore struct output *next; /* link node */
9601335b0dSGarrett D'Amore } output_t;
9701335b0dSGarrett D'Amore
9801335b0dSGarrett D'Amore /*
9901335b0dSGarrett D'Amore * Specifiers
10001335b0dSGarrett D'Amore */
10101335b0dSGarrett D'Amore
10201335b0dSGarrett D'Amore typedef unsigned char u8;
10301335b0dSGarrett D'Amore typedef unsigned short u16;
10401335b0dSGarrett D'Amore typedef unsigned int u32;
10501335b0dSGarrett D'Amore typedef unsigned long long u64;
10601335b0dSGarrett D'Amore typedef char s8;
10701335b0dSGarrett D'Amore typedef short s16;
10801335b0dSGarrett D'Amore typedef int s32;
10901335b0dSGarrett D'Amore typedef long long s64;
11001335b0dSGarrett D'Amore typedef float fF;
11101335b0dSGarrett D'Amore typedef double fD;
11201335b0dSGarrett D'Amore typedef long double fL;
11301335b0dSGarrett D'Amore
11401335b0dSGarrett D'Amore static void
usage(void)11501335b0dSGarrett D'Amore usage(void)
11601335b0dSGarrett D'Amore {
11701335b0dSGarrett D'Amore (void) fprintf(stderr, _("usage: od [-bcCdDfFoOsSvxX] "
11801335b0dSGarrett D'Amore "[-t types ]... [-A base] [-j skip] [-N count] [file]...\n"));
11901335b0dSGarrett D'Amore exit(1);
12001335b0dSGarrett D'Amore }
12101335b0dSGarrett D'Amore
12201335b0dSGarrett D'Amore #define DECL_GET(typ) \
12301335b0dSGarrett D'Amore static typ \
12401335b0dSGarrett D'Amore get_ ## typ(buffer_t *b, int index) \
12501335b0dSGarrett D'Amore { \
12601335b0dSGarrett D'Amore typ val = *(typ *)(void *)(b->data + index); \
12701335b0dSGarrett D'Amore return (val); \
12801335b0dSGarrett D'Amore }
12901335b0dSGarrett D'Amore DECL_GET(u8)
13001335b0dSGarrett D'Amore DECL_GET(u16)
13101335b0dSGarrett D'Amore DECL_GET(u32)
13201335b0dSGarrett D'Amore DECL_GET(u64)
13301335b0dSGarrett D'Amore DECL_GET(s8)
13401335b0dSGarrett D'Amore DECL_GET(s16)
13501335b0dSGarrett D'Amore DECL_GET(s32)
13601335b0dSGarrett D'Amore DECL_GET(s64)
13701335b0dSGarrett D'Amore DECL_GET(fF)
13801335b0dSGarrett D'Amore DECL_GET(fD)
13901335b0dSGarrett D'Amore DECL_GET(fL)
14001335b0dSGarrett D'Amore
14101335b0dSGarrett D'Amore #define DECL_OUT(nm, typ, fmt) \
14201335b0dSGarrett D'Amore static void \
14301335b0dSGarrett D'Amore do_ ## nm(buffer_t *buf, int index) \
14401335b0dSGarrett D'Amore { \
14501335b0dSGarrett D'Amore typ v = get_ ## typ(buf, index); \
14601335b0dSGarrett D'Amore (void) printf(fmt, v); \
14701335b0dSGarrett D'Amore } \
14801335b0dSGarrett D'Amore \
14901335b0dSGarrett D'Amore static output_t output_ ## nm = { \
15001335b0dSGarrett D'Amore sizeof (typ), do_ ## nm \
15101335b0dSGarrett D'Amore };
15201335b0dSGarrett D'Amore
15301335b0dSGarrett D'Amore DECL_OUT(oct_b, u8, " %03o")
15401335b0dSGarrett D'Amore DECL_OUT(oct_w, u16, " %06ho")
15501335b0dSGarrett D'Amore DECL_OUT(oct_d, u32, " %011o")
15601335b0dSGarrett D'Amore DECL_OUT(oct_q, u64, " %022llo")
15701335b0dSGarrett D'Amore DECL_OUT(dec_b, u8, " %03u")
15801335b0dSGarrett D'Amore DECL_OUT(dec_w, u16, " %05hu")
15901335b0dSGarrett D'Amore DECL_OUT(dec_d, u32, " %010u")
16001335b0dSGarrett D'Amore DECL_OUT(dec_q, u64, " %020llu")
16101335b0dSGarrett D'Amore DECL_OUT(sig_b, s8, " %03d")
16201335b0dSGarrett D'Amore DECL_OUT(sig_w, s16, " %6.05hd")
16301335b0dSGarrett D'Amore DECL_OUT(sig_d, s32, " %11.010d")
16401335b0dSGarrett D'Amore DECL_OUT(sig_q, s64, " %20.019lld")
16501335b0dSGarrett D'Amore DECL_OUT(hex_b, u8, " %02x")
16601335b0dSGarrett D'Amore DECL_OUT(hex_w, u16, " %04hx")
16701335b0dSGarrett D'Amore DECL_OUT(hex_d, s32, " %08x")
16801335b0dSGarrett D'Amore DECL_OUT(hex_q, s64, " %016llx")
16901335b0dSGarrett D'Amore DECL_OUT(float, fF, " %14.7e")
17001335b0dSGarrett D'Amore DECL_OUT(double, fD, " %21.14e")
17101335b0dSGarrett D'Amore DECL_OUT(ldouble, fL, " %24.14Le")
17201335b0dSGarrett D'Amore
17301335b0dSGarrett D'Amore static char *ascii[] = {
17401335b0dSGarrett D'Amore "nul", "soh", "stx", "etx", "eot", "enq", "ack", " be",
17501335b0dSGarrett D'Amore " bs", " ht", " lf", " vt", " ff", " cr", " so", " si",
17601335b0dSGarrett D'Amore "dle", "dc1", "dc2", "dc3", "dc4", "nak", "syn", "etb",
17701335b0dSGarrett D'Amore "can", " em", "sub", "esc", " fs", " gs", " rs", " us",
17801335b0dSGarrett D'Amore " sp", " !", " \"", " #", " $", " %", " &", " '",
17901335b0dSGarrett D'Amore " (", " )", " *", " +", " ,", " -", " .", " /",
18001335b0dSGarrett D'Amore " 0", " 1", " 2", " 3", " 4", " 5", " 6", " 7",
18101335b0dSGarrett D'Amore " 8", " 9", " :", " ;", " <", " =", " >", " ?",
18201335b0dSGarrett D'Amore " @", " A", " B", " C", " D", " E", " F", " G",
18301335b0dSGarrett D'Amore " H", " I", " J", " K", " L", " M", " N", " O",
18401335b0dSGarrett D'Amore " P", " Q", " R", " S", " T", " U", " V", " W",
18501335b0dSGarrett D'Amore " X", " Y", " Z", " [", " \\", " ]", " ^", " _",
18601335b0dSGarrett D'Amore " `", " a", " b", " c", " d", " e", " f", " g",
18701335b0dSGarrett D'Amore " h", " i", " j", " k", " l", " m", " n", " o",
18801335b0dSGarrett D'Amore " p", " q", " r", " s", " t", " u", " v", " w",
18901335b0dSGarrett D'Amore " x", " y", " z", " {", " |", " }", " ~", "del"
19001335b0dSGarrett D'Amore };
19101335b0dSGarrett D'Amore
19201335b0dSGarrett D'Amore static void
do_ascii(buffer_t * buf,int index)19301335b0dSGarrett D'Amore do_ascii(buffer_t *buf, int index)
19401335b0dSGarrett D'Amore {
19501335b0dSGarrett D'Amore uint8_t v = get_u8(buf, index);
19601335b0dSGarrett D'Amore
19701335b0dSGarrett D'Amore (void) fputc(' ', stdout);
19801335b0dSGarrett D'Amore (void) fputs(ascii[v & 0x7f], stdout);
19901335b0dSGarrett D'Amore }
20001335b0dSGarrett D'Amore
20101335b0dSGarrett D'Amore static output_t output_ascii = {
20201335b0dSGarrett D'Amore 1, do_ascii,
20301335b0dSGarrett D'Amore };
20401335b0dSGarrett D'Amore
20501335b0dSGarrett D'Amore static void
do_char(buffer_t * buf,int index)20601335b0dSGarrett D'Amore do_char(buffer_t *buf, int index)
20701335b0dSGarrett D'Amore {
20801335b0dSGarrett D'Amore static int nresid = 0;
20901335b0dSGarrett D'Amore static int printable = 0;
21001335b0dSGarrett D'Amore int cnt;
21101335b0dSGarrett D'Amore int avail;
21201335b0dSGarrett D'Amore int nb;
21301335b0dSGarrett D'Amore char scratch[10];
21401335b0dSGarrett D'Amore wchar_t wc;
21501335b0dSGarrett D'Amore int which;
21601335b0dSGarrett D'Amore
21701335b0dSGarrett D'Amore uint8_t v = get_u8(buf, index);
21801335b0dSGarrett D'Amore
21901335b0dSGarrett D'Amore /*
22001335b0dSGarrett D'Amore * If there were residual bytes from an earlier
22101335b0dSGarrett D'Amore * character, then just display the ** continuation
22201335b0dSGarrett D'Amore * indication.
22301335b0dSGarrett D'Amore */
22401335b0dSGarrett D'Amore if (nresid) {
22501335b0dSGarrett D'Amore if (printable) {
22601335b0dSGarrett D'Amore (void) fputs(" **", stdout);
22701335b0dSGarrett D'Amore } else {
22801335b0dSGarrett D'Amore (void) printf(" %03o", v);
22901335b0dSGarrett D'Amore }
23001335b0dSGarrett D'Amore nresid--;
23101335b0dSGarrett D'Amore return;
23201335b0dSGarrett D'Amore }
23301335b0dSGarrett D'Amore
23401335b0dSGarrett D'Amore /*
23501335b0dSGarrett D'Amore * Peek ahead up to MB_CUR_MAX characters. This has to be
23601335b0dSGarrett D'Amore * done carefully because we might need to look into the next
23701335b0dSGarrett D'Amore * block to really know for sure.
23801335b0dSGarrett D'Amore */
23901335b0dSGarrett D'Amore scratch[0] = v;
24001335b0dSGarrett D'Amore avail = buf->navail;
24101335b0dSGarrett D'Amore if (avail > MB_CUR_MAX)
24201335b0dSGarrett D'Amore avail = MB_CUR_MAX;
24301335b0dSGarrett D'Amore for (cnt = 1, which = index + 1; cnt < avail; cnt++, which++) {
24401335b0dSGarrett D'Amore scratch[cnt] = buf->data[which & buf->mask];
24501335b0dSGarrett D'Amore }
24601335b0dSGarrett D'Amore
24701335b0dSGarrett D'Amore /* now see if the value is a real character */
24801335b0dSGarrett D'Amore nresid = 0;
24901335b0dSGarrett D'Amore wc = 0;
25001335b0dSGarrett D'Amore nb = mbtowc(&wc, scratch, avail);
25101335b0dSGarrett D'Amore if (nb < 0) {
25201335b0dSGarrett D'Amore (void) printf(" %03o", v);
25301335b0dSGarrett D'Amore return;
25401335b0dSGarrett D'Amore }
25501335b0dSGarrett D'Amore if (nb == 0) {
25601335b0dSGarrett D'Amore (void) fputs(" \\0", stdout);
25701335b0dSGarrett D'Amore return;
25801335b0dSGarrett D'Amore }
25901335b0dSGarrett D'Amore nresid = nb - 1;
26001335b0dSGarrett D'Amore if (nb && iswprint(wc)) {
26101335b0dSGarrett D'Amore scratch[nb] = 0;
26201335b0dSGarrett D'Amore (void) fputs(" ", stdout);
26301335b0dSGarrett D'Amore (void) fputs(scratch, stdout);
26401335b0dSGarrett D'Amore printable = 1;
26501335b0dSGarrett D'Amore return;
26601335b0dSGarrett D'Amore }
26701335b0dSGarrett D'Amore printable = 0;
26801335b0dSGarrett D'Amore if (wc == 0) {
26901335b0dSGarrett D'Amore (void) fputs(" \\0", stdout);
27001335b0dSGarrett D'Amore } else if (wc == '\b') {
27101335b0dSGarrett D'Amore (void) fputs(" \\b", stdout);
27201335b0dSGarrett D'Amore } else if (wc == '\f') {
27301335b0dSGarrett D'Amore (void) fputs(" \\f", stdout);
27401335b0dSGarrett D'Amore } else if (wc == '\n') {
27501335b0dSGarrett D'Amore (void) fputs(" \\n", stdout);
27601335b0dSGarrett D'Amore } else if (wc == '\r') {
27701335b0dSGarrett D'Amore (void) fputs(" \\r", stdout);
27801335b0dSGarrett D'Amore } else if (wc == '\t') {
27901335b0dSGarrett D'Amore (void) fputs(" \\t", stdout);
28001335b0dSGarrett D'Amore } else {
28101335b0dSGarrett D'Amore (void) printf(" %03o", v);
28201335b0dSGarrett D'Amore }
28301335b0dSGarrett D'Amore }
28401335b0dSGarrett D'Amore
28501335b0dSGarrett D'Amore static output_t output_char = {
28601335b0dSGarrett D'Amore 1, do_char,
28701335b0dSGarrett D'Amore };
28801335b0dSGarrett D'Amore
28901335b0dSGarrett D'Amore /*
29001335b0dSGarrett D'Amore * List of output formatting structures.
29101335b0dSGarrett D'Amore */
29201335b0dSGarrett D'Amore static output_t *head = NULL;
29301335b0dSGarrett D'Amore static output_t **tailp = &head;
29401335b0dSGarrett D'Amore
29501335b0dSGarrett D'Amore static void
add_out(output_t * src)29601335b0dSGarrett D'Amore add_out(output_t *src)
29701335b0dSGarrett D'Amore {
29801335b0dSGarrett D'Amore output_t *out;
29901335b0dSGarrett D'Amore int m;
30001335b0dSGarrett D'Amore
30101335b0dSGarrett D'Amore if ((out = calloc(1, sizeof (*src))) == NULL) {
30201335b0dSGarrett D'Amore err(1, "malloc");
30301335b0dSGarrett D'Amore }
30401335b0dSGarrett D'Amore
30501335b0dSGarrett D'Amore m = lcm;
30601335b0dSGarrett D'Amore while ((m % src->width) != 0) {
30701335b0dSGarrett D'Amore m += lcm;
30801335b0dSGarrett D'Amore }
30901335b0dSGarrett D'Amore lcm = m;
31001335b0dSGarrett D'Amore blocksize = lcm;
31101335b0dSGarrett D'Amore while (blocksize < 16)
31201335b0dSGarrett D'Amore blocksize *= 2;
31301335b0dSGarrett D'Amore
31401335b0dSGarrett D'Amore (void) memcpy(out, src, sizeof (*src));
31501335b0dSGarrett D'Amore *tailp = out;
31601335b0dSGarrett D'Amore tailp = &out->next;
31701335b0dSGarrett D'Amore }
31801335b0dSGarrett D'Amore
31901335b0dSGarrett D'Amore static FILE *
next_input(void)32001335b0dSGarrett D'Amore next_input(void)
32101335b0dSGarrett D'Amore {
32201335b0dSGarrett D'Amore for (;;) {
32301335b0dSGarrett D'Amore if (curfile >= numfiles)
32401335b0dSGarrett D'Amore return (NULL);
32501335b0dSGarrett D'Amore
32601335b0dSGarrett D'Amore if (input != NULL) {
32701335b0dSGarrett D'Amore if ((input = freopen(files[curfile], "r", input)) !=
32801335b0dSGarrett D'Amore NULL) {
32901335b0dSGarrett D'Amore curfile++;
33001335b0dSGarrett D'Amore return (input);
33101335b0dSGarrett D'Amore }
33201335b0dSGarrett D'Amore } else {
33301335b0dSGarrett D'Amore if ((input = fopen(files[curfile], "r")) != NULL) {
33401335b0dSGarrett D'Amore curfile++;
33501335b0dSGarrett D'Amore return (input);
33601335b0dSGarrett D'Amore }
33701335b0dSGarrett D'Amore }
33801335b0dSGarrett D'Amore warn("open: %s", files[curfile]);
33901335b0dSGarrett D'Amore curfile++;
34001335b0dSGarrett D'Amore }
34101335b0dSGarrett D'Amore }
34201335b0dSGarrett D'Amore
34301335b0dSGarrett D'Amore static void
refill(buffer_t * b)34401335b0dSGarrett D'Amore refill(buffer_t *b)
34501335b0dSGarrett D'Amore {
34601335b0dSGarrett D'Amore int n;
34701335b0dSGarrett D'Amore int want;
34801335b0dSGarrett D'Amore int zero;
34901335b0dSGarrett D'Amore
35001335b0dSGarrett D'Amore /*
35101335b0dSGarrett D'Amore * If we have 2 blocks of bytes available, we're done. Note
35201335b0dSGarrett D'Amore * that each iteration usually loads up 16 bytes, unless we
35301335b0dSGarrett D'Amore * run out of data.
35401335b0dSGarrett D'Amore */
35501335b0dSGarrett D'Amore while ((input != NULL) && (b->navail < (2 * blocksize))) {
35601335b0dSGarrett D'Amore
35701335b0dSGarrett D'Amore /* we preload the next one in advance */
35801335b0dSGarrett D'Amore
35901335b0dSGarrett D'Amore if (limit == 0) {
36001335b0dSGarrett D'Amore (void) fclose(input);
36101335b0dSGarrett D'Amore input = NULL;
36201335b0dSGarrett D'Amore continue;
36301335b0dSGarrett D'Amore }
36401335b0dSGarrett D'Amore
36501335b0dSGarrett D'Amore /* we want to read a whole block if possible */
36601335b0dSGarrett D'Amore want = blocksize;
36701335b0dSGarrett D'Amore if ((limit >= 0) && (want > limit)) {
36801335b0dSGarrett D'Amore want = limit;
36901335b0dSGarrett D'Amore }
37001335b0dSGarrett D'Amore zero = blocksize;
37101335b0dSGarrett D'Amore
37201335b0dSGarrett D'Amore while (want && input) {
37301335b0dSGarrett D'Amore int c;
37401335b0dSGarrett D'Amore b->prod &= b->mask;
37501335b0dSGarrett D'Amore c = (b->prod + want > (b->mask + 1)) ?
37601335b0dSGarrett D'Amore b->mask - b->prod :
37701335b0dSGarrett D'Amore want;
37801335b0dSGarrett D'Amore
37901335b0dSGarrett D'Amore n = fread(b->data + b->prod, 1, c, input);
38001335b0dSGarrett D'Amore if (n < 0) {
38101335b0dSGarrett D'Amore warn("read: %s",
38201335b0dSGarrett D'Amore files ? files[curfile-1] : "stdin");
38301335b0dSGarrett D'Amore input = next_input();
38401335b0dSGarrett D'Amore continue;
38501335b0dSGarrett D'Amore }
38601335b0dSGarrett D'Amore if (n == 0) {
38701335b0dSGarrett D'Amore input = next_input();
38801335b0dSGarrett D'Amore continue;
38901335b0dSGarrett D'Amore }
39001335b0dSGarrett D'Amore if (limit >= 0)
39101335b0dSGarrett D'Amore limit -= n;
39201335b0dSGarrett D'Amore b->navail += n;
39301335b0dSGarrett D'Amore b->prod += n;
39401335b0dSGarrett D'Amore want -= n;
39501335b0dSGarrett D'Amore zero -= n;
39601335b0dSGarrett D'Amore }
39701335b0dSGarrett D'Amore
39801335b0dSGarrett D'Amore while (zero) {
39901335b0dSGarrett D'Amore b->data[b->prod & b->mask] = 0;
40001335b0dSGarrett D'Amore b->prod++;
40101335b0dSGarrett D'Amore b->prod &= b->mask;
40201335b0dSGarrett D'Amore zero--;
40301335b0dSGarrett D'Amore }
40401335b0dSGarrett D'Amore }
40501335b0dSGarrett D'Amore }
40601335b0dSGarrett D'Amore
40701335b0dSGarrett D'Amore #define STR1 "C1"
40801335b0dSGarrett D'Amore #define STR2 "S2"
40901335b0dSGarrett D'Amore #ifdef _LP64
41001335b0dSGarrett D'Amore #define STR8 "L8"
41101335b0dSGarrett D'Amore #define STR4 "I4"
41201335b0dSGarrett D'Amore #else
41301335b0dSGarrett D'Amore #define STR8 "8"
41401335b0dSGarrett D'Amore #define STR4 "IL4"
41501335b0dSGarrett D'Amore #endif
41601335b0dSGarrett D'Amore
41701335b0dSGarrett D'Amore static void
do_type_string(char * typestr)41801335b0dSGarrett D'Amore do_type_string(char *typestr)
41901335b0dSGarrett D'Amore {
42001335b0dSGarrett D'Amore if (*typestr == 0) {
42101335b0dSGarrett D'Amore errx(1, _("missing type string"));
42201335b0dSGarrett D'Amore }
42301335b0dSGarrett D'Amore while (*typestr) {
42401335b0dSGarrett D'Amore switch (*typestr) {
42501335b0dSGarrett D'Amore case 'a':
42601335b0dSGarrett D'Amore typestr++;
42701335b0dSGarrett D'Amore add_out(&output_ascii);
42801335b0dSGarrett D'Amore break;
42901335b0dSGarrett D'Amore case 'c':
43001335b0dSGarrett D'Amore add_out(&output_char);
43101335b0dSGarrett D'Amore typestr++;
43201335b0dSGarrett D'Amore break;
43301335b0dSGarrett D'Amore case 'f':
43401335b0dSGarrett D'Amore typestr++;
43501335b0dSGarrett D'Amore switch (*typestr) {
43601335b0dSGarrett D'Amore case 'F':
43701335b0dSGarrett D'Amore case '4':
43801335b0dSGarrett D'Amore add_out(&output_float);
43901335b0dSGarrett D'Amore typestr++;
44001335b0dSGarrett D'Amore break;
44101335b0dSGarrett D'Amore case '8':
44201335b0dSGarrett D'Amore case 'D':
44301335b0dSGarrett D'Amore add_out(&output_double);
44401335b0dSGarrett D'Amore typestr++;
44501335b0dSGarrett D'Amore break;
44601335b0dSGarrett D'Amore case 'L':
44701335b0dSGarrett D'Amore add_out(&output_ldouble);
44801335b0dSGarrett D'Amore typestr++;
44901335b0dSGarrett D'Amore break;
45001335b0dSGarrett D'Amore default:
45101335b0dSGarrett D'Amore add_out(&output_float);
45201335b0dSGarrett D'Amore break;
45301335b0dSGarrett D'Amore }
45401335b0dSGarrett D'Amore break;
45501335b0dSGarrett D'Amore
45601335b0dSGarrett D'Amore
45701335b0dSGarrett D'Amore case 'd':
45801335b0dSGarrett D'Amore typestr++;
45901335b0dSGarrett D'Amore if (strchr(STR1, *typestr)) {
46001335b0dSGarrett D'Amore typestr++;
46101335b0dSGarrett D'Amore add_out(&output_sig_b);
46201335b0dSGarrett D'Amore } else if (strchr(STR2, *typestr)) {
46301335b0dSGarrett D'Amore typestr++;
46401335b0dSGarrett D'Amore add_out(&output_sig_w);
46501335b0dSGarrett D'Amore } else if (strchr(STR4, *typestr)) {
46601335b0dSGarrett D'Amore typestr++;
46701335b0dSGarrett D'Amore add_out(&output_sig_d);
46801335b0dSGarrett D'Amore } else if (strchr(STR8, *typestr)) {
46901335b0dSGarrett D'Amore typestr++;
47001335b0dSGarrett D'Amore add_out(&output_sig_q);
47101335b0dSGarrett D'Amore } else {
47201335b0dSGarrett D'Amore add_out(&output_sig_d);
47301335b0dSGarrett D'Amore }
47401335b0dSGarrett D'Amore break;
47501335b0dSGarrett D'Amore
47601335b0dSGarrett D'Amore case 'u':
47701335b0dSGarrett D'Amore typestr++;
47801335b0dSGarrett D'Amore if (strchr(STR1, *typestr)) {
47901335b0dSGarrett D'Amore typestr++;
48001335b0dSGarrett D'Amore add_out(&output_dec_b);
48101335b0dSGarrett D'Amore } else if (strchr(STR2, *typestr)) {
48201335b0dSGarrett D'Amore typestr++;
48301335b0dSGarrett D'Amore add_out(&output_dec_w);
48401335b0dSGarrett D'Amore } else if (strchr(STR4, *typestr)) {
48501335b0dSGarrett D'Amore typestr++;
48601335b0dSGarrett D'Amore add_out(&output_dec_d);
48701335b0dSGarrett D'Amore } else if (strchr(STR8, *typestr)) {
48801335b0dSGarrett D'Amore typestr++;
48901335b0dSGarrett D'Amore add_out(&output_dec_q);
49001335b0dSGarrett D'Amore } else {
49101335b0dSGarrett D'Amore add_out(&output_dec_d);
49201335b0dSGarrett D'Amore }
49301335b0dSGarrett D'Amore break;
49401335b0dSGarrett D'Amore
49501335b0dSGarrett D'Amore case 'o':
49601335b0dSGarrett D'Amore typestr++;
49701335b0dSGarrett D'Amore if (strchr(STR1, *typestr)) {
49801335b0dSGarrett D'Amore typestr++;
49901335b0dSGarrett D'Amore add_out(&output_oct_b);
50001335b0dSGarrett D'Amore } else if (strchr(STR2, *typestr)) {
50101335b0dSGarrett D'Amore typestr++;
50201335b0dSGarrett D'Amore add_out(&output_oct_w);
50301335b0dSGarrett D'Amore } else if (strchr(STR4, *typestr)) {
50401335b0dSGarrett D'Amore typestr++;
50501335b0dSGarrett D'Amore add_out(&output_oct_d);
50601335b0dSGarrett D'Amore } else if (strchr(STR8, *typestr)) {
50701335b0dSGarrett D'Amore typestr++;
50801335b0dSGarrett D'Amore add_out(&output_oct_q);
50901335b0dSGarrett D'Amore } else {
51001335b0dSGarrett D'Amore add_out(&output_oct_d);
51101335b0dSGarrett D'Amore }
51201335b0dSGarrett D'Amore break;
51301335b0dSGarrett D'Amore
51401335b0dSGarrett D'Amore case 'x':
51501335b0dSGarrett D'Amore typestr++;
51601335b0dSGarrett D'Amore if (strchr(STR1, *typestr)) {
51701335b0dSGarrett D'Amore typestr++;
51801335b0dSGarrett D'Amore add_out(&output_hex_b);
51901335b0dSGarrett D'Amore } else if (strchr(STR2, *typestr)) {
52001335b0dSGarrett D'Amore typestr++;
52101335b0dSGarrett D'Amore add_out(&output_hex_w);
52201335b0dSGarrett D'Amore } else if (strchr(STR4, *typestr)) {
52301335b0dSGarrett D'Amore typestr++;
52401335b0dSGarrett D'Amore add_out(&output_hex_d);
52501335b0dSGarrett D'Amore } else if (strchr(STR8, *typestr)) {
52601335b0dSGarrett D'Amore typestr++;
52701335b0dSGarrett D'Amore add_out(&output_hex_q);
52801335b0dSGarrett D'Amore } else {
52901335b0dSGarrett D'Amore add_out(&output_hex_d);
53001335b0dSGarrett D'Amore }
53101335b0dSGarrett D'Amore break;
53201335b0dSGarrett D'Amore
53301335b0dSGarrett D'Amore default:
53401335b0dSGarrett D'Amore errx(1, _("unrecognized type string character: %c"),
53501335b0dSGarrett D'Amore *typestr);
53601335b0dSGarrett D'Amore exit(1);
53701335b0dSGarrett D'Amore }
53801335b0dSGarrett D'Amore }
53901335b0dSGarrett D'Amore }
54001335b0dSGarrett D'Amore
54101335b0dSGarrett D'Amore int
main(int argc,char ** argv)54201335b0dSGarrett D'Amore main(int argc, char **argv)
54301335b0dSGarrett D'Amore {
54401335b0dSGarrett D'Amore int c;
54501335b0dSGarrett D'Amore int i;
54601335b0dSGarrett D'Amore buffer_t buffer;
54701335b0dSGarrett D'Amore boolean_t first = B_TRUE;
54801335b0dSGarrett D'Amore boolean_t doall = B_FALSE;
54901335b0dSGarrett D'Amore boolean_t same = B_FALSE;
55001335b0dSGarrett D'Amore boolean_t newarg = B_FALSE;
55101335b0dSGarrett D'Amore off_t offset = 0;
55201335b0dSGarrett D'Amore off_t skip = 0;
55301335b0dSGarrett D'Amore char *eptr;
55401335b0dSGarrett D'Amore char *offstr = 0;
55501335b0dSGarrett D'Amore
55601335b0dSGarrett D'Amore input = stdin;
55701335b0dSGarrett D'Amore
55801335b0dSGarrett D'Amore (void) setlocale(LC_ALL, "");
559*84441f85SGarrett D'Amore (void) textdomain(TEXT_DOMAIN);
56001335b0dSGarrett D'Amore
56101335b0dSGarrett D'Amore while ((c = getopt(argc, argv, "A:bCcdDfFj:N:oOsSxXvt:")) != EOF) {
56201335b0dSGarrett D'Amore switch (c) {
56301335b0dSGarrett D'Amore case 'A':
56401335b0dSGarrett D'Amore newarg = B_TRUE;
56501335b0dSGarrett D'Amore if (strlen(optarg) > 1) {
56601335b0dSGarrett D'Amore afmt = NULL;
56701335b0dSGarrett D'Amore }
56801335b0dSGarrett D'Amore switch (*optarg) {
56901335b0dSGarrett D'Amore case 'o':
57001335b0dSGarrett D'Amore afmt = "%07llo";
57101335b0dSGarrett D'Amore cfmt = " ";
57201335b0dSGarrett D'Amore break;
57301335b0dSGarrett D'Amore case 'd':
57401335b0dSGarrett D'Amore afmt = "%07lld";
57501335b0dSGarrett D'Amore cfmt = " ";
57601335b0dSGarrett D'Amore break;
57701335b0dSGarrett D'Amore case 'x':
57801335b0dSGarrett D'Amore afmt = "%07llx";
57901335b0dSGarrett D'Amore cfmt = " ";
58001335b0dSGarrett D'Amore break;
58101335b0dSGarrett D'Amore case 'n':
58201335b0dSGarrett D'Amore /*
58301335b0dSGarrett D'Amore * You could argue that the code should
58401335b0dSGarrett D'Amore * use the same 7 spaces. Legacy uses 8
58501335b0dSGarrett D'Amore * though. Oh well. Better to avoid
58601335b0dSGarrett D'Amore * gratuitous change.
58701335b0dSGarrett D'Amore */
58801335b0dSGarrett D'Amore afmt = " ";
58901335b0dSGarrett D'Amore cfmt = " ";
59001335b0dSGarrett D'Amore break;
59101335b0dSGarrett D'Amore default:
59201335b0dSGarrett D'Amore afmt = NULL;
59301335b0dSGarrett D'Amore break;
59401335b0dSGarrett D'Amore }
59501335b0dSGarrett D'Amore if (strlen(optarg) != 1) {
59601335b0dSGarrett D'Amore afmt = NULL;
59701335b0dSGarrett D'Amore }
59801335b0dSGarrett D'Amore if (afmt == NULL)
59901335b0dSGarrett D'Amore warnx(_("invalid address base, "
60001335b0dSGarrett D'Amore "must be o, d, x, or n"));
60101335b0dSGarrett D'Amore break;
60201335b0dSGarrett D'Amore
60301335b0dSGarrett D'Amore case 'b':
60401335b0dSGarrett D'Amore add_out(&output_oct_b);
60501335b0dSGarrett D'Amore break;
60601335b0dSGarrett D'Amore
60701335b0dSGarrett D'Amore case 'c':
60801335b0dSGarrett D'Amore case 'C':
60901335b0dSGarrett D'Amore add_out(&output_char);
61001335b0dSGarrett D'Amore break;
61101335b0dSGarrett D'Amore
61201335b0dSGarrett D'Amore case 'f':
61301335b0dSGarrett D'Amore add_out(&output_float);
61401335b0dSGarrett D'Amore break;
61501335b0dSGarrett D'Amore
61601335b0dSGarrett D'Amore case 'F':
61701335b0dSGarrett D'Amore add_out(&output_double);
61801335b0dSGarrett D'Amore break;
61901335b0dSGarrett D'Amore
62001335b0dSGarrett D'Amore case 'd':
62101335b0dSGarrett D'Amore add_out(&output_dec_w);
62201335b0dSGarrett D'Amore break;
62301335b0dSGarrett D'Amore
62401335b0dSGarrett D'Amore case 'D':
62501335b0dSGarrett D'Amore add_out(&output_dec_d);
62601335b0dSGarrett D'Amore break;
62701335b0dSGarrett D'Amore
62801335b0dSGarrett D'Amore case 't':
62901335b0dSGarrett D'Amore newarg = B_TRUE;
63001335b0dSGarrett D'Amore do_type_string(optarg);
63101335b0dSGarrett D'Amore break;
63201335b0dSGarrett D'Amore
63301335b0dSGarrett D'Amore case 'o':
63401335b0dSGarrett D'Amore add_out(&output_oct_w);
63501335b0dSGarrett D'Amore break;
63601335b0dSGarrett D'Amore
63701335b0dSGarrett D'Amore case 'O':
63801335b0dSGarrett D'Amore add_out(&output_oct_d);
63901335b0dSGarrett D'Amore break;
64001335b0dSGarrett D'Amore
64101335b0dSGarrett D'Amore case 's':
64201335b0dSGarrett D'Amore add_out(&output_sig_w);
64301335b0dSGarrett D'Amore break;
64401335b0dSGarrett D'Amore
64501335b0dSGarrett D'Amore case 'S':
64601335b0dSGarrett D'Amore add_out(&output_sig_d);
64701335b0dSGarrett D'Amore break;
64801335b0dSGarrett D'Amore
64901335b0dSGarrett D'Amore case 'x':
65001335b0dSGarrett D'Amore add_out(&output_hex_w);
65101335b0dSGarrett D'Amore break;
65201335b0dSGarrett D'Amore
65301335b0dSGarrett D'Amore case 'X':
65401335b0dSGarrett D'Amore add_out(&output_hex_d);
65501335b0dSGarrett D'Amore break;
65601335b0dSGarrett D'Amore
65701335b0dSGarrett D'Amore case 'v':
65801335b0dSGarrett D'Amore doall = B_TRUE;
65901335b0dSGarrett D'Amore break;
66001335b0dSGarrett D'Amore
66101335b0dSGarrett D'Amore case 'j':
66201335b0dSGarrett D'Amore newarg = B_TRUE;
66301335b0dSGarrett D'Amore skip = strtoll(optarg, &eptr, 0);
66401335b0dSGarrett D'Amore if (*eptr == 'b') {
66501335b0dSGarrett D'Amore skip <<= 9; /* 512 bytes */
66601335b0dSGarrett D'Amore eptr++;
66701335b0dSGarrett D'Amore } else if (*eptr == 'k') {
66801335b0dSGarrett D'Amore skip <<= 10; /* 1k */
66901335b0dSGarrett D'Amore eptr++;
67001335b0dSGarrett D'Amore } else if (*eptr == 'm') {
67101335b0dSGarrett D'Amore skip <<= 20; /* 1m */
67201335b0dSGarrett D'Amore eptr++;
67301335b0dSGarrett D'Amore } else if (*eptr == 'g') {
67401335b0dSGarrett D'Amore skip <<= 30; /* 1g */
67501335b0dSGarrett D'Amore eptr++;
67601335b0dSGarrett D'Amore }
67701335b0dSGarrett D'Amore if ((skip < 0) || (eptr[0] != 0)) {
67801335b0dSGarrett D'Amore warnx(_("invalid skip count '%s' specified"),
67901335b0dSGarrett D'Amore optarg);
68001335b0dSGarrett D'Amore exit(1);
68101335b0dSGarrett D'Amore }
68201335b0dSGarrett D'Amore break;
68301335b0dSGarrett D'Amore
68401335b0dSGarrett D'Amore case 'N':
68501335b0dSGarrett D'Amore newarg = B_TRUE;
68601335b0dSGarrett D'Amore limit = strtoll(optarg, &eptr, 0);
68701335b0dSGarrett D'Amore /*
68801335b0dSGarrett D'Amore * POSIX doesn't specify this, but I think these
68901335b0dSGarrett D'Amore * may be helpful.
69001335b0dSGarrett D'Amore */
69101335b0dSGarrett D'Amore if (*eptr == 'b') {
69201335b0dSGarrett D'Amore limit <<= 9;
69301335b0dSGarrett D'Amore eptr++;
69401335b0dSGarrett D'Amore } else if (*eptr == 'k') {
69501335b0dSGarrett D'Amore limit <<= 10;
69601335b0dSGarrett D'Amore eptr++;
69701335b0dSGarrett D'Amore } else if (*eptr == 'm') {
69801335b0dSGarrett D'Amore limit <<= 20;
69901335b0dSGarrett D'Amore eptr++;
70001335b0dSGarrett D'Amore } else if (*eptr == 'g') {
70101335b0dSGarrett D'Amore limit <<= 30;
70201335b0dSGarrett D'Amore eptr++;
70301335b0dSGarrett D'Amore }
70401335b0dSGarrett D'Amore if ((limit < 0) || (eptr[0] != 0)) {
70501335b0dSGarrett D'Amore warnx(_("invalid byte count '%s' specified"),
70601335b0dSGarrett D'Amore optarg);
70701335b0dSGarrett D'Amore exit(1);
70801335b0dSGarrett D'Amore }
70901335b0dSGarrett D'Amore break;
71001335b0dSGarrett D'Amore
71101335b0dSGarrett D'Amore default:
71201335b0dSGarrett D'Amore usage();
71301335b0dSGarrett D'Amore break;
71401335b0dSGarrett D'Amore }
71501335b0dSGarrett D'Amore }
71601335b0dSGarrett D'Amore
71701335b0dSGarrett D'Amore /* this finds the smallest power of two size we can use */
71801335b0dSGarrett D'Amore buffer.mask = (1 << (ffs(blocksize * 3) + 1)) - 1;
71901335b0dSGarrett D'Amore buffer.data = memalign(16, buffer.mask + 1);
72001335b0dSGarrett D'Amore if (buffer.data == NULL) {
72101335b0dSGarrett D'Amore err(1, "memalign");
72201335b0dSGarrett D'Amore }
72301335b0dSGarrett D'Amore
72401335b0dSGarrett D'Amore
72501335b0dSGarrett D'Amore /*
72601335b0dSGarrett D'Amore * Wow. This option parsing is hideous.
72701335b0dSGarrett D'Amore *
72801335b0dSGarrett D'Amore * If the we've not seen a new option, and there is just one
72901335b0dSGarrett D'Amore * operand, if it starts with a "+", then treat it as an
73001335b0dSGarrett D'Amore * offset. Otherwise if two operands, and the second operand
73101335b0dSGarrett D'Amore * starts with + or a digit, then it is an offset.
73201335b0dSGarrett D'Amore */
73301335b0dSGarrett D'Amore if (!newarg) {
73401335b0dSGarrett D'Amore if (((argc - optind) == 1) && (argv[optind][0] == '+')) {
73501335b0dSGarrett D'Amore offstr = argv[optind];
73601335b0dSGarrett D'Amore argc--;
73701335b0dSGarrett D'Amore } else if (((argc - optind) == 2) &&
73801335b0dSGarrett D'Amore (strchr("+0123456789", (argv[optind + 1][0])) != NULL)) {
73901335b0dSGarrett D'Amore offstr = argv[optind + 1];
74001335b0dSGarrett D'Amore argc--;
74101335b0dSGarrett D'Amore }
74201335b0dSGarrett D'Amore }
74301335b0dSGarrett D'Amore if (offstr) {
74401335b0dSGarrett D'Amore int base = 0;
74501335b0dSGarrett D'Amore int mult = 1;
74601335b0dSGarrett D'Amore int l;
74701335b0dSGarrett D'Amore if (*offstr == '+') {
74801335b0dSGarrett D'Amore offstr++;
74901335b0dSGarrett D'Amore }
75001335b0dSGarrett D'Amore l = strlen(offstr);
75101335b0dSGarrett D'Amore if ((strncmp(offstr, "0x", 2) == 0)) {
75201335b0dSGarrett D'Amore afmt = "%07llx";
75301335b0dSGarrett D'Amore base = 16;
75401335b0dSGarrett D'Amore offstr += 2;
75501335b0dSGarrett D'Amore if (offstr[l - 1] == 'B') {
75601335b0dSGarrett D'Amore offstr[l - 1] = 0;
75701335b0dSGarrett D'Amore l--;
75801335b0dSGarrett D'Amore mult = 512;
75901335b0dSGarrett D'Amore }
76001335b0dSGarrett D'Amore } else {
76101335b0dSGarrett D'Amore base = 8;
76201335b0dSGarrett D'Amore afmt = "%07llo";
76301335b0dSGarrett D'Amore if ((offstr[l - 1] == 'B') || (offstr[l - 1] == 'b')) {
76401335b0dSGarrett D'Amore offstr[l - 1] = 0;
76501335b0dSGarrett D'Amore l--;
76601335b0dSGarrett D'Amore mult = 512;
76701335b0dSGarrett D'Amore }
76801335b0dSGarrett D'Amore if (offstr[l - 1] == '.') {
76901335b0dSGarrett D'Amore offstr[l - 1] = 0;
77001335b0dSGarrett D'Amore base = 10;
77101335b0dSGarrett D'Amore afmt = "%07lld";
77201335b0dSGarrett D'Amore }
77301335b0dSGarrett D'Amore }
77401335b0dSGarrett D'Amore skip = strtoll(offstr, &eptr, base);
77501335b0dSGarrett D'Amore if (*eptr != '\0') {
77601335b0dSGarrett D'Amore errx(1, _("invalid offset string specified"));
77701335b0dSGarrett D'Amore }
77801335b0dSGarrett D'Amore skip *= mult;
77901335b0dSGarrett D'Amore offset += skip;
78001335b0dSGarrett D'Amore }
78101335b0dSGarrett D'Amore
78201335b0dSGarrett D'Amore /*
78301335b0dSGarrett D'Amore * Allocate an array for all the input files.
78401335b0dSGarrett D'Amore */
78501335b0dSGarrett D'Amore if (argc > optind) {
78601335b0dSGarrett D'Amore files = calloc(sizeof (char *), argc - optind);
78701335b0dSGarrett D'Amore for (i = 0; i < argc - optind; i++) {
78801335b0dSGarrett D'Amore files[i] = argv[optind + i];
78901335b0dSGarrett D'Amore numfiles++;
79001335b0dSGarrett D'Amore }
79101335b0dSGarrett D'Amore input = next_input();
79201335b0dSGarrett D'Amore } else {
79301335b0dSGarrett D'Amore input = stdin;
79401335b0dSGarrett D'Amore }
79501335b0dSGarrett D'Amore
79601335b0dSGarrett D'Amore /*
79701335b0dSGarrett D'Amore * We need to seek ahead. fseek would be faster.
79801335b0dSGarrett D'Amore */
79901335b0dSGarrett D'Amore while (skip && (input != NULL)) {
80001335b0dSGarrett D'Amore struct stat sbuf;
80101335b0dSGarrett D'Amore
80201335b0dSGarrett D'Amore /*
80301335b0dSGarrett D'Amore * Only fseek() on regular files. (Others
80401335b0dSGarrett D'Amore * we have to read().
80501335b0dSGarrett D'Amore */
80601335b0dSGarrett D'Amore if (fstat(fileno(input), &sbuf) < 0) {
80701335b0dSGarrett D'Amore warn("fstat: %s", files[curfile-1]);
80801335b0dSGarrett D'Amore input = next_input();
80901335b0dSGarrett D'Amore continue;
81001335b0dSGarrett D'Amore }
81101335b0dSGarrett D'Amore if (S_ISREG(sbuf.st_mode)) {
81201335b0dSGarrett D'Amore /*
81301335b0dSGarrett D'Amore * No point in seeking a file that is too
81401335b0dSGarrett D'Amore * short to begin with.
81501335b0dSGarrett D'Amore */
81601335b0dSGarrett D'Amore if (sbuf.st_size < skip) {
81701335b0dSGarrett D'Amore skip -= sbuf.st_size;
81801335b0dSGarrett D'Amore input = next_input();
81901335b0dSGarrett D'Amore continue;
82001335b0dSGarrett D'Amore }
82101335b0dSGarrett D'Amore if (fseeko(input, skip, SEEK_SET) < 0) {
82201335b0dSGarrett D'Amore err(1, "fseek:%s", files[curfile-1]);
82301335b0dSGarrett D'Amore }
82401335b0dSGarrett D'Amore /* Done seeking. */
82501335b0dSGarrett D'Amore skip = 0;
82601335b0dSGarrett D'Amore break;
82701335b0dSGarrett D'Amore }
82801335b0dSGarrett D'Amore
82901335b0dSGarrett D'Amore /*
83001335b0dSGarrett D'Amore * fgetc seems like it would be slow, but it uses
83101335b0dSGarrett D'Amore * buffered I/O, so it should be fast enough.
83201335b0dSGarrett D'Amore */
83301335b0dSGarrett D'Amore flockfile(input);
83401335b0dSGarrett D'Amore while (skip) {
83501335b0dSGarrett D'Amore if (getc_unlocked(input) == EOF) {
83601335b0dSGarrett D'Amore funlockfile(input);
83701335b0dSGarrett D'Amore if (ferror(input)) {
83801335b0dSGarrett D'Amore warn("read: %s", files[curfile-1]);
83901335b0dSGarrett D'Amore }
84001335b0dSGarrett D'Amore input = next_input();
84101335b0dSGarrett D'Amore if (input != NULL) {
84201335b0dSGarrett D'Amore flockfile(input);
84301335b0dSGarrett D'Amore }
84401335b0dSGarrett D'Amore break;
84501335b0dSGarrett D'Amore }
84601335b0dSGarrett D'Amore skip--;
84701335b0dSGarrett D'Amore }
84801335b0dSGarrett D'Amore if (input != NULL)
84901335b0dSGarrett D'Amore funlockfile(input);
85001335b0dSGarrett D'Amore }
85101335b0dSGarrett D'Amore
85201335b0dSGarrett D'Amore if (head == NULL) {
85301335b0dSGarrett D'Amore add_out(&output_oct_w);
85401335b0dSGarrett D'Amore }
85501335b0dSGarrett D'Amore
85601335b0dSGarrett D'Amore buffer.navail = 0;
85701335b0dSGarrett D'Amore buffer.prod = 0;
85801335b0dSGarrett D'Amore buffer.cons = 0;
85901335b0dSGarrett D'Amore
86001335b0dSGarrett D'Amore for (refill(&buffer); buffer.navail > 0; refill(&buffer)) {
86101335b0dSGarrett D'Amore output_t *out;
86201335b0dSGarrett D'Amore int mx;
86301335b0dSGarrett D'Amore int j, k;
86401335b0dSGarrett D'Amore
86501335b0dSGarrett D'Amore /*
86601335b0dSGarrett D'Amore * If this buffer was the same as last, then just
86701335b0dSGarrett D'Amore * dump an asterisk.
86801335b0dSGarrett D'Amore */
86901335b0dSGarrett D'Amore if ((!first) && (buffer.navail >= blocksize) && (!doall)) {
87001335b0dSGarrett D'Amore j = buffer.cons;
87101335b0dSGarrett D'Amore k = j - blocksize;
87201335b0dSGarrett D'Amore for (i = 0; i < blocksize; i++) {
87301335b0dSGarrett D'Amore if (buffer.data[j & buffer.mask] !=
87401335b0dSGarrett D'Amore buffer.data[k & buffer.mask]) {
87501335b0dSGarrett D'Amore break;
87601335b0dSGarrett D'Amore }
87701335b0dSGarrett D'Amore j++;
87801335b0dSGarrett D'Amore k++;
87901335b0dSGarrett D'Amore }
88001335b0dSGarrett D'Amore if (i == blocksize) {
88101335b0dSGarrett D'Amore if (!same) {
88201335b0dSGarrett D'Amore (void) fputs("*\n", stdout);
88301335b0dSGarrett D'Amore same = B_TRUE;
88401335b0dSGarrett D'Amore }
88501335b0dSGarrett D'Amore buffer.navail -= blocksize;
88601335b0dSGarrett D'Amore offset += blocksize;
88701335b0dSGarrett D'Amore buffer.cons += blocksize;
88801335b0dSGarrett D'Amore buffer.cons &= buffer.mask;
88901335b0dSGarrett D'Amore continue;
89001335b0dSGarrett D'Amore }
89101335b0dSGarrett D'Amore }
89201335b0dSGarrett D'Amore
89301335b0dSGarrett D'Amore first = B_FALSE;
89401335b0dSGarrett D'Amore same = B_FALSE;
89501335b0dSGarrett D'Amore mx = (buffer.navail > blocksize) ? blocksize : buffer.navail;
89601335b0dSGarrett D'Amore
89701335b0dSGarrett D'Amore for (out = head; out != NULL; out = out->next) {
89801335b0dSGarrett D'Amore
89901335b0dSGarrett D'Amore if (out == head) {
90001335b0dSGarrett D'Amore /*LINTED E_SEC_PRINTF_VAR_FMT*/
90101335b0dSGarrett D'Amore (void) printf(afmt, offset);
90201335b0dSGarrett D'Amore } else {
90301335b0dSGarrett D'Amore (void) fputs(cfmt, stdout);
90401335b0dSGarrett D'Amore }
90501335b0dSGarrett D'Amore for (i = 0, j = buffer.cons; i < mx; i += out->width) {
90601335b0dSGarrett D'Amore out->func(&buffer, j);
90701335b0dSGarrett D'Amore j += out->width;
90801335b0dSGarrett D'Amore j &= buffer.mask;
90901335b0dSGarrett D'Amore }
91001335b0dSGarrett D'Amore (void) fputs("\n", stdout);
91101335b0dSGarrett D'Amore }
91201335b0dSGarrett D'Amore buffer.cons += mx;
91301335b0dSGarrett D'Amore buffer.cons &= buffer.mask;
91401335b0dSGarrett D'Amore offset += mx;
91501335b0dSGarrett D'Amore buffer.navail -= mx;
91601335b0dSGarrett D'Amore }
91701335b0dSGarrett D'Amore /*LINTED E_SEC_PRINTF_VAR_FMT*/
91801335b0dSGarrett D'Amore (void) printf(afmt, offset);
91901335b0dSGarrett D'Amore (void) fputs("\n", stdout);
92001335b0dSGarrett D'Amore return (0);
92101335b0dSGarrett D'Amore }
922