xref: /titanic_52/usr/src/lib/libofmt/common/ofmt.c (revision 071ca66ea5e45078699d3a2ade8c6ca5c00bba2c)
1*071ca66eSYuri Pankov /*
2*071ca66eSYuri Pankov  * CDDL HEADER START
3*071ca66eSYuri Pankov  *
4*071ca66eSYuri Pankov  * The contents of this file are subject to the terms of the
5*071ca66eSYuri Pankov  * Common Development and Distribution License (the "License").
6*071ca66eSYuri Pankov  * You may not use this file except in compliance with the License.
7*071ca66eSYuri Pankov  *
8*071ca66eSYuri Pankov  * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
9*071ca66eSYuri Pankov  * or http://www.opensolaris.org/os/licensing.
10*071ca66eSYuri Pankov  * See the License for the specific language governing permissions
11*071ca66eSYuri Pankov  * and limitations under the License.
12*071ca66eSYuri Pankov  *
13*071ca66eSYuri Pankov  * When distributing Covered Code, include this CDDL HEADER in each
14*071ca66eSYuri Pankov  * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
15*071ca66eSYuri Pankov  * If applicable, add the following below this CDDL HEADER, with the
16*071ca66eSYuri Pankov  * fields enclosed by brackets "[]" replaced with your own identifying
17*071ca66eSYuri Pankov  * information: Portions Copyright [yyyy] [name of copyright owner]
18*071ca66eSYuri Pankov  *
19*071ca66eSYuri Pankov  * CDDL HEADER END
20*071ca66eSYuri Pankov  */
21*071ca66eSYuri Pankov 
22*071ca66eSYuri Pankov /*
23*071ca66eSYuri Pankov  * Copyright 2010 Sun Microsystems, Inc.  All rights reserved.
24*071ca66eSYuri Pankov  * Use is subject to license terms.
25*071ca66eSYuri Pankov  */
26*071ca66eSYuri Pankov #include <errno.h>
27*071ca66eSYuri Pankov #include <sys/types.h>
28*071ca66eSYuri Pankov #include <stdlib.h>
29*071ca66eSYuri Pankov #include <string.h>
30*071ca66eSYuri Pankov #include <strings.h>
31*071ca66eSYuri Pankov #include <stdio.h>
32*071ca66eSYuri Pankov #include <ofmt.h>
33*071ca66eSYuri Pankov #include <sys/termios.h>
34*071ca66eSYuri Pankov #include <unistd.h>
35*071ca66eSYuri Pankov #include <sys/sysmacros.h>
36*071ca66eSYuri Pankov #include <libintl.h>
37*071ca66eSYuri Pankov 
38*071ca66eSYuri Pankov /*
39*071ca66eSYuri Pankov  * functions and structures to internally process a comma-separated string
40*071ca66eSYuri Pankov  * of fields selected for output.
41*071ca66eSYuri Pankov  */
42*071ca66eSYuri Pankov typedef struct {
43*071ca66eSYuri Pankov 	char	*s_buf;
44*071ca66eSYuri Pankov 	const char **s_fields;	/* array of pointers to the fields in s_buf */
45*071ca66eSYuri Pankov 	uint_t	s_nfields;	/* the number of fields in s_buf */
46*071ca66eSYuri Pankov 	uint_t	s_currfield;	/* the current field being processed */
47*071ca66eSYuri Pankov } split_t;
48*071ca66eSYuri Pankov static void splitfree(split_t *);
49*071ca66eSYuri Pankov static split_t *split_str(const char *, uint_t);
50*071ca66eSYuri Pankov static split_t *split_fields(const ofmt_field_t *, uint_t, uint_t);
51*071ca66eSYuri Pankov 
52*071ca66eSYuri Pankov /*
53*071ca66eSYuri Pankov  * The state of the output is tracked in a ofmt_state_t structure.
54*071ca66eSYuri Pankov  * Each os_fields[i] entry points at an ofmt_field_t array for
55*071ca66eSYuri Pankov  * the sub-command whose contents are provided by the caller, with
56*071ca66eSYuri Pankov  * os_nfields set to the number of requested fields.
57*071ca66eSYuri Pankov  */
58*071ca66eSYuri Pankov typedef struct ofmt_state_s {
59*071ca66eSYuri Pankov 	ofmt_field_t  	*os_fields;
60*071ca66eSYuri Pankov 	uint_t		os_nfields;
61*071ca66eSYuri Pankov 	boolean_t	os_lastfield;
62*071ca66eSYuri Pankov 	uint_t		os_overflow;
63*071ca66eSYuri Pankov 	struct winsize	os_winsize;
64*071ca66eSYuri Pankov 	int		os_nrow;
65*071ca66eSYuri Pankov 	uint_t		os_flags;
66*071ca66eSYuri Pankov 	int		os_nbad;
67*071ca66eSYuri Pankov 	char		**os_badfields;
68*071ca66eSYuri Pankov 	int		os_maxnamelen;	/* longest name (f. multiline) */
69*071ca66eSYuri Pankov } ofmt_state_t;
70*071ca66eSYuri Pankov /*
71*071ca66eSYuri Pankov  * A B_TRUE return value from the callback function will print out the contents
72*071ca66eSYuri Pankov  * of the output buffer, except when the buffer is returned with the empty
73*071ca66eSYuri Pankov  * string "", in which case the  OFMT_VAL_UNDEF will be printed.
74*071ca66eSYuri Pankov  *
75*071ca66eSYuri Pankov  * If the callback function returns B_FALSE, the "?" string will be emitted.
76*071ca66eSYuri Pankov  */
77*071ca66eSYuri Pankov #define	OFMT_VAL_UNDEF		"--"
78*071ca66eSYuri Pankov #define	OFMT_VAL_UNKNOWN	"?"
79*071ca66eSYuri Pankov 
80*071ca66eSYuri Pankov /*
81*071ca66eSYuri Pankov  * The maximum number of rows supported by the OFMT_WRAP option.
82*071ca66eSYuri Pankov  */
83*071ca66eSYuri Pankov #define	OFMT_MAX_ROWS		128
84*071ca66eSYuri Pankov 
85*071ca66eSYuri Pankov static void ofmt_print_header(ofmt_state_t *);
86*071ca66eSYuri Pankov static void ofmt_print_field(ofmt_state_t *, ofmt_field_t *, const char *,
87*071ca66eSYuri Pankov     boolean_t);
88*071ca66eSYuri Pankov 
89*071ca66eSYuri Pankov /*
90*071ca66eSYuri Pankov  * Split `str' into at most `maxfields' fields, Return a pointer to a
91*071ca66eSYuri Pankov  * split_t containing the split fields, or NULL on failure.
92*071ca66eSYuri Pankov  */
93*071ca66eSYuri Pankov static split_t *
94*071ca66eSYuri Pankov split_str(const char *str, uint_t maxfields)
95*071ca66eSYuri Pankov {
96*071ca66eSYuri Pankov 	char	*field, *token, *lasts = NULL;
97*071ca66eSYuri Pankov 	split_t	*sp;
98*071ca66eSYuri Pankov 
99*071ca66eSYuri Pankov 	if (*str == '\0' || maxfields == 0)
100*071ca66eSYuri Pankov 		return (NULL);
101*071ca66eSYuri Pankov 
102*071ca66eSYuri Pankov 	sp = calloc(sizeof (split_t), 1);
103*071ca66eSYuri Pankov 	if (sp == NULL)
104*071ca66eSYuri Pankov 		return (NULL);
105*071ca66eSYuri Pankov 
106*071ca66eSYuri Pankov 	sp->s_buf = strdup(str);
107*071ca66eSYuri Pankov 	sp->s_fields = malloc(sizeof (char *) * maxfields);
108*071ca66eSYuri Pankov 	if (sp->s_buf == NULL || sp->s_fields == NULL)
109*071ca66eSYuri Pankov 		goto fail;
110*071ca66eSYuri Pankov 
111*071ca66eSYuri Pankov 	token = sp->s_buf;
112*071ca66eSYuri Pankov 	while ((field = strtok_r(token, ",", &lasts)) != NULL) {
113*071ca66eSYuri Pankov 		if (sp->s_nfields == maxfields)
114*071ca66eSYuri Pankov 			goto fail;
115*071ca66eSYuri Pankov 		token = NULL;
116*071ca66eSYuri Pankov 		sp->s_fields[sp->s_nfields++] = field;
117*071ca66eSYuri Pankov 	}
118*071ca66eSYuri Pankov 	return (sp);
119*071ca66eSYuri Pankov fail:
120*071ca66eSYuri Pankov 	splitfree(sp);
121*071ca66eSYuri Pankov 	return (NULL);
122*071ca66eSYuri Pankov }
123*071ca66eSYuri Pankov 
124*071ca66eSYuri Pankov /*
125*071ca66eSYuri Pankov  * Split `fields' into at most `maxfields' fields. Return a pointer to
126*071ca66eSYuri Pankov  * a split_t containing the split fields, or NULL on failure. Invoked
127*071ca66eSYuri Pankov  * when all fields are implicitly selected at handle creation by
128*071ca66eSYuri Pankov  * passing in a NULL fields_str
129*071ca66eSYuri Pankov  */
130*071ca66eSYuri Pankov static split_t *
131*071ca66eSYuri Pankov split_fields(const ofmt_field_t *template, uint_t maxfields, uint_t maxcols)
132*071ca66eSYuri Pankov {
133*071ca66eSYuri Pankov 	split_t	*sp;
134*071ca66eSYuri Pankov 	int i, cols;
135*071ca66eSYuri Pankov 
136*071ca66eSYuri Pankov 	sp = calloc(sizeof (split_t), 1);
137*071ca66eSYuri Pankov 	if (sp == NULL)
138*071ca66eSYuri Pankov 		return (NULL);
139*071ca66eSYuri Pankov 
140*071ca66eSYuri Pankov 	sp->s_fields = malloc(sizeof (char *) * maxfields);
141*071ca66eSYuri Pankov 	if (sp->s_fields == NULL)
142*071ca66eSYuri Pankov 		goto fail;
143*071ca66eSYuri Pankov 	cols = 0;
144*071ca66eSYuri Pankov 	for (i = 0; i < maxfields; i++) {
145*071ca66eSYuri Pankov 		cols += template[i].of_width;
146*071ca66eSYuri Pankov 		/*
147*071ca66eSYuri Pankov 		 * If all fields are implied without explicitly passing
148*071ca66eSYuri Pankov 		 * in a fields_str, build a list of field names, stopping
149*071ca66eSYuri Pankov 		 * when we run out of columns.
150*071ca66eSYuri Pankov 		 */
151*071ca66eSYuri Pankov 		if (maxcols > 0 && cols > maxcols)
152*071ca66eSYuri Pankov 			break;
153*071ca66eSYuri Pankov 		sp->s_fields[sp->s_nfields++] = template[i].of_name;
154*071ca66eSYuri Pankov 	}
155*071ca66eSYuri Pankov 	return (sp);
156*071ca66eSYuri Pankov fail:
157*071ca66eSYuri Pankov 	splitfree(sp);
158*071ca66eSYuri Pankov 	return (NULL);
159*071ca66eSYuri Pankov }
160*071ca66eSYuri Pankov 
161*071ca66eSYuri Pankov /*
162*071ca66eSYuri Pankov  * Free the split_t structure pointed to by `sp'.
163*071ca66eSYuri Pankov  */
164*071ca66eSYuri Pankov static void
165*071ca66eSYuri Pankov splitfree(split_t *sp)
166*071ca66eSYuri Pankov {
167*071ca66eSYuri Pankov 	if (sp == NULL)
168*071ca66eSYuri Pankov 		return;
169*071ca66eSYuri Pankov 	free(sp->s_buf);
170*071ca66eSYuri Pankov 	free(sp->s_fields);
171*071ca66eSYuri Pankov 	free(sp);
172*071ca66eSYuri Pankov }
173*071ca66eSYuri Pankov 
174*071ca66eSYuri Pankov /*
175*071ca66eSYuri Pankov  * Open a handle to be used for printing formatted output.
176*071ca66eSYuri Pankov  */
177*071ca66eSYuri Pankov ofmt_status_t
178*071ca66eSYuri Pankov ofmt_open(const char *str, const ofmt_field_t *template, uint_t flags,
179*071ca66eSYuri Pankov     uint_t maxcols, ofmt_handle_t *ofmt)
180*071ca66eSYuri Pankov {
181*071ca66eSYuri Pankov 	split_t		*sp;
182*071ca66eSYuri Pankov 	uint_t		i, j, of_index;
183*071ca66eSYuri Pankov 	const ofmt_field_t *ofp;
184*071ca66eSYuri Pankov 	ofmt_field_t	*of;
185*071ca66eSYuri Pankov 	ofmt_state_t	*os = NULL;
186*071ca66eSYuri Pankov 	int		nfields = 0;
187*071ca66eSYuri Pankov 	ofmt_status_t	error = OFMT_SUCCESS;
188*071ca66eSYuri Pankov 	boolean_t	parsable = (flags & OFMT_PARSABLE);
189*071ca66eSYuri Pankov 	boolean_t	wrap = (flags & OFMT_WRAP);
190*071ca66eSYuri Pankov 	boolean_t	multiline = (flags & OFMT_MULTILINE);
191*071ca66eSYuri Pankov 
192*071ca66eSYuri Pankov 	*ofmt = NULL;
193*071ca66eSYuri Pankov 	if (parsable) {
194*071ca66eSYuri Pankov 		if (multiline)
195*071ca66eSYuri Pankov 			return (OFMT_EPARSEMULTI);
196*071ca66eSYuri Pankov 		/*
197*071ca66eSYuri Pankov 		 * For parsable output mode, the caller always needs
198*071ca66eSYuri Pankov 		 * to specify precisely which fields are to be selected,
199*071ca66eSYuri Pankov 		 * since the set of fields may change over time.
200*071ca66eSYuri Pankov 		 */
201*071ca66eSYuri Pankov 		if (str == NULL || str[0] == '\0')
202*071ca66eSYuri Pankov 			return (OFMT_EPARSENONE);
203*071ca66eSYuri Pankov 		if (strcasecmp(str, "all") == 0)
204*071ca66eSYuri Pankov 			return (OFMT_EPARSEALL);
205*071ca66eSYuri Pankov 		if (wrap)
206*071ca66eSYuri Pankov 			return (OFMT_EPARSEWRAP);
207*071ca66eSYuri Pankov 	}
208*071ca66eSYuri Pankov 	if (template == NULL)
209*071ca66eSYuri Pankov 		return (OFMT_ENOTEMPLATE);
210*071ca66eSYuri Pankov 	for (ofp = template; ofp->of_name != NULL; ofp++)
211*071ca66eSYuri Pankov 		nfields++;
212*071ca66eSYuri Pankov 	/*
213*071ca66eSYuri Pankov 	 * split str into the columns selected, or construct the
214*071ca66eSYuri Pankov 	 * full set of columns (equivalent to -o all).
215*071ca66eSYuri Pankov 	 */
216*071ca66eSYuri Pankov 	if (str != NULL && strcasecmp(str, "all") != 0) {
217*071ca66eSYuri Pankov 		sp = split_str(str, nfields);
218*071ca66eSYuri Pankov 	} else {
219*071ca66eSYuri Pankov 		if (parsable || (str != NULL && strcasecmp(str, "all") == 0))
220*071ca66eSYuri Pankov 			maxcols = 0;
221*071ca66eSYuri Pankov 		sp = split_fields(template, nfields, maxcols);
222*071ca66eSYuri Pankov 	}
223*071ca66eSYuri Pankov 	if (sp == NULL)
224*071ca66eSYuri Pankov 		goto nomem;
225*071ca66eSYuri Pankov 
226*071ca66eSYuri Pankov 	os = calloc(sizeof (ofmt_state_t) +
227*071ca66eSYuri Pankov 	    sp->s_nfields * sizeof (ofmt_field_t), 1);
228*071ca66eSYuri Pankov 	if (os == NULL)
229*071ca66eSYuri Pankov 		goto nomem;
230*071ca66eSYuri Pankov 	*ofmt = os;
231*071ca66eSYuri Pankov 	os->os_fields = (ofmt_field_t *)&os[1];
232*071ca66eSYuri Pankov 	os->os_flags = flags;
233*071ca66eSYuri Pankov 
234*071ca66eSYuri Pankov 	of = os->os_fields;
235*071ca66eSYuri Pankov 	of_index = 0;
236*071ca66eSYuri Pankov 	/*
237*071ca66eSYuri Pankov 	 * sp->s_nfields is the number of fields requested in fields_str.
238*071ca66eSYuri Pankov 	 * nfields is the number of fields in template.
239*071ca66eSYuri Pankov 	 */
240*071ca66eSYuri Pankov 	for (i = 0; i < sp->s_nfields; i++) {
241*071ca66eSYuri Pankov 		for (j = 0; j < nfields; j++) {
242*071ca66eSYuri Pankov 			if (strcasecmp(sp->s_fields[i],
243*071ca66eSYuri Pankov 			    template[j].of_name) == 0) {
244*071ca66eSYuri Pankov 				break;
245*071ca66eSYuri Pankov 			}
246*071ca66eSYuri Pankov 		}
247*071ca66eSYuri Pankov 		if (j == nfields) {
248*071ca66eSYuri Pankov 			int nbad = os->os_nbad++;
249*071ca66eSYuri Pankov 
250*071ca66eSYuri Pankov 			error = OFMT_EBADFIELDS;
251*071ca66eSYuri Pankov 			if (os->os_badfields == NULL) {
252*071ca66eSYuri Pankov 				os->os_badfields = malloc(sp->s_nfields *
253*071ca66eSYuri Pankov 				    sizeof (char *));
254*071ca66eSYuri Pankov 				if (os->os_badfields == NULL)
255*071ca66eSYuri Pankov 					goto nomem;
256*071ca66eSYuri Pankov 			}
257*071ca66eSYuri Pankov 			os->os_badfields[nbad] = strdup(sp->s_fields[i]);
258*071ca66eSYuri Pankov 			if (os->os_badfields[nbad] == NULL)
259*071ca66eSYuri Pankov 				goto nomem;
260*071ca66eSYuri Pankov 			continue;
261*071ca66eSYuri Pankov 		}
262*071ca66eSYuri Pankov 		of[of_index].of_name = strdup(template[j].of_name);
263*071ca66eSYuri Pankov 		if (of[of_index].of_name == NULL)
264*071ca66eSYuri Pankov 			goto nomem;
265*071ca66eSYuri Pankov 		if (multiline) {
266*071ca66eSYuri Pankov 			int n = strlen(of[of_index].of_name);
267*071ca66eSYuri Pankov 
268*071ca66eSYuri Pankov 			os->os_maxnamelen = MAX(n, os->os_maxnamelen);
269*071ca66eSYuri Pankov 		}
270*071ca66eSYuri Pankov 		of[of_index].of_width = template[j].of_width;
271*071ca66eSYuri Pankov 		of[of_index].of_id = template[j].of_id;
272*071ca66eSYuri Pankov 		of[of_index].of_cb = template[j].of_cb;
273*071ca66eSYuri Pankov 		of_index++;
274*071ca66eSYuri Pankov 	}
275*071ca66eSYuri Pankov 	splitfree(sp);
276*071ca66eSYuri Pankov 	if (of_index == 0) /* all values in str are bogus */
277*071ca66eSYuri Pankov 		return (OFMT_ENOFIELDS);
278*071ca66eSYuri Pankov 	os->os_nfields = of_index; /* actual number of fields printed */
279*071ca66eSYuri Pankov 	ofmt_update_winsize(*ofmt);
280*071ca66eSYuri Pankov 	return (error);
281*071ca66eSYuri Pankov nomem:
282*071ca66eSYuri Pankov 	error = OFMT_ENOMEM;
283*071ca66eSYuri Pankov 	if (os != NULL)
284*071ca66eSYuri Pankov 		ofmt_close(os);
285*071ca66eSYuri Pankov 	*ofmt = NULL;
286*071ca66eSYuri Pankov 	splitfree(sp);
287*071ca66eSYuri Pankov 	return (error);
288*071ca66eSYuri Pankov }
289*071ca66eSYuri Pankov 
290*071ca66eSYuri Pankov /*
291*071ca66eSYuri Pankov  * free resources associated with the ofmt_handle_t
292*071ca66eSYuri Pankov  */
293*071ca66eSYuri Pankov void
294*071ca66eSYuri Pankov ofmt_close(ofmt_handle_t ofmt)
295*071ca66eSYuri Pankov {
296*071ca66eSYuri Pankov 	ofmt_state_t *os = ofmt;
297*071ca66eSYuri Pankov 	int i;
298*071ca66eSYuri Pankov 
299*071ca66eSYuri Pankov 	if (os == NULL)
300*071ca66eSYuri Pankov 		return;
301*071ca66eSYuri Pankov 	for (i = 0; i < os->os_nfields; i++)
302*071ca66eSYuri Pankov 		free(os->os_fields[i].of_name);
303*071ca66eSYuri Pankov 	for (i = 0; i < os->os_nbad; i++)
304*071ca66eSYuri Pankov 		free(os->os_badfields[i]);
305*071ca66eSYuri Pankov 	free(os->os_badfields);
306*071ca66eSYuri Pankov 	free(os);
307*071ca66eSYuri Pankov }
308*071ca66eSYuri Pankov 
309*071ca66eSYuri Pankov /*
310*071ca66eSYuri Pankov  * Print the value for the selected field by calling the callback-function
311*071ca66eSYuri Pankov  * registered for the field.
312*071ca66eSYuri Pankov  */
313*071ca66eSYuri Pankov static void
314*071ca66eSYuri Pankov ofmt_print_field(ofmt_state_t *os, ofmt_field_t *ofp, const char *value,
315*071ca66eSYuri Pankov     boolean_t escsep)
316*071ca66eSYuri Pankov {
317*071ca66eSYuri Pankov 	uint_t	width = ofp->of_width;
318*071ca66eSYuri Pankov 	uint_t	valwidth;
319*071ca66eSYuri Pankov 	uint_t	compress;
320*071ca66eSYuri Pankov 	boolean_t parsable = (os->os_flags & OFMT_PARSABLE);
321*071ca66eSYuri Pankov 	boolean_t multiline = (os->os_flags & OFMT_MULTILINE);
322*071ca66eSYuri Pankov 	boolean_t rightjust = (os->os_flags & OFMT_RIGHTJUST);
323*071ca66eSYuri Pankov 	char	c;
324*071ca66eSYuri Pankov 
325*071ca66eSYuri Pankov 	/*
326*071ca66eSYuri Pankov 	 * Parsable fields are separated by ':'. If such a field contains
327*071ca66eSYuri Pankov 	 * a ':' or '\', this character is prefixed by a '\'.
328*071ca66eSYuri Pankov 	 */
329*071ca66eSYuri Pankov 	if (parsable) {
330*071ca66eSYuri Pankov 		if (os->os_nfields == 1) {
331*071ca66eSYuri Pankov 			(void) printf("%s", value);
332*071ca66eSYuri Pankov 			return;
333*071ca66eSYuri Pankov 		}
334*071ca66eSYuri Pankov 		while ((c = *value++) != '\0') {
335*071ca66eSYuri Pankov 			if (escsep && ((c == ':' || c == '\\')))
336*071ca66eSYuri Pankov 				(void) putchar('\\');
337*071ca66eSYuri Pankov 			(void) putchar(c);
338*071ca66eSYuri Pankov 		}
339*071ca66eSYuri Pankov 		if (!os->os_lastfield)
340*071ca66eSYuri Pankov 			(void) putchar(':');
341*071ca66eSYuri Pankov 	} else if (multiline) {
342*071ca66eSYuri Pankov 		if (value[0] == '\0')
343*071ca66eSYuri Pankov 			value = OFMT_VAL_UNDEF;
344*071ca66eSYuri Pankov 		(void) printf("%*.*s: %s", os->os_maxnamelen,
345*071ca66eSYuri Pankov 		    os->os_maxnamelen, ofp->of_name, value);
346*071ca66eSYuri Pankov 		if (!os->os_lastfield)
347*071ca66eSYuri Pankov 			(void) putchar('\n');
348*071ca66eSYuri Pankov 	} else {
349*071ca66eSYuri Pankov 		if (os->os_lastfield) {
350*071ca66eSYuri Pankov 			if (rightjust)
351*071ca66eSYuri Pankov 				(void) printf("%*s", width, value);
352*071ca66eSYuri Pankov 			else
353*071ca66eSYuri Pankov 				(void) printf("%s", value);
354*071ca66eSYuri Pankov 			os->os_overflow = 0;
355*071ca66eSYuri Pankov 			return;
356*071ca66eSYuri Pankov 		}
357*071ca66eSYuri Pankov 
358*071ca66eSYuri Pankov 		valwidth = strlen(value);
359*071ca66eSYuri Pankov 		if (valwidth + os->os_overflow >= width) {
360*071ca66eSYuri Pankov 			os->os_overflow += valwidth - width + 1;
361*071ca66eSYuri Pankov 			if (rightjust)
362*071ca66eSYuri Pankov 				(void) printf("%*s ", width, value);
363*071ca66eSYuri Pankov 			else
364*071ca66eSYuri Pankov 				(void) printf("%s ", value);
365*071ca66eSYuri Pankov 			return;
366*071ca66eSYuri Pankov 		}
367*071ca66eSYuri Pankov 
368*071ca66eSYuri Pankov 		if (os->os_overflow > 0) {
369*071ca66eSYuri Pankov 			compress = MIN(os->os_overflow, width - valwidth);
370*071ca66eSYuri Pankov 			os->os_overflow -= compress;
371*071ca66eSYuri Pankov 			width -= compress;
372*071ca66eSYuri Pankov 		}
373*071ca66eSYuri Pankov 		if (rightjust)
374*071ca66eSYuri Pankov 			(void) printf("%*s ", width, value);
375*071ca66eSYuri Pankov 		else
376*071ca66eSYuri Pankov 			(void) printf("%-*s", width, value);
377*071ca66eSYuri Pankov 	}
378*071ca66eSYuri Pankov }
379*071ca66eSYuri Pankov 
380*071ca66eSYuri Pankov /*
381*071ca66eSYuri Pankov  * Print enough to fit the field width.
382*071ca66eSYuri Pankov  */
383*071ca66eSYuri Pankov static void
384*071ca66eSYuri Pankov ofmt_fit_width(split_t **spp, uint_t width, char *value, uint_t bufsize)
385*071ca66eSYuri Pankov {
386*071ca66eSYuri Pankov 	split_t		*sp = *spp;
387*071ca66eSYuri Pankov 	char		*ptr = value, *lim = ptr + bufsize;
388*071ca66eSYuri Pankov 	int		i, nextlen;
389*071ca66eSYuri Pankov 
390*071ca66eSYuri Pankov 	if (sp == NULL) {
391*071ca66eSYuri Pankov 		sp = split_str(value, OFMT_MAX_ROWS);
392*071ca66eSYuri Pankov 		if (sp == NULL)
393*071ca66eSYuri Pankov 			return;
394*071ca66eSYuri Pankov 
395*071ca66eSYuri Pankov 		*spp = sp;
396*071ca66eSYuri Pankov 	}
397*071ca66eSYuri Pankov 	for (i = sp->s_currfield; i < sp->s_nfields; i++) {
398*071ca66eSYuri Pankov 		ptr += snprintf(ptr, lim - ptr, "%s,", sp->s_fields[i]);
399*071ca66eSYuri Pankov 		if (i + 1 == sp->s_nfields) {
400*071ca66eSYuri Pankov 			nextlen = 0;
401*071ca66eSYuri Pankov 			if (ptr > value)
402*071ca66eSYuri Pankov 				ptr[-1] = '\0';
403*071ca66eSYuri Pankov 		} else {
404*071ca66eSYuri Pankov 			nextlen = strlen(sp->s_fields[i + 1]);
405*071ca66eSYuri Pankov 		}
406*071ca66eSYuri Pankov 
407*071ca66eSYuri Pankov 		if (strlen(value) + nextlen > width || ptr >= lim) {
408*071ca66eSYuri Pankov 			i++;
409*071ca66eSYuri Pankov 			break;
410*071ca66eSYuri Pankov 		}
411*071ca66eSYuri Pankov 	}
412*071ca66eSYuri Pankov 	sp->s_currfield = i;
413*071ca66eSYuri Pankov }
414*071ca66eSYuri Pankov 
415*071ca66eSYuri Pankov /*
416*071ca66eSYuri Pankov  * Print one or more rows of output values for the selected columns.
417*071ca66eSYuri Pankov  */
418*071ca66eSYuri Pankov void
419*071ca66eSYuri Pankov ofmt_print(ofmt_handle_t ofmt, void *arg)
420*071ca66eSYuri Pankov {
421*071ca66eSYuri Pankov 	ofmt_state_t *os = ofmt;
422*071ca66eSYuri Pankov 	int i;
423*071ca66eSYuri Pankov 	char value[1024];
424*071ca66eSYuri Pankov 	ofmt_field_t *of;
425*071ca66eSYuri Pankov 	boolean_t escsep, more_rows;
426*071ca66eSYuri Pankov 	ofmt_arg_t ofarg;
427*071ca66eSYuri Pankov 	split_t **sp = NULL;
428*071ca66eSYuri Pankov 	boolean_t parsable = (os->os_flags & OFMT_PARSABLE);
429*071ca66eSYuri Pankov 	boolean_t multiline = (os->os_flags & OFMT_MULTILINE);
430*071ca66eSYuri Pankov 	boolean_t wrap = (os->os_flags & OFMT_WRAP);
431*071ca66eSYuri Pankov 
432*071ca66eSYuri Pankov 	if (wrap) {
433*071ca66eSYuri Pankov 		sp = calloc(sizeof (split_t *), os->os_nfields);
434*071ca66eSYuri Pankov 		if (sp == NULL)
435*071ca66eSYuri Pankov 			return;
436*071ca66eSYuri Pankov 	}
437*071ca66eSYuri Pankov 
438*071ca66eSYuri Pankov 	if ((os->os_nrow++ % os->os_winsize.ws_row) == 0 &&
439*071ca66eSYuri Pankov 	    !parsable && !multiline) {
440*071ca66eSYuri Pankov 		ofmt_print_header(os);
441*071ca66eSYuri Pankov 		os->os_nrow++;
442*071ca66eSYuri Pankov 	}
443*071ca66eSYuri Pankov 
444*071ca66eSYuri Pankov 	if (multiline && os->os_nrow > 1)
445*071ca66eSYuri Pankov 		(void) putchar('\n');
446*071ca66eSYuri Pankov 
447*071ca66eSYuri Pankov 	of = os->os_fields;
448*071ca66eSYuri Pankov 	escsep = (os->os_nfields > 1);
449*071ca66eSYuri Pankov 	more_rows = B_FALSE;
450*071ca66eSYuri Pankov 	for (i = 0; i < os->os_nfields; i++) {
451*071ca66eSYuri Pankov 		os->os_lastfield = (i + 1 == os->os_nfields);
452*071ca66eSYuri Pankov 		value[0] = '\0';
453*071ca66eSYuri Pankov 		ofarg.ofmt_id = of[i].of_id;
454*071ca66eSYuri Pankov 		ofarg.ofmt_cbarg = arg;
455*071ca66eSYuri Pankov 
456*071ca66eSYuri Pankov 		if ((*of[i].of_cb)(&ofarg, value, sizeof (value))) {
457*071ca66eSYuri Pankov 			if (wrap) {
458*071ca66eSYuri Pankov 				/*
459*071ca66eSYuri Pankov 				 * 'value' will be split at comma boundaries
460*071ca66eSYuri Pankov 				 * and stored into sp[i].
461*071ca66eSYuri Pankov 				 */
462*071ca66eSYuri Pankov 				ofmt_fit_width(&sp[i], of[i].of_width, value,
463*071ca66eSYuri Pankov 				    sizeof (value));
464*071ca66eSYuri Pankov 				if (sp[i] != NULL &&
465*071ca66eSYuri Pankov 				    sp[i]->s_currfield < sp[i]->s_nfields)
466*071ca66eSYuri Pankov 					more_rows = B_TRUE;
467*071ca66eSYuri Pankov 			}
468*071ca66eSYuri Pankov 
469*071ca66eSYuri Pankov 			ofmt_print_field(os, &of[i],
470*071ca66eSYuri Pankov 			    (*value == '\0' && !parsable) ?
471*071ca66eSYuri Pankov 			    OFMT_VAL_UNDEF : value, escsep);
472*071ca66eSYuri Pankov 		} else {
473*071ca66eSYuri Pankov 			ofmt_print_field(os, &of[i], OFMT_VAL_UNKNOWN, escsep);
474*071ca66eSYuri Pankov 		}
475*071ca66eSYuri Pankov 	}
476*071ca66eSYuri Pankov 	(void) putchar('\n');
477*071ca66eSYuri Pankov 
478*071ca66eSYuri Pankov 	while (more_rows) {
479*071ca66eSYuri Pankov 		more_rows = B_FALSE;
480*071ca66eSYuri Pankov 		for (i = 0; i < os->os_nfields; i++) {
481*071ca66eSYuri Pankov 			os->os_lastfield = (i + 1 == os->os_nfields);
482*071ca66eSYuri Pankov 			value[0] = '\0';
483*071ca66eSYuri Pankov 
484*071ca66eSYuri Pankov 			ofmt_fit_width(&sp[i], of[i].of_width,
485*071ca66eSYuri Pankov 			    value, sizeof (value));
486*071ca66eSYuri Pankov 			if (sp[i] != NULL &&
487*071ca66eSYuri Pankov 			    sp[i]->s_currfield < sp[i]->s_nfields)
488*071ca66eSYuri Pankov 				more_rows = B_TRUE;
489*071ca66eSYuri Pankov 
490*071ca66eSYuri Pankov 			ofmt_print_field(os, &of[i], value, escsep);
491*071ca66eSYuri Pankov 		}
492*071ca66eSYuri Pankov 		(void) putchar('\n');
493*071ca66eSYuri Pankov 	}
494*071ca66eSYuri Pankov 	(void) fflush(stdout);
495*071ca66eSYuri Pankov 
496*071ca66eSYuri Pankov 	if (sp != NULL) {
497*071ca66eSYuri Pankov 		for (i = 0; i < os->os_nfields; i++)
498*071ca66eSYuri Pankov 			splitfree(sp[i]);
499*071ca66eSYuri Pankov 		free(sp);
500*071ca66eSYuri Pankov 	}
501*071ca66eSYuri Pankov }
502*071ca66eSYuri Pankov 
503*071ca66eSYuri Pankov /*
504*071ca66eSYuri Pankov  * Print the field headers
505*071ca66eSYuri Pankov  */
506*071ca66eSYuri Pankov static void
507*071ca66eSYuri Pankov ofmt_print_header(ofmt_state_t *os)
508*071ca66eSYuri Pankov {
509*071ca66eSYuri Pankov 	int i;
510*071ca66eSYuri Pankov 	ofmt_field_t *of = os->os_fields;
511*071ca66eSYuri Pankov 	boolean_t escsep = (os->os_nfields > 1);
512*071ca66eSYuri Pankov 
513*071ca66eSYuri Pankov 	for (i = 0; i < os->os_nfields; i++) {
514*071ca66eSYuri Pankov 		os->os_lastfield = (i + 1 == os->os_nfields);
515*071ca66eSYuri Pankov 		ofmt_print_field(os, &of[i], of[i].of_name, escsep);
516*071ca66eSYuri Pankov 	}
517*071ca66eSYuri Pankov 	(void) putchar('\n');
518*071ca66eSYuri Pankov }
519*071ca66eSYuri Pankov 
520*071ca66eSYuri Pankov /*
521*071ca66eSYuri Pankov  * Update the current window size.
522*071ca66eSYuri Pankov  */
523*071ca66eSYuri Pankov void
524*071ca66eSYuri Pankov ofmt_update_winsize(ofmt_handle_t ofmt)
525*071ca66eSYuri Pankov {
526*071ca66eSYuri Pankov 	ofmt_state_t *os = ofmt;
527*071ca66eSYuri Pankov 	struct winsize *winsize = &os->os_winsize;
528*071ca66eSYuri Pankov 
529*071ca66eSYuri Pankov 	if (ioctl(1, TIOCGWINSZ, winsize) == -1 ||
530*071ca66eSYuri Pankov 	    winsize->ws_col == 0 || winsize->ws_row == 0) {
531*071ca66eSYuri Pankov 		winsize->ws_col = 80;
532*071ca66eSYuri Pankov 		winsize->ws_row = 24;
533*071ca66eSYuri Pankov 	}
534*071ca66eSYuri Pankov }
535*071ca66eSYuri Pankov 
536*071ca66eSYuri Pankov /*
537*071ca66eSYuri Pankov  * Return error diagnostics using the information in the ofmt_handle_t
538*071ca66eSYuri Pankov  */
539*071ca66eSYuri Pankov char *
540*071ca66eSYuri Pankov ofmt_strerror(ofmt_handle_t ofmt, ofmt_status_t error, char *buf,
541*071ca66eSYuri Pankov     uint_t bufsize)
542*071ca66eSYuri Pankov {
543*071ca66eSYuri Pankov 	ofmt_state_t *os = ofmt;
544*071ca66eSYuri Pankov 	int i;
545*071ca66eSYuri Pankov 	const char *s;
546*071ca66eSYuri Pankov 	char ebuf[OFMT_BUFSIZE];
547*071ca66eSYuri Pankov 	boolean_t parsable;
548*071ca66eSYuri Pankov 
549*071ca66eSYuri Pankov 	/*
550*071ca66eSYuri Pankov 	 * ebuf is intended for optional error-specific data to be appended
551*071ca66eSYuri Pankov 	 * after the internationalized error string for an error code.
552*071ca66eSYuri Pankov 	 */
553*071ca66eSYuri Pankov 	ebuf[0] = '\0';
554*071ca66eSYuri Pankov 
555*071ca66eSYuri Pankov 	switch (error) {
556*071ca66eSYuri Pankov 	case OFMT_SUCCESS:
557*071ca66eSYuri Pankov 		s = "success";
558*071ca66eSYuri Pankov 		break;
559*071ca66eSYuri Pankov 	case OFMT_EBADFIELDS:
560*071ca66eSYuri Pankov 		/*
561*071ca66eSYuri Pankov 		 * Enumerate the singular/plural version of the warning
562*071ca66eSYuri Pankov 		 * and error to simplify and improve localization.
563*071ca66eSYuri Pankov 		 */
564*071ca66eSYuri Pankov 		parsable = (os->os_flags & OFMT_PARSABLE);
565*071ca66eSYuri Pankov 		if (!parsable) {
566*071ca66eSYuri Pankov 			if (os->os_nbad > 1)
567*071ca66eSYuri Pankov 				s = "ignoring unknown output fields:";
568*071ca66eSYuri Pankov 			else
569*071ca66eSYuri Pankov 				s = "ignoring unknown output field:";
570*071ca66eSYuri Pankov 		} else {
571*071ca66eSYuri Pankov 			if (os->os_nbad > 1)
572*071ca66eSYuri Pankov 				s = "unknown output fields:";
573*071ca66eSYuri Pankov 			else
574*071ca66eSYuri Pankov 				s = "unknown output field:";
575*071ca66eSYuri Pankov 		}
576*071ca66eSYuri Pankov 		/* set up the bad fields in ebuf */
577*071ca66eSYuri Pankov 		for (i = 0; i < os->os_nbad; i++) {
578*071ca66eSYuri Pankov 			(void) strlcat(ebuf, " `", sizeof (ebuf));
579*071ca66eSYuri Pankov 			(void) strlcat(ebuf, os->os_badfields[i],
580*071ca66eSYuri Pankov 			    sizeof (ebuf));
581*071ca66eSYuri Pankov 			(void) strlcat(ebuf, "'", sizeof (ebuf));
582*071ca66eSYuri Pankov 		}
583*071ca66eSYuri Pankov 		break;
584*071ca66eSYuri Pankov 	case OFMT_ENOFIELDS:
585*071ca66eSYuri Pankov 		s = "no valid output fields";
586*071ca66eSYuri Pankov 		break;
587*071ca66eSYuri Pankov 	case OFMT_EPARSEMULTI:
588*071ca66eSYuri Pankov 		s = "multiline mode incompatible with parsable mode";
589*071ca66eSYuri Pankov 		break;
590*071ca66eSYuri Pankov 	case OFMT_EPARSEALL:
591*071ca66eSYuri Pankov 		s = "output field `all' invalid in parsable mode";
592*071ca66eSYuri Pankov 		break;
593*071ca66eSYuri Pankov 	case OFMT_EPARSENONE:
594*071ca66eSYuri Pankov 		s = "output fields must be specified in parsable mode";
595*071ca66eSYuri Pankov 		break;
596*071ca66eSYuri Pankov 	case OFMT_EPARSEWRAP:
597*071ca66eSYuri Pankov 		s = "parsable mode is incompatible with wrap mode";
598*071ca66eSYuri Pankov 		break;
599*071ca66eSYuri Pankov 	case OFMT_ENOTEMPLATE:
600*071ca66eSYuri Pankov 		s = "no template provided for fields";
601*071ca66eSYuri Pankov 		break;
602*071ca66eSYuri Pankov 	case OFMT_ENOMEM:
603*071ca66eSYuri Pankov 		s = strerror(ENOMEM);
604*071ca66eSYuri Pankov 		break;
605*071ca66eSYuri Pankov 	default:
606*071ca66eSYuri Pankov 		(void) snprintf(buf, bufsize,
607*071ca66eSYuri Pankov 		    dgettext(TEXT_DOMAIN, "unknown ofmt error (%d)"),
608*071ca66eSYuri Pankov 		    error);
609*071ca66eSYuri Pankov 		return (buf);
610*071ca66eSYuri Pankov 	}
611*071ca66eSYuri Pankov 	(void) snprintf(buf, bufsize, dgettext(TEXT_DOMAIN, s));
612*071ca66eSYuri Pankov 	(void) strlcat(buf, ebuf, bufsize);
613*071ca66eSYuri Pankov 	return (buf);
614*071ca66eSYuri Pankov }
615