1 /* 2 * CDDL HEADER START 3 * 4 * The contents of this file are subject to the terms of the 5 * Common Development and Distribution License (the "License"). 6 * You may not use this file except in compliance with the License. 7 * 8 * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE 9 * or http://www.opensolaris.org/os/licensing. 10 * See the License for the specific language governing permissions 11 * and limitations under the License. 12 * 13 * When distributing Covered Code, include this CDDL HEADER in each 14 * file and include the License file at usr/src/OPENSOLARIS.LICENSE. 15 * If applicable, add the following below this CDDL HEADER, with the 16 * fields enclosed by brackets "[]" replaced with your own identifying 17 * information: Portions Copyright [yyyy] [name of copyright owner] 18 * 19 * CDDL HEADER END 20 */ 21 /* 22 * Copyright 2010 Sun Microsystems, Inc. All rights reserved. 23 * Use is subject to license terms. 24 */ 25 26 /* 27 * Internal utility routines for the ZFS library. 28 */ 29 30 #include <errno.h> 31 #include <fcntl.h> 32 #include <libintl.h> 33 #include <stdarg.h> 34 #include <stdio.h> 35 #include <stdlib.h> 36 #include <strings.h> 37 #include <unistd.h> 38 #include <ctype.h> 39 #include <math.h> 40 #include <sys/mnttab.h> 41 #include <sys/mntent.h> 42 #include <sys/types.h> 43 44 #include <libzfs.h> 45 46 #include "libzfs_impl.h" 47 #include "zfs_prop.h" 48 49 int 50 libzfs_errno(libzfs_handle_t *hdl) 51 { 52 return (hdl->libzfs_error); 53 } 54 55 const char * 56 libzfs_error_action(libzfs_handle_t *hdl) 57 { 58 return (hdl->libzfs_action); 59 } 60 61 const char * 62 libzfs_error_description(libzfs_handle_t *hdl) 63 { 64 if (hdl->libzfs_desc[0] != '\0') 65 return (hdl->libzfs_desc); 66 67 switch (hdl->libzfs_error) { 68 case EZFS_NOMEM: 69 return (dgettext(TEXT_DOMAIN, "out of memory")); 70 case EZFS_BADPROP: 71 return (dgettext(TEXT_DOMAIN, "invalid property value")); 72 case EZFS_PROPREADONLY: 73 return (dgettext(TEXT_DOMAIN, "read only property")); 74 case EZFS_PROPTYPE: 75 return (dgettext(TEXT_DOMAIN, "property doesn't apply to " 76 "datasets of this type")); 77 case EZFS_PROPNONINHERIT: 78 return (dgettext(TEXT_DOMAIN, "property cannot be inherited")); 79 case EZFS_PROPSPACE: 80 return (dgettext(TEXT_DOMAIN, "invalid quota or reservation")); 81 case EZFS_BADTYPE: 82 return (dgettext(TEXT_DOMAIN, "operation not applicable to " 83 "datasets of this type")); 84 case EZFS_BUSY: 85 return (dgettext(TEXT_DOMAIN, "pool or dataset is busy")); 86 case EZFS_EXISTS: 87 return (dgettext(TEXT_DOMAIN, "pool or dataset exists")); 88 case EZFS_NOENT: 89 return (dgettext(TEXT_DOMAIN, "no such pool or dataset")); 90 case EZFS_BADSTREAM: 91 return (dgettext(TEXT_DOMAIN, "invalid backup stream")); 92 case EZFS_DSREADONLY: 93 return (dgettext(TEXT_DOMAIN, "dataset is read only")); 94 case EZFS_VOLTOOBIG: 95 return (dgettext(TEXT_DOMAIN, "volume size exceeds limit for " 96 "this system")); 97 case EZFS_INVALIDNAME: 98 return (dgettext(TEXT_DOMAIN, "invalid name")); 99 case EZFS_BADRESTORE: 100 return (dgettext(TEXT_DOMAIN, "unable to restore to " 101 "destination")); 102 case EZFS_BADBACKUP: 103 return (dgettext(TEXT_DOMAIN, "backup failed")); 104 case EZFS_BADTARGET: 105 return (dgettext(TEXT_DOMAIN, "invalid target vdev")); 106 case EZFS_NODEVICE: 107 return (dgettext(TEXT_DOMAIN, "no such device in pool")); 108 case EZFS_BADDEV: 109 return (dgettext(TEXT_DOMAIN, "invalid device")); 110 case EZFS_NOREPLICAS: 111 return (dgettext(TEXT_DOMAIN, "no valid replicas")); 112 case EZFS_RESILVERING: 113 return (dgettext(TEXT_DOMAIN, "currently resilvering")); 114 case EZFS_BADVERSION: 115 return (dgettext(TEXT_DOMAIN, "unsupported version")); 116 case EZFS_POOLUNAVAIL: 117 return (dgettext(TEXT_DOMAIN, "pool is unavailable")); 118 case EZFS_DEVOVERFLOW: 119 return (dgettext(TEXT_DOMAIN, "too many devices in one vdev")); 120 case EZFS_BADPATH: 121 return (dgettext(TEXT_DOMAIN, "must be an absolute path")); 122 case EZFS_CROSSTARGET: 123 return (dgettext(TEXT_DOMAIN, "operation crosses datasets or " 124 "pools")); 125 case EZFS_ZONED: 126 return (dgettext(TEXT_DOMAIN, "dataset in use by local zone")); 127 case EZFS_MOUNTFAILED: 128 return (dgettext(TEXT_DOMAIN, "mount failed")); 129 case EZFS_UMOUNTFAILED: 130 return (dgettext(TEXT_DOMAIN, "umount failed")); 131 case EZFS_UNSHARENFSFAILED: 132 return (dgettext(TEXT_DOMAIN, "unshare(1M) failed")); 133 case EZFS_SHARENFSFAILED: 134 return (dgettext(TEXT_DOMAIN, "share(1M) failed")); 135 case EZFS_UNSHARESMBFAILED: 136 return (dgettext(TEXT_DOMAIN, "smb remove share failed")); 137 case EZFS_SHARESMBFAILED: 138 return (dgettext(TEXT_DOMAIN, "smb add share failed")); 139 case EZFS_ISCSISVCUNAVAIL: 140 return (dgettext(TEXT_DOMAIN, 141 "iscsitgt service need to be enabled by " 142 "a privileged user")); 143 case EZFS_PERM: 144 return (dgettext(TEXT_DOMAIN, "permission denied")); 145 case EZFS_NOSPC: 146 return (dgettext(TEXT_DOMAIN, "out of space")); 147 case EZFS_IO: 148 return (dgettext(TEXT_DOMAIN, "I/O error")); 149 case EZFS_INTR: 150 return (dgettext(TEXT_DOMAIN, "signal received")); 151 case EZFS_ISSPARE: 152 return (dgettext(TEXT_DOMAIN, "device is reserved as a hot " 153 "spare")); 154 case EZFS_INVALCONFIG: 155 return (dgettext(TEXT_DOMAIN, "invalid vdev configuration")); 156 case EZFS_RECURSIVE: 157 return (dgettext(TEXT_DOMAIN, "recursive dataset dependency")); 158 case EZFS_NOHISTORY: 159 return (dgettext(TEXT_DOMAIN, "no history available")); 160 case EZFS_UNSHAREISCSIFAILED: 161 return (dgettext(TEXT_DOMAIN, 162 "iscsitgtd failed request to unshare")); 163 case EZFS_SHAREISCSIFAILED: 164 return (dgettext(TEXT_DOMAIN, 165 "iscsitgtd failed request to share")); 166 case EZFS_POOLPROPS: 167 return (dgettext(TEXT_DOMAIN, "failed to retrieve " 168 "pool properties")); 169 case EZFS_POOL_NOTSUP: 170 return (dgettext(TEXT_DOMAIN, "operation not supported " 171 "on this type of pool")); 172 case EZFS_POOL_INVALARG: 173 return (dgettext(TEXT_DOMAIN, "invalid argument for " 174 "this pool operation")); 175 case EZFS_NAMETOOLONG: 176 return (dgettext(TEXT_DOMAIN, "dataset name is too long")); 177 case EZFS_OPENFAILED: 178 return (dgettext(TEXT_DOMAIN, "open failed")); 179 case EZFS_NOCAP: 180 return (dgettext(TEXT_DOMAIN, 181 "disk capacity information could not be retrieved")); 182 case EZFS_LABELFAILED: 183 return (dgettext(TEXT_DOMAIN, "write of label failed")); 184 case EZFS_BADWHO: 185 return (dgettext(TEXT_DOMAIN, "invalid user/group")); 186 case EZFS_BADPERM: 187 return (dgettext(TEXT_DOMAIN, "invalid permission")); 188 case EZFS_BADPERMSET: 189 return (dgettext(TEXT_DOMAIN, "invalid permission set name")); 190 case EZFS_NODELEGATION: 191 return (dgettext(TEXT_DOMAIN, "delegated administration is " 192 "disabled on pool")); 193 case EZFS_PERMRDONLY: 194 return (dgettext(TEXT_DOMAIN, "snapshot permissions cannot be" 195 " modified")); 196 case EZFS_BADCACHE: 197 return (dgettext(TEXT_DOMAIN, "invalid or missing cache file")); 198 case EZFS_ISL2CACHE: 199 return (dgettext(TEXT_DOMAIN, "device is in use as a cache")); 200 case EZFS_VDEVNOTSUP: 201 return (dgettext(TEXT_DOMAIN, "vdev specification is not " 202 "supported")); 203 case EZFS_NOTSUP: 204 return (dgettext(TEXT_DOMAIN, "operation not supported " 205 "on this dataset")); 206 case EZFS_ACTIVE_SPARE: 207 return (dgettext(TEXT_DOMAIN, "pool has active shared spare " 208 "device")); 209 case EZFS_UNPLAYED_LOGS: 210 return (dgettext(TEXT_DOMAIN, "log device has unplayed intent " 211 "logs")); 212 case EZFS_REFTAG_RELE: 213 return (dgettext(TEXT_DOMAIN, "no such tag on this dataset")); 214 case EZFS_REFTAG_HOLD: 215 return (dgettext(TEXT_DOMAIN, "tag already exists on this " 216 "dataset")); 217 case EZFS_TAGTOOLONG: 218 return (dgettext(TEXT_DOMAIN, "tag too long")); 219 case EZFS_PIPEFAILED: 220 return (dgettext(TEXT_DOMAIN, "pipe create failed")); 221 case EZFS_THREADCREATEFAILED: 222 return (dgettext(TEXT_DOMAIN, "thread create failed")); 223 case EZFS_POSTSPLIT_ONLINE: 224 return (dgettext(TEXT_DOMAIN, "disk was split from this pool " 225 "into a new one")); 226 case EZFS_UNKNOWN: 227 return (dgettext(TEXT_DOMAIN, "unknown error")); 228 default: 229 assert(hdl->libzfs_error == 0); 230 return (dgettext(TEXT_DOMAIN, "no error")); 231 } 232 } 233 234 /*PRINTFLIKE2*/ 235 void 236 zfs_error_aux(libzfs_handle_t *hdl, const char *fmt, ...) 237 { 238 va_list ap; 239 240 va_start(ap, fmt); 241 242 (void) vsnprintf(hdl->libzfs_desc, sizeof (hdl->libzfs_desc), 243 fmt, ap); 244 hdl->libzfs_desc_active = 1; 245 246 va_end(ap); 247 } 248 249 static void 250 zfs_verror(libzfs_handle_t *hdl, int error, const char *fmt, va_list ap) 251 { 252 (void) vsnprintf(hdl->libzfs_action, sizeof (hdl->libzfs_action), 253 fmt, ap); 254 hdl->libzfs_error = error; 255 256 if (hdl->libzfs_desc_active) 257 hdl->libzfs_desc_active = 0; 258 else 259 hdl->libzfs_desc[0] = '\0'; 260 261 if (hdl->libzfs_printerr) { 262 if (error == EZFS_UNKNOWN) { 263 (void) fprintf(stderr, dgettext(TEXT_DOMAIN, "internal " 264 "error: %s\n"), libzfs_error_description(hdl)); 265 abort(); 266 } 267 268 (void) fprintf(stderr, "%s: %s\n", hdl->libzfs_action, 269 libzfs_error_description(hdl)); 270 if (error == EZFS_NOMEM) 271 exit(1); 272 } 273 } 274 275 int 276 zfs_error(libzfs_handle_t *hdl, int error, const char *msg) 277 { 278 return (zfs_error_fmt(hdl, error, "%s", msg)); 279 } 280 281 /*PRINTFLIKE3*/ 282 int 283 zfs_error_fmt(libzfs_handle_t *hdl, int error, const char *fmt, ...) 284 { 285 va_list ap; 286 287 va_start(ap, fmt); 288 289 zfs_verror(hdl, error, fmt, ap); 290 291 va_end(ap); 292 293 return (-1); 294 } 295 296 static int 297 zfs_common_error(libzfs_handle_t *hdl, int error, const char *fmt, 298 va_list ap) 299 { 300 switch (error) { 301 case EPERM: 302 case EACCES: 303 zfs_verror(hdl, EZFS_PERM, fmt, ap); 304 return (-1); 305 306 case ECANCELED: 307 zfs_verror(hdl, EZFS_NODELEGATION, fmt, ap); 308 return (-1); 309 310 case EIO: 311 zfs_verror(hdl, EZFS_IO, fmt, ap); 312 return (-1); 313 314 case EINTR: 315 zfs_verror(hdl, EZFS_INTR, fmt, ap); 316 return (-1); 317 } 318 319 return (0); 320 } 321 322 int 323 zfs_standard_error(libzfs_handle_t *hdl, int error, const char *msg) 324 { 325 return (zfs_standard_error_fmt(hdl, error, "%s", msg)); 326 } 327 328 /*PRINTFLIKE3*/ 329 int 330 zfs_standard_error_fmt(libzfs_handle_t *hdl, int error, const char *fmt, ...) 331 { 332 va_list ap; 333 334 va_start(ap, fmt); 335 336 if (zfs_common_error(hdl, error, fmt, ap) != 0) { 337 va_end(ap); 338 return (-1); 339 } 340 341 switch (error) { 342 case ENXIO: 343 case ENODEV: 344 zfs_verror(hdl, EZFS_IO, fmt, ap); 345 break; 346 347 case ENOENT: 348 zfs_error_aux(hdl, dgettext(TEXT_DOMAIN, 349 "dataset does not exist")); 350 zfs_verror(hdl, EZFS_NOENT, fmt, ap); 351 break; 352 353 case ENOSPC: 354 case EDQUOT: 355 zfs_verror(hdl, EZFS_NOSPC, fmt, ap); 356 return (-1); 357 358 case EEXIST: 359 zfs_error_aux(hdl, dgettext(TEXT_DOMAIN, 360 "dataset already exists")); 361 zfs_verror(hdl, EZFS_EXISTS, fmt, ap); 362 break; 363 364 case EBUSY: 365 zfs_error_aux(hdl, dgettext(TEXT_DOMAIN, 366 "dataset is busy")); 367 zfs_verror(hdl, EZFS_BUSY, fmt, ap); 368 break; 369 case EROFS: 370 zfs_error_aux(hdl, dgettext(TEXT_DOMAIN, 371 "snapshot permissions cannot be modified")); 372 zfs_verror(hdl, EZFS_PERMRDONLY, fmt, ap); 373 break; 374 case ENAMETOOLONG: 375 zfs_verror(hdl, EZFS_NAMETOOLONG, fmt, ap); 376 break; 377 case ENOTSUP: 378 zfs_verror(hdl, EZFS_BADVERSION, fmt, ap); 379 break; 380 case EAGAIN: 381 zfs_error_aux(hdl, dgettext(TEXT_DOMAIN, 382 "pool I/O is currently suspended")); 383 zfs_verror(hdl, EZFS_POOLUNAVAIL, fmt, ap); 384 break; 385 default: 386 zfs_error_aux(hdl, strerror(error)); 387 zfs_verror(hdl, EZFS_UNKNOWN, fmt, ap); 388 break; 389 } 390 391 va_end(ap); 392 return (-1); 393 } 394 395 int 396 zpool_standard_error(libzfs_handle_t *hdl, int error, const char *msg) 397 { 398 return (zpool_standard_error_fmt(hdl, error, "%s", msg)); 399 } 400 401 /*PRINTFLIKE3*/ 402 int 403 zpool_standard_error_fmt(libzfs_handle_t *hdl, int error, const char *fmt, ...) 404 { 405 va_list ap; 406 407 va_start(ap, fmt); 408 409 if (zfs_common_error(hdl, error, fmt, ap) != 0) { 410 va_end(ap); 411 return (-1); 412 } 413 414 switch (error) { 415 case ENODEV: 416 zfs_verror(hdl, EZFS_NODEVICE, fmt, ap); 417 break; 418 419 case ENOENT: 420 zfs_error_aux(hdl, 421 dgettext(TEXT_DOMAIN, "no such pool or dataset")); 422 zfs_verror(hdl, EZFS_NOENT, fmt, ap); 423 break; 424 425 case EEXIST: 426 zfs_error_aux(hdl, dgettext(TEXT_DOMAIN, 427 "pool already exists")); 428 zfs_verror(hdl, EZFS_EXISTS, fmt, ap); 429 break; 430 431 case EBUSY: 432 zfs_error_aux(hdl, dgettext(TEXT_DOMAIN, "pool is busy")); 433 zfs_verror(hdl, EZFS_BUSY, fmt, ap); 434 break; 435 436 case ENXIO: 437 zfs_error_aux(hdl, dgettext(TEXT_DOMAIN, 438 "one or more devices is currently unavailable")); 439 zfs_verror(hdl, EZFS_BADDEV, fmt, ap); 440 break; 441 442 case ENAMETOOLONG: 443 zfs_verror(hdl, EZFS_DEVOVERFLOW, fmt, ap); 444 break; 445 446 case ENOTSUP: 447 zfs_verror(hdl, EZFS_POOL_NOTSUP, fmt, ap); 448 break; 449 450 case EINVAL: 451 zfs_verror(hdl, EZFS_POOL_INVALARG, fmt, ap); 452 break; 453 454 case ENOSPC: 455 case EDQUOT: 456 zfs_verror(hdl, EZFS_NOSPC, fmt, ap); 457 return (-1); 458 case EAGAIN: 459 zfs_error_aux(hdl, dgettext(TEXT_DOMAIN, 460 "pool I/O is currently suspended")); 461 zfs_verror(hdl, EZFS_POOLUNAVAIL, fmt, ap); 462 break; 463 464 default: 465 zfs_error_aux(hdl, strerror(error)); 466 zfs_verror(hdl, EZFS_UNKNOWN, fmt, ap); 467 } 468 469 va_end(ap); 470 return (-1); 471 } 472 473 /* 474 * Display an out of memory error message and abort the current program. 475 */ 476 int 477 no_memory(libzfs_handle_t *hdl) 478 { 479 return (zfs_error(hdl, EZFS_NOMEM, "internal error")); 480 } 481 482 /* 483 * A safe form of malloc() which will die if the allocation fails. 484 */ 485 void * 486 zfs_alloc(libzfs_handle_t *hdl, size_t size) 487 { 488 void *data; 489 490 if ((data = calloc(1, size)) == NULL) 491 (void) no_memory(hdl); 492 493 return (data); 494 } 495 496 /* 497 * A safe form of realloc(), which also zeroes newly allocated space. 498 */ 499 void * 500 zfs_realloc(libzfs_handle_t *hdl, void *ptr, size_t oldsize, size_t newsize) 501 { 502 void *ret; 503 504 if ((ret = realloc(ptr, newsize)) == NULL) { 505 (void) no_memory(hdl); 506 return (NULL); 507 } 508 509 bzero((char *)ret + oldsize, (newsize - oldsize)); 510 return (ret); 511 } 512 513 /* 514 * A safe form of strdup() which will die if the allocation fails. 515 */ 516 char * 517 zfs_strdup(libzfs_handle_t *hdl, const char *str) 518 { 519 char *ret; 520 521 if ((ret = strdup(str)) == NULL) 522 (void) no_memory(hdl); 523 524 return (ret); 525 } 526 527 /* 528 * Convert a number to an appropriately human-readable output. 529 */ 530 void 531 zfs_nicenum(uint64_t num, char *buf, size_t buflen) 532 { 533 uint64_t n = num; 534 int index = 0; 535 char u; 536 537 while (n >= 1024) { 538 n /= 1024; 539 index++; 540 } 541 542 u = " KMGTPE"[index]; 543 544 if (index == 0) { 545 (void) snprintf(buf, buflen, "%llu", n); 546 } else if ((num & ((1ULL << 10 * index) - 1)) == 0) { 547 /* 548 * If this is an even multiple of the base, always display 549 * without any decimal precision. 550 */ 551 (void) snprintf(buf, buflen, "%llu%c", n, u); 552 } else { 553 /* 554 * We want to choose a precision that reflects the best choice 555 * for fitting in 5 characters. This can get rather tricky when 556 * we have numbers that are very close to an order of magnitude. 557 * For example, when displaying 10239 (which is really 9.999K), 558 * we want only a single place of precision for 10.0K. We could 559 * develop some complex heuristics for this, but it's much 560 * easier just to try each combination in turn. 561 */ 562 int i; 563 for (i = 2; i >= 0; i--) { 564 if (snprintf(buf, buflen, "%.*f%c", i, 565 (double)num / (1ULL << 10 * index), u) <= 5) 566 break; 567 } 568 } 569 } 570 571 void 572 libzfs_print_on_error(libzfs_handle_t *hdl, boolean_t printerr) 573 { 574 hdl->libzfs_printerr = printerr; 575 } 576 577 libzfs_handle_t * 578 libzfs_init(void) 579 { 580 libzfs_handle_t *hdl; 581 582 if ((hdl = calloc(sizeof (libzfs_handle_t), 1)) == NULL) { 583 return (NULL); 584 } 585 586 if ((hdl->libzfs_fd = open(ZFS_DEV, O_RDWR)) < 0) { 587 free(hdl); 588 return (NULL); 589 } 590 591 if ((hdl->libzfs_mnttab = fopen(MNTTAB, "r")) == NULL) { 592 (void) close(hdl->libzfs_fd); 593 free(hdl); 594 return (NULL); 595 } 596 597 hdl->libzfs_sharetab = fopen("/etc/dfs/sharetab", "r"); 598 599 zfs_prop_init(); 600 zpool_prop_init(); 601 libzfs_mnttab_init(hdl); 602 603 return (hdl); 604 } 605 606 void 607 libzfs_fini(libzfs_handle_t *hdl) 608 { 609 (void) close(hdl->libzfs_fd); 610 if (hdl->libzfs_mnttab) 611 (void) fclose(hdl->libzfs_mnttab); 612 if (hdl->libzfs_sharetab) 613 (void) fclose(hdl->libzfs_sharetab); 614 zfs_uninit_libshare(hdl); 615 if (hdl->libzfs_log_str) 616 (void) free(hdl->libzfs_log_str); 617 zpool_free_handles(hdl); 618 #ifdef PORT_SOLARIS 619 libzfs_fru_clear(hdl, B_TRUE); 620 #endif 621 namespace_clear(hdl); 622 libzfs_mnttab_fini(hdl); 623 free(hdl); 624 } 625 626 libzfs_handle_t * 627 zpool_get_handle(zpool_handle_t *zhp) 628 { 629 return (zhp->zpool_hdl); 630 } 631 632 libzfs_handle_t * 633 zfs_get_handle(zfs_handle_t *zhp) 634 { 635 return (zhp->zfs_hdl); 636 } 637 638 zpool_handle_t * 639 zfs_get_pool_handle(const zfs_handle_t *zhp) 640 { 641 return (zhp->zpool_hdl); 642 } 643 644 /* 645 * Given a name, determine whether or not it's a valid path 646 * (starts with '/' or "./"). If so, walk the mnttab trying 647 * to match the device number. If not, treat the path as an 648 * fs/vol/snap name. 649 */ 650 zfs_handle_t * 651 zfs_path_to_zhandle(libzfs_handle_t *hdl, char *path, zfs_type_t argtype) 652 { 653 struct statvfs statbuf; 654 int ret; 655 656 if (path[0] != '/' && strncmp(path, "./", strlen("./")) != 0) { 657 /* 658 * It's not a valid path, assume it's a name of type 'argtype'. 659 */ 660 return (zfs_open(hdl, path, argtype)); 661 } 662 663 if (getstatfs(&statbuf, path) != 0) { 664 (void) fprintf(stderr, "%s: %s\n", path, strerror(errno)); 665 return (NULL); 666 } 667 668 if (strcmp(statbuf.f_fstypename, MNTTYPE_ZFS) != 0) { 669 (void) fprintf(stderr, gettext("'%s': not a ZFS filesystem\n"), 670 path); 671 return (NULL); 672 } 673 674 return (zfs_open(hdl, statbuf.f_mntfromname, ZFS_TYPE_FILESYSTEM)); 675 } 676 677 /* 678 * Initialize the zc_nvlist_dst member to prepare for receiving an nvlist from 679 * an ioctl(). 680 */ 681 int 682 zcmd_alloc_dst_nvlist(libzfs_handle_t *hdl, zfs_cmd_t *zc, size_t len) 683 { 684 if (len == 0) 685 len = 4*1024; 686 zc->zc_nvlist_dst_size = len; 687 if ((zc->zc_nvlist_dst = (uint64_t)(uintptr_t) 688 zfs_alloc(hdl, zc->zc_nvlist_dst_size)) == NULL) 689 return (-1); 690 691 return (0); 692 } 693 694 /* 695 * Called when an ioctl() which returns an nvlist fails with ENOMEM. This will 696 * expand the nvlist to the size specified in 'zc_nvlist_dst_size', which was 697 * filled in by the kernel to indicate the actual required size. 698 */ 699 int 700 zcmd_expand_dst_nvlist(libzfs_handle_t *hdl, zfs_cmd_t *zc) 701 { 702 free((void *)(uintptr_t)zc->zc_nvlist_dst); 703 if ((zc->zc_nvlist_dst = (uint64_t)(uintptr_t) 704 zfs_alloc(hdl, zc->zc_nvlist_dst_size)) 705 == NULL) 706 return (-1); 707 708 return (0); 709 } 710 711 /* 712 * Called to free the src and dst nvlists stored in the command structure. 713 */ 714 void 715 zcmd_free_nvlists(zfs_cmd_t *zc) 716 { 717 free((void *)(uintptr_t)zc->zc_nvlist_conf); 718 free((void *)(uintptr_t)zc->zc_nvlist_src); 719 free((void *)(uintptr_t)zc->zc_nvlist_dst); 720 } 721 722 static int 723 zcmd_write_nvlist_com(libzfs_handle_t *hdl, uint64_t *outnv, uint64_t *outlen, 724 nvlist_t *nvl) 725 { 726 char *packed; 727 size_t len; 728 729 verify(nvlist_size(nvl, &len, NV_ENCODE_NATIVE) == 0); 730 731 if ((packed = zfs_alloc(hdl, len)) == NULL) 732 return (-1); 733 734 verify(nvlist_pack(nvl, &packed, &len, NV_ENCODE_NATIVE, 0) == 0); 735 736 *outnv = (uint64_t)(uintptr_t)packed; 737 *outlen = len; 738 739 return (0); 740 } 741 742 int 743 zcmd_write_conf_nvlist(libzfs_handle_t *hdl, zfs_cmd_t *zc, nvlist_t *nvl) 744 { 745 return (zcmd_write_nvlist_com(hdl, &zc->zc_nvlist_conf, 746 &zc->zc_nvlist_conf_size, nvl)); 747 } 748 749 int 750 zcmd_write_src_nvlist(libzfs_handle_t *hdl, zfs_cmd_t *zc, nvlist_t *nvl) 751 { 752 return (zcmd_write_nvlist_com(hdl, &zc->zc_nvlist_src, 753 &zc->zc_nvlist_src_size, nvl)); 754 } 755 756 /* 757 * Unpacks an nvlist from the ZFS ioctl command structure. 758 */ 759 int 760 zcmd_read_dst_nvlist(libzfs_handle_t *hdl, zfs_cmd_t *zc, nvlist_t **nvlp) 761 { 762 if (nvlist_unpack((void *)(uintptr_t)zc->zc_nvlist_dst, 763 zc->zc_nvlist_dst_size, nvlp, 0) != 0) 764 return (no_memory(hdl)); 765 766 return (0); 767 } 768 769 int 770 zfs_ioctl(libzfs_handle_t *hdl, int request, zfs_cmd_t *zc) 771 { 772 int error; 773 774 zc->zc_history = (uint64_t)(uintptr_t)hdl->libzfs_log_str; 775 error = ioctl(hdl->libzfs_fd, request, zc); 776 if (hdl->libzfs_log_str) { 777 free(hdl->libzfs_log_str); 778 hdl->libzfs_log_str = NULL; 779 } 780 zc->zc_history = 0; 781 782 return (error); 783 } 784 785 /* 786 * ================================================================ 787 * API shared by zfs and zpool property management 788 * ================================================================ 789 */ 790 791 static void 792 zprop_print_headers(zprop_get_cbdata_t *cbp, zfs_type_t type) 793 { 794 zprop_list_t *pl = cbp->cb_proplist; 795 int i; 796 char *title; 797 size_t len; 798 799 cbp->cb_first = B_FALSE; 800 if (cbp->cb_scripted) 801 return; 802 803 /* 804 * Start with the length of the column headers. 805 */ 806 cbp->cb_colwidths[GET_COL_NAME] = strlen(dgettext(TEXT_DOMAIN, "NAME")); 807 cbp->cb_colwidths[GET_COL_PROPERTY] = strlen(dgettext(TEXT_DOMAIN, 808 "PROPERTY")); 809 cbp->cb_colwidths[GET_COL_VALUE] = strlen(dgettext(TEXT_DOMAIN, 810 "VALUE")); 811 cbp->cb_colwidths[GET_COL_RECVD] = strlen(dgettext(TEXT_DOMAIN, 812 "RECEIVED")); 813 cbp->cb_colwidths[GET_COL_SOURCE] = strlen(dgettext(TEXT_DOMAIN, 814 "SOURCE")); 815 816 /* first property is always NAME */ 817 assert(cbp->cb_proplist->pl_prop == 818 ((type == ZFS_TYPE_POOL) ? ZPOOL_PROP_NAME : ZFS_PROP_NAME)); 819 820 /* 821 * Go through and calculate the widths for each column. For the 822 * 'source' column, we kludge it up by taking the worst-case scenario of 823 * inheriting from the longest name. This is acceptable because in the 824 * majority of cases 'SOURCE' is the last column displayed, and we don't 825 * use the width anyway. Note that the 'VALUE' column can be oversized, 826 * if the name of the property is much longer than any values we find. 827 */ 828 for (pl = cbp->cb_proplist; pl != NULL; pl = pl->pl_next) { 829 /* 830 * 'PROPERTY' column 831 */ 832 if (pl->pl_prop != ZPROP_INVAL) { 833 const char *propname = (type == ZFS_TYPE_POOL) ? 834 zpool_prop_to_name(pl->pl_prop) : 835 zfs_prop_to_name(pl->pl_prop); 836 837 len = strlen(propname); 838 if (len > cbp->cb_colwidths[GET_COL_PROPERTY]) 839 cbp->cb_colwidths[GET_COL_PROPERTY] = len; 840 } else { 841 len = strlen(pl->pl_user_prop); 842 if (len > cbp->cb_colwidths[GET_COL_PROPERTY]) 843 cbp->cb_colwidths[GET_COL_PROPERTY] = len; 844 } 845 846 /* 847 * 'VALUE' column. The first property is always the 'name' 848 * property that was tacked on either by /sbin/zfs's 849 * zfs_do_get() or when calling zprop_expand_list(), so we 850 * ignore its width. If the user specified the name property 851 * to display, then it will be later in the list in any case. 852 */ 853 if (pl != cbp->cb_proplist && 854 pl->pl_width > cbp->cb_colwidths[GET_COL_VALUE]) 855 cbp->cb_colwidths[GET_COL_VALUE] = pl->pl_width; 856 857 /* 'RECEIVED' column. */ 858 if (pl != cbp->cb_proplist && 859 pl->pl_recvd_width > cbp->cb_colwidths[GET_COL_RECVD]) 860 cbp->cb_colwidths[GET_COL_RECVD] = pl->pl_recvd_width; 861 862 /* 863 * 'NAME' and 'SOURCE' columns 864 */ 865 if (pl->pl_prop == (type == ZFS_TYPE_POOL ? ZPOOL_PROP_NAME : 866 ZFS_PROP_NAME) && 867 pl->pl_width > cbp->cb_colwidths[GET_COL_NAME]) { 868 cbp->cb_colwidths[GET_COL_NAME] = pl->pl_width; 869 cbp->cb_colwidths[GET_COL_SOURCE] = pl->pl_width + 870 strlen(dgettext(TEXT_DOMAIN, "inherited from")); 871 } 872 } 873 874 /* 875 * Now go through and print the headers. 876 */ 877 for (i = 0; i < ZFS_GET_NCOLS; i++) { 878 switch (cbp->cb_columns[i]) { 879 case GET_COL_NAME: 880 title = dgettext(TEXT_DOMAIN, "NAME"); 881 break; 882 case GET_COL_PROPERTY: 883 title = dgettext(TEXT_DOMAIN, "PROPERTY"); 884 break; 885 case GET_COL_VALUE: 886 title = dgettext(TEXT_DOMAIN, "VALUE"); 887 break; 888 case GET_COL_RECVD: 889 title = dgettext(TEXT_DOMAIN, "RECEIVED"); 890 break; 891 case GET_COL_SOURCE: 892 title = dgettext(TEXT_DOMAIN, "SOURCE"); 893 break; 894 default: 895 title = NULL; 896 } 897 898 if (title != NULL) { 899 if (i == (ZFS_GET_NCOLS - 1) || 900 cbp->cb_columns[i + 1] == GET_COL_NONE) 901 (void) printf("%s", title); 902 else 903 (void) printf("%-*s ", 904 cbp->cb_colwidths[cbp->cb_columns[i]], 905 title); 906 } 907 } 908 (void) printf("\n"); 909 } 910 911 /* 912 * Display a single line of output, according to the settings in the callback 913 * structure. 914 */ 915 void 916 zprop_print_one_property(const char *name, zprop_get_cbdata_t *cbp, 917 const char *propname, const char *value, zprop_source_t sourcetype, 918 const char *source, const char *recvd_value) 919 { 920 int i; 921 const char *str; 922 char buf[128]; 923 924 /* 925 * Ignore those source types that the user has chosen to ignore. 926 */ 927 if ((sourcetype & cbp->cb_sources) == 0) 928 return; 929 930 if (cbp->cb_first) 931 zprop_print_headers(cbp, cbp->cb_type); 932 933 for (i = 0; i < ZFS_GET_NCOLS; i++) { 934 switch (cbp->cb_columns[i]) { 935 case GET_COL_NAME: 936 str = name; 937 break; 938 939 case GET_COL_PROPERTY: 940 str = propname; 941 break; 942 943 case GET_COL_VALUE: 944 str = value; 945 break; 946 947 case GET_COL_SOURCE: 948 switch (sourcetype) { 949 case ZPROP_SRC_NONE: 950 str = "-"; 951 break; 952 953 case ZPROP_SRC_DEFAULT: 954 str = "default"; 955 break; 956 957 case ZPROP_SRC_LOCAL: 958 str = "local"; 959 break; 960 961 case ZPROP_SRC_TEMPORARY: 962 str = "temporary"; 963 break; 964 965 case ZPROP_SRC_INHERITED: 966 (void) snprintf(buf, sizeof (buf), 967 "inherited from %s", source); 968 str = buf; 969 break; 970 case ZPROP_SRC_RECEIVED: 971 str = "received"; 972 break; 973 } 974 break; 975 976 case GET_COL_RECVD: 977 str = (recvd_value == NULL ? "-" : recvd_value); 978 break; 979 980 default: 981 continue; 982 } 983 984 if (cbp->cb_columns[i + 1] == GET_COL_NONE) 985 (void) printf("%s", str); 986 else if (cbp->cb_scripted) 987 (void) printf("%s\t", str); 988 else 989 (void) printf("%-*s ", 990 cbp->cb_colwidths[cbp->cb_columns[i]], 991 str); 992 } 993 994 (void) printf("\n"); 995 } 996 997 /* 998 * Given a numeric suffix, convert the value into a number of bits that the 999 * resulting value must be shifted. 1000 */ 1001 static int 1002 str2shift(libzfs_handle_t *hdl, const char *buf) 1003 { 1004 const char *ends = "BKMGTPEZ"; 1005 int i; 1006 1007 if (buf[0] == '\0') 1008 return (0); 1009 for (i = 0; i < strlen(ends); i++) { 1010 if (toupper(buf[0]) == ends[i]) 1011 break; 1012 } 1013 if (i == strlen(ends)) { 1014 zfs_error_aux(hdl, dgettext(TEXT_DOMAIN, 1015 "invalid numeric suffix '%s'"), buf); 1016 return (-1); 1017 } 1018 1019 /* 1020 * We want to allow trailing 'b' characters for 'GB' or 'Mb'. But don't 1021 * allow 'BB' - that's just weird. 1022 */ 1023 if (buf[1] == '\0' || (toupper(buf[1]) == 'B' && buf[2] == '\0' && 1024 toupper(buf[0]) != 'B')) 1025 return (10*i); 1026 1027 zfs_error_aux(hdl, dgettext(TEXT_DOMAIN, 1028 "invalid numeric suffix '%s'"), buf); 1029 return (-1); 1030 } 1031 1032 /* 1033 * Convert a string of the form '100G' into a real number. Used when setting 1034 * properties or creating a volume. 'buf' is used to place an extended error 1035 * message for the caller to use. 1036 */ 1037 int 1038 zfs_nicestrtonum(libzfs_handle_t *hdl, const char *value, uint64_t *num) 1039 { 1040 char *end; 1041 int shift; 1042 1043 *num = 0; 1044 1045 /* Check to see if this looks like a number. */ 1046 if ((value[0] < '0' || value[0] > '9') && value[0] != '.') { 1047 if (hdl) 1048 zfs_error_aux(hdl, dgettext(TEXT_DOMAIN, 1049 "bad numeric value '%s'"), value); 1050 return (-1); 1051 } 1052 1053 /* Rely on strtoull() to process the numeric portion. */ 1054 errno = 0; 1055 *num = strtoull(value, &end, 10); 1056 1057 /* 1058 * Check for ERANGE, which indicates that the value is too large to fit 1059 * in a 64-bit value. 1060 */ 1061 if (errno == ERANGE) { 1062 if (hdl) 1063 zfs_error_aux(hdl, dgettext(TEXT_DOMAIN, 1064 "numeric value is too large")); 1065 return (-1); 1066 } 1067 1068 /* 1069 * If we have a decimal value, then do the computation with floating 1070 * point arithmetic. Otherwise, use standard arithmetic. 1071 */ 1072 if (*end == '.') { 1073 double fval = strtod(value, &end); 1074 1075 if ((shift = str2shift(hdl, end)) == -1) 1076 return (-1); 1077 1078 fval *= pow(2, shift); 1079 1080 if (fval > UINT64_MAX) { 1081 if (hdl) 1082 zfs_error_aux(hdl, dgettext(TEXT_DOMAIN, 1083 "numeric value is too large")); 1084 return (-1); 1085 } 1086 1087 *num = (uint64_t)fval; 1088 } else { 1089 if ((shift = str2shift(hdl, end)) == -1) 1090 return (-1); 1091 1092 /* Check for overflow */ 1093 if (shift >= 64 || (*num << shift) >> shift != *num) { 1094 if (hdl) 1095 zfs_error_aux(hdl, dgettext(TEXT_DOMAIN, 1096 "numeric value is too large")); 1097 return (-1); 1098 } 1099 1100 *num <<= shift; 1101 } 1102 1103 return (0); 1104 } 1105 1106 /* 1107 * Given a propname=value nvpair to set, parse any numeric properties 1108 * (index, boolean, etc) if they are specified as strings and add the 1109 * resulting nvpair to the returned nvlist. 1110 * 1111 * At the DSL layer, all properties are either 64-bit numbers or strings. 1112 * We want the user to be able to ignore this fact and specify properties 1113 * as native values (numbers, for example) or as strings (to simplify 1114 * command line utilities). This also handles converting index types 1115 * (compression, checksum, etc) from strings to their on-disk index. 1116 */ 1117 int 1118 zprop_parse_value(libzfs_handle_t *hdl, nvpair_t *elem, int prop, 1119 zfs_type_t type, nvlist_t *ret, char **svalp, uint64_t *ivalp, 1120 const char *errbuf) 1121 { 1122 data_type_t datatype = nvpair_type(elem); 1123 zprop_type_t proptype; 1124 const char *propname; 1125 char *value; 1126 boolean_t isnone = B_FALSE; 1127 1128 if (type == ZFS_TYPE_POOL) { 1129 proptype = zpool_prop_get_type(prop); 1130 propname = zpool_prop_to_name(prop); 1131 } else { 1132 proptype = zfs_prop_get_type(prop); 1133 propname = zfs_prop_to_name(prop); 1134 } 1135 1136 /* 1137 * Convert any properties to the internal DSL value types. 1138 */ 1139 *svalp = NULL; 1140 *ivalp = 0; 1141 1142 switch (proptype) { 1143 case PROP_TYPE_STRING: 1144 if (datatype != DATA_TYPE_STRING) { 1145 zfs_error_aux(hdl, dgettext(TEXT_DOMAIN, 1146 "'%s' must be a string"), nvpair_name(elem)); 1147 goto error; 1148 } 1149 (void) nvpair_value_string(elem, svalp); 1150 if (strlen(*svalp) >= ZFS_MAXPROPLEN) { 1151 zfs_error_aux(hdl, dgettext(TEXT_DOMAIN, 1152 "'%s' is too long"), nvpair_name(elem)); 1153 goto error; 1154 } 1155 break; 1156 1157 case PROP_TYPE_NUMBER: 1158 if (datatype == DATA_TYPE_STRING) { 1159 (void) nvpair_value_string(elem, &value); 1160 if (strcmp(value, "none") == 0) { 1161 isnone = B_TRUE; 1162 } else if (zfs_nicestrtonum(hdl, value, ivalp) 1163 != 0) { 1164 goto error; 1165 } 1166 } else if (datatype == DATA_TYPE_UINT64) { 1167 (void) nvpair_value_uint64(elem, ivalp); 1168 } else { 1169 zfs_error_aux(hdl, dgettext(TEXT_DOMAIN, 1170 "'%s' must be a number"), nvpair_name(elem)); 1171 goto error; 1172 } 1173 1174 /* 1175 * Quota special: force 'none' and don't allow 0. 1176 */ 1177 if ((type & ZFS_TYPE_DATASET) && *ivalp == 0 && !isnone && 1178 (prop == ZFS_PROP_QUOTA || prop == ZFS_PROP_REFQUOTA)) { 1179 zfs_error_aux(hdl, dgettext(TEXT_DOMAIN, 1180 "use 'none' to disable quota/refquota")); 1181 goto error; 1182 } 1183 break; 1184 1185 case PROP_TYPE_INDEX: 1186 if (datatype != DATA_TYPE_STRING) { 1187 zfs_error_aux(hdl, dgettext(TEXT_DOMAIN, 1188 "'%s' must be a string"), nvpair_name(elem)); 1189 goto error; 1190 } 1191 1192 (void) nvpair_value_string(elem, &value); 1193 1194 if (zprop_string_to_index(prop, value, ivalp, type) != 0) { 1195 zfs_error_aux(hdl, dgettext(TEXT_DOMAIN, 1196 "'%s' must be one of '%s'"), propname, 1197 zprop_values(prop, type)); 1198 goto error; 1199 } 1200 break; 1201 1202 default: 1203 abort(); 1204 } 1205 1206 /* 1207 * Add the result to our return set of properties. 1208 */ 1209 if (*svalp != NULL) { 1210 if (nvlist_add_string(ret, propname, *svalp) != 0) { 1211 (void) no_memory(hdl); 1212 return (-1); 1213 } 1214 } else { 1215 if (nvlist_add_uint64(ret, propname, *ivalp) != 0) { 1216 (void) no_memory(hdl); 1217 return (-1); 1218 } 1219 } 1220 1221 return (0); 1222 error: 1223 (void) zfs_error(hdl, EZFS_BADPROP, errbuf); 1224 return (-1); 1225 } 1226 1227 static int 1228 addlist(libzfs_handle_t *hdl, char *propname, zprop_list_t **listp, 1229 zfs_type_t type) 1230 { 1231 int prop; 1232 zprop_list_t *entry; 1233 1234 prop = zprop_name_to_prop(propname, type); 1235 1236 if (prop != ZPROP_INVAL && !zprop_valid_for_type(prop, type)) 1237 prop = ZPROP_INVAL; 1238 1239 /* 1240 * When no property table entry can be found, return failure if 1241 * this is a pool property or if this isn't a user-defined 1242 * dataset property, 1243 */ 1244 if (prop == ZPROP_INVAL && (type == ZFS_TYPE_POOL || 1245 (!zfs_prop_user(propname) && !zfs_prop_userquota(propname)))) { 1246 zfs_error_aux(hdl, dgettext(TEXT_DOMAIN, 1247 "invalid property '%s'"), propname); 1248 return (zfs_error(hdl, EZFS_BADPROP, 1249 dgettext(TEXT_DOMAIN, "bad property list"))); 1250 } 1251 1252 if ((entry = zfs_alloc(hdl, sizeof (zprop_list_t))) == NULL) 1253 return (-1); 1254 1255 entry->pl_prop = prop; 1256 if (prop == ZPROP_INVAL) { 1257 if ((entry->pl_user_prop = zfs_strdup(hdl, propname)) == NULL) { 1258 free(entry); 1259 return (-1); 1260 } 1261 entry->pl_width = strlen(propname); 1262 } else { 1263 entry->pl_width = zprop_width(prop, &entry->pl_fixed, 1264 type); 1265 } 1266 1267 *listp = entry; 1268 1269 return (0); 1270 } 1271 1272 /* 1273 * Given a comma-separated list of properties, construct a property list 1274 * containing both user-defined and native properties. This function will 1275 * return a NULL list if 'all' is specified, which can later be expanded 1276 * by zprop_expand_list(). 1277 */ 1278 int 1279 zprop_get_list(libzfs_handle_t *hdl, char *props, zprop_list_t **listp, 1280 zfs_type_t type) 1281 { 1282 *listp = NULL; 1283 1284 /* 1285 * If 'all' is specified, return a NULL list. 1286 */ 1287 if (strcmp(props, "all") == 0) 1288 return (0); 1289 1290 /* 1291 * If no props were specified, return an error. 1292 */ 1293 if (props[0] == '\0') { 1294 zfs_error_aux(hdl, dgettext(TEXT_DOMAIN, 1295 "no properties specified")); 1296 return (zfs_error(hdl, EZFS_BADPROP, dgettext(TEXT_DOMAIN, 1297 "bad property list"))); 1298 } 1299 1300 /* 1301 * It would be nice to use getsubopt() here, but the inclusion of column 1302 * aliases makes this more effort than it's worth. 1303 */ 1304 while (*props != '\0') { 1305 size_t len; 1306 char *p; 1307 char c; 1308 1309 if ((p = strchr(props, ',')) == NULL) { 1310 len = strlen(props); 1311 p = props + len; 1312 } else { 1313 len = p - props; 1314 } 1315 1316 /* 1317 * Check for empty options. 1318 */ 1319 if (len == 0) { 1320 zfs_error_aux(hdl, dgettext(TEXT_DOMAIN, 1321 "empty property name")); 1322 return (zfs_error(hdl, EZFS_BADPROP, 1323 dgettext(TEXT_DOMAIN, "bad property list"))); 1324 } 1325 1326 /* 1327 * Check all regular property names. 1328 */ 1329 c = props[len]; 1330 props[len] = '\0'; 1331 1332 if (strcmp(props, "space") == 0) { 1333 static char *spaceprops[] = { 1334 "name", "avail", "used", "usedbysnapshots", 1335 "usedbydataset", "usedbyrefreservation", 1336 "usedbychildren", NULL 1337 }; 1338 int i; 1339 1340 for (i = 0; spaceprops[i]; i++) { 1341 if (addlist(hdl, spaceprops[i], listp, type)) 1342 return (-1); 1343 listp = &(*listp)->pl_next; 1344 } 1345 } else { 1346 if (addlist(hdl, props, listp, type)) 1347 return (-1); 1348 listp = &(*listp)->pl_next; 1349 } 1350 1351 props = p; 1352 if (c == ',') 1353 props++; 1354 } 1355 1356 return (0); 1357 } 1358 1359 void 1360 zprop_free_list(zprop_list_t *pl) 1361 { 1362 zprop_list_t *next; 1363 1364 while (pl != NULL) { 1365 next = pl->pl_next; 1366 free(pl->pl_user_prop); 1367 free(pl); 1368 pl = next; 1369 } 1370 } 1371 1372 typedef struct expand_data { 1373 zprop_list_t **last; 1374 libzfs_handle_t *hdl; 1375 zfs_type_t type; 1376 } expand_data_t; 1377 1378 int 1379 zprop_expand_list_cb(int prop, void *cb) 1380 { 1381 zprop_list_t *entry; 1382 expand_data_t *edp = cb; 1383 1384 if ((entry = zfs_alloc(edp->hdl, sizeof (zprop_list_t))) == NULL) 1385 return (ZPROP_INVAL); 1386 1387 entry->pl_prop = prop; 1388 entry->pl_width = zprop_width(prop, &entry->pl_fixed, edp->type); 1389 entry->pl_all = B_TRUE; 1390 1391 *(edp->last) = entry; 1392 edp->last = &entry->pl_next; 1393 1394 return (ZPROP_CONT); 1395 } 1396 1397 int 1398 zprop_expand_list(libzfs_handle_t *hdl, zprop_list_t **plp, zfs_type_t type) 1399 { 1400 zprop_list_t *entry; 1401 zprop_list_t **last; 1402 expand_data_t exp; 1403 1404 if (*plp == NULL) { 1405 /* 1406 * If this is the very first time we've been called for an 'all' 1407 * specification, expand the list to include all native 1408 * properties. 1409 */ 1410 last = plp; 1411 1412 exp.last = last; 1413 exp.hdl = hdl; 1414 exp.type = type; 1415 1416 if (zprop_iter_common(zprop_expand_list_cb, &exp, B_FALSE, 1417 B_FALSE, type) == ZPROP_INVAL) 1418 return (-1); 1419 1420 /* 1421 * Add 'name' to the beginning of the list, which is handled 1422 * specially. 1423 */ 1424 if ((entry = zfs_alloc(hdl, sizeof (zprop_list_t))) == NULL) 1425 return (-1); 1426 1427 entry->pl_prop = (type == ZFS_TYPE_POOL) ? ZPOOL_PROP_NAME : 1428 ZFS_PROP_NAME; 1429 entry->pl_width = zprop_width(entry->pl_prop, 1430 &entry->pl_fixed, type); 1431 entry->pl_all = B_TRUE; 1432 entry->pl_next = *plp; 1433 *plp = entry; 1434 } 1435 return (0); 1436 } 1437 1438 int 1439 zprop_iter(zprop_func func, void *cb, boolean_t show_all, boolean_t ordered, 1440 zfs_type_t type) 1441 { 1442 return (zprop_iter_common(func, cb, show_all, ordered, type)); 1443 } 1444