xref: /titanic_51/usr/src/cmd/od/od.c (revision 84441f85b19f6b8080883f30109e58e43c893709)
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
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
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
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
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 *
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
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
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
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