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