12db6d663SJoshua M. Clulow /*
22db6d663SJoshua M. Clulow  * This file and its contents are supplied under the terms of the
32db6d663SJoshua M. Clulow  * Common Development and Distribution License ("CDDL"), version 1.0.
42db6d663SJoshua M. Clulow  * You may only use this file in accordance with the terms of version
52db6d663SJoshua M. Clulow  * 1.0 of the CDDL.
62db6d663SJoshua M. Clulow  *
72db6d663SJoshua M. Clulow  * A full copy of the text of the CDDL should have accompanied this
82db6d663SJoshua M. Clulow  * source.  A copy of the CDDL is also available via the Internet at
92db6d663SJoshua M. Clulow  * http://www.illumos.org/license/CDDL.
102db6d663SJoshua M. Clulow  */
112db6d663SJoshua M. Clulow /*
1237c79205SJoshua M. Clulow  * Copyright (c) 2014, Joyent, Inc.
13*2ec7644aSSerapheim Dimitropoulos  * Copyright (c) 2017 by Delphix. All rights reserved.
142db6d663SJoshua M. Clulow  */
152db6d663SJoshua M. Clulow 
162db6d663SJoshua M. Clulow #include <stdio.h>
172db6d663SJoshua M. Clulow #include <stdlib.h>
182db6d663SJoshua M. Clulow #include <strings.h>
192db6d663SJoshua M. Clulow #include <wchar.h>
202db6d663SJoshua M. Clulow #include <sys/debug.h>
212db6d663SJoshua M. Clulow 
222db6d663SJoshua M. Clulow #include "libnvpair.h"
232db6d663SJoshua M. Clulow 
2437c79205SJoshua M. Clulow #define	FPRINTF(fp, ...)				\
2537c79205SJoshua M. Clulow 	do {						\
2637c79205SJoshua M. Clulow 		if (fprintf(fp, __VA_ARGS__) < 0)	\
2737c79205SJoshua M. Clulow 			return (-1);			\
2837c79205SJoshua M. Clulow 	} while (0)
292db6d663SJoshua M. Clulow 
302db6d663SJoshua M. Clulow /*
312db6d663SJoshua M. Clulow  * When formatting a string for JSON output we must escape certain characters,
322db6d663SJoshua M. Clulow  * as described in RFC4627.  This applies to both member names and
332db6d663SJoshua M. Clulow  * DATA_TYPE_STRING values.
342db6d663SJoshua M. Clulow  *
352db6d663SJoshua M. Clulow  * This function will only operate correctly if the following conditions are
362db6d663SJoshua M. Clulow  * met:
372db6d663SJoshua M. Clulow  *
382db6d663SJoshua M. Clulow  *       1. The input String is encoded in the current locale.
392db6d663SJoshua M. Clulow  *
402db6d663SJoshua M. Clulow  *       2. The current locale includes the Basic Multilingual Plane (plane 0)
412db6d663SJoshua M. Clulow  *          as defined in the Unicode standard.
422db6d663SJoshua M. Clulow  *
432db6d663SJoshua M. Clulow  * The output will be entirely 7-bit ASCII (as a subset of UTF-8) with all
442db6d663SJoshua M. Clulow  * representable Unicode characters included in their escaped numeric form.
452db6d663SJoshua M. Clulow  */
462db6d663SJoshua M. Clulow static int
nvlist_print_json_string(FILE * fp,const char * input)472db6d663SJoshua M. Clulow nvlist_print_json_string(FILE *fp, const char *input)
482db6d663SJoshua M. Clulow {
492db6d663SJoshua M. Clulow 	mbstate_t mbr;
502db6d663SJoshua M. Clulow 	wchar_t c;
512db6d663SJoshua M. Clulow 	size_t sz;
522db6d663SJoshua M. Clulow 
532db6d663SJoshua M. Clulow 	bzero(&mbr, sizeof (mbr));
542db6d663SJoshua M. Clulow 
552db6d663SJoshua M. Clulow 	FPRINTF(fp, "\"");
562db6d663SJoshua M. Clulow 	while ((sz = mbrtowc(&c, input, MB_CUR_MAX, &mbr)) > 0) {
572db6d663SJoshua M. Clulow 		switch (c) {
582db6d663SJoshua M. Clulow 		case '"':
592db6d663SJoshua M. Clulow 			FPRINTF(fp, "\\\"");
602db6d663SJoshua M. Clulow 			break;
612db6d663SJoshua M. Clulow 		case '\n':
622db6d663SJoshua M. Clulow 			FPRINTF(fp, "\\n");
632db6d663SJoshua M. Clulow 			break;
642db6d663SJoshua M. Clulow 		case '\r':
652db6d663SJoshua M. Clulow 			FPRINTF(fp, "\\r");
662db6d663SJoshua M. Clulow 			break;
672db6d663SJoshua M. Clulow 		case '\\':
682db6d663SJoshua M. Clulow 			FPRINTF(fp, "\\\\");
692db6d663SJoshua M. Clulow 			break;
702db6d663SJoshua M. Clulow 		case '\f':
712db6d663SJoshua M. Clulow 			FPRINTF(fp, "\\f");
722db6d663SJoshua M. Clulow 			break;
732db6d663SJoshua M. Clulow 		case '\t':
742db6d663SJoshua M. Clulow 			FPRINTF(fp, "\\t");
752db6d663SJoshua M. Clulow 			break;
762db6d663SJoshua M. Clulow 		case '\b':
772db6d663SJoshua M. Clulow 			FPRINTF(fp, "\\b");
782db6d663SJoshua M. Clulow 			break;
792db6d663SJoshua M. Clulow 		default:
802db6d663SJoshua M. Clulow 			if ((c >= 0x00 && c <= 0x1f) ||
812db6d663SJoshua M. Clulow 			    (c > 0x7f && c <= 0xffff)) {
822db6d663SJoshua M. Clulow 				/*
832db6d663SJoshua M. Clulow 				 * Render both Control Characters and Unicode
842db6d663SJoshua M. Clulow 				 * characters in the Basic Multilingual Plane
852db6d663SJoshua M. Clulow 				 * as JSON-escaped multibyte characters.
862db6d663SJoshua M. Clulow 				 */
872db6d663SJoshua M. Clulow 				FPRINTF(fp, "\\u%04x", (int)(0xffff & c));
882db6d663SJoshua M. Clulow 			} else if (c >= 0x20 && c <= 0x7f) {
892db6d663SJoshua M. Clulow 				/*
902db6d663SJoshua M. Clulow 				 * Render other 7-bit ASCII characters directly
912db6d663SJoshua M. Clulow 				 * and drop other, unrepresentable characters.
922db6d663SJoshua M. Clulow 				 */
932db6d663SJoshua M. Clulow 				FPRINTF(fp, "%c", (int)(0xff & c));
942db6d663SJoshua M. Clulow 			}
952db6d663SJoshua M. Clulow 			break;
962db6d663SJoshua M. Clulow 		}
972db6d663SJoshua M. Clulow 		input += sz;
982db6d663SJoshua M. Clulow 	}
992db6d663SJoshua M. Clulow 
1002db6d663SJoshua M. Clulow 	if (sz == (size_t)-1 || sz == (size_t)-2) {
1012db6d663SJoshua M. Clulow 		/*
1022db6d663SJoshua M. Clulow 		 * We last read an invalid multibyte character sequence,
1032db6d663SJoshua M. Clulow 		 * so return an error.
1042db6d663SJoshua M. Clulow 		 */
1052db6d663SJoshua M. Clulow 		return (-1);
1062db6d663SJoshua M. Clulow 	}
1072db6d663SJoshua M. Clulow 
1082db6d663SJoshua M. Clulow 	FPRINTF(fp, "\"");
1092db6d663SJoshua M. Clulow 	return (0);
1102db6d663SJoshua M. Clulow }
1112db6d663SJoshua M. Clulow 
1122db6d663SJoshua M. Clulow /*
1132db6d663SJoshua M. Clulow  * Dump a JSON-formatted representation of an nvlist to the provided FILE *.
1142db6d663SJoshua M. Clulow  * This routine does not output any new-lines or additional whitespace other
1152db6d663SJoshua M. Clulow  * than that contained in strings, nor does it call fflush(3C).
1162db6d663SJoshua M. Clulow  */
1172db6d663SJoshua M. Clulow int
nvlist_print_json(FILE * fp,nvlist_t * nvl)1182db6d663SJoshua M. Clulow nvlist_print_json(FILE *fp, nvlist_t *nvl)
1192db6d663SJoshua M. Clulow {
1202db6d663SJoshua M. Clulow 	nvpair_t *curr;
1212db6d663SJoshua M. Clulow 	boolean_t first = B_TRUE;
1222db6d663SJoshua M. Clulow 
1232db6d663SJoshua M. Clulow 	FPRINTF(fp, "{");
1242db6d663SJoshua M. Clulow 
1252db6d663SJoshua M. Clulow 	for (curr = nvlist_next_nvpair(nvl, NULL); curr;
1262db6d663SJoshua M. Clulow 	    curr = nvlist_next_nvpair(nvl, curr)) {
1272db6d663SJoshua M. Clulow 		data_type_t type = nvpair_type(curr);
1282db6d663SJoshua M. Clulow 
1292db6d663SJoshua M. Clulow 		if (!first)
1302db6d663SJoshua M. Clulow 			FPRINTF(fp, ",");
1312db6d663SJoshua M. Clulow 		else
1322db6d663SJoshua M. Clulow 			first = B_FALSE;
1332db6d663SJoshua M. Clulow 
1342db6d663SJoshua M. Clulow 		if (nvlist_print_json_string(fp, nvpair_name(curr)) == -1)
1352db6d663SJoshua M. Clulow 			return (-1);
1362db6d663SJoshua M. Clulow 		FPRINTF(fp, ":");
1372db6d663SJoshua M. Clulow 
1382db6d663SJoshua M. Clulow 		switch (type) {
1392db6d663SJoshua M. Clulow 		case DATA_TYPE_STRING: {
1402db6d663SJoshua M. Clulow 			char *string = fnvpair_value_string(curr);
1412db6d663SJoshua M. Clulow 			if (nvlist_print_json_string(fp, string) == -1)
1422db6d663SJoshua M. Clulow 				return (-1);
1432db6d663SJoshua M. Clulow 			break;
1442db6d663SJoshua M. Clulow 		}
1452db6d663SJoshua M. Clulow 
1462db6d663SJoshua M. Clulow 		case DATA_TYPE_BOOLEAN: {
1472db6d663SJoshua M. Clulow 			FPRINTF(fp, "true");
1482db6d663SJoshua M. Clulow 			break;
1492db6d663SJoshua M. Clulow 		}
1502db6d663SJoshua M. Clulow 
1512db6d663SJoshua M. Clulow 		case DATA_TYPE_BOOLEAN_VALUE: {
1522db6d663SJoshua M. Clulow 			FPRINTF(fp, "%s", fnvpair_value_boolean_value(curr) ==
1532db6d663SJoshua M. Clulow 			    B_TRUE ? "true" : "false");
1542db6d663SJoshua M. Clulow 			break;
1552db6d663SJoshua M. Clulow 		}
1562db6d663SJoshua M. Clulow 
1572db6d663SJoshua M. Clulow 		case DATA_TYPE_BYTE: {
1582db6d663SJoshua M. Clulow 			FPRINTF(fp, "%hhu", fnvpair_value_byte(curr));
1592db6d663SJoshua M. Clulow 			break;
1602db6d663SJoshua M. Clulow 		}
1612db6d663SJoshua M. Clulow 
1622db6d663SJoshua M. Clulow 		case DATA_TYPE_INT8: {
1632db6d663SJoshua M. Clulow 			FPRINTF(fp, "%hhd", fnvpair_value_int8(curr));
1642db6d663SJoshua M. Clulow 			break;
1652db6d663SJoshua M. Clulow 		}
1662db6d663SJoshua M. Clulow 
1672db6d663SJoshua M. Clulow 		case DATA_TYPE_UINT8: {
1682db6d663SJoshua M. Clulow 			FPRINTF(fp, "%hhu", fnvpair_value_uint8_t(curr));
1692db6d663SJoshua M. Clulow 			break;
1702db6d663SJoshua M. Clulow 		}
1712db6d663SJoshua M. Clulow 
1722db6d663SJoshua M. Clulow 		case DATA_TYPE_INT16: {
1732db6d663SJoshua M. Clulow 			FPRINTF(fp, "%hd", fnvpair_value_int16(curr));
1742db6d663SJoshua M. Clulow 			break;
1752db6d663SJoshua M. Clulow 		}
1762db6d663SJoshua M. Clulow 
1772db6d663SJoshua M. Clulow 		case DATA_TYPE_UINT16: {
1782db6d663SJoshua M. Clulow 			FPRINTF(fp, "%hu", fnvpair_value_uint16(curr));
1792db6d663SJoshua M. Clulow 			break;
1802db6d663SJoshua M. Clulow 		}
1812db6d663SJoshua M. Clulow 
1822db6d663SJoshua M. Clulow 		case DATA_TYPE_INT32: {
1832db6d663SJoshua M. Clulow 			FPRINTF(fp, "%d", fnvpair_value_int32(curr));
1842db6d663SJoshua M. Clulow 			break;
1852db6d663SJoshua M. Clulow 		}
1862db6d663SJoshua M. Clulow 
1872db6d663SJoshua M. Clulow 		case DATA_TYPE_UINT32: {
1882db6d663SJoshua M. Clulow 			FPRINTF(fp, "%u", fnvpair_value_uint32(curr));
1892db6d663SJoshua M. Clulow 			break;
1902db6d663SJoshua M. Clulow 		}
1912db6d663SJoshua M. Clulow 
1922db6d663SJoshua M. Clulow 		case DATA_TYPE_INT64: {
1932db6d663SJoshua M. Clulow 			FPRINTF(fp, "%lld",
1942db6d663SJoshua M. Clulow 			    (long long)fnvpair_value_int64(curr));
1952db6d663SJoshua M. Clulow 			break;
1962db6d663SJoshua M. Clulow 		}
1972db6d663SJoshua M. Clulow 
1982db6d663SJoshua M. Clulow 		case DATA_TYPE_UINT64: {
1992db6d663SJoshua M. Clulow 			FPRINTF(fp, "%llu",
2002db6d663SJoshua M. Clulow 			    (unsigned long long)fnvpair_value_uint64(curr));
2012db6d663SJoshua M. Clulow 			break;
2022db6d663SJoshua M. Clulow 		}
2032db6d663SJoshua M. Clulow 
2042db6d663SJoshua M. Clulow 		case DATA_TYPE_HRTIME: {
2052db6d663SJoshua M. Clulow 			hrtime_t val;
2062db6d663SJoshua M. Clulow 			VERIFY0(nvpair_value_hrtime(curr, &val));
2072db6d663SJoshua M. Clulow 			FPRINTF(fp, "%llu", (unsigned long long)val);
2082db6d663SJoshua M. Clulow 			break;
2092db6d663SJoshua M. Clulow 		}
2102db6d663SJoshua M. Clulow 
2112db6d663SJoshua M. Clulow 		case DATA_TYPE_DOUBLE: {
2122db6d663SJoshua M. Clulow 			double val;
2132db6d663SJoshua M. Clulow 			VERIFY0(nvpair_value_double(curr, &val));
2142db6d663SJoshua M. Clulow 			FPRINTF(fp, "%f", val);
2152db6d663SJoshua M. Clulow 			break;
2162db6d663SJoshua M. Clulow 		}
2172db6d663SJoshua M. Clulow 
2182db6d663SJoshua M. Clulow 		case DATA_TYPE_NVLIST: {
2192db6d663SJoshua M. Clulow 			if (nvlist_print_json(fp,
2202db6d663SJoshua M. Clulow 			    fnvpair_value_nvlist(curr)) == -1)
2212db6d663SJoshua M. Clulow 				return (-1);
2222db6d663SJoshua M. Clulow 			break;
2232db6d663SJoshua M. Clulow 		}
2242db6d663SJoshua M. Clulow 
2252db6d663SJoshua M. Clulow 		case DATA_TYPE_STRING_ARRAY: {
2262db6d663SJoshua M. Clulow 			char **val;
2272db6d663SJoshua M. Clulow 			uint_t valsz, i;
2282db6d663SJoshua M. Clulow 			VERIFY0(nvpair_value_string_array(curr, &val, &valsz));
2292db6d663SJoshua M. Clulow 			FPRINTF(fp, "[");
2302db6d663SJoshua M. Clulow 			for (i = 0; i < valsz; i++) {
2312db6d663SJoshua M. Clulow 				if (i > 0)
2322db6d663SJoshua M. Clulow 					FPRINTF(fp, ",");
2332db6d663SJoshua M. Clulow 				if (nvlist_print_json_string(fp, val[i]) == -1)
2342db6d663SJoshua M. Clulow 					return (-1);
2352db6d663SJoshua M. Clulow 			}
2362db6d663SJoshua M. Clulow 			FPRINTF(fp, "]");
2372db6d663SJoshua M. Clulow 			break;
2382db6d663SJoshua M. Clulow 		}
2392db6d663SJoshua M. Clulow 
2402db6d663SJoshua M. Clulow 		case DATA_TYPE_NVLIST_ARRAY: {
2412db6d663SJoshua M. Clulow 			nvlist_t **val;
2422db6d663SJoshua M. Clulow 			uint_t valsz, i;
2432db6d663SJoshua M. Clulow 			VERIFY0(nvpair_value_nvlist_array(curr, &val, &valsz));
2442db6d663SJoshua M. Clulow 			FPRINTF(fp, "[");
2452db6d663SJoshua M. Clulow 			for (i = 0; i < valsz; i++) {
2462db6d663SJoshua M. Clulow 				if (i > 0)
2472db6d663SJoshua M. Clulow 					FPRINTF(fp, ",");
2482db6d663SJoshua M. Clulow 				if (nvlist_print_json(fp, val[i]) == -1)
2492db6d663SJoshua M. Clulow 					return (-1);
2502db6d663SJoshua M. Clulow 			}
2512db6d663SJoshua M. Clulow 			FPRINTF(fp, "]");
2522db6d663SJoshua M. Clulow 			break;
2532db6d663SJoshua M. Clulow 		}
2542db6d663SJoshua M. Clulow 
2552db6d663SJoshua M. Clulow 		case DATA_TYPE_BOOLEAN_ARRAY: {
2562db6d663SJoshua M. Clulow 			boolean_t *val;
2572db6d663SJoshua M. Clulow 			uint_t valsz, i;
2582db6d663SJoshua M. Clulow 			VERIFY0(nvpair_value_boolean_array(curr, &val, &valsz));
2592db6d663SJoshua M. Clulow 			FPRINTF(fp, "[");
2602db6d663SJoshua M. Clulow 			for (i = 0; i < valsz; i++) {
2612db6d663SJoshua M. Clulow 				if (i > 0)
2622db6d663SJoshua M. Clulow 					FPRINTF(fp, ",");
2632db6d663SJoshua M. Clulow 				FPRINTF(fp, val[i] == B_TRUE ?
2642db6d663SJoshua M. Clulow 				    "true" : "false");
2652db6d663SJoshua M. Clulow 			}
2662db6d663SJoshua M. Clulow 			FPRINTF(fp, "]");
2672db6d663SJoshua M. Clulow 			break;
2682db6d663SJoshua M. Clulow 		}
2692db6d663SJoshua M. Clulow 
2702db6d663SJoshua M. Clulow 		case DATA_TYPE_BYTE_ARRAY: {
2712db6d663SJoshua M. Clulow 			uchar_t *val;
2722db6d663SJoshua M. Clulow 			uint_t valsz, i;
2732db6d663SJoshua M. Clulow 			VERIFY0(nvpair_value_byte_array(curr, &val, &valsz));
2742db6d663SJoshua M. Clulow 			FPRINTF(fp, "[");
2752db6d663SJoshua M. Clulow 			for (i = 0; i < valsz; i++) {
2762db6d663SJoshua M. Clulow 				if (i > 0)
2772db6d663SJoshua M. Clulow 					FPRINTF(fp, ",");
2782db6d663SJoshua M. Clulow 				FPRINTF(fp, "%hhu", val[i]);
2792db6d663SJoshua M. Clulow 			}
2802db6d663SJoshua M. Clulow 			FPRINTF(fp, "]");
2812db6d663SJoshua M. Clulow 			break;
2822db6d663SJoshua M. Clulow 		}
2832db6d663SJoshua M. Clulow 
2842db6d663SJoshua M. Clulow 		case DATA_TYPE_UINT8_ARRAY: {
2852db6d663SJoshua M. Clulow 			uint8_t *val;
2862db6d663SJoshua M. Clulow 			uint_t valsz, i;
2872db6d663SJoshua M. Clulow 			VERIFY0(nvpair_value_uint8_array(curr, &val, &valsz));
2882db6d663SJoshua M. Clulow 			FPRINTF(fp, "[");
2892db6d663SJoshua M. Clulow 			for (i = 0; i < valsz; i++) {
2902db6d663SJoshua M. Clulow 				if (i > 0)
2912db6d663SJoshua M. Clulow 					FPRINTF(fp, ",");
2922db6d663SJoshua M. Clulow 				FPRINTF(fp, "%hhu", val[i]);
2932db6d663SJoshua M. Clulow 			}
2942db6d663SJoshua M. Clulow 			FPRINTF(fp, "]");
2952db6d663SJoshua M. Clulow 			break;
2962db6d663SJoshua M. Clulow 		}
2972db6d663SJoshua M. Clulow 
2982db6d663SJoshua M. Clulow 		case DATA_TYPE_INT8_ARRAY: {
2992db6d663SJoshua M. Clulow 			int8_t *val;
3002db6d663SJoshua M. Clulow 			uint_t valsz, i;
3012db6d663SJoshua M. Clulow 			VERIFY0(nvpair_value_int8_array(curr, &val, &valsz));
3022db6d663SJoshua M. Clulow 			FPRINTF(fp, "[");
3032db6d663SJoshua M. Clulow 			for (i = 0; i < valsz; i++) {
3042db6d663SJoshua M. Clulow 				if (i > 0)
3052db6d663SJoshua M. Clulow 					FPRINTF(fp, ",");
3062db6d663SJoshua M. Clulow 				FPRINTF(fp, "%hd", val[i]);
3072db6d663SJoshua M. Clulow 			}
3082db6d663SJoshua M. Clulow 			FPRINTF(fp, "]");
3092db6d663SJoshua M. Clulow 			break;
3102db6d663SJoshua M. Clulow 		}
3112db6d663SJoshua M. Clulow 
3122db6d663SJoshua M. Clulow 		case DATA_TYPE_UINT16_ARRAY: {
3132db6d663SJoshua M. Clulow 			uint16_t *val;
3142db6d663SJoshua M. Clulow 			uint_t valsz, i;
3152db6d663SJoshua M. Clulow 			VERIFY0(nvpair_value_uint16_array(curr, &val, &valsz));
3162db6d663SJoshua M. Clulow 			FPRINTF(fp, "[");
3172db6d663SJoshua M. Clulow 			for (i = 0; i < valsz; i++) {
3182db6d663SJoshua M. Clulow 				if (i > 0)
3192db6d663SJoshua M. Clulow 					FPRINTF(fp, ",");
3202db6d663SJoshua M. Clulow 				FPRINTF(fp, "%hu", val[i]);
3212db6d663SJoshua M. Clulow 			}
3222db6d663SJoshua M. Clulow 			FPRINTF(fp, "]");
3232db6d663SJoshua M. Clulow 			break;
3242db6d663SJoshua M. Clulow 		}
3252db6d663SJoshua M. Clulow 
3262db6d663SJoshua M. Clulow 		case DATA_TYPE_INT16_ARRAY: {
3272db6d663SJoshua M. Clulow 			int16_t *val;
3282db6d663SJoshua M. Clulow 			uint_t valsz, i;
3292db6d663SJoshua M. Clulow 			VERIFY0(nvpair_value_int16_array(curr, &val, &valsz));
3302db6d663SJoshua M. Clulow 			FPRINTF(fp, "[");
3312db6d663SJoshua M. Clulow 			for (i = 0; i < valsz; i++) {
3322db6d663SJoshua M. Clulow 				if (i > 0)
3332db6d663SJoshua M. Clulow 					FPRINTF(fp, ",");
33437c79205SJoshua M. Clulow 				FPRINTF(fp, "%hd", val[i]);
3352db6d663SJoshua M. Clulow 			}
3362db6d663SJoshua M. Clulow 			FPRINTF(fp, "]");
3372db6d663SJoshua M. Clulow 			break;
3382db6d663SJoshua M. Clulow 		}
3392db6d663SJoshua M. Clulow 
3402db6d663SJoshua M. Clulow 		case DATA_TYPE_UINT32_ARRAY: {
3412db6d663SJoshua M. Clulow 			uint32_t *val;
3422db6d663SJoshua M. Clulow 			uint_t valsz, i;
3432db6d663SJoshua M. Clulow 			VERIFY0(nvpair_value_uint32_array(curr, &val, &valsz));
3442db6d663SJoshua M. Clulow 			FPRINTF(fp, "[");
3452db6d663SJoshua M. Clulow 			for (i = 0; i < valsz; i++) {
3462db6d663SJoshua M. Clulow 				if (i > 0)
3472db6d663SJoshua M. Clulow 					FPRINTF(fp, ",");
3482db6d663SJoshua M. Clulow 				FPRINTF(fp, "%u", val[i]);
3492db6d663SJoshua M. Clulow 			}
3502db6d663SJoshua M. Clulow 			FPRINTF(fp, "]");
3512db6d663SJoshua M. Clulow 			break;
3522db6d663SJoshua M. Clulow 		}
3532db6d663SJoshua M. Clulow 
3542db6d663SJoshua M. Clulow 		case DATA_TYPE_INT32_ARRAY: {
3552db6d663SJoshua M. Clulow 			int32_t *val;
3562db6d663SJoshua M. Clulow 			uint_t valsz, i;
3572db6d663SJoshua M. Clulow 			VERIFY0(nvpair_value_int32_array(curr, &val, &valsz));
3582db6d663SJoshua M. Clulow 			FPRINTF(fp, "[");
3592db6d663SJoshua M. Clulow 			for (i = 0; i < valsz; i++) {
3602db6d663SJoshua M. Clulow 				if (i > 0)
3612db6d663SJoshua M. Clulow 					FPRINTF(fp, ",");
3622db6d663SJoshua M. Clulow 				FPRINTF(fp, "%d", val[i]);
3632db6d663SJoshua M. Clulow 			}
3642db6d663SJoshua M. Clulow 			FPRINTF(fp, "]");
3652db6d663SJoshua M. Clulow 			break;
3662db6d663SJoshua M. Clulow 		}
3672db6d663SJoshua M. Clulow 
3682db6d663SJoshua M. Clulow 		case DATA_TYPE_UINT64_ARRAY: {
3692db6d663SJoshua M. Clulow 			uint64_t *val;
3702db6d663SJoshua M. Clulow 			uint_t valsz, i;
3712db6d663SJoshua M. Clulow 			VERIFY0(nvpair_value_uint64_array(curr, &val, &valsz));
3722db6d663SJoshua M. Clulow 			FPRINTF(fp, "[");
3732db6d663SJoshua M. Clulow 			for (i = 0; i < valsz; i++) {
3742db6d663SJoshua M. Clulow 				if (i > 0)
3752db6d663SJoshua M. Clulow 					FPRINTF(fp, ",");
3762db6d663SJoshua M. Clulow 				FPRINTF(fp, "%llu",
3772db6d663SJoshua M. Clulow 				    (unsigned long long)val[i]);
3782db6d663SJoshua M. Clulow 			}
3792db6d663SJoshua M. Clulow 			FPRINTF(fp, "]");
3802db6d663SJoshua M. Clulow 			break;
3812db6d663SJoshua M. Clulow 		}
3822db6d663SJoshua M. Clulow 
3832db6d663SJoshua M. Clulow 		case DATA_TYPE_INT64_ARRAY: {
3842db6d663SJoshua M. Clulow 			int64_t *val;
3852db6d663SJoshua M. Clulow 			uint_t valsz, i;
3862db6d663SJoshua M. Clulow 			VERIFY0(nvpair_value_int64_array(curr, &val, &valsz));
3872db6d663SJoshua M. Clulow 			FPRINTF(fp, "[");
3882db6d663SJoshua M. Clulow 			for (i = 0; i < valsz; i++) {
3892db6d663SJoshua M. Clulow 				if (i > 0)
3902db6d663SJoshua M. Clulow 					FPRINTF(fp, ",");
3912db6d663SJoshua M. Clulow 				FPRINTF(fp, "%lld", (long long)val[i]);
3922db6d663SJoshua M. Clulow 			}
3932db6d663SJoshua M. Clulow 			FPRINTF(fp, "]");
3942db6d663SJoshua M. Clulow 			break;
3952db6d663SJoshua M. Clulow 		}
3962db6d663SJoshua M. Clulow 
3972db6d663SJoshua M. Clulow 		case DATA_TYPE_UNKNOWN:
398*2ec7644aSSerapheim Dimitropoulos 		case DATA_TYPE_DONTCARE:
3992db6d663SJoshua M. Clulow 			return (-1);
4002db6d663SJoshua M. Clulow 		}
401*2ec7644aSSerapheim Dimitropoulos 
4022db6d663SJoshua M. Clulow 	}
4032db6d663SJoshua M. Clulow 
4042db6d663SJoshua M. Clulow 	FPRINTF(fp, "}");
4052db6d663SJoshua M. Clulow 	return (0);
4062db6d663SJoshua M. Clulow }
407