18002d411SSowmini Varadhan /* 28002d411SSowmini Varadhan * CDDL HEADER START 38002d411SSowmini Varadhan * 48002d411SSowmini Varadhan * The contents of this file are subject to the terms of the 58002d411SSowmini Varadhan * Common Development and Distribution License (the "License"). 68002d411SSowmini Varadhan * You may not use this file except in compliance with the License. 78002d411SSowmini Varadhan * 88002d411SSowmini Varadhan * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE 98002d411SSowmini Varadhan * or http://www.opensolaris.org/os/licensing. 108002d411SSowmini Varadhan * See the License for the specific language governing permissions 118002d411SSowmini Varadhan * and limitations under the License. 128002d411SSowmini Varadhan * 138002d411SSowmini Varadhan * When distributing Covered Code, include this CDDL HEADER in each 148002d411SSowmini Varadhan * file and include the License file at usr/src/OPENSOLARIS.LICENSE. 158002d411SSowmini Varadhan * If applicable, add the following below this CDDL HEADER, with the 168002d411SSowmini Varadhan * fields enclosed by brackets "[]" replaced with your own identifying 178002d411SSowmini Varadhan * information: Portions Copyright [yyyy] [name of copyright owner] 188002d411SSowmini Varadhan * 198002d411SSowmini Varadhan * CDDL HEADER END 208002d411SSowmini Varadhan */ 218002d411SSowmini Varadhan 228002d411SSowmini Varadhan /* 23*0dc2366fSVenugopal Iyer * Copyright 2010 Sun Microsystems, Inc. All rights reserved. 248002d411SSowmini Varadhan * Use is subject to license terms. 258002d411SSowmini Varadhan */ 268002d411SSowmini Varadhan #include <errno.h> 278002d411SSowmini Varadhan #include <sys/types.h> 288002d411SSowmini Varadhan #include <stdlib.h> 298002d411SSowmini Varadhan #include <string.h> 308002d411SSowmini Varadhan #include <strings.h> 318002d411SSowmini Varadhan #include <stdio.h> 328002d411SSowmini Varadhan #include <ofmt.h> 338002d411SSowmini Varadhan #include <sys/termios.h> 348002d411SSowmini Varadhan #include <unistd.h> 358002d411SSowmini Varadhan #include <sys/sysmacros.h> 368002d411SSowmini Varadhan #include <libintl.h> 378002d411SSowmini Varadhan 388002d411SSowmini Varadhan /* 398002d411SSowmini Varadhan * functions and structures to internally process a comma-separated string 408002d411SSowmini Varadhan * of fields selected for output. 418002d411SSowmini Varadhan */ 428002d411SSowmini Varadhan typedef struct { 438002d411SSowmini Varadhan char *s_buf; 448002d411SSowmini Varadhan const char **s_fields; /* array of pointers to the fields in s_buf */ 458002d411SSowmini Varadhan uint_t s_nfields; /* the number of fields in s_buf */ 4625ec3e3dSEric Cheng uint_t s_currfield; /* the current field being processed */ 478002d411SSowmini Varadhan } split_t; 488002d411SSowmini Varadhan static void splitfree(split_t *); 498002d411SSowmini Varadhan static split_t *split_str(const char *, uint_t); 508002d411SSowmini Varadhan static split_t *split_fields(const ofmt_field_t *, uint_t, uint_t); 518002d411SSowmini Varadhan 528002d411SSowmini Varadhan /* 538002d411SSowmini Varadhan * The state of the output is tracked in a ofmt_state_t structure. 548002d411SSowmini Varadhan * Each os_fields[i] entry points at an ofmt_field_t array for 558002d411SSowmini Varadhan * the sub-command whose contents are provided by the caller, with 568002d411SSowmini Varadhan * os_nfields set to the number of requested fields. 578002d411SSowmini Varadhan */ 588002d411SSowmini Varadhan typedef struct ofmt_state_s { 598002d411SSowmini Varadhan ofmt_field_t *os_fields; 608002d411SSowmini Varadhan uint_t os_nfields; 618002d411SSowmini Varadhan boolean_t os_lastfield; 628002d411SSowmini Varadhan uint_t os_overflow; 638002d411SSowmini Varadhan struct winsize os_winsize; 648002d411SSowmini Varadhan int os_nrow; 65*0dc2366fSVenugopal Iyer uint_t os_flags; 668002d411SSowmini Varadhan int os_nbad; 678002d411SSowmini Varadhan char **os_badfields; 68dbed73cbSSangeeta Misra int os_maxnamelen; /* longest name (f. multiline) */ 698002d411SSowmini Varadhan } ofmt_state_t; 708002d411SSowmini Varadhan /* 718002d411SSowmini Varadhan * A B_TRUE return value from the callback function will print out the contents 728002d411SSowmini Varadhan * of the output buffer, except when the buffer is returned with the empty 738002d411SSowmini Varadhan * string "", in which case the OFMT_VAL_UNDEF will be printed. 748002d411SSowmini Varadhan * 758002d411SSowmini Varadhan * If the callback function returns B_FALSE, the "?" string will be emitted. 768002d411SSowmini Varadhan */ 778002d411SSowmini Varadhan #define OFMT_VAL_UNDEF "--" 788002d411SSowmini Varadhan #define OFMT_VAL_UNKNOWN "?" 7925ec3e3dSEric Cheng 8025ec3e3dSEric Cheng /* 8125ec3e3dSEric Cheng * The maximum number of rows supported by the OFMT_WRAP option. 8225ec3e3dSEric Cheng */ 8325ec3e3dSEric Cheng #define OFMT_MAX_ROWS 128 8425ec3e3dSEric Cheng 858002d411SSowmini Varadhan static void ofmt_print_header(ofmt_state_t *); 868002d411SSowmini Varadhan static void ofmt_print_field(ofmt_state_t *, ofmt_field_t *, const char *, 878002d411SSowmini Varadhan boolean_t); 888002d411SSowmini Varadhan 898002d411SSowmini Varadhan /* 908002d411SSowmini Varadhan * Split `str' into at most `maxfields' fields, Return a pointer to a 918002d411SSowmini Varadhan * split_t containing the split fields, or NULL on failure. 928002d411SSowmini Varadhan */ 938002d411SSowmini Varadhan static split_t * 948002d411SSowmini Varadhan split_str(const char *str, uint_t maxfields) 958002d411SSowmini Varadhan { 968002d411SSowmini Varadhan char *field, *token, *lasts = NULL; 978002d411SSowmini Varadhan split_t *sp; 988002d411SSowmini Varadhan 998002d411SSowmini Varadhan if (*str == '\0' || maxfields == 0) 1008002d411SSowmini Varadhan return (NULL); 1018002d411SSowmini Varadhan 1028002d411SSowmini Varadhan sp = calloc(sizeof (split_t), 1); 1038002d411SSowmini Varadhan if (sp == NULL) 1048002d411SSowmini Varadhan return (NULL); 1058002d411SSowmini Varadhan 1068002d411SSowmini Varadhan sp->s_buf = strdup(str); 1078002d411SSowmini Varadhan sp->s_fields = malloc(sizeof (char *) * maxfields); 1088002d411SSowmini Varadhan if (sp->s_buf == NULL || sp->s_fields == NULL) 1098002d411SSowmini Varadhan goto fail; 1108002d411SSowmini Varadhan 1118002d411SSowmini Varadhan token = sp->s_buf; 1128002d411SSowmini Varadhan while ((field = strtok_r(token, ",", &lasts)) != NULL) { 1138002d411SSowmini Varadhan if (sp->s_nfields == maxfields) 1148002d411SSowmini Varadhan goto fail; 1158002d411SSowmini Varadhan token = NULL; 1168002d411SSowmini Varadhan sp->s_fields[sp->s_nfields++] = field; 1178002d411SSowmini Varadhan } 1188002d411SSowmini Varadhan return (sp); 1198002d411SSowmini Varadhan fail: 1208002d411SSowmini Varadhan splitfree(sp); 1218002d411SSowmini Varadhan return (NULL); 1228002d411SSowmini Varadhan } 1238002d411SSowmini Varadhan 1248002d411SSowmini Varadhan /* 1258002d411SSowmini Varadhan * Split `fields' into at most `maxfields' fields. Return a pointer to 1268002d411SSowmini Varadhan * a split_t containing the split fields, or NULL on failure. Invoked 1278002d411SSowmini Varadhan * when all fields are implicitly selected at handle creation by 1288002d411SSowmini Varadhan * passing in a NULL fields_str 1298002d411SSowmini Varadhan */ 1308002d411SSowmini Varadhan static split_t * 1318002d411SSowmini Varadhan split_fields(const ofmt_field_t *template, uint_t maxfields, uint_t maxcols) 1328002d411SSowmini Varadhan { 1338002d411SSowmini Varadhan split_t *sp; 1348002d411SSowmini Varadhan int i, cols; 1358002d411SSowmini Varadhan 1368002d411SSowmini Varadhan sp = calloc(sizeof (split_t), 1); 1378002d411SSowmini Varadhan if (sp == NULL) 1388002d411SSowmini Varadhan return (NULL); 1398002d411SSowmini Varadhan 1408002d411SSowmini Varadhan sp->s_fields = malloc(sizeof (char *) * maxfields); 1418002d411SSowmini Varadhan if (sp->s_fields == NULL) 1428002d411SSowmini Varadhan goto fail; 1438002d411SSowmini Varadhan cols = 0; 1448002d411SSowmini Varadhan for (i = 0; i < maxfields; i++) { 1458002d411SSowmini Varadhan cols += template[i].of_width; 1468002d411SSowmini Varadhan /* 1478002d411SSowmini Varadhan * If all fields are implied without explicitly passing 1488002d411SSowmini Varadhan * in a fields_str, build a list of field names, stopping 1498002d411SSowmini Varadhan * when we run out of columns. 1508002d411SSowmini Varadhan */ 1518002d411SSowmini Varadhan if (maxcols > 0 && cols > maxcols) 1528002d411SSowmini Varadhan break; 1538002d411SSowmini Varadhan sp->s_fields[sp->s_nfields++] = template[i].of_name; 1548002d411SSowmini Varadhan } 1558002d411SSowmini Varadhan return (sp); 1568002d411SSowmini Varadhan fail: 1578002d411SSowmini Varadhan splitfree(sp); 1588002d411SSowmini Varadhan return (NULL); 1598002d411SSowmini Varadhan } 1608002d411SSowmini Varadhan 1618002d411SSowmini Varadhan /* 1628002d411SSowmini Varadhan * Free the split_t structure pointed to by `sp'. 1638002d411SSowmini Varadhan */ 1648002d411SSowmini Varadhan static void 1658002d411SSowmini Varadhan splitfree(split_t *sp) 1668002d411SSowmini Varadhan { 1678002d411SSowmini Varadhan if (sp == NULL) 1688002d411SSowmini Varadhan return; 1698002d411SSowmini Varadhan free(sp->s_buf); 1708002d411SSowmini Varadhan free(sp->s_fields); 1718002d411SSowmini Varadhan free(sp); 1728002d411SSowmini Varadhan } 1738002d411SSowmini Varadhan 1748002d411SSowmini Varadhan /* 1758002d411SSowmini Varadhan * Open a handle to be used for printing formatted output. 1768002d411SSowmini Varadhan */ 1778002d411SSowmini Varadhan ofmt_status_t 1782b24ab6bSSebastien Roy ofmt_open(const char *str, const ofmt_field_t *template, uint_t flags, 1798002d411SSowmini Varadhan uint_t maxcols, ofmt_handle_t *ofmt) 1808002d411SSowmini Varadhan { 1818002d411SSowmini Varadhan split_t *sp; 1828002d411SSowmini Varadhan uint_t i, j, of_index; 1832b24ab6bSSebastien Roy const ofmt_field_t *ofp; 1848002d411SSowmini Varadhan ofmt_field_t *of; 1858002d411SSowmini Varadhan ofmt_state_t *os; 1868002d411SSowmini Varadhan int nfields = 0; 1878002d411SSowmini Varadhan ofmt_status_t err = OFMT_SUCCESS; 188*0dc2366fSVenugopal Iyer boolean_t parsable = (flags & OFMT_PARSABLE); 189*0dc2366fSVenugopal Iyer boolean_t wrap = (flags & OFMT_WRAP); 190dbed73cbSSangeeta Misra boolean_t multiline = (flags & OFMT_MULTILINE); 1918002d411SSowmini Varadhan 1928002d411SSowmini Varadhan *ofmt = NULL; 1938002d411SSowmini Varadhan if (parsable) { 194dbed73cbSSangeeta Misra if (multiline) 195dbed73cbSSangeeta Misra return (OFMT_EPARSEMULTI); 1968002d411SSowmini Varadhan /* 1978002d411SSowmini Varadhan * For parsable output mode, the caller always needs 1988002d411SSowmini Varadhan * to specify precisely which fields are to be selected, 1998002d411SSowmini Varadhan * since the set of fields may change over time. 2008002d411SSowmini Varadhan */ 2018002d411SSowmini Varadhan if (str == NULL || str[0] == '\0') 2028002d411SSowmini Varadhan return (OFMT_EPARSENONE); 2038002d411SSowmini Varadhan if (strcmp(str, "all") == 0) 2048002d411SSowmini Varadhan return (OFMT_EPARSEALL); 20525ec3e3dSEric Cheng if (wrap) 20625ec3e3dSEric Cheng return (OFMT_EPARSEWRAP); 2078002d411SSowmini Varadhan } 2088002d411SSowmini Varadhan if (template == NULL) 2098002d411SSowmini Varadhan return (OFMT_ENOTEMPLATE); 2102b24ab6bSSebastien Roy for (ofp = template; ofp->of_name != NULL; ofp++) 2118002d411SSowmini Varadhan nfields++; 2128002d411SSowmini Varadhan /* 2138002d411SSowmini Varadhan * split str into the columns selected, or construct the 2148002d411SSowmini Varadhan * full set of columns (equivalent to -o all). 2158002d411SSowmini Varadhan */ 2168002d411SSowmini Varadhan if (str != NULL && strcmp(str, "all") != 0) { 2178002d411SSowmini Varadhan sp = split_str(str, nfields); 2188002d411SSowmini Varadhan } else { 2198002d411SSowmini Varadhan if (parsable || (str != NULL && strcmp(str, "all") == 0)) 2208002d411SSowmini Varadhan maxcols = 0; 2218002d411SSowmini Varadhan sp = split_fields(template, nfields, maxcols); 2228002d411SSowmini Varadhan } 2238002d411SSowmini Varadhan if (sp == NULL) 2248002d411SSowmini Varadhan goto nomem; 2258002d411SSowmini Varadhan 2268002d411SSowmini Varadhan os = calloc(sizeof (ofmt_state_t) + 2278002d411SSowmini Varadhan sp->s_nfields * sizeof (ofmt_field_t), 1); 2288002d411SSowmini Varadhan if (os == NULL) 2298002d411SSowmini Varadhan goto nomem; 2308002d411SSowmini Varadhan *ofmt = os; 2318002d411SSowmini Varadhan os->os_fields = (ofmt_field_t *)&os[1]; 232*0dc2366fSVenugopal Iyer os->os_flags = flags; 23325ec3e3dSEric Cheng 2348002d411SSowmini Varadhan of = os->os_fields; 2358002d411SSowmini Varadhan of_index = 0; 2368002d411SSowmini Varadhan /* 2378002d411SSowmini Varadhan * sp->s_nfields is the number of fields requested in fields_str. 2388002d411SSowmini Varadhan * nfields is the number of fields in template. 2398002d411SSowmini Varadhan */ 2408002d411SSowmini Varadhan for (i = 0; i < sp->s_nfields; i++) { 2418002d411SSowmini Varadhan for (j = 0; j < nfields; j++) { 2428002d411SSowmini Varadhan if (strcasecmp(sp->s_fields[i], 2438002d411SSowmini Varadhan template[j].of_name) == 0) { 2448002d411SSowmini Varadhan break; 2458002d411SSowmini Varadhan } 2468002d411SSowmini Varadhan } 2478002d411SSowmini Varadhan if (j == nfields) { 2488002d411SSowmini Varadhan int nbad = os->os_nbad++; 2498002d411SSowmini Varadhan 2508002d411SSowmini Varadhan err = OFMT_EBADFIELDS; 2518002d411SSowmini Varadhan if (os->os_badfields == NULL) { 2528002d411SSowmini Varadhan os->os_badfields = malloc(sp->s_nfields * 2538002d411SSowmini Varadhan sizeof (char *)); 2548002d411SSowmini Varadhan if (os->os_badfields == NULL) 2558002d411SSowmini Varadhan goto nomem; 2568002d411SSowmini Varadhan } 2578002d411SSowmini Varadhan os->os_badfields[nbad] = strdup(sp->s_fields[i]); 2588002d411SSowmini Varadhan if (os->os_badfields[nbad] == NULL) 2598002d411SSowmini Varadhan goto nomem; 2608002d411SSowmini Varadhan continue; 2618002d411SSowmini Varadhan } 2628002d411SSowmini Varadhan of[of_index].of_name = strdup(template[j].of_name); 2638002d411SSowmini Varadhan if (of[of_index].of_name == NULL) 2648002d411SSowmini Varadhan goto nomem; 265dbed73cbSSangeeta Misra if (multiline) { 266dbed73cbSSangeeta Misra int n = strlen(of[of_index].of_name); 267dbed73cbSSangeeta Misra 268dbed73cbSSangeeta Misra os->os_maxnamelen = MAX(n, os->os_maxnamelen); 269dbed73cbSSangeeta Misra } 2708002d411SSowmini Varadhan of[of_index].of_width = template[j].of_width; 2718002d411SSowmini Varadhan of[of_index].of_id = template[j].of_id; 2728002d411SSowmini Varadhan of[of_index].of_cb = template[j].of_cb; 2738002d411SSowmini Varadhan of_index++; 2748002d411SSowmini Varadhan } 2758002d411SSowmini Varadhan splitfree(sp); 2768002d411SSowmini Varadhan if (of_index == 0) /* all values in str are bogus */ 2778002d411SSowmini Varadhan return (OFMT_ENOFIELDS); 2788002d411SSowmini Varadhan os->os_nfields = of_index; /* actual number of fields printed */ 2798002d411SSowmini Varadhan ofmt_update_winsize(*ofmt); 2808002d411SSowmini Varadhan return (err); 2818002d411SSowmini Varadhan nomem: 2828002d411SSowmini Varadhan err = OFMT_ENOMEM; 2838002d411SSowmini Varadhan if (os != NULL) 2848002d411SSowmini Varadhan ofmt_close(os); 2858002d411SSowmini Varadhan *ofmt = NULL; 2868002d411SSowmini Varadhan splitfree(sp); 2878002d411SSowmini Varadhan return (err); 2888002d411SSowmini Varadhan } 2898002d411SSowmini Varadhan 2908002d411SSowmini Varadhan /* 2918002d411SSowmini Varadhan * free resources associated with the ofmt_handle_t 2928002d411SSowmini Varadhan */ 2938002d411SSowmini Varadhan void 2948002d411SSowmini Varadhan ofmt_close(ofmt_handle_t ofmt) 2958002d411SSowmini Varadhan { 2968002d411SSowmini Varadhan ofmt_state_t *os = ofmt; 2978002d411SSowmini Varadhan int i; 2988002d411SSowmini Varadhan 2998002d411SSowmini Varadhan if (os == NULL) 3008002d411SSowmini Varadhan return; 3018002d411SSowmini Varadhan for (i = 0; i < os->os_nfields; i++) 3028002d411SSowmini Varadhan free(os->os_fields[i].of_name); 3038002d411SSowmini Varadhan for (i = 0; i < os->os_nbad; i++) 3048002d411SSowmini Varadhan free(os->os_badfields[i]); 3058002d411SSowmini Varadhan free(os->os_badfields); 3068002d411SSowmini Varadhan free(os); 3078002d411SSowmini Varadhan } 3088002d411SSowmini Varadhan 3098002d411SSowmini Varadhan /* 3108002d411SSowmini Varadhan * Print the value for the selected field by calling the callback-function 3118002d411SSowmini Varadhan * registered for the field. 3128002d411SSowmini Varadhan */ 3138002d411SSowmini Varadhan static void 3148002d411SSowmini Varadhan ofmt_print_field(ofmt_state_t *os, ofmt_field_t *ofp, const char *value, 3158002d411SSowmini Varadhan boolean_t escsep) 3168002d411SSowmini Varadhan { 3178002d411SSowmini Varadhan uint_t width = ofp->of_width; 3188002d411SSowmini Varadhan uint_t valwidth; 3198002d411SSowmini Varadhan uint_t compress; 320*0dc2366fSVenugopal Iyer boolean_t parsable = (os->os_flags & OFMT_PARSABLE); 321*0dc2366fSVenugopal Iyer boolean_t multiline = (os->os_flags & OFMT_MULTILINE); 322*0dc2366fSVenugopal Iyer boolean_t rightjust = (os->os_flags & OFMT_RIGHTJUST); 3238002d411SSowmini Varadhan char c; 3248002d411SSowmini Varadhan 3258002d411SSowmini Varadhan /* 3268002d411SSowmini Varadhan * Parsable fields are separated by ':'. If such a field contains 3278002d411SSowmini Varadhan * a ':' or '\', this character is prefixed by a '\'. 3288002d411SSowmini Varadhan */ 329*0dc2366fSVenugopal Iyer if (parsable) { 3308002d411SSowmini Varadhan if (os->os_nfields == 1) { 3318002d411SSowmini Varadhan (void) printf("%s", value); 3328002d411SSowmini Varadhan return; 3338002d411SSowmini Varadhan } 3348002d411SSowmini Varadhan while ((c = *value++) != '\0') { 3358002d411SSowmini Varadhan if (escsep && ((c == ':' || c == '\\'))) 3368002d411SSowmini Varadhan (void) putchar('\\'); 3378002d411SSowmini Varadhan (void) putchar(c); 3388002d411SSowmini Varadhan } 3398002d411SSowmini Varadhan if (!os->os_lastfield) 3408002d411SSowmini Varadhan (void) putchar(':'); 341*0dc2366fSVenugopal Iyer } else if (multiline) { 342dbed73cbSSangeeta Misra if (value[0] == '\0') 343dbed73cbSSangeeta Misra value = OFMT_VAL_UNDEF; 344dbed73cbSSangeeta Misra (void) printf("%*.*s: %s", os->os_maxnamelen, 345dbed73cbSSangeeta Misra os->os_maxnamelen, ofp->of_name, value); 346dbed73cbSSangeeta Misra if (!os->os_lastfield) 347dbed73cbSSangeeta Misra (void) putchar('\n'); 3488002d411SSowmini Varadhan } else { 3498002d411SSowmini Varadhan if (os->os_lastfield) { 350*0dc2366fSVenugopal Iyer if (rightjust) 351*0dc2366fSVenugopal Iyer (void) printf("%*s", width, value); 352*0dc2366fSVenugopal Iyer else 3538002d411SSowmini Varadhan (void) printf("%s", value); 3548002d411SSowmini Varadhan os->os_overflow = 0; 3558002d411SSowmini Varadhan return; 3568002d411SSowmini Varadhan } 3578002d411SSowmini Varadhan 3588002d411SSowmini Varadhan valwidth = strlen(value); 3598002d411SSowmini Varadhan if (valwidth + os->os_overflow >= width) { 3608002d411SSowmini Varadhan os->os_overflow += valwidth - width + 1; 361*0dc2366fSVenugopal Iyer if (rightjust) 362*0dc2366fSVenugopal Iyer (void) printf("%*s ", width, value); 363*0dc2366fSVenugopal Iyer else 3648002d411SSowmini Varadhan (void) printf("%s ", value); 3658002d411SSowmini Varadhan return; 3668002d411SSowmini Varadhan } 3678002d411SSowmini Varadhan 3688002d411SSowmini Varadhan if (os->os_overflow > 0) { 3698002d411SSowmini Varadhan compress = MIN(os->os_overflow, width - valwidth); 3708002d411SSowmini Varadhan os->os_overflow -= compress; 3718002d411SSowmini Varadhan width -= compress; 3728002d411SSowmini Varadhan } 373*0dc2366fSVenugopal Iyer if (rightjust) 374*0dc2366fSVenugopal Iyer (void) printf("%*s ", width, value); 375*0dc2366fSVenugopal Iyer else 3768002d411SSowmini Varadhan (void) printf("%-*s", width, value); 3778002d411SSowmini Varadhan } 3788002d411SSowmini Varadhan } 3798002d411SSowmini Varadhan 3808002d411SSowmini Varadhan /* 38125ec3e3dSEric Cheng * Print enough to fit the field width. 38225ec3e3dSEric Cheng */ 38325ec3e3dSEric Cheng static void 38425ec3e3dSEric Cheng ofmt_fit_width(split_t **spp, uint_t width, char *value, uint_t bufsize) 38525ec3e3dSEric Cheng { 38625ec3e3dSEric Cheng split_t *sp = *spp; 38725ec3e3dSEric Cheng char *ptr = value, *lim = ptr + bufsize; 38825ec3e3dSEric Cheng int i, nextlen; 38925ec3e3dSEric Cheng 39025ec3e3dSEric Cheng if (sp == NULL) { 39125ec3e3dSEric Cheng sp = split_str(value, OFMT_MAX_ROWS); 39225ec3e3dSEric Cheng if (sp == NULL) 39325ec3e3dSEric Cheng return; 39425ec3e3dSEric Cheng 39525ec3e3dSEric Cheng *spp = sp; 39625ec3e3dSEric Cheng } 39725ec3e3dSEric Cheng for (i = sp->s_currfield; i < sp->s_nfields; i++) { 39825ec3e3dSEric Cheng ptr += snprintf(ptr, lim - ptr, "%s,", sp->s_fields[i]); 39925ec3e3dSEric Cheng if (i + 1 == sp->s_nfields) { 40025ec3e3dSEric Cheng nextlen = 0; 40125ec3e3dSEric Cheng if (ptr > value) 40225ec3e3dSEric Cheng ptr[-1] = '\0'; 40325ec3e3dSEric Cheng } else { 40425ec3e3dSEric Cheng nextlen = strlen(sp->s_fields[i + 1]); 40525ec3e3dSEric Cheng } 40625ec3e3dSEric Cheng 40725ec3e3dSEric Cheng if (strlen(value) + nextlen > width || ptr >= lim) { 40825ec3e3dSEric Cheng i++; 40925ec3e3dSEric Cheng break; 41025ec3e3dSEric Cheng } 41125ec3e3dSEric Cheng } 41225ec3e3dSEric Cheng sp->s_currfield = i; 41325ec3e3dSEric Cheng } 41425ec3e3dSEric Cheng 41525ec3e3dSEric Cheng /* 41625ec3e3dSEric Cheng * Print one or more rows of output values for the selected columns. 4178002d411SSowmini Varadhan */ 4188002d411SSowmini Varadhan void 4198002d411SSowmini Varadhan ofmt_print(ofmt_handle_t ofmt, void *arg) 4208002d411SSowmini Varadhan { 4218002d411SSowmini Varadhan ofmt_state_t *os = ofmt; 4228002d411SSowmini Varadhan int i; 4238002d411SSowmini Varadhan char value[1024]; 4248002d411SSowmini Varadhan ofmt_field_t *of; 42525ec3e3dSEric Cheng boolean_t escsep, more_rows; 4268002d411SSowmini Varadhan ofmt_arg_t ofarg; 42725ec3e3dSEric Cheng split_t **sp = NULL; 428*0dc2366fSVenugopal Iyer boolean_t parsable = (os->os_flags & OFMT_PARSABLE); 429*0dc2366fSVenugopal Iyer boolean_t multiline = (os->os_flags & OFMT_MULTILINE); 430*0dc2366fSVenugopal Iyer boolean_t wrap = (os->os_flags & OFMT_WRAP); 43125ec3e3dSEric Cheng 432*0dc2366fSVenugopal Iyer if (wrap) { 43325ec3e3dSEric Cheng sp = calloc(sizeof (split_t *), os->os_nfields); 43425ec3e3dSEric Cheng if (sp == NULL) 43525ec3e3dSEric Cheng return; 43625ec3e3dSEric Cheng } 4378002d411SSowmini Varadhan 438*0dc2366fSVenugopal Iyer if ((os->os_nrow++ % os->os_winsize.ws_row) == 0 && 439*0dc2366fSVenugopal Iyer !parsable && !multiline) { 4408002d411SSowmini Varadhan ofmt_print_header(os); 4418002d411SSowmini Varadhan os->os_nrow++; 4428002d411SSowmini Varadhan } 4438002d411SSowmini Varadhan 444*0dc2366fSVenugopal Iyer if (multiline && os->os_nrow > 1) 445dbed73cbSSangeeta Misra (void) putchar('\n'); 446dbed73cbSSangeeta Misra 4478002d411SSowmini Varadhan of = os->os_fields; 4488002d411SSowmini Varadhan escsep = (os->os_nfields > 1); 44925ec3e3dSEric Cheng more_rows = B_FALSE; 4508002d411SSowmini Varadhan for (i = 0; i < os->os_nfields; i++) { 4518002d411SSowmini Varadhan os->os_lastfield = (i + 1 == os->os_nfields); 4528002d411SSowmini Varadhan value[0] = '\0'; 4538002d411SSowmini Varadhan ofarg.ofmt_id = of[i].of_id; 4548002d411SSowmini Varadhan ofarg.ofmt_cbarg = arg; 45525ec3e3dSEric Cheng 45625ec3e3dSEric Cheng if ((*of[i].of_cb)(&ofarg, value, sizeof (value))) { 457*0dc2366fSVenugopal Iyer if (wrap) { 45825ec3e3dSEric Cheng /* 45925ec3e3dSEric Cheng * 'value' will be split at comma boundaries 46025ec3e3dSEric Cheng * and stored into sp[i]. 46125ec3e3dSEric Cheng */ 46225ec3e3dSEric Cheng ofmt_fit_width(&sp[i], of[i].of_width, value, 46325ec3e3dSEric Cheng sizeof (value)); 46425ec3e3dSEric Cheng if (sp[i] != NULL && 46525ec3e3dSEric Cheng sp[i]->s_currfield < sp[i]->s_nfields) 46625ec3e3dSEric Cheng more_rows = B_TRUE; 46725ec3e3dSEric Cheng } 468*0dc2366fSVenugopal Iyer 46925ec3e3dSEric Cheng ofmt_print_field(os, &of[i], 470*0dc2366fSVenugopal Iyer (*value == '\0' && !parsable) ? 47125ec3e3dSEric Cheng OFMT_VAL_UNDEF : value, escsep); 47225ec3e3dSEric Cheng } else { 4738002d411SSowmini Varadhan ofmt_print_field(os, &of[i], OFMT_VAL_UNKNOWN, escsep); 4748002d411SSowmini Varadhan } 47525ec3e3dSEric Cheng } 4768002d411SSowmini Varadhan (void) putchar('\n'); 47725ec3e3dSEric Cheng 47825ec3e3dSEric Cheng while (more_rows) { 47925ec3e3dSEric Cheng more_rows = B_FALSE; 48025ec3e3dSEric Cheng for (i = 0; i < os->os_nfields; i++) { 48125ec3e3dSEric Cheng os->os_lastfield = (i + 1 == os->os_nfields); 48225ec3e3dSEric Cheng value[0] = '\0'; 48325ec3e3dSEric Cheng 48425ec3e3dSEric Cheng ofmt_fit_width(&sp[i], of[i].of_width, 48525ec3e3dSEric Cheng value, sizeof (value)); 48625ec3e3dSEric Cheng if (sp[i] != NULL && 48725ec3e3dSEric Cheng sp[i]->s_currfield < sp[i]->s_nfields) 48825ec3e3dSEric Cheng more_rows = B_TRUE; 48925ec3e3dSEric Cheng 49025ec3e3dSEric Cheng ofmt_print_field(os, &of[i], value, escsep); 49125ec3e3dSEric Cheng } 49225ec3e3dSEric Cheng (void) putchar('\n'); 49325ec3e3dSEric Cheng } 4948002d411SSowmini Varadhan (void) fflush(stdout); 49525ec3e3dSEric Cheng 49625ec3e3dSEric Cheng if (sp != NULL) { 49725ec3e3dSEric Cheng for (i = 0; i < os->os_nfields; i++) 49825ec3e3dSEric Cheng splitfree(sp[i]); 49925ec3e3dSEric Cheng free(sp); 50025ec3e3dSEric Cheng } 5018002d411SSowmini Varadhan } 5028002d411SSowmini Varadhan 5038002d411SSowmini Varadhan /* 5048002d411SSowmini Varadhan * Print the field headers 5058002d411SSowmini Varadhan */ 5068002d411SSowmini Varadhan static void 5078002d411SSowmini Varadhan ofmt_print_header(ofmt_state_t *os) 5088002d411SSowmini Varadhan { 5098002d411SSowmini Varadhan int i; 5108002d411SSowmini Varadhan ofmt_field_t *of = os->os_fields; 5118002d411SSowmini Varadhan boolean_t escsep = (os->os_nfields > 1); 5128002d411SSowmini Varadhan 5138002d411SSowmini Varadhan for (i = 0; i < os->os_nfields; i++) { 5148002d411SSowmini Varadhan os->os_lastfield = (i + 1 == os->os_nfields); 5158002d411SSowmini Varadhan ofmt_print_field(os, &of[i], of[i].of_name, escsep); 5168002d411SSowmini Varadhan } 5178002d411SSowmini Varadhan (void) putchar('\n'); 5188002d411SSowmini Varadhan } 5198002d411SSowmini Varadhan 5208002d411SSowmini Varadhan /* 5218002d411SSowmini Varadhan * Update the current window size. 5228002d411SSowmini Varadhan */ 5238002d411SSowmini Varadhan void 5248002d411SSowmini Varadhan ofmt_update_winsize(ofmt_handle_t ofmt) 5258002d411SSowmini Varadhan { 5268002d411SSowmini Varadhan ofmt_state_t *os = ofmt; 5278002d411SSowmini Varadhan struct winsize *winsize = &os->os_winsize; 5288002d411SSowmini Varadhan 5298002d411SSowmini Varadhan if (ioctl(1, TIOCGWINSZ, winsize) == -1 || 5308002d411SSowmini Varadhan winsize->ws_col == 0 || winsize->ws_row == 0) { 5318002d411SSowmini Varadhan winsize->ws_col = 80; 5328002d411SSowmini Varadhan winsize->ws_row = 24; 5338002d411SSowmini Varadhan } 5348002d411SSowmini Varadhan } 5358002d411SSowmini Varadhan 5368002d411SSowmini Varadhan /* 5378002d411SSowmini Varadhan * Return error diagnostics using the information in the ofmt_handle_t 5388002d411SSowmini Varadhan */ 5398002d411SSowmini Varadhan char * 5408002d411SSowmini Varadhan ofmt_strerror(ofmt_handle_t ofmt, ofmt_status_t err, char *buf, uint_t bufsize) 5418002d411SSowmini Varadhan { 5428002d411SSowmini Varadhan ofmt_state_t *os = ofmt; 5438002d411SSowmini Varadhan int i; 5448002d411SSowmini Varadhan const char *s; 5458002d411SSowmini Varadhan char ebuf[OFMT_BUFSIZE]; 546*0dc2366fSVenugopal Iyer boolean_t parsable; 5478002d411SSowmini Varadhan 5488002d411SSowmini Varadhan /* 5498002d411SSowmini Varadhan * ebuf is intended for optional error-specific data to be appended 5508002d411SSowmini Varadhan * after the internationalized error string for an error code. 5518002d411SSowmini Varadhan */ 5528002d411SSowmini Varadhan ebuf[0] = '\0'; 5538002d411SSowmini Varadhan 5548002d411SSowmini Varadhan switch (err) { 5558002d411SSowmini Varadhan case OFMT_SUCCESS: 5568002d411SSowmini Varadhan s = "success"; 5578002d411SSowmini Varadhan break; 5588002d411SSowmini Varadhan case OFMT_EBADFIELDS: 5598002d411SSowmini Varadhan /* 5608002d411SSowmini Varadhan * Enumerate the singular/plural version of the warning 5618002d411SSowmini Varadhan * and error to simplify and improve localization. 5628002d411SSowmini Varadhan */ 563*0dc2366fSVenugopal Iyer parsable = (os->os_flags & OFMT_PARSABLE); 564*0dc2366fSVenugopal Iyer if (!parsable) { 5658002d411SSowmini Varadhan if (os->os_nbad > 1) 5668002d411SSowmini Varadhan s = "ignoring unknown output fields:"; 5678002d411SSowmini Varadhan else 5688002d411SSowmini Varadhan s = "ignoring unknown output field:"; 5698002d411SSowmini Varadhan } else { 5708002d411SSowmini Varadhan if (os->os_nbad > 1) 5718002d411SSowmini Varadhan s = "unknown output fields:"; 5728002d411SSowmini Varadhan else 5738002d411SSowmini Varadhan s = "unknown output field:"; 5748002d411SSowmini Varadhan } 5758002d411SSowmini Varadhan /* set up the bad fields in ebuf */ 5768002d411SSowmini Varadhan for (i = 0; i < os->os_nbad; i++) { 5778002d411SSowmini Varadhan (void) strlcat(ebuf, " `", sizeof (ebuf)); 5788002d411SSowmini Varadhan (void) strlcat(ebuf, os->os_badfields[i], 5798002d411SSowmini Varadhan sizeof (ebuf)); 5808002d411SSowmini Varadhan (void) strlcat(ebuf, "'", sizeof (ebuf)); 5818002d411SSowmini Varadhan } 5828002d411SSowmini Varadhan break; 5838002d411SSowmini Varadhan case OFMT_ENOFIELDS: 5848002d411SSowmini Varadhan s = "no valid output fields"; 5858002d411SSowmini Varadhan break; 586dbed73cbSSangeeta Misra case OFMT_EPARSEMULTI: 587dbed73cbSSangeeta Misra s = "multiline mode incompatible with parsable mode"; 588dbed73cbSSangeeta Misra break; 5898002d411SSowmini Varadhan case OFMT_EPARSEALL: 5908002d411SSowmini Varadhan s = "output field `all' invalid in parsable mode"; 5918002d411SSowmini Varadhan break; 5928002d411SSowmini Varadhan case OFMT_EPARSENONE: 5938002d411SSowmini Varadhan s = "output fields must be specified in parsable mode"; 5948002d411SSowmini Varadhan break; 59525ec3e3dSEric Cheng case OFMT_EPARSEWRAP: 59625ec3e3dSEric Cheng s = "parsable mode is incompatible with wrap mode"; 59725ec3e3dSEric Cheng break; 5988002d411SSowmini Varadhan case OFMT_ENOTEMPLATE: 5998002d411SSowmini Varadhan s = "no template provided for fields"; 6008002d411SSowmini Varadhan break; 6018002d411SSowmini Varadhan case OFMT_ENOMEM: 6028002d411SSowmini Varadhan s = strerror(ENOMEM); 6038002d411SSowmini Varadhan break; 6048002d411SSowmini Varadhan default: 6058002d411SSowmini Varadhan (void) snprintf(buf, bufsize, 6068002d411SSowmini Varadhan dgettext(TEXT_DOMAIN, "unknown ofmt error (%d)"), 6078002d411SSowmini Varadhan err); 6088002d411SSowmini Varadhan return (buf); 6098002d411SSowmini Varadhan } 6108002d411SSowmini Varadhan (void) snprintf(buf, bufsize, dgettext(TEXT_DOMAIN, s)); 6118002d411SSowmini Varadhan (void) strlcat(buf, ebuf, bufsize); 6128002d411SSowmini Varadhan return (buf); 6138002d411SSowmini Varadhan } 614