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