1*eda14cbcSMatt Macy /* 2*eda14cbcSMatt Macy * CDDL HEADER START 3*eda14cbcSMatt Macy * 4*eda14cbcSMatt Macy * The contents of this file are subject to the terms of the 5*eda14cbcSMatt Macy * Common Development and Distribution License (the "License"). 6*eda14cbcSMatt Macy * You may not use this file except in compliance with the License. 7*eda14cbcSMatt Macy * 8*eda14cbcSMatt Macy * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE 9*eda14cbcSMatt Macy * or http://www.opensolaris.org/os/licensing. 10*eda14cbcSMatt Macy * See the License for the specific language governing permissions 11*eda14cbcSMatt Macy * and limitations under the License. 12*eda14cbcSMatt Macy * 13*eda14cbcSMatt Macy * When distributing Covered Code, include this CDDL HEADER in each 14*eda14cbcSMatt Macy * file and include the License file at usr/src/OPENSOLARIS.LICENSE. 15*eda14cbcSMatt Macy * If applicable, add the following below this CDDL HEADER, with the 16*eda14cbcSMatt Macy * fields enclosed by brackets "[]" replaced with your own identifying 17*eda14cbcSMatt Macy * information: Portions Copyright [yyyy] [name of copyright owner] 18*eda14cbcSMatt Macy * 19*eda14cbcSMatt Macy * CDDL HEADER END 20*eda14cbcSMatt Macy */ 21*eda14cbcSMatt Macy 22*eda14cbcSMatt Macy /* 23*eda14cbcSMatt Macy * Copyright (c) 2005, 2010, Oracle and/or its affiliates. All rights reserved. 24*eda14cbcSMatt Macy */ 25*eda14cbcSMatt Macy 26*eda14cbcSMatt Macy #include <stdio.h> 27*eda14cbcSMatt Macy #include <stdlib.h> 28*eda14cbcSMatt Macy #include <string.h> 29*eda14cbcSMatt Macy #include <sys/nvpair.h> 30*eda14cbcSMatt Macy #include <sys/fs/zfs.h> 31*eda14cbcSMatt Macy 32*eda14cbcSMatt Macy #include <libzutil.h> 33*eda14cbcSMatt Macy 34*eda14cbcSMatt Macy static void 35*eda14cbcSMatt Macy dump_ddt_stat(const ddt_stat_t *dds, int h) 36*eda14cbcSMatt Macy { 37*eda14cbcSMatt Macy char refcnt[6]; 38*eda14cbcSMatt Macy char blocks[6], lsize[6], psize[6], dsize[6]; 39*eda14cbcSMatt Macy char ref_blocks[6], ref_lsize[6], ref_psize[6], ref_dsize[6]; 40*eda14cbcSMatt Macy 41*eda14cbcSMatt Macy if (dds == NULL || dds->dds_blocks == 0) 42*eda14cbcSMatt Macy return; 43*eda14cbcSMatt Macy 44*eda14cbcSMatt Macy if (h == -1) 45*eda14cbcSMatt Macy (void) strcpy(refcnt, "Total"); 46*eda14cbcSMatt Macy else 47*eda14cbcSMatt Macy zfs_nicenum(1ULL << h, refcnt, sizeof (refcnt)); 48*eda14cbcSMatt Macy 49*eda14cbcSMatt Macy zfs_nicenum(dds->dds_blocks, blocks, sizeof (blocks)); 50*eda14cbcSMatt Macy zfs_nicebytes(dds->dds_lsize, lsize, sizeof (lsize)); 51*eda14cbcSMatt Macy zfs_nicebytes(dds->dds_psize, psize, sizeof (psize)); 52*eda14cbcSMatt Macy zfs_nicebytes(dds->dds_dsize, dsize, sizeof (dsize)); 53*eda14cbcSMatt Macy zfs_nicenum(dds->dds_ref_blocks, ref_blocks, sizeof (ref_blocks)); 54*eda14cbcSMatt Macy zfs_nicebytes(dds->dds_ref_lsize, ref_lsize, sizeof (ref_lsize)); 55*eda14cbcSMatt Macy zfs_nicebytes(dds->dds_ref_psize, ref_psize, sizeof (ref_psize)); 56*eda14cbcSMatt Macy zfs_nicebytes(dds->dds_ref_dsize, ref_dsize, sizeof (ref_dsize)); 57*eda14cbcSMatt Macy 58*eda14cbcSMatt Macy (void) printf("%6s %6s %5s %5s %5s %6s %5s %5s %5s\n", 59*eda14cbcSMatt Macy refcnt, 60*eda14cbcSMatt Macy blocks, lsize, psize, dsize, 61*eda14cbcSMatt Macy ref_blocks, ref_lsize, ref_psize, ref_dsize); 62*eda14cbcSMatt Macy } 63*eda14cbcSMatt Macy 64*eda14cbcSMatt Macy /* 65*eda14cbcSMatt Macy * Print the DDT histogram and the column totals. 66*eda14cbcSMatt Macy */ 67*eda14cbcSMatt Macy void 68*eda14cbcSMatt Macy zpool_dump_ddt(const ddt_stat_t *dds_total, const ddt_histogram_t *ddh) 69*eda14cbcSMatt Macy { 70*eda14cbcSMatt Macy int h; 71*eda14cbcSMatt Macy 72*eda14cbcSMatt Macy (void) printf("\n"); 73*eda14cbcSMatt Macy 74*eda14cbcSMatt Macy (void) printf("bucket " 75*eda14cbcSMatt Macy " allocated " 76*eda14cbcSMatt Macy " referenced \n"); 77*eda14cbcSMatt Macy (void) printf("______ " 78*eda14cbcSMatt Macy "______________________________ " 79*eda14cbcSMatt Macy "______________________________\n"); 80*eda14cbcSMatt Macy 81*eda14cbcSMatt Macy (void) printf("%6s %6s %5s %5s %5s %6s %5s %5s %5s\n", 82*eda14cbcSMatt Macy "refcnt", 83*eda14cbcSMatt Macy "blocks", "LSIZE", "PSIZE", "DSIZE", 84*eda14cbcSMatt Macy "blocks", "LSIZE", "PSIZE", "DSIZE"); 85*eda14cbcSMatt Macy 86*eda14cbcSMatt Macy (void) printf("%6s %6s %5s %5s %5s %6s %5s %5s %5s\n", 87*eda14cbcSMatt Macy "------", 88*eda14cbcSMatt Macy "------", "-----", "-----", "-----", 89*eda14cbcSMatt Macy "------", "-----", "-----", "-----"); 90*eda14cbcSMatt Macy 91*eda14cbcSMatt Macy for (h = 0; h < 64; h++) 92*eda14cbcSMatt Macy dump_ddt_stat(&ddh->ddh_stat[h], h); 93*eda14cbcSMatt Macy 94*eda14cbcSMatt Macy dump_ddt_stat(dds_total, -1); 95*eda14cbcSMatt Macy 96*eda14cbcSMatt Macy (void) printf("\n"); 97*eda14cbcSMatt Macy } 98*eda14cbcSMatt Macy 99*eda14cbcSMatt Macy /* 100*eda14cbcSMatt Macy * Process the buffer of nvlists, unpacking and storing each nvlist record 101*eda14cbcSMatt Macy * into 'records'. 'leftover' is set to the number of bytes that weren't 102*eda14cbcSMatt Macy * processed as there wasn't a complete record. 103*eda14cbcSMatt Macy */ 104*eda14cbcSMatt Macy int 105*eda14cbcSMatt Macy zpool_history_unpack(char *buf, uint64_t bytes_read, uint64_t *leftover, 106*eda14cbcSMatt Macy nvlist_t ***records, uint_t *numrecords) 107*eda14cbcSMatt Macy { 108*eda14cbcSMatt Macy uint64_t reclen; 109*eda14cbcSMatt Macy nvlist_t *nv; 110*eda14cbcSMatt Macy int i; 111*eda14cbcSMatt Macy void *tmp; 112*eda14cbcSMatt Macy 113*eda14cbcSMatt Macy while (bytes_read > sizeof (reclen)) { 114*eda14cbcSMatt Macy 115*eda14cbcSMatt Macy /* get length of packed record (stored as little endian) */ 116*eda14cbcSMatt Macy for (i = 0, reclen = 0; i < sizeof (reclen); i++) 117*eda14cbcSMatt Macy reclen += (uint64_t)(((uchar_t *)buf)[i]) << (8*i); 118*eda14cbcSMatt Macy 119*eda14cbcSMatt Macy if (bytes_read < sizeof (reclen) + reclen) 120*eda14cbcSMatt Macy break; 121*eda14cbcSMatt Macy 122*eda14cbcSMatt Macy /* unpack record */ 123*eda14cbcSMatt Macy if (nvlist_unpack(buf + sizeof (reclen), reclen, &nv, 0) != 0) 124*eda14cbcSMatt Macy return (ENOMEM); 125*eda14cbcSMatt Macy bytes_read -= sizeof (reclen) + reclen; 126*eda14cbcSMatt Macy buf += sizeof (reclen) + reclen; 127*eda14cbcSMatt Macy 128*eda14cbcSMatt Macy /* add record to nvlist array */ 129*eda14cbcSMatt Macy (*numrecords)++; 130*eda14cbcSMatt Macy if (ISP2(*numrecords + 1)) { 131*eda14cbcSMatt Macy tmp = realloc(*records, 132*eda14cbcSMatt Macy *numrecords * 2 * sizeof (nvlist_t *)); 133*eda14cbcSMatt Macy if (tmp == NULL) { 134*eda14cbcSMatt Macy nvlist_free(nv); 135*eda14cbcSMatt Macy (*numrecords)--; 136*eda14cbcSMatt Macy return (ENOMEM); 137*eda14cbcSMatt Macy } 138*eda14cbcSMatt Macy *records = tmp; 139*eda14cbcSMatt Macy } 140*eda14cbcSMatt Macy (*records)[*numrecords - 1] = nv; 141*eda14cbcSMatt Macy } 142*eda14cbcSMatt Macy 143*eda14cbcSMatt Macy *leftover = bytes_read; 144*eda14cbcSMatt Macy return (0); 145*eda14cbcSMatt Macy } 146