libzfs_pool.c revision 2a6b87f07ac0c0b819179c84afe5a60afa04cfa5
fa9e4066f08beec538e775443c5be79dd423fcabahrens/*
fa9e4066f08beec538e775443c5be79dd423fcabahrens * CDDL HEADER START
fa9e4066f08beec538e775443c5be79dd423fcabahrens *
fa9e4066f08beec538e775443c5be79dd423fcabahrens * The contents of this file are subject to the terms of the
441d80aa4f613b6298fc8bd3151f4be02dbf84fclling * Common Development and Distribution License (the "License").
441d80aa4f613b6298fc8bd3151f4be02dbf84fclling * You may not use this file except in compliance with the License.
fa9e4066f08beec538e775443c5be79dd423fcabahrens *
fa9e4066f08beec538e775443c5be79dd423fcabahrens * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
fa9e4066f08beec538e775443c5be79dd423fcabahrens * or http://www.opensolaris.org/os/licensing.
fa9e4066f08beec538e775443c5be79dd423fcabahrens * See the License for the specific language governing permissions
fa9e4066f08beec538e775443c5be79dd423fcabahrens * and limitations under the License.
fa9e4066f08beec538e775443c5be79dd423fcabahrens *
fa9e4066f08beec538e775443c5be79dd423fcabahrens * When distributing Covered Code, include this CDDL HEADER in each
fa9e4066f08beec538e775443c5be79dd423fcabahrens * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
fa9e4066f08beec538e775443c5be79dd423fcabahrens * If applicable, add the following below this CDDL HEADER, with the
fa9e4066f08beec538e775443c5be79dd423fcabahrens * fields enclosed by brackets "[]" replaced with your own identifying
fa9e4066f08beec538e775443c5be79dd423fcabahrens * information: Portions Copyright [yyyy] [name of copyright owner]
fa9e4066f08beec538e775443c5be79dd423fcabahrens *
fa9e4066f08beec538e775443c5be79dd423fcabahrens * CDDL HEADER END
fa9e4066f08beec538e775443c5be79dd423fcabahrens */
99653d4ee642c6528e88224f12409a5f23060994eschrock
fa9e4066f08beec538e775443c5be79dd423fcabahrens/*
39c23413b8df94a95f67b34cfd4a4dfc3fd0b48deschrock * Copyright 2007 Sun Microsystems, Inc. All rights reserved.
fa9e4066f08beec538e775443c5be79dd423fcabahrens * Use is subject to license terms.
fa9e4066f08beec538e775443c5be79dd423fcabahrens */
fa9e4066f08beec538e775443c5be79dd423fcabahrens
fa9e4066f08beec538e775443c5be79dd423fcabahrens#pragma ident "%Z%%M% %I% %E% SMI"
fa9e4066f08beec538e775443c5be79dd423fcabahrens
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl#include <alloca.h>
fa9e4066f08beec538e775443c5be79dd423fcabahrens#include <assert.h>
fa9e4066f08beec538e775443c5be79dd423fcabahrens#include <ctype.h>
fa9e4066f08beec538e775443c5be79dd423fcabahrens#include <errno.h>
fa9e4066f08beec538e775443c5be79dd423fcabahrens#include <devid.h>
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl#include <dirent.h>
fa9e4066f08beec538e775443c5be79dd423fcabahrens#include <fcntl.h>
fa9e4066f08beec538e775443c5be79dd423fcabahrens#include <libintl.h>
fa9e4066f08beec538e775443c5be79dd423fcabahrens#include <stdio.h>
fa9e4066f08beec538e775443c5be79dd423fcabahrens#include <stdlib.h>
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl#include <strings.h>
fa9e4066f08beec538e775443c5be79dd423fcabahrens#include <unistd.h>
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor#include <sys/efi_partition.h>
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor#include <sys/vtoc.h>
fa9e4066f08beec538e775443c5be79dd423fcabahrens#include <sys/zfs_ioctl.h>
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock#include <sys/zio.h>
06eeb2ad640ce72d394ac521094bed7681044408ek#include <strings.h>
fa9e4066f08beec538e775443c5be79dd423fcabahrens
fa9e4066f08beec538e775443c5be79dd423fcabahrens#include "zfs_namecheck.h"
b1b8ab34de515a5e83206da22c3d7e563241b021lling#include "zfs_prop.h"
fa9e4066f08beec538e775443c5be79dd423fcabahrens#include "libzfs_impl.h"
fa9e4066f08beec538e775443c5be79dd423fcabahrens
fa9e4066f08beec538e775443c5be79dd423fcabahrens/*
fa9e4066f08beec538e775443c5be79dd423fcabahrens * Validate the given pool name, optionally putting an extended error message in
fa9e4066f08beec538e775443c5be79dd423fcabahrens * 'buf'.
fa9e4066f08beec538e775443c5be79dd423fcabahrens */
99653d4ee642c6528e88224f12409a5f23060994eschrockstatic boolean_t
99653d4ee642c6528e88224f12409a5f23060994eschrockzpool_name_valid(libzfs_handle_t *hdl, boolean_t isopen, const char *pool)
fa9e4066f08beec538e775443c5be79dd423fcabahrens{
fa9e4066f08beec538e775443c5be79dd423fcabahrens namecheck_err_t why;
fa9e4066f08beec538e775443c5be79dd423fcabahrens char what;
b468a217b67dc26ce21da5d5a2ca09bb6249e4faeschrock int ret;
b468a217b67dc26ce21da5d5a2ca09bb6249e4faeschrock
b468a217b67dc26ce21da5d5a2ca09bb6249e4faeschrock ret = pool_namecheck(pool, &why, &what);
b468a217b67dc26ce21da5d5a2ca09bb6249e4faeschrock
b468a217b67dc26ce21da5d5a2ca09bb6249e4faeschrock /*
b468a217b67dc26ce21da5d5a2ca09bb6249e4faeschrock * The rules for reserved pool names were extended at a later point.
b468a217b67dc26ce21da5d5a2ca09bb6249e4faeschrock * But we need to support users with existing pools that may now be
b468a217b67dc26ce21da5d5a2ca09bb6249e4faeschrock * invalid. So we only check for this expanded set of names during a
b468a217b67dc26ce21da5d5a2ca09bb6249e4faeschrock * create (or import), and only in userland.
b468a217b67dc26ce21da5d5a2ca09bb6249e4faeschrock */
b468a217b67dc26ce21da5d5a2ca09bb6249e4faeschrock if (ret == 0 && !isopen &&
b468a217b67dc26ce21da5d5a2ca09bb6249e4faeschrock (strncmp(pool, "mirror", 6) == 0 ||
b468a217b67dc26ce21da5d5a2ca09bb6249e4faeschrock strncmp(pool, "raidz", 5) == 0 ||
8654d0253136055bd4cc2423d87378e8a37f2eb5perrin strncmp(pool, "spare", 5) == 0 ||
8654d0253136055bd4cc2423d87378e8a37f2eb5perrin strcmp(pool, "log") == 0)) {
99653d4ee642c6528e88224f12409a5f23060994eschrock zfs_error_aux(hdl,
99653d4ee642c6528e88224f12409a5f23060994eschrock dgettext(TEXT_DOMAIN, "name is reserved"));
99653d4ee642c6528e88224f12409a5f23060994eschrock return (B_FALSE);
b468a217b67dc26ce21da5d5a2ca09bb6249e4faeschrock }
b468a217b67dc26ce21da5d5a2ca09bb6249e4faeschrock
fa9e4066f08beec538e775443c5be79dd423fcabahrens
b468a217b67dc26ce21da5d5a2ca09bb6249e4faeschrock if (ret != 0) {
99653d4ee642c6528e88224f12409a5f23060994eschrock if (hdl != NULL) {
fa9e4066f08beec538e775443c5be79dd423fcabahrens switch (why) {
b81d61a68b235e0529ebadc18e14d9d1dd52a258lling case NAME_ERR_TOOLONG:
99653d4ee642c6528e88224f12409a5f23060994eschrock zfs_error_aux(hdl,
b81d61a68b235e0529ebadc18e14d9d1dd52a258lling dgettext(TEXT_DOMAIN, "name is too long"));
b81d61a68b235e0529ebadc18e14d9d1dd52a258lling break;
b81d61a68b235e0529ebadc18e14d9d1dd52a258lling
fa9e4066f08beec538e775443c5be79dd423fcabahrens case NAME_ERR_INVALCHAR:
99653d4ee642c6528e88224f12409a5f23060994eschrock zfs_error_aux(hdl,
fa9e4066f08beec538e775443c5be79dd423fcabahrens dgettext(TEXT_DOMAIN, "invalid character "
fa9e4066f08beec538e775443c5be79dd423fcabahrens "'%c' in pool name"), what);
fa9e4066f08beec538e775443c5be79dd423fcabahrens break;
fa9e4066f08beec538e775443c5be79dd423fcabahrens
fa9e4066f08beec538e775443c5be79dd423fcabahrens case NAME_ERR_NOLETTER:
99653d4ee642c6528e88224f12409a5f23060994eschrock zfs_error_aux(hdl, dgettext(TEXT_DOMAIN,
99653d4ee642c6528e88224f12409a5f23060994eschrock "name must begin with a letter"));
fa9e4066f08beec538e775443c5be79dd423fcabahrens break;
fa9e4066f08beec538e775443c5be79dd423fcabahrens
fa9e4066f08beec538e775443c5be79dd423fcabahrens case NAME_ERR_RESERVED:
99653d4ee642c6528e88224f12409a5f23060994eschrock zfs_error_aux(hdl, dgettext(TEXT_DOMAIN,
99653d4ee642c6528e88224f12409a5f23060994eschrock "name is reserved"));
fa9e4066f08beec538e775443c5be79dd423fcabahrens break;
fa9e4066f08beec538e775443c5be79dd423fcabahrens
fa9e4066f08beec538e775443c5be79dd423fcabahrens case NAME_ERR_DISKLIKE:
99653d4ee642c6528e88224f12409a5f23060994eschrock zfs_error_aux(hdl, dgettext(TEXT_DOMAIN,
99653d4ee642c6528e88224f12409a5f23060994eschrock "pool name is reserved"));
fa9e4066f08beec538e775443c5be79dd423fcabahrens break;
5ad820458efd0fdb914baff9c1447c22b819fa23nd
5ad820458efd0fdb914baff9c1447c22b819fa23nd case NAME_ERR_LEADING_SLASH:
5ad820458efd0fdb914baff9c1447c22b819fa23nd zfs_error_aux(hdl, dgettext(TEXT_DOMAIN,
5ad820458efd0fdb914baff9c1447c22b819fa23nd "leading slash in name"));
5ad820458efd0fdb914baff9c1447c22b819fa23nd break;
5ad820458efd0fdb914baff9c1447c22b819fa23nd
5ad820458efd0fdb914baff9c1447c22b819fa23nd case NAME_ERR_EMPTY_COMPONENT:
5ad820458efd0fdb914baff9c1447c22b819fa23nd zfs_error_aux(hdl, dgettext(TEXT_DOMAIN,
5ad820458efd0fdb914baff9c1447c22b819fa23nd "empty component in name"));
5ad820458efd0fdb914baff9c1447c22b819fa23nd break;
5ad820458efd0fdb914baff9c1447c22b819fa23nd
5ad820458efd0fdb914baff9c1447c22b819fa23nd case NAME_ERR_TRAILING_SLASH:
5ad820458efd0fdb914baff9c1447c22b819fa23nd zfs_error_aux(hdl, dgettext(TEXT_DOMAIN,
5ad820458efd0fdb914baff9c1447c22b819fa23nd "trailing slash in name"));
5ad820458efd0fdb914baff9c1447c22b819fa23nd break;
5ad820458efd0fdb914baff9c1447c22b819fa23nd
5ad820458efd0fdb914baff9c1447c22b819fa23nd case NAME_ERR_MULTIPLE_AT:
5ad820458efd0fdb914baff9c1447c22b819fa23nd zfs_error_aux(hdl, dgettext(TEXT_DOMAIN,
5ad820458efd0fdb914baff9c1447c22b819fa23nd "multiple '@' delimiters in name"));
5ad820458efd0fdb914baff9c1447c22b819fa23nd break;
5ad820458efd0fdb914baff9c1447c22b819fa23nd
fa9e4066f08beec538e775443c5be79dd423fcabahrens }
fa9e4066f08beec538e775443c5be79dd423fcabahrens }
99653d4ee642c6528e88224f12409a5f23060994eschrock return (B_FALSE);
fa9e4066f08beec538e775443c5be79dd423fcabahrens }
fa9e4066f08beec538e775443c5be79dd423fcabahrens
99653d4ee642c6528e88224f12409a5f23060994eschrock return (B_TRUE);
fa9e4066f08beec538e775443c5be79dd423fcabahrens}
fa9e4066f08beec538e775443c5be79dd423fcabahrens
b1b8ab34de515a5e83206da22c3d7e563241b021llingstatic int
b1b8ab34de515a5e83206da22c3d7e563241b021llingzpool_get_all_props(zpool_handle_t *zhp)
b1b8ab34de515a5e83206da22c3d7e563241b021lling{
b1b8ab34de515a5e83206da22c3d7e563241b021lling zfs_cmd_t zc = { 0 };
b1b8ab34de515a5e83206da22c3d7e563241b021lling libzfs_handle_t *hdl = zhp->zpool_hdl;
b1b8ab34de515a5e83206da22c3d7e563241b021lling
b1b8ab34de515a5e83206da22c3d7e563241b021lling (void) strlcpy(zc.zc_name, zhp->zpool_name, sizeof (zc.zc_name));
b1b8ab34de515a5e83206da22c3d7e563241b021lling
b1b8ab34de515a5e83206da22c3d7e563241b021lling if (zcmd_alloc_dst_nvlist(hdl, &zc, 0) != 0)
b1b8ab34de515a5e83206da22c3d7e563241b021lling return (-1);
b1b8ab34de515a5e83206da22c3d7e563241b021lling
b1b8ab34de515a5e83206da22c3d7e563241b021lling while (ioctl(hdl->libzfs_fd, ZFS_IOC_POOL_GET_PROPS, &zc) != 0) {
b1b8ab34de515a5e83206da22c3d7e563241b021lling if (errno == ENOMEM) {
b1b8ab34de515a5e83206da22c3d7e563241b021lling if (zcmd_expand_dst_nvlist(hdl, &zc) != 0) {
b1b8ab34de515a5e83206da22c3d7e563241b021lling zcmd_free_nvlists(&zc);
b1b8ab34de515a5e83206da22c3d7e563241b021lling return (-1);
b1b8ab34de515a5e83206da22c3d7e563241b021lling }
b1b8ab34de515a5e83206da22c3d7e563241b021lling } else {
b1b8ab34de515a5e83206da22c3d7e563241b021lling zcmd_free_nvlists(&zc);
b1b8ab34de515a5e83206da22c3d7e563241b021lling return (-1);
b1b8ab34de515a5e83206da22c3d7e563241b021lling }
b1b8ab34de515a5e83206da22c3d7e563241b021lling }
b1b8ab34de515a5e83206da22c3d7e563241b021lling
b1b8ab34de515a5e83206da22c3d7e563241b021lling if (zcmd_read_dst_nvlist(hdl, &zc, &zhp->zpool_props) != 0) {
b1b8ab34de515a5e83206da22c3d7e563241b021lling zcmd_free_nvlists(&zc);
b1b8ab34de515a5e83206da22c3d7e563241b021lling return (-1);
b1b8ab34de515a5e83206da22c3d7e563241b021lling }
b1b8ab34de515a5e83206da22c3d7e563241b021lling
b1b8ab34de515a5e83206da22c3d7e563241b021lling zcmd_free_nvlists(&zc);
b1b8ab34de515a5e83206da22c3d7e563241b021lling
b1b8ab34de515a5e83206da22c3d7e563241b021lling return (0);
b1b8ab34de515a5e83206da22c3d7e563241b021lling}
b1b8ab34de515a5e83206da22c3d7e563241b021lling
fa9e4066f08beec538e775443c5be79dd423fcabahrens/*
fa9e4066f08beec538e775443c5be79dd423fcabahrens * Open a handle to the given pool, even if the pool is currently in the FAULTED
fa9e4066f08beec538e775443c5be79dd423fcabahrens * state.
fa9e4066f08beec538e775443c5be79dd423fcabahrens */
fa9e4066f08beec538e775443c5be79dd423fcabahrenszpool_handle_t *
99653d4ee642c6528e88224f12409a5f23060994eschrockzpool_open_canfail(libzfs_handle_t *hdl, const char *pool)
fa9e4066f08beec538e775443c5be79dd423fcabahrens{
fa9e4066f08beec538e775443c5be79dd423fcabahrens zpool_handle_t *zhp;
94de1d4cf6ec0a3bf040dcc4b8df107c4ed36b51eschrock boolean_t missing;
fa9e4066f08beec538e775443c5be79dd423fcabahrens
fa9e4066f08beec538e775443c5be79dd423fcabahrens /*
fa9e4066f08beec538e775443c5be79dd423fcabahrens * Make sure the pool name is valid.
fa9e4066f08beec538e775443c5be79dd423fcabahrens */
99653d4ee642c6528e88224f12409a5f23060994eschrock if (!zpool_name_valid(hdl, B_TRUE, pool)) {
ece3d9b3bacef51a5f34d993935eedbb7bb87059lling (void) zfs_error_fmt(hdl, EZFS_INVALIDNAME,
99653d4ee642c6528e88224f12409a5f23060994eschrock dgettext(TEXT_DOMAIN, "cannot open '%s'"),
99653d4ee642c6528e88224f12409a5f23060994eschrock pool);
fa9e4066f08beec538e775443c5be79dd423fcabahrens return (NULL);
fa9e4066f08beec538e775443c5be79dd423fcabahrens }
fa9e4066f08beec538e775443c5be79dd423fcabahrens
99653d4ee642c6528e88224f12409a5f23060994eschrock if ((zhp = zfs_alloc(hdl, sizeof (zpool_handle_t))) == NULL)
99653d4ee642c6528e88224f12409a5f23060994eschrock return (NULL);
fa9e4066f08beec538e775443c5be79dd423fcabahrens
99653d4ee642c6528e88224f12409a5f23060994eschrock zhp->zpool_hdl = hdl;
fa9e4066f08beec538e775443c5be79dd423fcabahrens (void) strlcpy(zhp->zpool_name, pool, sizeof (zhp->zpool_name));
fa9e4066f08beec538e775443c5be79dd423fcabahrens
94de1d4cf6ec0a3bf040dcc4b8df107c4ed36b51eschrock if (zpool_refresh_stats(zhp, &missing) != 0) {
94de1d4cf6ec0a3bf040dcc4b8df107c4ed36b51eschrock zpool_close(zhp);
94de1d4cf6ec0a3bf040dcc4b8df107c4ed36b51eschrock return (NULL);
94de1d4cf6ec0a3bf040dcc4b8df107c4ed36b51eschrock }
94de1d4cf6ec0a3bf040dcc4b8df107c4ed36b51eschrock
94de1d4cf6ec0a3bf040dcc4b8df107c4ed36b51eschrock if (missing) {
94de1d4cf6ec0a3bf040dcc4b8df107c4ed36b51eschrock zfs_error_aux(hdl, dgettext(TEXT_DOMAIN,
94de1d4cf6ec0a3bf040dcc4b8df107c4ed36b51eschrock "no such pool"));
ece3d9b3bacef51a5f34d993935eedbb7bb87059lling (void) zfs_error_fmt(hdl, EZFS_NOENT,
94de1d4cf6ec0a3bf040dcc4b8df107c4ed36b51eschrock dgettext(TEXT_DOMAIN, "cannot open '%s'"),
94de1d4cf6ec0a3bf040dcc4b8df107c4ed36b51eschrock pool);
94de1d4cf6ec0a3bf040dcc4b8df107c4ed36b51eschrock zpool_close(zhp);
94de1d4cf6ec0a3bf040dcc4b8df107c4ed36b51eschrock return (NULL);
fa9e4066f08beec538e775443c5be79dd423fcabahrens }
fa9e4066f08beec538e775443c5be79dd423fcabahrens
fa9e4066f08beec538e775443c5be79dd423fcabahrens return (zhp);
fa9e4066f08beec538e775443c5be79dd423fcabahrens}
fa9e4066f08beec538e775443c5be79dd423fcabahrens
fa9e4066f08beec538e775443c5be79dd423fcabahrens/*
fa9e4066f08beec538e775443c5be79dd423fcabahrens * Like the above, but silent on error. Used when iterating over pools (because
fa9e4066f08beec538e775443c5be79dd423fcabahrens * the configuration cache may be out of date).
fa9e4066f08beec538e775443c5be79dd423fcabahrens */
94de1d4cf6ec0a3bf040dcc4b8df107c4ed36b51eschrockint
94de1d4cf6ec0a3bf040dcc4b8df107c4ed36b51eschrockzpool_open_silent(libzfs_handle_t *hdl, const char *pool, zpool_handle_t **ret)
fa9e4066f08beec538e775443c5be79dd423fcabahrens{
fa9e4066f08beec538e775443c5be79dd423fcabahrens zpool_handle_t *zhp;
94de1d4cf6ec0a3bf040dcc4b8df107c4ed36b51eschrock boolean_t missing;
fa9e4066f08beec538e775443c5be79dd423fcabahrens
94de1d4cf6ec0a3bf040dcc4b8df107c4ed36b51eschrock if ((zhp = zfs_alloc(hdl, sizeof (zpool_handle_t))) == NULL)
94de1d4cf6ec0a3bf040dcc4b8df107c4ed36b51eschrock return (-1);
fa9e4066f08beec538e775443c5be79dd423fcabahrens
99653d4ee642c6528e88224f12409a5f23060994eschrock zhp->zpool_hdl = hdl;
fa9e4066f08beec538e775443c5be79dd423fcabahrens (void) strlcpy(zhp->zpool_name, pool, sizeof (zhp->zpool_name));
fa9e4066f08beec538e775443c5be79dd423fcabahrens
94de1d4cf6ec0a3bf040dcc4b8df107c4ed36b51eschrock if (zpool_refresh_stats(zhp, &missing) != 0) {
94de1d4cf6ec0a3bf040dcc4b8df107c4ed36b51eschrock zpool_close(zhp);
94de1d4cf6ec0a3bf040dcc4b8df107c4ed36b51eschrock return (-1);
fa9e4066f08beec538e775443c5be79dd423fcabahrens }
fa9e4066f08beec538e775443c5be79dd423fcabahrens
94de1d4cf6ec0a3bf040dcc4b8df107c4ed36b51eschrock if (missing) {
94de1d4cf6ec0a3bf040dcc4b8df107c4ed36b51eschrock zpool_close(zhp);
94de1d4cf6ec0a3bf040dcc4b8df107c4ed36b51eschrock *ret = NULL;
94de1d4cf6ec0a3bf040dcc4b8df107c4ed36b51eschrock return (0);
94de1d4cf6ec0a3bf040dcc4b8df107c4ed36b51eschrock }
94de1d4cf6ec0a3bf040dcc4b8df107c4ed36b51eschrock
94de1d4cf6ec0a3bf040dcc4b8df107c4ed36b51eschrock *ret = zhp;
94de1d4cf6ec0a3bf040dcc4b8df107c4ed36b51eschrock return (0);
fa9e4066f08beec538e775443c5be79dd423fcabahrens}
fa9e4066f08beec538e775443c5be79dd423fcabahrens
fa9e4066f08beec538e775443c5be79dd423fcabahrens/*
fa9e4066f08beec538e775443c5be79dd423fcabahrens * Similar to zpool_open_canfail(), but refuses to open pools in the faulted
fa9e4066f08beec538e775443c5be79dd423fcabahrens * state.
fa9e4066f08beec538e775443c5be79dd423fcabahrens */
fa9e4066f08beec538e775443c5be79dd423fcabahrenszpool_handle_t *
99653d4ee642c6528e88224f12409a5f23060994eschrockzpool_open(libzfs_handle_t *hdl, const char *pool)
fa9e4066f08beec538e775443c5be79dd423fcabahrens{
fa9e4066f08beec538e775443c5be79dd423fcabahrens zpool_handle_t *zhp;
fa9e4066f08beec538e775443c5be79dd423fcabahrens
99653d4ee642c6528e88224f12409a5f23060994eschrock if ((zhp = zpool_open_canfail(hdl, pool)) == NULL)
fa9e4066f08beec538e775443c5be79dd423fcabahrens return (NULL);
fa9e4066f08beec538e775443c5be79dd423fcabahrens
fa9e4066f08beec538e775443c5be79dd423fcabahrens if (zhp->zpool_state == POOL_STATE_UNAVAIL) {
ece3d9b3bacef51a5f34d993935eedbb7bb87059lling (void) zfs_error_fmt(hdl, EZFS_POOLUNAVAIL,
99653d4ee642c6528e88224f12409a5f23060994eschrock dgettext(TEXT_DOMAIN, "cannot open '%s'"), zhp->zpool_name);
fa9e4066f08beec538e775443c5be79dd423fcabahrens zpool_close(zhp);
fa9e4066f08beec538e775443c5be79dd423fcabahrens return (NULL);
fa9e4066f08beec538e775443c5be79dd423fcabahrens }
fa9e4066f08beec538e775443c5be79dd423fcabahrens
fa9e4066f08beec538e775443c5be79dd423fcabahrens return (zhp);
fa9e4066f08beec538e775443c5be79dd423fcabahrens}
fa9e4066f08beec538e775443c5be79dd423fcabahrens
fa9e4066f08beec538e775443c5be79dd423fcabahrens/*
fa9e4066f08beec538e775443c5be79dd423fcabahrens * Close the handle. Simply frees the memory associated with the handle.
fa9e4066f08beec538e775443c5be79dd423fcabahrens */
fa9e4066f08beec538e775443c5be79dd423fcabahrensvoid
fa9e4066f08beec538e775443c5be79dd423fcabahrenszpool_close(zpool_handle_t *zhp)
fa9e4066f08beec538e775443c5be79dd423fcabahrens{
fa9e4066f08beec538e775443c5be79dd423fcabahrens if (zhp->zpool_config)
fa9e4066f08beec538e775443c5be79dd423fcabahrens nvlist_free(zhp->zpool_config);
088e9d477eee66081e407fbc5a33c4da25f66f6aeschrock if (zhp->zpool_old_config)
088e9d477eee66081e407fbc5a33c4da25f66f6aeschrock nvlist_free(zhp->zpool_old_config);
b1b8ab34de515a5e83206da22c3d7e563241b021lling if (zhp->zpool_props)
b1b8ab34de515a5e83206da22c3d7e563241b021lling nvlist_free(zhp->zpool_props);
fa9e4066f08beec538e775443c5be79dd423fcabahrens free(zhp);
fa9e4066f08beec538e775443c5be79dd423fcabahrens}
fa9e4066f08beec538e775443c5be79dd423fcabahrens
fa9e4066f08beec538e775443c5be79dd423fcabahrens/*
fa9e4066f08beec538e775443c5be79dd423fcabahrens * Return the name of the pool.
fa9e4066f08beec538e775443c5be79dd423fcabahrens */
fa9e4066f08beec538e775443c5be79dd423fcabahrensconst char *
fa9e4066f08beec538e775443c5be79dd423fcabahrenszpool_get_name(zpool_handle_t *zhp)
fa9e4066f08beec538e775443c5be79dd423fcabahrens{
fa9e4066f08beec538e775443c5be79dd423fcabahrens return (zhp->zpool_name);
fa9e4066f08beec538e775443c5be79dd423fcabahrens}
fa9e4066f08beec538e775443c5be79dd423fcabahrens
fa9e4066f08beec538e775443c5be79dd423fcabahrens/*
fa9e4066f08beec538e775443c5be79dd423fcabahrens * Return the GUID of the pool.
fa9e4066f08beec538e775443c5be79dd423fcabahrens */
fa9e4066f08beec538e775443c5be79dd423fcabahrensuint64_t
fa9e4066f08beec538e775443c5be79dd423fcabahrenszpool_get_guid(zpool_handle_t *zhp)
fa9e4066f08beec538e775443c5be79dd423fcabahrens{
fa9e4066f08beec538e775443c5be79dd423fcabahrens uint64_t guid;
fa9e4066f08beec538e775443c5be79dd423fcabahrens
fa9e4066f08beec538e775443c5be79dd423fcabahrens verify(nvlist_lookup_uint64(zhp->zpool_config, ZPOOL_CONFIG_POOL_GUID,
fa9e4066f08beec538e775443c5be79dd423fcabahrens &guid) == 0);
fa9e4066f08beec538e775443c5be79dd423fcabahrens return (guid);
fa9e4066f08beec538e775443c5be79dd423fcabahrens}
fa9e4066f08beec538e775443c5be79dd423fcabahrens
99653d4ee642c6528e88224f12409a5f23060994eschrock/*
99653d4ee642c6528e88224f12409a5f23060994eschrock * Return the version of the pool.
99653d4ee642c6528e88224f12409a5f23060994eschrock */
99653d4ee642c6528e88224f12409a5f23060994eschrockuint64_t
99653d4ee642c6528e88224f12409a5f23060994eschrockzpool_get_version(zpool_handle_t *zhp)
99653d4ee642c6528e88224f12409a5f23060994eschrock{
99653d4ee642c6528e88224f12409a5f23060994eschrock uint64_t version;
99653d4ee642c6528e88224f12409a5f23060994eschrock
99653d4ee642c6528e88224f12409a5f23060994eschrock verify(nvlist_lookup_uint64(zhp->zpool_config, ZPOOL_CONFIG_VERSION,
99653d4ee642c6528e88224f12409a5f23060994eschrock &version) == 0);
99653d4ee642c6528e88224f12409a5f23060994eschrock
99653d4ee642c6528e88224f12409a5f23060994eschrock return (version);
99653d4ee642c6528e88224f12409a5f23060994eschrock}
99653d4ee642c6528e88224f12409a5f23060994eschrock
fa9e4066f08beec538e775443c5be79dd423fcabahrens/*
fa9e4066f08beec538e775443c5be79dd423fcabahrens * Return the amount of space currently consumed by the pool.
fa9e4066f08beec538e775443c5be79dd423fcabahrens */
fa9e4066f08beec538e775443c5be79dd423fcabahrensuint64_t
fa9e4066f08beec538e775443c5be79dd423fcabahrenszpool_get_space_used(zpool_handle_t *zhp)
fa9e4066f08beec538e775443c5be79dd423fcabahrens{
fa9e4066f08beec538e775443c5be79dd423fcabahrens nvlist_t *nvroot;
fa9e4066f08beec538e775443c5be79dd423fcabahrens vdev_stat_t *vs;
fa9e4066f08beec538e775443c5be79dd423fcabahrens uint_t vsc;
fa9e4066f08beec538e775443c5be79dd423fcabahrens
fa9e4066f08beec538e775443c5be79dd423fcabahrens verify(nvlist_lookup_nvlist(zhp->zpool_config, ZPOOL_CONFIG_VDEV_TREE,
fa9e4066f08beec538e775443c5be79dd423fcabahrens &nvroot) == 0);
fa9e4066f08beec538e775443c5be79dd423fcabahrens verify(nvlist_lookup_uint64_array(nvroot, ZPOOL_CONFIG_STATS,
fa9e4066f08beec538e775443c5be79dd423fcabahrens (uint64_t **)&vs, &vsc) == 0);
fa9e4066f08beec538e775443c5be79dd423fcabahrens
fa9e4066f08beec538e775443c5be79dd423fcabahrens return (vs->vs_alloc);
fa9e4066f08beec538e775443c5be79dd423fcabahrens}
fa9e4066f08beec538e775443c5be79dd423fcabahrens
fa9e4066f08beec538e775443c5be79dd423fcabahrens/*
fa9e4066f08beec538e775443c5be79dd423fcabahrens * Return the total space in the pool.
fa9e4066f08beec538e775443c5be79dd423fcabahrens */
fa9e4066f08beec538e775443c5be79dd423fcabahrensuint64_t
fa9e4066f08beec538e775443c5be79dd423fcabahrenszpool_get_space_total(zpool_handle_t *zhp)
fa9e4066f08beec538e775443c5be79dd423fcabahrens{
fa9e4066f08beec538e775443c5be79dd423fcabahrens nvlist_t *nvroot;
fa9e4066f08beec538e775443c5be79dd423fcabahrens vdev_stat_t *vs;
fa9e4066f08beec538e775443c5be79dd423fcabahrens uint_t vsc;
fa9e4066f08beec538e775443c5be79dd423fcabahrens
fa9e4066f08beec538e775443c5be79dd423fcabahrens verify(nvlist_lookup_nvlist(zhp->zpool_config, ZPOOL_CONFIG_VDEV_TREE,
fa9e4066f08beec538e775443c5be79dd423fcabahrens &nvroot) == 0);
fa9e4066f08beec538e775443c5be79dd423fcabahrens verify(nvlist_lookup_uint64_array(nvroot, ZPOOL_CONFIG_STATS,
fa9e4066f08beec538e775443c5be79dd423fcabahrens (uint64_t **)&vs, &vsc) == 0);
fa9e4066f08beec538e775443c5be79dd423fcabahrens
fa9e4066f08beec538e775443c5be79dd423fcabahrens return (vs->vs_space);
fa9e4066f08beec538e775443c5be79dd423fcabahrens}
fa9e4066f08beec538e775443c5be79dd423fcabahrens
fa9e4066f08beec538e775443c5be79dd423fcabahrens/*
fa9e4066f08beec538e775443c5be79dd423fcabahrens * Return the alternate root for this pool, if any.
fa9e4066f08beec538e775443c5be79dd423fcabahrens */
fa9e4066f08beec538e775443c5be79dd423fcabahrensint
fa9e4066f08beec538e775443c5be79dd423fcabahrenszpool_get_root(zpool_handle_t *zhp, char *buf, size_t buflen)
fa9e4066f08beec538e775443c5be79dd423fcabahrens{
fa9e4066f08beec538e775443c5be79dd423fcabahrens zfs_cmd_t zc = { 0 };
fa9e4066f08beec538e775443c5be79dd423fcabahrens
fa9e4066f08beec538e775443c5be79dd423fcabahrens (void) strlcpy(zc.zc_name, zhp->zpool_name, sizeof (zc.zc_name));
99653d4ee642c6528e88224f12409a5f23060994eschrock if (ioctl(zhp->zpool_hdl->libzfs_fd, ZFS_IOC_OBJSET_STATS, &zc) != 0 ||
e9dbad6f263d5570ed7ff5443ec5b958af8c24d7eschrock zc.zc_value[0] == '\0')
fa9e4066f08beec538e775443c5be79dd423fcabahrens return (-1);
fa9e4066f08beec538e775443c5be79dd423fcabahrens
e9dbad6f263d5570ed7ff5443ec5b958af8c24d7eschrock (void) strlcpy(buf, zc.zc_value, buflen);
fa9e4066f08beec538e775443c5be79dd423fcabahrens
fa9e4066f08beec538e775443c5be79dd423fcabahrens return (0);
fa9e4066f08beec538e775443c5be79dd423fcabahrens}
fa9e4066f08beec538e775443c5be79dd423fcabahrens
fa9e4066f08beec538e775443c5be79dd423fcabahrens/*
fa9e4066f08beec538e775443c5be79dd423fcabahrens * Return the state of the pool (ACTIVE or UNAVAILABLE)
fa9e4066f08beec538e775443c5be79dd423fcabahrens */
fa9e4066f08beec538e775443c5be79dd423fcabahrensint
fa9e4066f08beec538e775443c5be79dd423fcabahrenszpool_get_state(zpool_handle_t *zhp)
fa9e4066f08beec538e775443c5be79dd423fcabahrens{
fa9e4066f08beec538e775443c5be79dd423fcabahrens return (zhp->zpool_state);
fa9e4066f08beec538e775443c5be79dd423fcabahrens}
fa9e4066f08beec538e775443c5be79dd423fcabahrens
fa9e4066f08beec538e775443c5be79dd423fcabahrens/*
fa9e4066f08beec538e775443c5be79dd423fcabahrens * Create the named pool, using the provided vdev list. It is assumed
fa9e4066f08beec538e775443c5be79dd423fcabahrens * that the consumer has already validated the contents of the nvlist, so we
fa9e4066f08beec538e775443c5be79dd423fcabahrens * don't have to worry about error semantics.
fa9e4066f08beec538e775443c5be79dd423fcabahrens */
fa9e4066f08beec538e775443c5be79dd423fcabahrensint
99653d4ee642c6528e88224f12409a5f23060994eschrockzpool_create(libzfs_handle_t *hdl, const char *pool, nvlist_t *nvroot,
99653d4ee642c6528e88224f12409a5f23060994eschrock const char *altroot)
fa9e4066f08beec538e775443c5be79dd423fcabahrens{
fa9e4066f08beec538e775443c5be79dd423fcabahrens zfs_cmd_t zc = { 0 };
99653d4ee642c6528e88224f12409a5f23060994eschrock char msg[1024];
fa9e4066f08beec538e775443c5be79dd423fcabahrens
99653d4ee642c6528e88224f12409a5f23060994eschrock (void) snprintf(msg, sizeof (msg), dgettext(TEXT_DOMAIN,
99653d4ee642c6528e88224f12409a5f23060994eschrock "cannot create '%s'"), pool);
fa9e4066f08beec538e775443c5be79dd423fcabahrens
99653d4ee642c6528e88224f12409a5f23060994eschrock if (!zpool_name_valid(hdl, B_FALSE, pool))
99653d4ee642c6528e88224f12409a5f23060994eschrock return (zfs_error(hdl, EZFS_INVALIDNAME, msg));
fa9e4066f08beec538e775443c5be79dd423fcabahrens
99653d4ee642c6528e88224f12409a5f23060994eschrock if (altroot != NULL && altroot[0] != '/')
ece3d9b3bacef51a5f34d993935eedbb7bb87059lling return (zfs_error_fmt(hdl, EZFS_BADPATH,
99653d4ee642c6528e88224f12409a5f23060994eschrock dgettext(TEXT_DOMAIN, "bad alternate root '%s'"), altroot));
fa9e4066f08beec538e775443c5be79dd423fcabahrens
e9dbad6f263d5570ed7ff5443ec5b958af8c24d7eschrock if (zcmd_write_src_nvlist(hdl, &zc, nvroot, NULL) != 0)
99653d4ee642c6528e88224f12409a5f23060994eschrock return (-1);
99653d4ee642c6528e88224f12409a5f23060994eschrock
fa9e4066f08beec538e775443c5be79dd423fcabahrens (void) strlcpy(zc.zc_name, pool, sizeof (zc.zc_name));
fa9e4066f08beec538e775443c5be79dd423fcabahrens
fa9e4066f08beec538e775443c5be79dd423fcabahrens if (altroot != NULL)
e9dbad6f263d5570ed7ff5443ec5b958af8c24d7eschrock (void) strlcpy(zc.zc_value, altroot, sizeof (zc.zc_value));
fa9e4066f08beec538e775443c5be79dd423fcabahrens
ecd6cf800b63704be73fb264c3f5b6e0dafc068dmarks if (zfs_ioctl(hdl, ZFS_IOC_POOL_CREATE, &zc) != 0) {
e9dbad6f263d5570ed7ff5443ec5b958af8c24d7eschrock zcmd_free_nvlists(&zc);
fa9e4066f08beec538e775443c5be79dd423fcabahrens
99653d4ee642c6528e88224f12409a5f23060994eschrock switch (errno) {
fa9e4066f08beec538e775443c5be79dd423fcabahrens case EBUSY:
fa9e4066f08beec538e775443c5be79dd423fcabahrens /*
fa9e4066f08beec538e775443c5be79dd423fcabahrens * This can happen if the user has specified the same
fa9e4066f08beec538e775443c5be79dd423fcabahrens * device multiple times. We can't reliably detect this
fa9e4066f08beec538e775443c5be79dd423fcabahrens * until we try to add it and see we already have a
fa9e4066f08beec538e775443c5be79dd423fcabahrens * label.
fa9e4066f08beec538e775443c5be79dd423fcabahrens */
99653d4ee642c6528e88224f12409a5f23060994eschrock zfs_error_aux(hdl, dgettext(TEXT_DOMAIN,
99653d4ee642c6528e88224f12409a5f23060994eschrock "one or more vdevs refer to the same device"));
99653d4ee642c6528e88224f12409a5f23060994eschrock return (zfs_error(hdl, EZFS_BADDEV, msg));
fa9e4066f08beec538e775443c5be79dd423fcabahrens
fa9e4066f08beec538e775443c5be79dd423fcabahrens case EOVERFLOW:
fa9e4066f08beec538e775443c5be79dd423fcabahrens /*
99653d4ee642c6528e88224f12409a5f23060994eschrock * This occurs when one of the devices is below
fa9e4066f08beec538e775443c5be79dd423fcabahrens * SPA_MINDEVSIZE. Unfortunately, we can't detect which
fa9e4066f08beec538e775443c5be79dd423fcabahrens * device was the problem device since there's no
fa9e4066f08beec538e775443c5be79dd423fcabahrens * reliable way to determine device size from userland.
fa9e4066f08beec538e775443c5be79dd423fcabahrens */
fa9e4066f08beec538e775443c5be79dd423fcabahrens {
fa9e4066f08beec538e775443c5be79dd423fcabahrens char buf[64];
fa9e4066f08beec538e775443c5be79dd423fcabahrens
fa9e4066f08beec538e775443c5be79dd423fcabahrens zfs_nicenum(SPA_MINDEVSIZE, buf, sizeof (buf));
fa9e4066f08beec538e775443c5be79dd423fcabahrens
99653d4ee642c6528e88224f12409a5f23060994eschrock zfs_error_aux(hdl, dgettext(TEXT_DOMAIN,
99653d4ee642c6528e88224f12409a5f23060994eschrock "one or more devices is less than the "
99653d4ee642c6528e88224f12409a5f23060994eschrock "minimum size (%s)"), buf);
fa9e4066f08beec538e775443c5be79dd423fcabahrens }
99653d4ee642c6528e88224f12409a5f23060994eschrock return (zfs_error(hdl, EZFS_BADDEV, msg));
fa9e4066f08beec538e775443c5be79dd423fcabahrens
fa9e4066f08beec538e775443c5be79dd423fcabahrens case ENOSPC:
99653d4ee642c6528e88224f12409a5f23060994eschrock zfs_error_aux(hdl, dgettext(TEXT_DOMAIN,
99653d4ee642c6528e88224f12409a5f23060994eschrock "one or more devices is out of space"));
99653d4ee642c6528e88224f12409a5f23060994eschrock return (zfs_error(hdl, EZFS_BADDEV, msg));
fa9e4066f08beec538e775443c5be79dd423fcabahrens
fa9e4066f08beec538e775443c5be79dd423fcabahrens default:
99653d4ee642c6528e88224f12409a5f23060994eschrock return (zpool_standard_error(hdl, errno, msg));
fa9e4066f08beec538e775443c5be79dd423fcabahrens }
fa9e4066f08beec538e775443c5be79dd423fcabahrens }
e9dbad6f263d5570ed7ff5443ec5b958af8c24d7eschrock zcmd_free_nvlists(&zc);
fa9e4066f08beec538e775443c5be79dd423fcabahrens
fa9e4066f08beec538e775443c5be79dd423fcabahrens /*
fa9e4066f08beec538e775443c5be79dd423fcabahrens * If this is an alternate root pool, then we automatically set the
e9dbad6f263d5570ed7ff5443ec5b958af8c24d7eschrock * mountpoint of the root dataset to be '/'.
fa9e4066f08beec538e775443c5be79dd423fcabahrens */
fa9e4066f08beec538e775443c5be79dd423fcabahrens if (altroot != NULL) {
fa9e4066f08beec538e775443c5be79dd423fcabahrens zfs_handle_t *zhp;
fa9e4066f08beec538e775443c5be79dd423fcabahrens
99653d4ee642c6528e88224f12409a5f23060994eschrock verify((zhp = zfs_open(hdl, pool, ZFS_TYPE_ANY)) != NULL);
e9dbad6f263d5570ed7ff5443ec5b958af8c24d7eschrock verify(zfs_prop_set(zhp, zfs_prop_to_name(ZFS_PROP_MOUNTPOINT),
e9dbad6f263d5570ed7ff5443ec5b958af8c24d7eschrock "/") == 0);
fa9e4066f08beec538e775443c5be79dd423fcabahrens
fa9e4066f08beec538e775443c5be79dd423fcabahrens zfs_close(zhp);
fa9e4066f08beec538e775443c5be79dd423fcabahrens }
fa9e4066f08beec538e775443c5be79dd423fcabahrens
fa9e4066f08beec538e775443c5be79dd423fcabahrens return (0);
fa9e4066f08beec538e775443c5be79dd423fcabahrens}
fa9e4066f08beec538e775443c5be79dd423fcabahrens
fa9e4066f08beec538e775443c5be79dd423fcabahrens/*
fa9e4066f08beec538e775443c5be79dd423fcabahrens * Destroy the given pool. It is up to the caller to ensure that there are no
fa9e4066f08beec538e775443c5be79dd423fcabahrens * datasets left in the pool.
fa9e4066f08beec538e775443c5be79dd423fcabahrens */
fa9e4066f08beec538e775443c5be79dd423fcabahrensint
fa9e4066f08beec538e775443c5be79dd423fcabahrenszpool_destroy(zpool_handle_t *zhp)
fa9e4066f08beec538e775443c5be79dd423fcabahrens{
fa9e4066f08beec538e775443c5be79dd423fcabahrens zfs_cmd_t zc = { 0 };
fa9e4066f08beec538e775443c5be79dd423fcabahrens zfs_handle_t *zfp = NULL;
99653d4ee642c6528e88224f12409a5f23060994eschrock libzfs_handle_t *hdl = zhp->zpool_hdl;
99653d4ee642c6528e88224f12409a5f23060994eschrock char msg[1024];
fa9e4066f08beec538e775443c5be79dd423fcabahrens
fa9e4066f08beec538e775443c5be79dd423fcabahrens if (zhp->zpool_state == POOL_STATE_ACTIVE &&
99653d4ee642c6528e88224f12409a5f23060994eschrock (zfp = zfs_open(zhp->zpool_hdl, zhp->zpool_name,
99653d4ee642c6528e88224f12409a5f23060994eschrock ZFS_TYPE_FILESYSTEM)) == NULL)
fa9e4066f08beec538e775443c5be79dd423fcabahrens return (-1);
fa9e4066f08beec538e775443c5be79dd423fcabahrens
5ad820458efd0fdb914baff9c1447c22b819fa23nd if (zpool_remove_zvol_links(zhp) != 0)
fa9e4066f08beec538e775443c5be79dd423fcabahrens return (-1);
fa9e4066f08beec538e775443c5be79dd423fcabahrens
fa9e4066f08beec538e775443c5be79dd423fcabahrens (void) strlcpy(zc.zc_name, zhp->zpool_name, sizeof (zc.zc_name));
fa9e4066f08beec538e775443c5be79dd423fcabahrens
ecd6cf800b63704be73fb264c3f5b6e0dafc068dmarks if (zfs_ioctl(zhp->zpool_hdl, ZFS_IOC_POOL_DESTROY, &zc) != 0) {
99653d4ee642c6528e88224f12409a5f23060994eschrock (void) snprintf(msg, sizeof (msg), dgettext(TEXT_DOMAIN,
99653d4ee642c6528e88224f12409a5f23060994eschrock "cannot destroy '%s'"), zhp->zpool_name);
fa9e4066f08beec538e775443c5be79dd423fcabahrens
99653d4ee642c6528e88224f12409a5f23060994eschrock if (errno == EROFS) {
99653d4ee642c6528e88224f12409a5f23060994eschrock zfs_error_aux(hdl, dgettext(TEXT_DOMAIN,
99653d4ee642c6528e88224f12409a5f23060994eschrock "one or more devices is read only"));
99653d4ee642c6528e88224f12409a5f23060994eschrock (void) zfs_error(hdl, EZFS_BADDEV, msg);
99653d4ee642c6528e88224f12409a5f23060994eschrock } else {
99653d4ee642c6528e88224f12409a5f23060994eschrock (void) zpool_standard_error(hdl, errno, msg);
fa9e4066f08beec538e775443c5be79dd423fcabahrens }
fa9e4066f08beec538e775443c5be79dd423fcabahrens
fa9e4066f08beec538e775443c5be79dd423fcabahrens if (zfp)
fa9e4066f08beec538e775443c5be79dd423fcabahrens zfs_close(zfp);
fa9e4066f08beec538e775443c5be79dd423fcabahrens return (-1);
fa9e4066f08beec538e775443c5be79dd423fcabahrens }
fa9e4066f08beec538e775443c5be79dd423fcabahrens
fa9e4066f08beec538e775443c5be79dd423fcabahrens if (zfp) {
fa9e4066f08beec538e775443c5be79dd423fcabahrens remove_mountpoint(zfp);
fa9e4066f08beec538e775443c5be79dd423fcabahrens zfs_close(zfp);
fa9e4066f08beec538e775443c5be79dd423fcabahrens }
fa9e4066f08beec538e775443c5be79dd423fcabahrens
fa9e4066f08beec538e775443c5be79dd423fcabahrens return (0);
fa9e4066f08beec538e775443c5be79dd423fcabahrens}
fa9e4066f08beec538e775443c5be79dd423fcabahrens
fa9e4066f08beec538e775443c5be79dd423fcabahrens/*
fa9e4066f08beec538e775443c5be79dd423fcabahrens * Add the given vdevs to the pool. The caller must have already performed the
fa9e4066f08beec538e775443c5be79dd423fcabahrens * necessary verification to ensure that the vdev specification is well-formed.
fa9e4066f08beec538e775443c5be79dd423fcabahrens */
fa9e4066f08beec538e775443c5be79dd423fcabahrensint
fa9e4066f08beec538e775443c5be79dd423fcabahrenszpool_add(zpool_handle_t *zhp, nvlist_t *nvroot)
fa9e4066f08beec538e775443c5be79dd423fcabahrens{
e9dbad6f263d5570ed7ff5443ec5b958af8c24d7eschrock zfs_cmd_t zc = { 0 };
99653d4ee642c6528e88224f12409a5f23060994eschrock int ret;
99653d4ee642c6528e88224f12409a5f23060994eschrock libzfs_handle_t *hdl = zhp->zpool_hdl;
99653d4ee642c6528e88224f12409a5f23060994eschrock char msg[1024];
99653d4ee642c6528e88224f12409a5f23060994eschrock nvlist_t **spares;
99653d4ee642c6528e88224f12409a5f23060994eschrock uint_t nspares;
99653d4ee642c6528e88224f12409a5f23060994eschrock
99653d4ee642c6528e88224f12409a5f23060994eschrock (void) snprintf(msg, sizeof (msg), dgettext(TEXT_DOMAIN,
99653d4ee642c6528e88224f12409a5f23060994eschrock "cannot add to '%s'"), zhp->zpool_name);
99653d4ee642c6528e88224f12409a5f23060994eschrock
e7437265dc2a4920c197ed4337665539d358b22cahrens if (zpool_get_version(zhp) < SPA_VERSION_SPARES &&
99653d4ee642c6528e88224f12409a5f23060994eschrock nvlist_lookup_nvlist_array(nvroot, ZPOOL_CONFIG_SPARES,
99653d4ee642c6528e88224f12409a5f23060994eschrock &spares, &nspares) == 0) {
99653d4ee642c6528e88224f12409a5f23060994eschrock zfs_error_aux(hdl, dgettext(TEXT_DOMAIN, "pool must be "
99653d4ee642c6528e88224f12409a5f23060994eschrock "upgraded to add hot spares"));
99653d4ee642c6528e88224f12409a5f23060994eschrock return (zfs_error(hdl, EZFS_BADVERSION, msg));
99653d4ee642c6528e88224f12409a5f23060994eschrock }
fa9e4066f08beec538e775443c5be79dd423fcabahrens
e9dbad6f263d5570ed7ff5443ec5b958af8c24d7eschrock if (zcmd_write_src_nvlist(hdl, &zc, nvroot, NULL) != 0)
99653d4ee642c6528e88224f12409a5f23060994eschrock return (-1);
fa9e4066f08beec538e775443c5be79dd423fcabahrens (void) strlcpy(zc.zc_name, zhp->zpool_name, sizeof (zc.zc_name));
fa9e4066f08beec538e775443c5be79dd423fcabahrens
ecd6cf800b63704be73fb264c3f5b6e0dafc068dmarks if (zfs_ioctl(zhp->zpool_hdl, ZFS_IOC_VDEV_ADD, &zc) != 0) {
fa9e4066f08beec538e775443c5be79dd423fcabahrens switch (errno) {
fa9e4066f08beec538e775443c5be79dd423fcabahrens case EBUSY:
fa9e4066f08beec538e775443c5be79dd423fcabahrens /*
fa9e4066f08beec538e775443c5be79dd423fcabahrens * This can happen if the user has specified the same
fa9e4066f08beec538e775443c5be79dd423fcabahrens * device multiple times. We can't reliably detect this
fa9e4066f08beec538e775443c5be79dd423fcabahrens * until we try to add it and see we already have a
fa9e4066f08beec538e775443c5be79dd423fcabahrens * label.
fa9e4066f08beec538e775443c5be79dd423fcabahrens */
99653d4ee642c6528e88224f12409a5f23060994eschrock zfs_error_aux(hdl, dgettext(TEXT_DOMAIN,
99653d4ee642c6528e88224f12409a5f23060994eschrock "one or more vdevs refer to the same device"));
99653d4ee642c6528e88224f12409a5f23060994eschrock (void) zfs_error(hdl, EZFS_BADDEV, msg);
fa9e4066f08beec538e775443c5be79dd423fcabahrens break;
fa9e4066f08beec538e775443c5be79dd423fcabahrens
fa9e4066f08beec538e775443c5be79dd423fcabahrens case EOVERFLOW:
fa9e4066f08beec538e775443c5be79dd423fcabahrens /*
fa9e4066f08beec538e775443c5be79dd423fcabahrens * This occurrs when one of the devices is below
fa9e4066f08beec538e775443c5be79dd423fcabahrens * SPA_MINDEVSIZE. Unfortunately, we can't detect which
fa9e4066f08beec538e775443c5be79dd423fcabahrens * device was the problem device since there's no
fa9e4066f08beec538e775443c5be79dd423fcabahrens * reliable way to determine device size from userland.
fa9e4066f08beec538e775443c5be79dd423fcabahrens */
fa9e4066f08beec538e775443c5be79dd423fcabahrens {
fa9e4066f08beec538e775443c5be79dd423fcabahrens char buf[64];
fa9e4066f08beec538e775443c5be79dd423fcabahrens
fa9e4066f08beec538e775443c5be79dd423fcabahrens zfs_nicenum(SPA_MINDEVSIZE, buf, sizeof (buf));
fa9e4066f08beec538e775443c5be79dd423fcabahrens
99653d4ee642c6528e88224f12409a5f23060994eschrock zfs_error_aux(hdl, dgettext(TEXT_DOMAIN,
99653d4ee642c6528e88224f12409a5f23060994eschrock "device is less than the minimum "
99653d4ee642c6528e88224f12409a5f23060994eschrock "size (%s)"), buf);
fa9e4066f08beec538e775443c5be79dd423fcabahrens }
99653d4ee642c6528e88224f12409a5f23060994eschrock (void) zfs_error(hdl, EZFS_BADDEV, msg);
99653d4ee642c6528e88224f12409a5f23060994eschrock break;
99653d4ee642c6528e88224f12409a5f23060994eschrock
99653d4ee642c6528e88224f12409a5f23060994eschrock case ENOTSUP:
99653d4ee642c6528e88224f12409a5f23060994eschrock zfs_error_aux(hdl, dgettext(TEXT_DOMAIN,
8654d0253136055bd4cc2423d87378e8a37f2eb5perrin "pool must be upgraded to add these vdevs"));
99653d4ee642c6528e88224f12409a5f23060994eschrock (void) zfs_error(hdl, EZFS_BADVERSION, msg);
fa9e4066f08beec538e775443c5be79dd423fcabahrens break;
fa9e4066f08beec538e775443c5be79dd423fcabahrens
b1b8ab34de515a5e83206da22c3d7e563241b021lling case EDOM:
b1b8ab34de515a5e83206da22c3d7e563241b021lling zfs_error_aux(hdl, dgettext(TEXT_DOMAIN,
8654d0253136055bd4cc2423d87378e8a37f2eb5perrin "root pool can not have multiple vdevs"
8654d0253136055bd4cc2423d87378e8a37f2eb5perrin " or separate logs"));
b1b8ab34de515a5e83206da22c3d7e563241b021lling (void) zfs_error(hdl, EZFS_POOL_NOTSUP, msg);
b1b8ab34de515a5e83206da22c3d7e563241b021lling break;
b1b8ab34de515a5e83206da22c3d7e563241b021lling
fa9e4066f08beec538e775443c5be79dd423fcabahrens default:
99653d4ee642c6528e88224f12409a5f23060994eschrock (void) zpool_standard_error(hdl, errno, msg);
fa9e4066f08beec538e775443c5be79dd423fcabahrens }
fa9e4066f08beec538e775443c5be79dd423fcabahrens
99653d4ee642c6528e88224f12409a5f23060994eschrock ret = -1;
99653d4ee642c6528e88224f12409a5f23060994eschrock } else {
99653d4ee642c6528e88224f12409a5f23060994eschrock ret = 0;
fa9e4066f08beec538e775443c5be79dd423fcabahrens }
fa9e4066f08beec538e775443c5be79dd423fcabahrens
e9dbad6f263d5570ed7ff5443ec5b958af8c24d7eschrock zcmd_free_nvlists(&zc);
fa9e4066f08beec538e775443c5be79dd423fcabahrens
99653d4ee642c6528e88224f12409a5f23060994eschrock return (ret);
fa9e4066f08beec538e775443c5be79dd423fcabahrens}
fa9e4066f08beec538e775443c5be79dd423fcabahrens
fa9e4066f08beec538e775443c5be79dd423fcabahrens/*
fa9e4066f08beec538e775443c5be79dd423fcabahrens * Exports the pool from the system. The caller must ensure that there are no
fa9e4066f08beec538e775443c5be79dd423fcabahrens * mounted datasets in the pool.
fa9e4066f08beec538e775443c5be79dd423fcabahrens */
fa9e4066f08beec538e775443c5be79dd423fcabahrensint
fa9e4066f08beec538e775443c5be79dd423fcabahrenszpool_export(zpool_handle_t *zhp)
fa9e4066f08beec538e775443c5be79dd423fcabahrens{
fa9e4066f08beec538e775443c5be79dd423fcabahrens zfs_cmd_t zc = { 0 };
fa9e4066f08beec538e775443c5be79dd423fcabahrens
fa9e4066f08beec538e775443c5be79dd423fcabahrens if (zpool_remove_zvol_links(zhp) != 0)
fa9e4066f08beec538e775443c5be79dd423fcabahrens return (-1);
fa9e4066f08beec538e775443c5be79dd423fcabahrens
fa9e4066f08beec538e775443c5be79dd423fcabahrens (void) strlcpy(zc.zc_name, zhp->zpool_name, sizeof (zc.zc_name));
fa9e4066f08beec538e775443c5be79dd423fcabahrens
ecd6cf800b63704be73fb264c3f5b6e0dafc068dmarks if (zfs_ioctl(zhp->zpool_hdl, ZFS_IOC_POOL_EXPORT, &zc) != 0)
ece3d9b3bacef51a5f34d993935eedbb7bb87059lling return (zpool_standard_error_fmt(zhp->zpool_hdl, errno,
99653d4ee642c6528e88224f12409a5f23060994eschrock dgettext(TEXT_DOMAIN, "cannot export '%s'"),
99653d4ee642c6528e88224f12409a5f23060994eschrock zhp->zpool_name));
fa9e4066f08beec538e775443c5be79dd423fcabahrens return (0);
fa9e4066f08beec538e775443c5be79dd423fcabahrens}
fa9e4066f08beec538e775443c5be79dd423fcabahrens
fa9e4066f08beec538e775443c5be79dd423fcabahrens/*
fa9e4066f08beec538e775443c5be79dd423fcabahrens * Import the given pool using the known configuration. The configuration
fa9e4066f08beec538e775443c5be79dd423fcabahrens * should have come from zpool_find_import(). The 'newname' and 'altroot'
fa9e4066f08beec538e775443c5be79dd423fcabahrens * parameters control whether the pool is imported with a different name or with
fa9e4066f08beec538e775443c5be79dd423fcabahrens * an alternate root, respectively.
fa9e4066f08beec538e775443c5be79dd423fcabahrens */
fa9e4066f08beec538e775443c5be79dd423fcabahrensint
99653d4ee642c6528e88224f12409a5f23060994eschrockzpool_import(libzfs_handle_t *hdl, nvlist_t *config, const char *newname,
99653d4ee642c6528e88224f12409a5f23060994eschrock const char *altroot)
fa9e4066f08beec538e775443c5be79dd423fcabahrens{
e9dbad6f263d5570ed7ff5443ec5b958af8c24d7eschrock zfs_cmd_t zc = { 0 };
fa9e4066f08beec538e775443c5be79dd423fcabahrens char *thename;
fa9e4066f08beec538e775443c5be79dd423fcabahrens char *origname;
fa9e4066f08beec538e775443c5be79dd423fcabahrens int ret;
fa9e4066f08beec538e775443c5be79dd423fcabahrens
fa9e4066f08beec538e775443c5be79dd423fcabahrens verify(nvlist_lookup_string(config, ZPOOL_CONFIG_POOL_NAME,
fa9e4066f08beec538e775443c5be79dd423fcabahrens &origname) == 0);
fa9e4066f08beec538e775443c5be79dd423fcabahrens
fa9e4066f08beec538e775443c5be79dd423fcabahrens if (newname != NULL) {
99653d4ee642c6528e88224f12409a5f23060994eschrock if (!zpool_name_valid(hdl, B_FALSE, newname))
ece3d9b3bacef51a5f34d993935eedbb7bb87059lling return (zfs_error_fmt(hdl, EZFS_INVALIDNAME,
99653d4ee642c6528e88224f12409a5f23060994eschrock dgettext(TEXT_DOMAIN, "cannot import '%s'"),
99653d4ee642c6528e88224f12409a5f23060994eschrock newname));
fa9e4066f08beec538e775443c5be79dd423fcabahrens thename = (char *)newname;
fa9e4066f08beec538e775443c5be79dd423fcabahrens } else {
fa9e4066f08beec538e775443c5be79dd423fcabahrens thename = origname;
fa9e4066f08beec538e775443c5be79dd423fcabahrens }
fa9e4066f08beec538e775443c5be79dd423fcabahrens
99653d4ee642c6528e88224f12409a5f23060994eschrock if (altroot != NULL && altroot[0] != '/')
ece3d9b3bacef51a5f34d993935eedbb7bb87059lling return (zfs_error_fmt(hdl, EZFS_BADPATH,
99653d4ee642c6528e88224f12409a5f23060994eschrock dgettext(TEXT_DOMAIN, "bad alternate root '%s'"),
99653d4ee642c6528e88224f12409a5f23060994eschrock altroot));
fa9e4066f08beec538e775443c5be79dd423fcabahrens
fa9e4066f08beec538e775443c5be79dd423fcabahrens (void) strlcpy(zc.zc_name, thename, sizeof (zc.zc_name));
fa9e4066f08beec538e775443c5be79dd423fcabahrens
fa9e4066f08beec538e775443c5be79dd423fcabahrens if (altroot != NULL)
e9dbad6f263d5570ed7ff5443ec5b958af8c24d7eschrock (void) strlcpy(zc.zc_value, altroot, sizeof (zc.zc_value));
fa9e4066f08beec538e775443c5be79dd423fcabahrens else
e9dbad6f263d5570ed7ff5443ec5b958af8c24d7eschrock zc.zc_value[0] = '\0';
fa9e4066f08beec538e775443c5be79dd423fcabahrens
fa9e4066f08beec538e775443c5be79dd423fcabahrens verify(nvlist_lookup_uint64(config, ZPOOL_CONFIG_POOL_GUID,
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock &zc.zc_guid) == 0);
fa9e4066f08beec538e775443c5be79dd423fcabahrens
e9dbad6f263d5570ed7ff5443ec5b958af8c24d7eschrock if (zcmd_write_src_nvlist(hdl, &zc, config, NULL) != 0)
99653d4ee642c6528e88224f12409a5f23060994eschrock return (-1);
fa9e4066f08beec538e775443c5be79dd423fcabahrens
fa9e4066f08beec538e775443c5be79dd423fcabahrens ret = 0;
ecd6cf800b63704be73fb264c3f5b6e0dafc068dmarks if (zfs_ioctl(hdl, ZFS_IOC_POOL_IMPORT, &zc) != 0) {
fa9e4066f08beec538e775443c5be79dd423fcabahrens char desc[1024];
fa9e4066f08beec538e775443c5be79dd423fcabahrens if (newname == NULL)
fa9e4066f08beec538e775443c5be79dd423fcabahrens (void) snprintf(desc, sizeof (desc),
fa9e4066f08beec538e775443c5be79dd423fcabahrens dgettext(TEXT_DOMAIN, "cannot import '%s'"),
fa9e4066f08beec538e775443c5be79dd423fcabahrens thename);
fa9e4066f08beec538e775443c5be79dd423fcabahrens else
fa9e4066f08beec538e775443c5be79dd423fcabahrens (void) snprintf(desc, sizeof (desc),
fa9e4066f08beec538e775443c5be79dd423fcabahrens dgettext(TEXT_DOMAIN, "cannot import '%s' as '%s'"),
fa9e4066f08beec538e775443c5be79dd423fcabahrens origname, thename);
fa9e4066f08beec538e775443c5be79dd423fcabahrens
fa9e4066f08beec538e775443c5be79dd423fcabahrens switch (errno) {
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock case ENOTSUP:
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock /*
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock * Unsupported version.
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock */
99653d4ee642c6528e88224f12409a5f23060994eschrock (void) zfs_error(hdl, EZFS_BADVERSION, desc);
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock break;
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock
b5989ec7cd031c9bb066c68877fb5017e45f205beschrock case EINVAL:
b5989ec7cd031c9bb066c68877fb5017e45f205beschrock (void) zfs_error(hdl, EZFS_INVALCONFIG, desc);
b5989ec7cd031c9bb066c68877fb5017e45f205beschrock break;
b5989ec7cd031c9bb066c68877fb5017e45f205beschrock
fa9e4066f08beec538e775443c5be79dd423fcabahrens default:
99653d4ee642c6528e88224f12409a5f23060994eschrock (void) zpool_standard_error(hdl, errno, desc);
fa9e4066f08beec538e775443c5be79dd423fcabahrens }
fa9e4066f08beec538e775443c5be79dd423fcabahrens
fa9e4066f08beec538e775443c5be79dd423fcabahrens ret = -1;
fa9e4066f08beec538e775443c5be79dd423fcabahrens } else {
fa9e4066f08beec538e775443c5be79dd423fcabahrens zpool_handle_t *zhp;
ecd6cf800b63704be73fb264c3f5b6e0dafc068dmarks
fa9e4066f08beec538e775443c5be79dd423fcabahrens /*
fa9e4066f08beec538e775443c5be79dd423fcabahrens * This should never fail, but play it safe anyway.
fa9e4066f08beec538e775443c5be79dd423fcabahrens */
94de1d4cf6ec0a3bf040dcc4b8df107c4ed36b51eschrock if (zpool_open_silent(hdl, thename, &zhp) != 0) {
94de1d4cf6ec0a3bf040dcc4b8df107c4ed36b51eschrock ret = -1;
94de1d4cf6ec0a3bf040dcc4b8df107c4ed36b51eschrock } else if (zhp != NULL) {
fa9e4066f08beec538e775443c5be79dd423fcabahrens ret = zpool_create_zvol_links(zhp);
fa9e4066f08beec538e775443c5be79dd423fcabahrens zpool_close(zhp);
fa9e4066f08beec538e775443c5be79dd423fcabahrens }
ecd6cf800b63704be73fb264c3f5b6e0dafc068dmarks
fa9e4066f08beec538e775443c5be79dd423fcabahrens }
fa9e4066f08beec538e775443c5be79dd423fcabahrens
ecd6cf800b63704be73fb264c3f5b6e0dafc068dmarks
e9dbad6f263d5570ed7ff5443ec5b958af8c24d7eschrock zcmd_free_nvlists(&zc);
fa9e4066f08beec538e775443c5be79dd423fcabahrens return (ret);
fa9e4066f08beec538e775443c5be79dd423fcabahrens}
fa9e4066f08beec538e775443c5be79dd423fcabahrens
fa9e4066f08beec538e775443c5be79dd423fcabahrens/*
fa9e4066f08beec538e775443c5be79dd423fcabahrens * Scrub the pool.
fa9e4066f08beec538e775443c5be79dd423fcabahrens */
fa9e4066f08beec538e775443c5be79dd423fcabahrensint
fa9e4066f08beec538e775443c5be79dd423fcabahrenszpool_scrub(zpool_handle_t *zhp, pool_scrub_type_t type)
fa9e4066f08beec538e775443c5be79dd423fcabahrens{
fa9e4066f08beec538e775443c5be79dd423fcabahrens zfs_cmd_t zc = { 0 };
fa9e4066f08beec538e775443c5be79dd423fcabahrens char msg[1024];
99653d4ee642c6528e88224f12409a5f23060994eschrock libzfs_handle_t *hdl = zhp->zpool_hdl;
fa9e4066f08beec538e775443c5be79dd423fcabahrens
fa9e4066f08beec538e775443c5be79dd423fcabahrens (void) strlcpy(zc.zc_name, zhp->zpool_name, sizeof (zc.zc_name));
fa9e4066f08beec538e775443c5be79dd423fcabahrens zc.zc_cookie = type;
fa9e4066f08beec538e775443c5be79dd423fcabahrens
ecd6cf800b63704be73fb264c3f5b6e0dafc068dmarks if (zfs_ioctl(zhp->zpool_hdl, ZFS_IOC_POOL_SCRUB, &zc) == 0)
fa9e4066f08beec538e775443c5be79dd423fcabahrens return (0);
fa9e4066f08beec538e775443c5be79dd423fcabahrens
fa9e4066f08beec538e775443c5be79dd423fcabahrens (void) snprintf(msg, sizeof (msg),
fa9e4066f08beec538e775443c5be79dd423fcabahrens dgettext(TEXT_DOMAIN, "cannot scrub %s"), zc.zc_name);
fa9e4066f08beec538e775443c5be79dd423fcabahrens
99653d4ee642c6528e88224f12409a5f23060994eschrock if (errno == EBUSY)
99653d4ee642c6528e88224f12409a5f23060994eschrock return (zfs_error(hdl, EZFS_RESILVERING, msg));
99653d4ee642c6528e88224f12409a5f23060994eschrock else
99653d4ee642c6528e88224f12409a5f23060994eschrock return (zpool_standard_error(hdl, errno, msg));
fa9e4066f08beec538e775443c5be79dd423fcabahrens}
fa9e4066f08beec538e775443c5be79dd423fcabahrens
a43d325b828008a3ab54eed57fd7c00b6470172bek/*
a43d325b828008a3ab54eed57fd7c00b6470172bek * 'avail_spare' is set to TRUE if the provided guid refers to an AVAIL
a43d325b828008a3ab54eed57fd7c00b6470172bek * spare; but FALSE if its an INUSE spare.
a43d325b828008a3ab54eed57fd7c00b6470172bek */
99653d4ee642c6528e88224f12409a5f23060994eschrockstatic nvlist_t *
99653d4ee642c6528e88224f12409a5f23060994eschrockvdev_to_nvlist_iter(nvlist_t *nv, const char *search, uint64_t guid,
a43d325b828008a3ab54eed57fd7c00b6470172bek boolean_t *avail_spare)
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock{
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock uint_t c, children;
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock nvlist_t **child;
99653d4ee642c6528e88224f12409a5f23060994eschrock uint64_t theguid, present;
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock char *path;
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock uint64_t wholedisk = 0;
99653d4ee642c6528e88224f12409a5f23060994eschrock nvlist_t *ret;
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock
99653d4ee642c6528e88224f12409a5f23060994eschrock verify(nvlist_lookup_uint64(nv, ZPOOL_CONFIG_GUID, &theguid) == 0);
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock if (search == NULL &&
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock nvlist_lookup_uint64(nv, ZPOOL_CONFIG_NOT_PRESENT, &present) == 0) {
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock /*
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock * If the device has never been present since import, the only
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock * reliable way to match the vdev is by GUID.
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock */
99653d4ee642c6528e88224f12409a5f23060994eschrock if (theguid == guid)
99653d4ee642c6528e88224f12409a5f23060994eschrock return (nv);
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock } else if (search != NULL &&
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock nvlist_lookup_string(nv, ZPOOL_CONFIG_PATH, &path) == 0) {
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock (void) nvlist_lookup_uint64(nv, ZPOOL_CONFIG_WHOLE_DISK,
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock &wholedisk);
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock if (wholedisk) {
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock /*
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock * For whole disks, the internal path has 's0', but the
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock * path passed in by the user doesn't.
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock */
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock if (strlen(search) == strlen(path) - 2 &&
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock strncmp(search, path, strlen(search)) == 0)
99653d4ee642c6528e88224f12409a5f23060994eschrock return (nv);
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock } else if (strcmp(search, path) == 0) {
99653d4ee642c6528e88224f12409a5f23060994eschrock return (nv);
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock }
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock }
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock if (nvlist_lookup_nvlist_array(nv, ZPOOL_CONFIG_CHILDREN,
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock &child, &children) != 0)
99653d4ee642c6528e88224f12409a5f23060994eschrock return (NULL);
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock for (c = 0; c < children; c++)
99653d4ee642c6528e88224f12409a5f23060994eschrock if ((ret = vdev_to_nvlist_iter(child[c], search, guid,
a43d325b828008a3ab54eed57fd7c00b6470172bek avail_spare)) != NULL)
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock return (ret);
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock
99653d4ee642c6528e88224f12409a5f23060994eschrock if (nvlist_lookup_nvlist_array(nv, ZPOOL_CONFIG_SPARES,
99653d4ee642c6528e88224f12409a5f23060994eschrock &child, &children) == 0) {
99653d4ee642c6528e88224f12409a5f23060994eschrock for (c = 0; c < children; c++) {
99653d4ee642c6528e88224f12409a5f23060994eschrock if ((ret = vdev_to_nvlist_iter(child[c], search, guid,
a43d325b828008a3ab54eed57fd7c00b6470172bek avail_spare)) != NULL) {
a43d325b828008a3ab54eed57fd7c00b6470172bek *avail_spare = B_TRUE;
99653d4ee642c6528e88224f12409a5f23060994eschrock return (ret);
99653d4ee642c6528e88224f12409a5f23060994eschrock }
99653d4ee642c6528e88224f12409a5f23060994eschrock }
99653d4ee642c6528e88224f12409a5f23060994eschrock }
99653d4ee642c6528e88224f12409a5f23060994eschrock
99653d4ee642c6528e88224f12409a5f23060994eschrock return (NULL);
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock}
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock
99653d4ee642c6528e88224f12409a5f23060994eschrocknvlist_t *
a43d325b828008a3ab54eed57fd7c00b6470172bekzpool_find_vdev(zpool_handle_t *zhp, const char *path, boolean_t *avail_spare)
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock{
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock char buf[MAXPATHLEN];
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock const char *search;
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock char *end;
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock nvlist_t *nvroot;
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock uint64_t guid;
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock
0917b783fd655a0c943e0b8fb848db2301774947eschrock guid = strtoull(path, &end, 10);
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock if (guid != 0 && *end == '\0') {
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock search = NULL;
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock } else if (path[0] != '/') {
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock (void) snprintf(buf, sizeof (buf), "%s%s", "/dev/dsk/", path);
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock search = buf;
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock } else {
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock search = path;
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock }
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock verify(nvlist_lookup_nvlist(zhp->zpool_config, ZPOOL_CONFIG_VDEV_TREE,
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock &nvroot) == 0);
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock
a43d325b828008a3ab54eed57fd7c00b6470172bek *avail_spare = B_FALSE;
a43d325b828008a3ab54eed57fd7c00b6470172bek return (vdev_to_nvlist_iter(nvroot, search, guid, avail_spare));
a43d325b828008a3ab54eed57fd7c00b6470172bek}
a43d325b828008a3ab54eed57fd7c00b6470172bek
a43d325b828008a3ab54eed57fd7c00b6470172bek/*
a43d325b828008a3ab54eed57fd7c00b6470172bek * Returns TRUE if the given guid corresponds to a spare (INUSE or not).
a43d325b828008a3ab54eed57fd7c00b6470172bek */
a43d325b828008a3ab54eed57fd7c00b6470172bekstatic boolean_t
a43d325b828008a3ab54eed57fd7c00b6470172bekis_spare(zpool_handle_t *zhp, uint64_t guid)
a43d325b828008a3ab54eed57fd7c00b6470172bek{
a43d325b828008a3ab54eed57fd7c00b6470172bek uint64_t spare_guid;
a43d325b828008a3ab54eed57fd7c00b6470172bek nvlist_t *nvroot;
a43d325b828008a3ab54eed57fd7c00b6470172bek nvlist_t **spares;
a43d325b828008a3ab54eed57fd7c00b6470172bek uint_t nspares;
a43d325b828008a3ab54eed57fd7c00b6470172bek int i;
a43d325b828008a3ab54eed57fd7c00b6470172bek
a43d325b828008a3ab54eed57fd7c00b6470172bek verify(nvlist_lookup_nvlist(zhp->zpool_config, ZPOOL_CONFIG_VDEV_TREE,
a43d325b828008a3ab54eed57fd7c00b6470172bek &nvroot) == 0);
a43d325b828008a3ab54eed57fd7c00b6470172bek if (nvlist_lookup_nvlist_array(nvroot, ZPOOL_CONFIG_SPARES,
a43d325b828008a3ab54eed57fd7c00b6470172bek &spares, &nspares) == 0) {
a43d325b828008a3ab54eed57fd7c00b6470172bek for (i = 0; i < nspares; i++) {
a43d325b828008a3ab54eed57fd7c00b6470172bek verify(nvlist_lookup_uint64(spares[i],
a43d325b828008a3ab54eed57fd7c00b6470172bek ZPOOL_CONFIG_GUID, &spare_guid) == 0);
a43d325b828008a3ab54eed57fd7c00b6470172bek if (guid == spare_guid)
a43d325b828008a3ab54eed57fd7c00b6470172bek return (B_TRUE);
a43d325b828008a3ab54eed57fd7c00b6470172bek }
a43d325b828008a3ab54eed57fd7c00b6470172bek }
a43d325b828008a3ab54eed57fd7c00b6470172bek
a43d325b828008a3ab54eed57fd7c00b6470172bek return (B_FALSE);
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock}
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock
fa9e4066f08beec538e775443c5be79dd423fcabahrens/*
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock * Bring the specified vdev online. The 'flags' parameter is a set of the
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock * ZFS_ONLINE_* flags.
fa9e4066f08beec538e775443c5be79dd423fcabahrens */
fa9e4066f08beec538e775443c5be79dd423fcabahrensint
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrockzpool_vdev_online(zpool_handle_t *zhp, const char *path, int flags,
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock vdev_state_t *newstate)
fa9e4066f08beec538e775443c5be79dd423fcabahrens{
fa9e4066f08beec538e775443c5be79dd423fcabahrens zfs_cmd_t zc = { 0 };
fa9e4066f08beec538e775443c5be79dd423fcabahrens char msg[1024];
99653d4ee642c6528e88224f12409a5f23060994eschrock nvlist_t *tgt;
a43d325b828008a3ab54eed57fd7c00b6470172bek boolean_t avail_spare;
99653d4ee642c6528e88224f12409a5f23060994eschrock libzfs_handle_t *hdl = zhp->zpool_hdl;
fa9e4066f08beec538e775443c5be79dd423fcabahrens
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock (void) snprintf(msg, sizeof (msg),
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock dgettext(TEXT_DOMAIN, "cannot online %s"), path);
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock
fa9e4066f08beec538e775443c5be79dd423fcabahrens (void) strlcpy(zc.zc_name, zhp->zpool_name, sizeof (zc.zc_name));
a43d325b828008a3ab54eed57fd7c00b6470172bek if ((tgt = zpool_find_vdev(zhp, path, &avail_spare)) == NULL)
99653d4ee642c6528e88224f12409a5f23060994eschrock return (zfs_error(hdl, EZFS_NODEVICE, msg));
fa9e4066f08beec538e775443c5be79dd423fcabahrens
99653d4ee642c6528e88224f12409a5f23060994eschrock verify(nvlist_lookup_uint64(tgt, ZPOOL_CONFIG_GUID, &zc.zc_guid) == 0);
fa9e4066f08beec538e775443c5be79dd423fcabahrens
a43d325b828008a3ab54eed57fd7c00b6470172bek if (avail_spare || is_spare(zhp, zc.zc_guid) == B_TRUE)
a43d325b828008a3ab54eed57fd7c00b6470172bek return (zfs_error(hdl, EZFS_ISSPARE, msg));
a43d325b828008a3ab54eed57fd7c00b6470172bek
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock zc.zc_cookie = VDEV_STATE_ONLINE;
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock zc.zc_obj = flags;
fa9e4066f08beec538e775443c5be79dd423fcabahrens
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock
ecd6cf800b63704be73fb264c3f5b6e0dafc068dmarks if (zfs_ioctl(zhp->zpool_hdl, ZFS_IOC_VDEV_SET_STATE, &zc) != 0)
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock return (zpool_standard_error(hdl, errno, msg));
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock *newstate = zc.zc_cookie;
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock return (0);
fa9e4066f08beec538e775443c5be79dd423fcabahrens}
fa9e4066f08beec538e775443c5be79dd423fcabahrens
fa9e4066f08beec538e775443c5be79dd423fcabahrens/*
fa9e4066f08beec538e775443c5be79dd423fcabahrens * Take the specified vdev offline
fa9e4066f08beec538e775443c5be79dd423fcabahrens */
fa9e4066f08beec538e775443c5be79dd423fcabahrensint
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrockzpool_vdev_offline(zpool_handle_t *zhp, const char *path, boolean_t istmp)
fa9e4066f08beec538e775443c5be79dd423fcabahrens{
fa9e4066f08beec538e775443c5be79dd423fcabahrens zfs_cmd_t zc = { 0 };
fa9e4066f08beec538e775443c5be79dd423fcabahrens char msg[1024];
99653d4ee642c6528e88224f12409a5f23060994eschrock nvlist_t *tgt;
a43d325b828008a3ab54eed57fd7c00b6470172bek boolean_t avail_spare;
99653d4ee642c6528e88224f12409a5f23060994eschrock libzfs_handle_t *hdl = zhp->zpool_hdl;
fa9e4066f08beec538e775443c5be79dd423fcabahrens
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock (void) snprintf(msg, sizeof (msg),
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock dgettext(TEXT_DOMAIN, "cannot offline %s"), path);
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock
fa9e4066f08beec538e775443c5be79dd423fcabahrens (void) strlcpy(zc.zc_name, zhp->zpool_name, sizeof (zc.zc_name));
a43d325b828008a3ab54eed57fd7c00b6470172bek if ((tgt = zpool_find_vdev(zhp, path, &avail_spare)) == NULL)
99653d4ee642c6528e88224f12409a5f23060994eschrock return (zfs_error(hdl, EZFS_NODEVICE, msg));
99653d4ee642c6528e88224f12409a5f23060994eschrock
99653d4ee642c6528e88224f12409a5f23060994eschrock verify(nvlist_lookup_uint64(tgt, ZPOOL_CONFIG_GUID, &zc.zc_guid) == 0);
fa9e4066f08beec538e775443c5be79dd423fcabahrens
a43d325b828008a3ab54eed57fd7c00b6470172bek if (avail_spare || is_spare(zhp, zc.zc_guid) == B_TRUE)
a43d325b828008a3ab54eed57fd7c00b6470172bek return (zfs_error(hdl, EZFS_ISSPARE, msg));
a43d325b828008a3ab54eed57fd7c00b6470172bek
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock zc.zc_cookie = VDEV_STATE_OFFLINE;
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock zc.zc_obj = istmp ? ZFS_OFFLINE_TEMPORARY : 0;
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock
ecd6cf800b63704be73fb264c3f5b6e0dafc068dmarks if (zfs_ioctl(zhp->zpool_hdl, ZFS_IOC_VDEV_SET_STATE, &zc) == 0)
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock return (0);
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock switch (errno) {
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock case EBUSY:
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock /*
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock * There are no other replicas of this device.
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock */
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock return (zfs_error(hdl, EZFS_NOREPLICAS, msg));
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock default:
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock return (zpool_standard_error(hdl, errno, msg));
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock }
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock}
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock/*
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock * Mark the given vdev faulted.
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock */
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrockint
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrockzpool_vdev_fault(zpool_handle_t *zhp, uint64_t guid)
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock{
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock zfs_cmd_t zc = { 0 };
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock char msg[1024];
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock libzfs_handle_t *hdl = zhp->zpool_hdl;
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock (void) snprintf(msg, sizeof (msg),
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock dgettext(TEXT_DOMAIN, "cannot fault %llu"), guid);
441d80aa4f613b6298fc8bd3151f4be02dbf84fclling
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock (void) strlcpy(zc.zc_name, zhp->zpool_name, sizeof (zc.zc_name));
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock zc.zc_guid = guid;
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock zc.zc_cookie = VDEV_STATE_FAULTED;
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock if (ioctl(zhp->zpool_hdl->libzfs_fd, ZFS_IOC_VDEV_SET_STATE, &zc) == 0)
fa9e4066f08beec538e775443c5be79dd423fcabahrens return (0);
fa9e4066f08beec538e775443c5be79dd423fcabahrens
fa9e4066f08beec538e775443c5be79dd423fcabahrens switch (errno) {
99653d4ee642c6528e88224f12409a5f23060994eschrock case EBUSY:
fa9e4066f08beec538e775443c5be79dd423fcabahrens
fa9e4066f08beec538e775443c5be79dd423fcabahrens /*
fa9e4066f08beec538e775443c5be79dd423fcabahrens * There are no other replicas of this device.
fa9e4066f08beec538e775443c5be79dd423fcabahrens */
99653d4ee642c6528e88224f12409a5f23060994eschrock return (zfs_error(hdl, EZFS_NOREPLICAS, msg));
fa9e4066f08beec538e775443c5be79dd423fcabahrens
99653d4ee642c6528e88224f12409a5f23060994eschrock default:
99653d4ee642c6528e88224f12409a5f23060994eschrock return (zpool_standard_error(hdl, errno, msg));
fa9e4066f08beec538e775443c5be79dd423fcabahrens }
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock}
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock/*
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock * Mark the given vdev degraded.
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock */
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrockint
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrockzpool_vdev_degrade(zpool_handle_t *zhp, uint64_t guid)
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock{
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock zfs_cmd_t zc = { 0 };
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock char msg[1024];
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock libzfs_handle_t *hdl = zhp->zpool_hdl;
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock (void) snprintf(msg, sizeof (msg),
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock dgettext(TEXT_DOMAIN, "cannot degrade %llu"), guid);
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock (void) strlcpy(zc.zc_name, zhp->zpool_name, sizeof (zc.zc_name));
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock zc.zc_guid = guid;
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock zc.zc_cookie = VDEV_STATE_DEGRADED;
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock if (ioctl(zhp->zpool_hdl->libzfs_fd, ZFS_IOC_VDEV_SET_STATE, &zc) == 0)
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock return (0);
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock return (zpool_standard_error(hdl, errno, msg));
99653d4ee642c6528e88224f12409a5f23060994eschrock}
99653d4ee642c6528e88224f12409a5f23060994eschrock
99653d4ee642c6528e88224f12409a5f23060994eschrock/*
99653d4ee642c6528e88224f12409a5f23060994eschrock * Returns TRUE if the given nvlist is a vdev that was originally swapped in as
99653d4ee642c6528e88224f12409a5f23060994eschrock * a hot spare.
99653d4ee642c6528e88224f12409a5f23060994eschrock */
99653d4ee642c6528e88224f12409a5f23060994eschrockstatic boolean_t
99653d4ee642c6528e88224f12409a5f23060994eschrockis_replacing_spare(nvlist_t *search, nvlist_t *tgt, int which)
99653d4ee642c6528e88224f12409a5f23060994eschrock{
99653d4ee642c6528e88224f12409a5f23060994eschrock nvlist_t **child;
99653d4ee642c6528e88224f12409a5f23060994eschrock uint_t c, children;
99653d4ee642c6528e88224f12409a5f23060994eschrock char *type;
99653d4ee642c6528e88224f12409a5f23060994eschrock
99653d4ee642c6528e88224f12409a5f23060994eschrock if (nvlist_lookup_nvlist_array(search, ZPOOL_CONFIG_CHILDREN, &child,
99653d4ee642c6528e88224f12409a5f23060994eschrock &children) == 0) {
99653d4ee642c6528e88224f12409a5f23060994eschrock verify(nvlist_lookup_string(search, ZPOOL_CONFIG_TYPE,
99653d4ee642c6528e88224f12409a5f23060994eschrock &type) == 0);
99653d4ee642c6528e88224f12409a5f23060994eschrock
99653d4ee642c6528e88224f12409a5f23060994eschrock if (strcmp(type, VDEV_TYPE_SPARE) == 0 &&
99653d4ee642c6528e88224f12409a5f23060994eschrock children == 2 && child[which] == tgt)
99653d4ee642c6528e88224f12409a5f23060994eschrock return (B_TRUE);
99653d4ee642c6528e88224f12409a5f23060994eschrock
99653d4ee642c6528e88224f12409a5f23060994eschrock for (c = 0; c < children; c++)
99653d4ee642c6528e88224f12409a5f23060994eschrock if (is_replacing_spare(child[c], tgt, which))
99653d4ee642c6528e88224f12409a5f23060994eschrock return (B_TRUE);
99653d4ee642c6528e88224f12409a5f23060994eschrock }
99653d4ee642c6528e88224f12409a5f23060994eschrock
99653d4ee642c6528e88224f12409a5f23060994eschrock return (B_FALSE);
fa9e4066f08beec538e775443c5be79dd423fcabahrens}
fa9e4066f08beec538e775443c5be79dd423fcabahrens
fa9e4066f08beec538e775443c5be79dd423fcabahrens/*
fa9e4066f08beec538e775443c5be79dd423fcabahrens * Attach new_disk (fully described by nvroot) to old_disk.
8654d0253136055bd4cc2423d87378e8a37f2eb5perrin * If 'replacing' is specified, the new disk will replace the old one.
fa9e4066f08beec538e775443c5be79dd423fcabahrens */
fa9e4066f08beec538e775443c5be79dd423fcabahrensint
fa9e4066f08beec538e775443c5be79dd423fcabahrenszpool_vdev_attach(zpool_handle_t *zhp,
fa9e4066f08beec538e775443c5be79dd423fcabahrens const char *old_disk, const char *new_disk, nvlist_t *nvroot, int replacing)
fa9e4066f08beec538e775443c5be79dd423fcabahrens{
fa9e4066f08beec538e775443c5be79dd423fcabahrens zfs_cmd_t zc = { 0 };
fa9e4066f08beec538e775443c5be79dd423fcabahrens char msg[1024];
fa9e4066f08beec538e775443c5be79dd423fcabahrens int ret;
99653d4ee642c6528e88224f12409a5f23060994eschrock nvlist_t *tgt;
a43d325b828008a3ab54eed57fd7c00b6470172bek boolean_t avail_spare;
8654d0253136055bd4cc2423d87378e8a37f2eb5perrin uint64_t val, is_log;
99653d4ee642c6528e88224f12409a5f23060994eschrock char *path;
99653d4ee642c6528e88224f12409a5f23060994eschrock nvlist_t **child;
99653d4ee642c6528e88224f12409a5f23060994eschrock uint_t children;
99653d4ee642c6528e88224f12409a5f23060994eschrock nvlist_t *config_root;
99653d4ee642c6528e88224f12409a5f23060994eschrock libzfs_handle_t *hdl = zhp->zpool_hdl;
fa9e4066f08beec538e775443c5be79dd423fcabahrens
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock if (replacing)
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock (void) snprintf(msg, sizeof (msg), dgettext(TEXT_DOMAIN,
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock "cannot replace %s with %s"), old_disk, new_disk);
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock else
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock (void) snprintf(msg, sizeof (msg), dgettext(TEXT_DOMAIN,
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock "cannot attach %s to %s"), new_disk, old_disk);
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock
fa9e4066f08beec538e775443c5be79dd423fcabahrens (void) strlcpy(zc.zc_name, zhp->zpool_name, sizeof (zc.zc_name));
a43d325b828008a3ab54eed57fd7c00b6470172bek if ((tgt = zpool_find_vdev(zhp, old_disk, &avail_spare)) == 0)
99653d4ee642c6528e88224f12409a5f23060994eschrock return (zfs_error(hdl, EZFS_NODEVICE, msg));
99653d4ee642c6528e88224f12409a5f23060994eschrock
a43d325b828008a3ab54eed57fd7c00b6470172bek if (avail_spare)
99653d4ee642c6528e88224f12409a5f23060994eschrock return (zfs_error(hdl, EZFS_ISSPARE, msg));
99653d4ee642c6528e88224f12409a5f23060994eschrock
99653d4ee642c6528e88224f12409a5f23060994eschrock verify(nvlist_lookup_uint64(tgt, ZPOOL_CONFIG_GUID, &zc.zc_guid) == 0);
fa9e4066f08beec538e775443c5be79dd423fcabahrens zc.zc_cookie = replacing;
fa9e4066f08beec538e775443c5be79dd423fcabahrens
99653d4ee642c6528e88224f12409a5f23060994eschrock if (nvlist_lookup_nvlist_array(nvroot, ZPOOL_CONFIG_CHILDREN,
99653d4ee642c6528e88224f12409a5f23060994eschrock &child, &children) != 0 || children != 1) {
99653d4ee642c6528e88224f12409a5f23060994eschrock zfs_error_aux(hdl, dgettext(TEXT_DOMAIN,
99653d4ee642c6528e88224f12409a5f23060994eschrock "new device must be a single disk"));
99653d4ee642c6528e88224f12409a5f23060994eschrock return (zfs_error(hdl, EZFS_INVALCONFIG, msg));
99653d4ee642c6528e88224f12409a5f23060994eschrock }
99653d4ee642c6528e88224f12409a5f23060994eschrock
99653d4ee642c6528e88224f12409a5f23060994eschrock verify(nvlist_lookup_nvlist(zpool_get_config(zhp, NULL),
99653d4ee642c6528e88224f12409a5f23060994eschrock ZPOOL_CONFIG_VDEV_TREE, &config_root) == 0);
99653d4ee642c6528e88224f12409a5f23060994eschrock
99653d4ee642c6528e88224f12409a5f23060994eschrock /*
99653d4ee642c6528e88224f12409a5f23060994eschrock * If the target is a hot spare that has been swapped in, we can only
99653d4ee642c6528e88224f12409a5f23060994eschrock * replace it with another hot spare.
99653d4ee642c6528e88224f12409a5f23060994eschrock */
99653d4ee642c6528e88224f12409a5f23060994eschrock if (replacing &&
99653d4ee642c6528e88224f12409a5f23060994eschrock nvlist_lookup_uint64(tgt, ZPOOL_CONFIG_IS_SPARE, &val) == 0 &&
99653d4ee642c6528e88224f12409a5f23060994eschrock nvlist_lookup_string(child[0], ZPOOL_CONFIG_PATH, &path) == 0 &&
a43d325b828008a3ab54eed57fd7c00b6470172bek (zpool_find_vdev(zhp, path, &avail_spare) == NULL ||
a43d325b828008a3ab54eed57fd7c00b6470172bek !avail_spare) && is_replacing_spare(config_root, tgt, 1)) {
99653d4ee642c6528e88224f12409a5f23060994eschrock zfs_error_aux(hdl, dgettext(TEXT_DOMAIN,
99653d4ee642c6528e88224f12409a5f23060994eschrock "can only be replaced by another hot spare"));
99653d4ee642c6528e88224f12409a5f23060994eschrock return (zfs_error(hdl, EZFS_BADTARGET, msg));
99653d4ee642c6528e88224f12409a5f23060994eschrock }
99653d4ee642c6528e88224f12409a5f23060994eschrock
99653d4ee642c6528e88224f12409a5f23060994eschrock /*
99653d4ee642c6528e88224f12409a5f23060994eschrock * If we are attempting to replace a spare, it canot be applied to an
99653d4ee642c6528e88224f12409a5f23060994eschrock * already spared device.
99653d4ee642c6528e88224f12409a5f23060994eschrock */
99653d4ee642c6528e88224f12409a5f23060994eschrock if (replacing &&
99653d4ee642c6528e88224f12409a5f23060994eschrock nvlist_lookup_string(child[0], ZPOOL_CONFIG_PATH, &path) == 0 &&
a43d325b828008a3ab54eed57fd7c00b6470172bek zpool_find_vdev(zhp, path, &avail_spare) != NULL && avail_spare &&
99653d4ee642c6528e88224f12409a5f23060994eschrock is_replacing_spare(config_root, tgt, 0)) {
99653d4ee642c6528e88224f12409a5f23060994eschrock zfs_error_aux(hdl, dgettext(TEXT_DOMAIN,
99653d4ee642c6528e88224f12409a5f23060994eschrock "device has already been replaced with a spare"));
99653d4ee642c6528e88224f12409a5f23060994eschrock return (zfs_error(hdl, EZFS_BADTARGET, msg));
99653d4ee642c6528e88224f12409a5f23060994eschrock }
99653d4ee642c6528e88224f12409a5f23060994eschrock
e9dbad6f263d5570ed7ff5443ec5b958af8c24d7eschrock if (zcmd_write_src_nvlist(hdl, &zc, nvroot, NULL) != 0)
99653d4ee642c6528e88224f12409a5f23060994eschrock return (-1);
fa9e4066f08beec538e775443c5be79dd423fcabahrens
ecd6cf800b63704be73fb264c3f5b6e0dafc068dmarks ret = zfs_ioctl(zhp->zpool_hdl, ZFS_IOC_VDEV_ATTACH, &zc);
fa9e4066f08beec538e775443c5be79dd423fcabahrens
e9dbad6f263d5570ed7ff5443ec5b958af8c24d7eschrock zcmd_free_nvlists(&zc);
fa9e4066f08beec538e775443c5be79dd423fcabahrens
fa9e4066f08beec538e775443c5be79dd423fcabahrens if (ret == 0)
fa9e4066f08beec538e775443c5be79dd423fcabahrens return (0);
fa9e4066f08beec538e775443c5be79dd423fcabahrens
fa9e4066f08beec538e775443c5be79dd423fcabahrens switch (errno) {
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock case ENOTSUP:
fa9e4066f08beec538e775443c5be79dd423fcabahrens /*
fa9e4066f08beec538e775443c5be79dd423fcabahrens * Can't attach to or replace this type of vdev.
fa9e4066f08beec538e775443c5be79dd423fcabahrens */
8654d0253136055bd4cc2423d87378e8a37f2eb5perrin if (replacing) {
8654d0253136055bd4cc2423d87378e8a37f2eb5perrin is_log = B_FALSE;
8654d0253136055bd4cc2423d87378e8a37f2eb5perrin (void) nvlist_lookup_uint64(tgt, ZPOOL_CONFIG_IS_LOG,
8654d0253136055bd4cc2423d87378e8a37f2eb5perrin &is_log);
8654d0253136055bd4cc2423d87378e8a37f2eb5perrin if (is_log)
8654d0253136055bd4cc2423d87378e8a37f2eb5perrin zfs_error_aux(hdl, dgettext(TEXT_DOMAIN,
8654d0253136055bd4cc2423d87378e8a37f2eb5perrin "cannot replace a log with a spare"));
8654d0253136055bd4cc2423d87378e8a37f2eb5perrin else
8654d0253136055bd4cc2423d87378e8a37f2eb5perrin zfs_error_aux(hdl, dgettext(TEXT_DOMAIN,
8654d0253136055bd4cc2423d87378e8a37f2eb5perrin "cannot replace a replacing device"));
8654d0253136055bd4cc2423d87378e8a37f2eb5perrin } else {
99653d4ee642c6528e88224f12409a5f23060994eschrock zfs_error_aux(hdl, dgettext(TEXT_DOMAIN,
99653d4ee642c6528e88224f12409a5f23060994eschrock "can only attach to mirrors and top-level "
99653d4ee642c6528e88224f12409a5f23060994eschrock "disks"));
8654d0253136055bd4cc2423d87378e8a37f2eb5perrin }
99653d4ee642c6528e88224f12409a5f23060994eschrock (void) zfs_error(hdl, EZFS_BADTARGET, msg);
fa9e4066f08beec538e775443c5be79dd423fcabahrens break;
fa9e4066f08beec538e775443c5be79dd423fcabahrens
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock case EINVAL:
fa9e4066f08beec538e775443c5be79dd423fcabahrens /*
fa9e4066f08beec538e775443c5be79dd423fcabahrens * The new device must be a single disk.
fa9e4066f08beec538e775443c5be79dd423fcabahrens */
99653d4ee642c6528e88224f12409a5f23060994eschrock zfs_error_aux(hdl, dgettext(TEXT_DOMAIN,
99653d4ee642c6528e88224f12409a5f23060994eschrock "new device must be a single disk"));
99653d4ee642c6528e88224f12409a5f23060994eschrock (void) zfs_error(hdl, EZFS_INVALCONFIG, msg);
fa9e4066f08beec538e775443c5be79dd423fcabahrens break;
fa9e4066f08beec538e775443c5be79dd423fcabahrens
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock case EBUSY:
99653d4ee642c6528e88224f12409a5f23060994eschrock zfs_error_aux(hdl, dgettext(TEXT_DOMAIN, "%s is busy"),
99653d4ee642c6528e88224f12409a5f23060994eschrock new_disk);
99653d4ee642c6528e88224f12409a5f23060994eschrock (void) zfs_error(hdl, EZFS_BADDEV, msg);
fa9e4066f08beec538e775443c5be79dd423fcabahrens break;
fa9e4066f08beec538e775443c5be79dd423fcabahrens
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock case EOVERFLOW:
fa9e4066f08beec538e775443c5be79dd423fcabahrens /*
fa9e4066f08beec538e775443c5be79dd423fcabahrens * The new device is too small.
fa9e4066f08beec538e775443c5be79dd423fcabahrens */
99653d4ee642c6528e88224f12409a5f23060994eschrock zfs_error_aux(hdl, dgettext(TEXT_DOMAIN,
99653d4ee642c6528e88224f12409a5f23060994eschrock "device is too small"));
99653d4ee642c6528e88224f12409a5f23060994eschrock (void) zfs_error(hdl, EZFS_BADDEV, msg);
fa9e4066f08beec538e775443c5be79dd423fcabahrens break;
fa9e4066f08beec538e775443c5be79dd423fcabahrens
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock case EDOM:
fa9e4066f08beec538e775443c5be79dd423fcabahrens /*
fa9e4066f08beec538e775443c5be79dd423fcabahrens * The new device has a different alignment requirement.
fa9e4066f08beec538e775443c5be79dd423fcabahrens */
99653d4ee642c6528e88224f12409a5f23060994eschrock zfs_error_aux(hdl, dgettext(TEXT_DOMAIN,
99653d4ee642c6528e88224f12409a5f23060994eschrock "devices have different sector alignment"));
99653d4ee642c6528e88224f12409a5f23060994eschrock (void) zfs_error(hdl, EZFS_BADDEV, msg);
fa9e4066f08beec538e775443c5be79dd423fcabahrens break;
fa9e4066f08beec538e775443c5be79dd423fcabahrens
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock case ENAMETOOLONG:
fa9e4066f08beec538e775443c5be79dd423fcabahrens /*
fa9e4066f08beec538e775443c5be79dd423fcabahrens * The resulting top-level vdev spec won't fit in the label.
fa9e4066f08beec538e775443c5be79dd423fcabahrens */
99653d4ee642c6528e88224f12409a5f23060994eschrock (void) zfs_error(hdl, EZFS_DEVOVERFLOW, msg);
fa9e4066f08beec538e775443c5be79dd423fcabahrens break;
fa9e4066f08beec538e775443c5be79dd423fcabahrens
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock default:
99653d4ee642c6528e88224f12409a5f23060994eschrock (void) zpool_standard_error(hdl, errno, msg);
fa9e4066f08beec538e775443c5be79dd423fcabahrens }
fa9e4066f08beec538e775443c5be79dd423fcabahrens
99653d4ee642c6528e88224f12409a5f23060994eschrock return (-1);
fa9e4066f08beec538e775443c5be79dd423fcabahrens}
fa9e4066f08beec538e775443c5be79dd423fcabahrens
fa9e4066f08beec538e775443c5be79dd423fcabahrens/*
fa9e4066f08beec538e775443c5be79dd423fcabahrens * Detach the specified device.
fa9e4066f08beec538e775443c5be79dd423fcabahrens */
fa9e4066f08beec538e775443c5be79dd423fcabahrensint
fa9e4066f08beec538e775443c5be79dd423fcabahrenszpool_vdev_detach(zpool_handle_t *zhp, const char *path)
fa9e4066f08beec538e775443c5be79dd423fcabahrens{
fa9e4066f08beec538e775443c5be79dd423fcabahrens zfs_cmd_t zc = { 0 };
fa9e4066f08beec538e775443c5be79dd423fcabahrens char msg[1024];
99653d4ee642c6528e88224f12409a5f23060994eschrock nvlist_t *tgt;
a43d325b828008a3ab54eed57fd7c00b6470172bek boolean_t avail_spare;
99653d4ee642c6528e88224f12409a5f23060994eschrock libzfs_handle_t *hdl = zhp->zpool_hdl;
fa9e4066f08beec538e775443c5be79dd423fcabahrens
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock (void) snprintf(msg, sizeof (msg),
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock dgettext(TEXT_DOMAIN, "cannot detach %s"), path);
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock
fa9e4066f08beec538e775443c5be79dd423fcabahrens (void) strlcpy(zc.zc_name, zhp->zpool_name, sizeof (zc.zc_name));
a43d325b828008a3ab54eed57fd7c00b6470172bek if ((tgt = zpool_find_vdev(zhp, path, &avail_spare)) == 0)
99653d4ee642c6528e88224f12409a5f23060994eschrock return (zfs_error(hdl, EZFS_NODEVICE, msg));
fa9e4066f08beec538e775443c5be79dd423fcabahrens
a43d325b828008a3ab54eed57fd7c00b6470172bek if (avail_spare)
99653d4ee642c6528e88224f12409a5f23060994eschrock return (zfs_error(hdl, EZFS_ISSPARE, msg));
99653d4ee642c6528e88224f12409a5f23060994eschrock
99653d4ee642c6528e88224f12409a5f23060994eschrock verify(nvlist_lookup_uint64(tgt, ZPOOL_CONFIG_GUID, &zc.zc_guid) == 0);
99653d4ee642c6528e88224f12409a5f23060994eschrock
ecd6cf800b63704be73fb264c3f5b6e0dafc068dmarks if (zfs_ioctl(hdl, ZFS_IOC_VDEV_DETACH, &zc) == 0)
fa9e4066f08beec538e775443c5be79dd423fcabahrens return (0);
fa9e4066f08beec538e775443c5be79dd423fcabahrens
fa9e4066f08beec538e775443c5be79dd423fcabahrens switch (errno) {
fa9e4066f08beec538e775443c5be79dd423fcabahrens
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock case ENOTSUP:
fa9e4066f08beec538e775443c5be79dd423fcabahrens /*
fa9e4066f08beec538e775443c5be79dd423fcabahrens * Can't detach from this type of vdev.
fa9e4066f08beec538e775443c5be79dd423fcabahrens */
99653d4ee642c6528e88224f12409a5f23060994eschrock zfs_error_aux(hdl, dgettext(TEXT_DOMAIN, "only "
99653d4ee642c6528e88224f12409a5f23060994eschrock "applicable to mirror and replacing vdevs"));
99653d4ee642c6528e88224f12409a5f23060994eschrock (void) zfs_error(zhp->zpool_hdl, EZFS_BADTARGET, msg);
fa9e4066f08beec538e775443c5be79dd423fcabahrens break;
fa9e4066f08beec538e775443c5be79dd423fcabahrens
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock case EBUSY:
fa9e4066f08beec538e775443c5be79dd423fcabahrens /*
fa9e4066f08beec538e775443c5be79dd423fcabahrens * There are no other replicas of this device.
fa9e4066f08beec538e775443c5be79dd423fcabahrens */
99653d4ee642c6528e88224f12409a5f23060994eschrock (void) zfs_error(hdl, EZFS_NOREPLICAS, msg);
fa9e4066f08beec538e775443c5be79dd423fcabahrens break;
fa9e4066f08beec538e775443c5be79dd423fcabahrens
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock default:
99653d4ee642c6528e88224f12409a5f23060994eschrock (void) zpool_standard_error(hdl, errno, msg);
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock }
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock
99653d4ee642c6528e88224f12409a5f23060994eschrock return (-1);
99653d4ee642c6528e88224f12409a5f23060994eschrock}
99653d4ee642c6528e88224f12409a5f23060994eschrock
99653d4ee642c6528e88224f12409a5f23060994eschrock/*
99653d4ee642c6528e88224f12409a5f23060994eschrock * Remove the given device. Currently, this is supported only for hot spares.
99653d4ee642c6528e88224f12409a5f23060994eschrock */
99653d4ee642c6528e88224f12409a5f23060994eschrockint
99653d4ee642c6528e88224f12409a5f23060994eschrockzpool_vdev_remove(zpool_handle_t *zhp, const char *path)
99653d4ee642c6528e88224f12409a5f23060994eschrock{
99653d4ee642c6528e88224f12409a5f23060994eschrock zfs_cmd_t zc = { 0 };
99653d4ee642c6528e88224f12409a5f23060994eschrock char msg[1024];
99653d4ee642c6528e88224f12409a5f23060994eschrock nvlist_t *tgt;
a43d325b828008a3ab54eed57fd7c00b6470172bek boolean_t avail_spare;
99653d4ee642c6528e88224f12409a5f23060994eschrock libzfs_handle_t *hdl = zhp->zpool_hdl;
99653d4ee642c6528e88224f12409a5f23060994eschrock
99653d4ee642c6528e88224f12409a5f23060994eschrock (void) snprintf(msg, sizeof (msg),
99653d4ee642c6528e88224f12409a5f23060994eschrock dgettext(TEXT_DOMAIN, "cannot remove %s"), path);
99653d4ee642c6528e88224f12409a5f23060994eschrock
99653d4ee642c6528e88224f12409a5f23060994eschrock (void) strlcpy(zc.zc_name, zhp->zpool_name, sizeof (zc.zc_name));
a43d325b828008a3ab54eed57fd7c00b6470172bek if ((tgt = zpool_find_vdev(zhp, path, &avail_spare)) == 0)
99653d4ee642c6528e88224f12409a5f23060994eschrock return (zfs_error(hdl, EZFS_NODEVICE, msg));
99653d4ee642c6528e88224f12409a5f23060994eschrock
a43d325b828008a3ab54eed57fd7c00b6470172bek if (!avail_spare) {
99653d4ee642c6528e88224f12409a5f23060994eschrock zfs_error_aux(hdl, dgettext(TEXT_DOMAIN,
39c23413b8df94a95f67b34cfd4a4dfc3fd0b48deschrock "only inactive hot spares can be removed"));
99653d4ee642c6528e88224f12409a5f23060994eschrock return (zfs_error(hdl, EZFS_NODEVICE, msg));
99653d4ee642c6528e88224f12409a5f23060994eschrock }
99653d4ee642c6528e88224f12409a5f23060994eschrock
99653d4ee642c6528e88224f12409a5f23060994eschrock verify(nvlist_lookup_uint64(tgt, ZPOOL_CONFIG_GUID, &zc.zc_guid) == 0);
99653d4ee642c6528e88224f12409a5f23060994eschrock
ecd6cf800b63704be73fb264c3f5b6e0dafc068dmarks if (zfs_ioctl(hdl, ZFS_IOC_VDEV_REMOVE, &zc) == 0)
99653d4ee642c6528e88224f12409a5f23060994eschrock return (0);
99653d4ee642c6528e88224f12409a5f23060994eschrock
99653d4ee642c6528e88224f12409a5f23060994eschrock return (zpool_standard_error(hdl, errno, msg));
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock}
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock/*
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock * Clear the errors for the pool, or the particular device if specified.
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock */
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrockint
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrockzpool_clear(zpool_handle_t *zhp, const char *path)
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock{
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock zfs_cmd_t zc = { 0 };
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock char msg[1024];
99653d4ee642c6528e88224f12409a5f23060994eschrock nvlist_t *tgt;
a43d325b828008a3ab54eed57fd7c00b6470172bek boolean_t avail_spare;
99653d4ee642c6528e88224f12409a5f23060994eschrock libzfs_handle_t *hdl = zhp->zpool_hdl;
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock if (path)
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock (void) snprintf(msg, sizeof (msg),
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock dgettext(TEXT_DOMAIN, "cannot clear errors for %s"),
e9dbad6f263d5570ed7ff5443ec5b958af8c24d7eschrock path);
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock else
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock (void) snprintf(msg, sizeof (msg),
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock dgettext(TEXT_DOMAIN, "cannot clear errors for %s"),
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock zhp->zpool_name);
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock (void) strlcpy(zc.zc_name, zhp->zpool_name, sizeof (zc.zc_name));
99653d4ee642c6528e88224f12409a5f23060994eschrock if (path) {
a43d325b828008a3ab54eed57fd7c00b6470172bek if ((tgt = zpool_find_vdev(zhp, path, &avail_spare)) == 0)
99653d4ee642c6528e88224f12409a5f23060994eschrock return (zfs_error(hdl, EZFS_NODEVICE, msg));
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock
a43d325b828008a3ab54eed57fd7c00b6470172bek if (avail_spare)
99653d4ee642c6528e88224f12409a5f23060994eschrock return (zfs_error(hdl, EZFS_ISSPARE, msg));
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock
99653d4ee642c6528e88224f12409a5f23060994eschrock verify(nvlist_lookup_uint64(tgt, ZPOOL_CONFIG_GUID,
99653d4ee642c6528e88224f12409a5f23060994eschrock &zc.zc_guid) == 0);
fa9e4066f08beec538e775443c5be79dd423fcabahrens }
fa9e4066f08beec538e775443c5be79dd423fcabahrens
ecd6cf800b63704be73fb264c3f5b6e0dafc068dmarks if (zfs_ioctl(hdl, ZFS_IOC_CLEAR, &zc) == 0)
99653d4ee642c6528e88224f12409a5f23060994eschrock return (0);
99653d4ee642c6528e88224f12409a5f23060994eschrock
99653d4ee642c6528e88224f12409a5f23060994eschrock return (zpool_standard_error(hdl, errno, msg));
fa9e4066f08beec538e775443c5be79dd423fcabahrens}
fa9e4066f08beec538e775443c5be79dd423fcabahrens
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock/*
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock * Similar to zpool_clear(), but takes a GUID (used by fmd).
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock */
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrockint
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrockzpool_vdev_clear(zpool_handle_t *zhp, uint64_t guid)
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock{
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock zfs_cmd_t zc = { 0 };
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock char msg[1024];
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock libzfs_handle_t *hdl = zhp->zpool_hdl;
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock (void) snprintf(msg, sizeof (msg),
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock dgettext(TEXT_DOMAIN, "cannot clear errors for %llx"),
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock guid);
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock (void) strlcpy(zc.zc_name, zhp->zpool_name, sizeof (zc.zc_name));
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock zc.zc_guid = guid;
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock if (ioctl(hdl->libzfs_fd, ZFS_IOC_CLEAR, &zc) == 0)
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock return (0);
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock return (zpool_standard_error(hdl, errno, msg));
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock}
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl/*
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl * Iterate over all zvols in a given pool by walking the /dev/zvol/dsk/<pool>
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl * hierarchy.
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl */
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahlint
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahlzpool_iter_zvol(zpool_handle_t *zhp, int (*cb)(const char *, void *),
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl void *data)
fa9e4066f08beec538e775443c5be79dd423fcabahrens{
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl libzfs_handle_t *hdl = zhp->zpool_hdl;
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl char (*paths)[MAXPATHLEN];
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl size_t size = 4;
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl int curr, fd, base, ret = 0;
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl DIR *dirp;
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl struct dirent *dp;
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl struct stat st;
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl if ((base = open("/dev/zvol/dsk", O_RDONLY)) < 0)
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl return (errno == ENOENT ? 0 : -1);
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl if (fstatat(base, zhp->zpool_name, &st, 0) != 0) {
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl int err = errno;
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl (void) close(base);
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl return (err == ENOENT ? 0 : -1);
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl }
fa9e4066f08beec538e775443c5be79dd423fcabahrens
fa9e4066f08beec538e775443c5be79dd423fcabahrens /*
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl * Oddly this wasn't a directory -- ignore that failure since we
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl * know there are no links lower in the (non-existant) hierarchy.
fa9e4066f08beec538e775443c5be79dd423fcabahrens */
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl if (!S_ISDIR(st.st_mode)) {
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl (void) close(base);
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl return (0);
fa9e4066f08beec538e775443c5be79dd423fcabahrens }
fa9e4066f08beec538e775443c5be79dd423fcabahrens
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl if ((paths = zfs_alloc(hdl, size * sizeof (paths[0]))) == NULL) {
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl (void) close(base);
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl return (-1);
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl }
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl (void) strlcpy(paths[0], zhp->zpool_name, sizeof (paths[0]));
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl curr = 0;
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl while (curr >= 0) {
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl if (fstatat(base, paths[curr], &st, AT_SYMLINK_NOFOLLOW) != 0)
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl goto err;
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl if (S_ISDIR(st.st_mode)) {
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl if ((fd = openat(base, paths[curr], O_RDONLY)) < 0)
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl goto err;
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl if ((dirp = fdopendir(fd)) == NULL) {
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl (void) close(fd);
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl goto err;
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl }
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl while ((dp = readdir(dirp)) != NULL) {
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl if (dp->d_name[0] == '.')
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl continue;
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl if (curr + 1 == size) {
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl paths = zfs_realloc(hdl, paths,
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl size * sizeof (paths[0]),
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl size * 2 * sizeof (paths[0]));
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl if (paths == NULL) {
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl (void) closedir(dirp);
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl (void) close(fd);
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl goto err;
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl }
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl size *= 2;
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl }
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl (void) strlcpy(paths[curr + 1], paths[curr],
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl sizeof (paths[curr + 1]));
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl (void) strlcat(paths[curr], "/",
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl sizeof (paths[curr]));
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl (void) strlcat(paths[curr], dp->d_name,
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl sizeof (paths[curr]));
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl curr++;
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl }
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl (void) closedir(dirp);
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl } else {
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl if ((ret = cb(paths[curr], data)) != 0)
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl break;
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl }
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl curr--;
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl }
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl free(paths);
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl (void) close(base);
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl return (ret);
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahlerr:
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl free(paths);
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl (void) close(base);
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl return (-1);
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl}
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahltypedef struct zvol_cb {
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl zpool_handle_t *zcb_pool;
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl boolean_t zcb_create;
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl} zvol_cb_t;
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl/*ARGSUSED*/
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahlstatic int
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahldo_zvol_create(zfs_handle_t *zhp, void *data)
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl{
3aefe2c713c6309989772b3ff922a68037186a75ahrens int ret = 0;
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl
3aefe2c713c6309989772b3ff922a68037186a75ahrens if (ZFS_IS_VOLUME(zhp)) {
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl (void) zvol_create_link(zhp->zfs_hdl, zhp->zfs_name);
3aefe2c713c6309989772b3ff922a68037186a75ahrens ret = zfs_iter_snapshots(zhp, do_zvol_create, NULL);
3aefe2c713c6309989772b3ff922a68037186a75ahrens }
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl
3aefe2c713c6309989772b3ff922a68037186a75ahrens if (ret == 0)
3aefe2c713c6309989772b3ff922a68037186a75ahrens ret = zfs_iter_filesystems(zhp, do_zvol_create, NULL);
fa9e4066f08beec538e775443c5be79dd423fcabahrens
fa9e4066f08beec538e775443c5be79dd423fcabahrens zfs_close(zhp);
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl
fa9e4066f08beec538e775443c5be79dd423fcabahrens return (ret);
fa9e4066f08beec538e775443c5be79dd423fcabahrens}
fa9e4066f08beec538e775443c5be79dd423fcabahrens
fa9e4066f08beec538e775443c5be79dd423fcabahrens/*
fa9e4066f08beec538e775443c5be79dd423fcabahrens * Iterate over all zvols in the pool and make any necessary minor nodes.
fa9e4066f08beec538e775443c5be79dd423fcabahrens */
fa9e4066f08beec538e775443c5be79dd423fcabahrensint
fa9e4066f08beec538e775443c5be79dd423fcabahrenszpool_create_zvol_links(zpool_handle_t *zhp)
fa9e4066f08beec538e775443c5be79dd423fcabahrens{
fa9e4066f08beec538e775443c5be79dd423fcabahrens zfs_handle_t *zfp;
fa9e4066f08beec538e775443c5be79dd423fcabahrens int ret;
fa9e4066f08beec538e775443c5be79dd423fcabahrens
fa9e4066f08beec538e775443c5be79dd423fcabahrens /*
fa9e4066f08beec538e775443c5be79dd423fcabahrens * If the pool is unavailable, just return success.
fa9e4066f08beec538e775443c5be79dd423fcabahrens */
99653d4ee642c6528e88224f12409a5f23060994eschrock if ((zfp = make_dataset_handle(zhp->zpool_hdl,
99653d4ee642c6528e88224f12409a5f23060994eschrock zhp->zpool_name)) == NULL)
fa9e4066f08beec538e775443c5be79dd423fcabahrens return (0);
fa9e4066f08beec538e775443c5be79dd423fcabahrens
3aefe2c713c6309989772b3ff922a68037186a75ahrens ret = zfs_iter_filesystems(zfp, do_zvol_create, NULL);
fa9e4066f08beec538e775443c5be79dd423fcabahrens
fa9e4066f08beec538e775443c5be79dd423fcabahrens zfs_close(zfp);
fa9e4066f08beec538e775443c5be79dd423fcabahrens return (ret);
fa9e4066f08beec538e775443c5be79dd423fcabahrens}
fa9e4066f08beec538e775443c5be79dd423fcabahrens
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahlstatic int
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahldo_zvol_remove(const char *dataset, void *data)
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl{
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl zpool_handle_t *zhp = data;
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl return (zvol_remove_link(zhp->zpool_hdl, dataset));
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl}
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl
fa9e4066f08beec538e775443c5be79dd423fcabahrens/*
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl * Iterate over all zvols in the pool and remove any minor nodes. We iterate
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl * by examining the /dev links so that a corrupted pool doesn't impede this
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl * operation.
fa9e4066f08beec538e775443c5be79dd423fcabahrens */
fa9e4066f08beec538e775443c5be79dd423fcabahrensint
fa9e4066f08beec538e775443c5be79dd423fcabahrenszpool_remove_zvol_links(zpool_handle_t *zhp)
fa9e4066f08beec538e775443c5be79dd423fcabahrens{
f3861e1a2ceec23a5b699c24d814b7775a9e0b52ahl return (zpool_iter_zvol(zhp, do_zvol_remove, zhp));
fa9e4066f08beec538e775443c5be79dd423fcabahrens}
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock/*
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock * Convert from a devid string to a path.
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock */
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrockstatic char *
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrockdevid_to_path(char *devid_str)
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock{
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock ddi_devid_t devid;
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock char *minor;
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock char *path;
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock devid_nmlist_t *list = NULL;
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock int ret;
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock if (devid_str_decode(devid_str, &devid, &minor) != 0)
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock return (NULL);
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock ret = devid_deviceid_to_nmlist("/dev", devid, minor, &list);
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock devid_str_free(minor);
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock devid_free(devid);
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock if (ret != 0)
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock return (NULL);
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock
99653d4ee642c6528e88224f12409a5f23060994eschrock if ((path = strdup(list[0].devname)) == NULL)
99653d4ee642c6528e88224f12409a5f23060994eschrock return (NULL);
99653d4ee642c6528e88224f12409a5f23060994eschrock
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock devid_free_nmlist(list);
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock return (path);
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock}
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock/*
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock * Convert from a path to a devid string.
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock */
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrockstatic char *
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrockpath_to_devid(const char *path)
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock{
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock int fd;
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock ddi_devid_t devid;
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock char *minor, *ret;
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock if ((fd = open(path, O_RDONLY)) < 0)
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock return (NULL);
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock minor = NULL;
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock ret = NULL;
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock if (devid_get(fd, &devid) == 0) {
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock if (devid_get_minor_name(fd, &minor) == 0)
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock ret = devid_str_encode(devid, minor);
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock if (minor != NULL)
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock devid_str_free(minor);
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock devid_free(devid);
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock }
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock (void) close(fd);
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock return (ret);
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock}
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock/*
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock * Issue the necessary ioctl() to update the stored path value for the vdev. We
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock * ignore any failure here, since a common case is for an unprivileged user to
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock * type 'zpool status', and we'll display the correct information anyway.
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock */
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrockstatic void
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrockset_path(zpool_handle_t *zhp, nvlist_t *nv, const char *path)
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock{
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock zfs_cmd_t zc = { 0 };
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock (void) strncpy(zc.zc_name, zhp->zpool_name, sizeof (zc.zc_name));
e9dbad6f263d5570ed7ff5443ec5b958af8c24d7eschrock (void) strncpy(zc.zc_value, path, sizeof (zc.zc_value));
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock verify(nvlist_lookup_uint64(nv, ZPOOL_CONFIG_GUID,
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock &zc.zc_guid) == 0);
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock
99653d4ee642c6528e88224f12409a5f23060994eschrock (void) ioctl(zhp->zpool_hdl->libzfs_fd, ZFS_IOC_VDEV_SETPATH, &zc);
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock}
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock/*
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock * Given a vdev, return the name to display in iostat. If the vdev has a path,
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock * we use that, stripping off any leading "/dev/dsk/"; if not, we use the type.
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock * We also check if this is a whole disk, in which case we strip off the
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock * trailing 's0' slice name.
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock *
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock * This routine is also responsible for identifying when disks have been
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock * reconfigured in a new location. The kernel will have opened the device by
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock * devid, but the path will still refer to the old location. To catch this, we
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock * first do a path -> devid translation (which is fast for the common case). If
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock * the devid matches, we're done. If not, we do a reverse devid -> path
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock * translation and issue the appropriate ioctl() to update the path of the vdev.
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock * If 'zhp' is NULL, then this is an exported pool, and we don't need to do any
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock * of these checks.
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock */
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrockchar *
99653d4ee642c6528e88224f12409a5f23060994eschrockzpool_vdev_name(libzfs_handle_t *hdl, zpool_handle_t *zhp, nvlist_t *nv)
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock{
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock char *path, *devid;
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock uint64_t value;
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock char buf[64];
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock vdev_stat_t *vs;
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock uint_t vsc;
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock if (nvlist_lookup_uint64(nv, ZPOOL_CONFIG_NOT_PRESENT,
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock &value) == 0) {
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock verify(nvlist_lookup_uint64(nv, ZPOOL_CONFIG_GUID,
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock &value) == 0);
5ad820458efd0fdb914baff9c1447c22b819fa23nd (void) snprintf(buf, sizeof (buf), "%llu",
5ad820458efd0fdb914baff9c1447c22b819fa23nd (u_longlong_t)value);
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock path = buf;
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock } else if (nvlist_lookup_string(nv, ZPOOL_CONFIG_PATH, &path) == 0) {
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock /*
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock * If the device is dead (faulted, offline, etc) then don't
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock * bother opening it. Otherwise we may be forcing the user to
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock * open a misbehaving device, which can have undesirable
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock * effects.
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock */
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock if ((nvlist_lookup_uint64_array(nv, ZPOOL_CONFIG_STATS,
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock (uint64_t **)&vs, &vsc) != 0 ||
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock vs->vs_state >= VDEV_STATE_DEGRADED) &&
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock zhp != NULL &&
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock nvlist_lookup_string(nv, ZPOOL_CONFIG_DEVID, &devid) == 0) {
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock /*
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock * Determine if the current path is correct.
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock */
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock char *newdevid = path_to_devid(path);
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock if (newdevid == NULL ||
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock strcmp(devid, newdevid) != 0) {
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock char *newpath;
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock if ((newpath = devid_to_path(devid)) != NULL) {
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock /*
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock * Update the path appropriately.
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock */
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock set_path(zhp, nv, newpath);
99653d4ee642c6528e88224f12409a5f23060994eschrock if (nvlist_add_string(nv,
99653d4ee642c6528e88224f12409a5f23060994eschrock ZPOOL_CONFIG_PATH, newpath) == 0)
99653d4ee642c6528e88224f12409a5f23060994eschrock verify(nvlist_lookup_string(nv,
99653d4ee642c6528e88224f12409a5f23060994eschrock ZPOOL_CONFIG_PATH,
99653d4ee642c6528e88224f12409a5f23060994eschrock &path) == 0);
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock free(newpath);
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock }
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock }
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock
99653d4ee642c6528e88224f12409a5f23060994eschrock if (newdevid)
99653d4ee642c6528e88224f12409a5f23060994eschrock devid_str_free(newdevid);
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock }
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock if (strncmp(path, "/dev/dsk/", 9) == 0)
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock path += 9;
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock if (nvlist_lookup_uint64(nv, ZPOOL_CONFIG_WHOLE_DISK,
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock &value) == 0 && value) {
99653d4ee642c6528e88224f12409a5f23060994eschrock char *tmp = zfs_strdup(hdl, path);
99653d4ee642c6528e88224f12409a5f23060994eschrock if (tmp == NULL)
99653d4ee642c6528e88224f12409a5f23060994eschrock return (NULL);
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock tmp[strlen(path) - 2] = '\0';
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock return (tmp);
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock }
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock } else {
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock verify(nvlist_lookup_string(nv, ZPOOL_CONFIG_TYPE, &path) == 0);
99653d4ee642c6528e88224f12409a5f23060994eschrock
99653d4ee642c6528e88224f12409a5f23060994eschrock /*
99653d4ee642c6528e88224f12409a5f23060994eschrock * If it's a raidz device, we need to stick in the parity level.
99653d4ee642c6528e88224f12409a5f23060994eschrock */
99653d4ee642c6528e88224f12409a5f23060994eschrock if (strcmp(path, VDEV_TYPE_RAIDZ) == 0) {
99653d4ee642c6528e88224f12409a5f23060994eschrock verify(nvlist_lookup_uint64(nv, ZPOOL_CONFIG_NPARITY,
99653d4ee642c6528e88224f12409a5f23060994eschrock &value) == 0);
99653d4ee642c6528e88224f12409a5f23060994eschrock (void) snprintf(buf, sizeof (buf), "%s%llu", path,
5ad820458efd0fdb914baff9c1447c22b819fa23nd (u_longlong_t)value);
99653d4ee642c6528e88224f12409a5f23060994eschrock path = buf;
99653d4ee642c6528e88224f12409a5f23060994eschrock }
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock }
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock
99653d4ee642c6528e88224f12409a5f23060994eschrock return (zfs_strdup(hdl, path));
c67d9675bbc8392fe45f3a7dfbda1ad4daa1eb07eschrock}
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrockstatic int
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrockzbookmark_compare(const void *a, const void *b)
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock{
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock return (memcmp(a, b, sizeof (zbookmark_t)));
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock}
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock/*
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock * Retrieve the persistent error log, uniquify the members, and return to the
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock * caller.
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock */
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrockint
55434c770c89aa1b84474f2559a106803511aba0ekzpool_get_errlog(zpool_handle_t *zhp, nvlist_t **nverrlistp)
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock{
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock zfs_cmd_t zc = { 0 };
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock uint64_t count;
e9dbad6f263d5570ed7ff5443ec5b958af8c24d7eschrock zbookmark_t *zb = NULL;
55434c770c89aa1b84474f2559a106803511aba0ek int i;
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock /*
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock * Retrieve the raw error list from the kernel. If the number of errors
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock * has increased, allocate more space and continue until we get the
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock * entire list.
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock */
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock verify(nvlist_lookup_uint64(zhp->zpool_config, ZPOOL_CONFIG_ERRCOUNT,
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock &count) == 0);
75519f380eac71fe6d10b26e736f01567d6c13c9ek if (count == 0)
75519f380eac71fe6d10b26e736f01567d6c13c9ek return (0);
e9dbad6f263d5570ed7ff5443ec5b958af8c24d7eschrock if ((zc.zc_nvlist_dst = (uintptr_t)zfs_alloc(zhp->zpool_hdl,
5ad820458efd0fdb914baff9c1447c22b819fa23nd count * sizeof (zbookmark_t))) == (uintptr_t)NULL)
99653d4ee642c6528e88224f12409a5f23060994eschrock return (-1);
e9dbad6f263d5570ed7ff5443ec5b958af8c24d7eschrock zc.zc_nvlist_dst_size = count;
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock (void) strcpy(zc.zc_name, zhp->zpool_name);
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock for (;;) {
99653d4ee642c6528e88224f12409a5f23060994eschrock if (ioctl(zhp->zpool_hdl->libzfs_fd, ZFS_IOC_ERROR_LOG,
99653d4ee642c6528e88224f12409a5f23060994eschrock &zc) != 0) {
e9dbad6f263d5570ed7ff5443ec5b958af8c24d7eschrock free((void *)(uintptr_t)zc.zc_nvlist_dst);
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock if (errno == ENOMEM) {
bf561db0030e7d3435bcc1e1f5889fac05a18814vb count = zc.zc_nvlist_dst_size;
e9dbad6f263d5570ed7ff5443ec5b958af8c24d7eschrock if ((zc.zc_nvlist_dst = (uintptr_t)
bf561db0030e7d3435bcc1e1f5889fac05a18814vb zfs_alloc(zhp->zpool_hdl, count *
bf561db0030e7d3435bcc1e1f5889fac05a18814vb sizeof (zbookmark_t))) == (uintptr_t)NULL)
99653d4ee642c6528e88224f12409a5f23060994eschrock return (-1);
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock } else {
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock return (-1);
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock }
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock } else {
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock break;
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock }
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock }
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock /*
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock * Sort the resulting bookmarks. This is a little confusing due to the
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock * implementation of ZFS_IOC_ERROR_LOG. The bookmarks are copied last
e9dbad6f263d5570ed7ff5443ec5b958af8c24d7eschrock * to first, and 'zc_nvlist_dst_size' indicates the number of boomarks
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock * _not_ copied as part of the process. So we point the start of our
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock * array appropriate and decrement the total number of elements.
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock */
e9dbad6f263d5570ed7ff5443ec5b958af8c24d7eschrock zb = ((zbookmark_t *)(uintptr_t)zc.zc_nvlist_dst) +
e9dbad6f263d5570ed7ff5443ec5b958af8c24d7eschrock zc.zc_nvlist_dst_size;
e9dbad6f263d5570ed7ff5443ec5b958af8c24d7eschrock count -= zc.zc_nvlist_dst_size;
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock qsort(zb, count, sizeof (zbookmark_t), zbookmark_compare);
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock
55434c770c89aa1b84474f2559a106803511aba0ek verify(nvlist_alloc(nverrlistp, 0, KM_SLEEP) == 0);
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock /*
55434c770c89aa1b84474f2559a106803511aba0ek * Fill in the nverrlistp with nvlist's of dataset and object numbers.
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock */
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock for (i = 0; i < count; i++) {
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock nvlist_t *nv;
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock
c0a81264b59ba24de8701436570c3aae5689dc89ek /* ignoring zb_blkid and zb_level for now */
c0a81264b59ba24de8701436570c3aae5689dc89ek if (i > 0 && zb[i-1].zb_objset == zb[i].zb_objset &&
c0a81264b59ba24de8701436570c3aae5689dc89ek zb[i-1].zb_object == zb[i].zb_object)
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock continue;
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock
55434c770c89aa1b84474f2559a106803511aba0ek if (nvlist_alloc(&nv, NV_UNIQUE_NAME, KM_SLEEP) != 0)
55434c770c89aa1b84474f2559a106803511aba0ek goto nomem;
55434c770c89aa1b84474f2559a106803511aba0ek if (nvlist_add_uint64(nv, ZPOOL_ERR_DATASET,
55434c770c89aa1b84474f2559a106803511aba0ek zb[i].zb_objset) != 0) {
55434c770c89aa1b84474f2559a106803511aba0ek nvlist_free(nv);
99653d4ee642c6528e88224f12409a5f23060994eschrock goto nomem;
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock }
55434c770c89aa1b84474f2559a106803511aba0ek if (nvlist_add_uint64(nv, ZPOOL_ERR_OBJECT,
55434c770c89aa1b84474f2559a106803511aba0ek zb[i].zb_object) != 0) {
55434c770c89aa1b84474f2559a106803511aba0ek nvlist_free(nv);
55434c770c89aa1b84474f2559a106803511aba0ek goto nomem;
55434c770c89aa1b84474f2559a106803511aba0ek }
55434c770c89aa1b84474f2559a106803511aba0ek if (nvlist_add_nvlist(*nverrlistp, "ejk", nv) != 0) {
55434c770c89aa1b84474f2559a106803511aba0ek nvlist_free(nv);
55434c770c89aa1b84474f2559a106803511aba0ek goto nomem;
55434c770c89aa1b84474f2559a106803511aba0ek }
55434c770c89aa1b84474f2559a106803511aba0ek nvlist_free(nv);
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock }
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock
3ccfa83cd9cddd1e34808ba18082c156758c5ec8ahrens free((void *)(uintptr_t)zc.zc_nvlist_dst);
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock return (0);
99653d4ee642c6528e88224f12409a5f23060994eschrock
99653d4ee642c6528e88224f12409a5f23060994eschrocknomem:
e9dbad6f263d5570ed7ff5443ec5b958af8c24d7eschrock free((void *)(uintptr_t)zc.zc_nvlist_dst);
99653d4ee642c6528e88224f12409a5f23060994eschrock return (no_memory(zhp->zpool_hdl));
ea8dc4b6d2251b437950c0056bc626b311c73c27eschrock}
eaca9bbd5f5d1e4e554da4c7108e8a03c8c33481eschrock
eaca9bbd5f5d1e4e554da4c7108e8a03c8c33481eschrock/*
eaca9bbd5f5d1e4e554da4c7108e8a03c8c33481eschrock * Upgrade a ZFS pool to the latest on-disk version.
eaca9bbd5f5d1e4e554da4c7108e8a03c8c33481eschrock */
eaca9bbd5f5d1e4e554da4c7108e8a03c8c33481eschrockint
eaca9bbd5f5d1e4e554da4c7108e8a03c8c33481eschrockzpool_upgrade(zpool_handle_t *zhp)
eaca9bbd5f5d1e4e554da4c7108e8a03c8c33481eschrock{
eaca9bbd5f5d1e4e554da4c7108e8a03c8c33481eschrock zfs_cmd_t zc = { 0 };
99653d4ee642c6528e88224f12409a5f23060994eschrock libzfs_handle_t *hdl = zhp->zpool_hdl;
eaca9bbd5f5d1e4e554da4c7108e8a03c8c33481eschrock
eaca9bbd5f5d1e4e554da4c7108e8a03c8c33481eschrock (void) strcpy(zc.zc_name, zhp->zpool_name);
ecd6cf800b63704be73fb264c3f5b6e0dafc068dmarks if (zfs_ioctl(hdl, ZFS_IOC_POOL_UPGRADE, &zc) != 0)
ece3d9b3bacef51a5f34d993935eedbb7bb87059lling return (zpool_standard_error_fmt(hdl, errno,
99653d4ee642c6528e88224f12409a5f23060994eschrock dgettext(TEXT_DOMAIN, "cannot upgrade '%s'"),
99653d4ee642c6528e88224f12409a5f23060994eschrock zhp->zpool_name));
eaca9bbd5f5d1e4e554da4c7108e8a03c8c33481eschrock return (0);
eaca9bbd5f5d1e4e554da4c7108e8a03c8c33481eschrock}
06eeb2ad640ce72d394ac521094bed7681044408ek
06eeb2ad640ce72d394ac521094bed7681044408ekvoid
2a6b87f07ac0c0b819179c84afe5a60afa04cfa5ekzpool_set_history_str(const char *subcommand, int argc, char **argv,
2a6b87f07ac0c0b819179c84afe5a60afa04cfa5ek char *history_str)
06eeb2ad640ce72d394ac521094bed7681044408ek{
06eeb2ad640ce72d394ac521094bed7681044408ek int i;
06eeb2ad640ce72d394ac521094bed7681044408ek
2a6b87f07ac0c0b819179c84afe5a60afa04cfa5ek (void) strlcpy(history_str, subcommand, HIS_MAX_RECORD_LEN);
2a6b87f07ac0c0b819179c84afe5a60afa04cfa5ek for (i = 1; i < argc; i++) {
2a6b87f07ac0c0b819179c84afe5a60afa04cfa5ek if (strlen(history_str) + 1 + strlen(argv[i]) >
2a6b87f07ac0c0b819179c84afe5a60afa04cfa5ek HIS_MAX_RECORD_LEN)
2a6b87f07ac0c0b819179c84afe5a60afa04cfa5ek break;
2a6b87f07ac0c0b819179c84afe5a60afa04cfa5ek (void) strlcat(history_str, " ", HIS_MAX_RECORD_LEN);
2a6b87f07ac0c0b819179c84afe5a60afa04cfa5ek (void) strlcat(history_str, argv[i], HIS_MAX_RECORD_LEN);
2a6b87f07ac0c0b819179c84afe5a60afa04cfa5ek }
2a6b87f07ac0c0b819179c84afe5a60afa04cfa5ek}
2a6b87f07ac0c0b819179c84afe5a60afa04cfa5ek
2a6b87f07ac0c0b819179c84afe5a60afa04cfa5ek/*
2a6b87f07ac0c0b819179c84afe5a60afa04cfa5ek * Stage command history for logging.
2a6b87f07ac0c0b819179c84afe5a60afa04cfa5ek */
2a6b87f07ac0c0b819179c84afe5a60afa04cfa5ekint
2a6b87f07ac0c0b819179c84afe5a60afa04cfa5ekzpool_stage_history(libzfs_handle_t *hdl, const char *history_str)
2a6b87f07ac0c0b819179c84afe5a60afa04cfa5ek{
2a6b87f07ac0c0b819179c84afe5a60afa04cfa5ek if (history_str == NULL)
2a6b87f07ac0c0b819179c84afe5a60afa04cfa5ek return (EINVAL);
2a6b87f07ac0c0b819179c84afe5a60afa04cfa5ek
2a6b87f07ac0c0b819179c84afe5a60afa04cfa5ek if (strlen(history_str) > HIS_MAX_RECORD_LEN)
2a6b87f07ac0c0b819179c84afe5a60afa04cfa5ek return (EINVAL);
2a6b87f07ac0c0b819179c84afe5a60afa04cfa5ek
228975cc44c2290cc190960e5894ac6ce0863855ek if (hdl->libzfs_log_str != NULL)
ecd6cf800b63704be73fb264c3f5b6e0dafc068dmarks free(hdl->libzfs_log_str);
06eeb2ad640ce72d394ac521094bed7681044408ek
2a6b87f07ac0c0b819179c84afe5a60afa04cfa5ek if ((hdl->libzfs_log_str = strdup(history_str)) == NULL)
2a6b87f07ac0c0b819179c84afe5a60afa04cfa5ek return (no_memory(hdl));
06eeb2ad640ce72d394ac521094bed7681044408ek
2a6b87f07ac0c0b819179c84afe5a60afa04cfa5ek return (0);
06eeb2ad640ce72d394ac521094bed7681044408ek}
06eeb2ad640ce72d394ac521094bed7681044408ek
06eeb2ad640ce72d394ac521094bed7681044408ek/*
06eeb2ad640ce72d394ac521094bed7681044408ek * Perform ioctl to get some command history of a pool.
06eeb2ad640ce72d394ac521094bed7681044408ek *
06eeb2ad640ce72d394ac521094bed7681044408ek * 'buf' is the buffer to fill up to 'len' bytes. 'off' is the
06eeb2ad640ce72d394ac521094bed7681044408ek * logical offset of the history buffer to start reading from.
06eeb2ad640ce72d394ac521094bed7681044408ek *
06eeb2ad640ce72d394ac521094bed7681044408ek * Upon return, 'off' is the next logical offset to read from and
06eeb2ad640ce72d394ac521094bed7681044408ek * 'len' is the actual amount of bytes read into 'buf'.
06eeb2ad640ce72d394ac521094bed7681044408ek */
06eeb2ad640ce72d394ac521094bed7681044408ekstatic int
06eeb2ad640ce72d394ac521094bed7681044408ekget_history(zpool_handle_t *zhp, char *buf, uint64_t *off, uint64_t *len)
06eeb2ad640ce72d394ac521094bed7681044408ek{
06eeb2ad640ce72d394ac521094bed7681044408ek zfs_cmd_t zc = { 0 };
06eeb2ad640ce72d394ac521094bed7681044408ek libzfs_handle_t *hdl = zhp->zpool_hdl;
06eeb2ad640ce72d394ac521094bed7681044408ek
06eeb2ad640ce72d394ac521094bed7681044408ek (void) strlcpy(zc.zc_name, zhp->zpool_name, sizeof (zc.zc_name));
06eeb2ad640ce72d394ac521094bed7681044408ek
06eeb2ad640ce72d394ac521094bed7681044408ek zc.zc_history = (uint64_t)(uintptr_t)buf;
06eeb2ad640ce72d394ac521094bed7681044408ek zc.zc_history_len = *len;
06eeb2ad640ce72d394ac521094bed7681044408ek zc.zc_history_offset = *off;
06eeb2ad640ce72d394ac521094bed7681044408ek
06eeb2ad640ce72d394ac521094bed7681044408ek if (ioctl(hdl->libzfs_fd, ZFS_IOC_POOL_GET_HISTORY, &zc) != 0) {
06eeb2ad640ce72d394ac521094bed7681044408ek switch (errno) {
06eeb2ad640ce72d394ac521094bed7681044408ek case EPERM:
ece3d9b3bacef51a5f34d993935eedbb7bb87059lling return (zfs_error_fmt(hdl, EZFS_PERM,
ece3d9b3bacef51a5f34d993935eedbb7bb87059lling dgettext(TEXT_DOMAIN,
06eeb2ad640ce72d394ac521094bed7681044408ek "cannot show history for pool '%s'"),
06eeb2ad640ce72d394ac521094bed7681044408ek zhp->zpool_name));
06eeb2ad640ce72d394ac521094bed7681044408ek case ENOENT:
ece3d9b3bacef51a5f34d993935eedbb7bb87059lling return (zfs_error_fmt(hdl, EZFS_NOHISTORY,
06eeb2ad640ce72d394ac521094bed7681044408ek dgettext(TEXT_DOMAIN, "cannot get history for pool "
06eeb2ad640ce72d394ac521094bed7681044408ek "'%s'"), zhp->zpool_name));
d7306b64c847d897abb9ece8624fca9cf28d358fek case ENOTSUP:
d7306b64c847d897abb9ece8624fca9cf28d358fek return (zfs_error_fmt(hdl, EZFS_BADVERSION,
d7306b64c847d897abb9ece8624fca9cf28d358fek dgettext(TEXT_DOMAIN, "cannot get history for pool "
d7306b64c847d897abb9ece8624fca9cf28d358fek "'%s', pool must be upgraded"), zhp->zpool_name));
06eeb2ad640ce72d394ac521094bed7681044408ek default:
ece3d9b3bacef51a5f34d993935eedbb7bb87059lling return (zpool_standard_error_fmt(hdl, errno,
06eeb2ad640ce72d394ac521094bed7681044408ek dgettext(TEXT_DOMAIN,
06eeb2ad640ce72d394ac521094bed7681044408ek "cannot get history for '%s'"), zhp->zpool_name));
06eeb2ad640ce72d394ac521094bed7681044408ek }
06eeb2ad640ce72d394ac521094bed7681044408ek }
06eeb2ad640ce72d394ac521094bed7681044408ek
06eeb2ad640ce72d394ac521094bed7681044408ek *len = zc.zc_history_len;
06eeb2ad640ce72d394ac521094bed7681044408ek *off = zc.zc_history_offset;
06eeb2ad640ce72d394ac521094bed7681044408ek
06eeb2ad640ce72d394ac521094bed7681044408ek return (0);
06eeb2ad640ce72d394ac521094bed7681044408ek}
06eeb2ad640ce72d394ac521094bed7681044408ek
06eeb2ad640ce72d394ac521094bed7681044408ek/*
06eeb2ad640ce72d394ac521094bed7681044408ek * Process the buffer of nvlists, unpacking and storing each nvlist record
06eeb2ad640ce72d394ac521094bed7681044408ek * into 'records'. 'leftover' is set to the number of bytes that weren't
06eeb2ad640ce72d394ac521094bed7681044408ek * processed as there wasn't a complete record.
06eeb2ad640ce72d394ac521094bed7681044408ek */
06eeb2ad640ce72d394ac521094bed7681044408ekstatic int
06eeb2ad640ce72d394ac521094bed7681044408ekzpool_history_unpack(char *buf, uint64_t bytes_read, uint64_t *leftover,
06eeb2ad640ce72d394ac521094bed7681044408ek nvlist_t ***records, uint_t *numrecords)
06eeb2ad640ce72d394ac521094bed7681044408ek{
06eeb2ad640ce72d394ac521094bed7681044408ek uint64_t reclen;
06eeb2ad640ce72d394ac521094bed7681044408ek nvlist_t *nv;
06eeb2ad640ce72d394ac521094bed7681044408ek int i;
06eeb2ad640ce72d394ac521094bed7681044408ek
06eeb2ad640ce72d394ac521094bed7681044408ek while (bytes_read > sizeof (reclen)) {
06eeb2ad640ce72d394ac521094bed7681044408ek
06eeb2ad640ce72d394ac521094bed7681044408ek /* get length of packed record (stored as little endian) */
06eeb2ad640ce72d394ac521094bed7681044408ek for (i = 0, reclen = 0; i < sizeof (reclen); i++)
06eeb2ad640ce72d394ac521094bed7681044408ek reclen += (uint64_t)(((uchar_t *)buf)[i]) << (8*i);
06eeb2ad640ce72d394ac521094bed7681044408ek
06eeb2ad640ce72d394ac521094bed7681044408ek if (bytes_read < sizeof (reclen) + reclen)
06eeb2ad640ce72d394ac521094bed7681044408ek break;
06eeb2ad640ce72d394ac521094bed7681044408ek
06eeb2ad640ce72d394ac521094bed7681044408ek /* unpack record */
06eeb2ad640ce72d394ac521094bed7681044408ek if (nvlist_unpack(buf + sizeof (reclen), reclen, &nv, 0) != 0)
06eeb2ad640ce72d394ac521094bed7681044408ek return (ENOMEM);
06eeb2ad640ce72d394ac521094bed7681044408ek bytes_read -= sizeof (reclen) + reclen;
06eeb2ad640ce72d394ac521094bed7681044408ek buf += sizeof (reclen) + reclen;
06eeb2ad640ce72d394ac521094bed7681044408ek
06eeb2ad640ce72d394ac521094bed7681044408ek /* add record to nvlist array */
06eeb2ad640ce72d394ac521094bed7681044408ek (*numrecords)++;
06eeb2ad640ce72d394ac521094bed7681044408ek if (ISP2(*numrecords + 1)) {
06eeb2ad640ce72d394ac521094bed7681044408ek *records = realloc(*records,
06eeb2ad640ce72d394ac521094bed7681044408ek *numrecords * 2 * sizeof (nvlist_t *));
06eeb2ad640ce72d394ac521094bed7681044408ek }
06eeb2ad640ce72d394ac521094bed7681044408ek (*records)[*numrecords - 1] = nv;
06eeb2ad640ce72d394ac521094bed7681044408ek }
06eeb2ad640ce72d394ac521094bed7681044408ek
06eeb2ad640ce72d394ac521094bed7681044408ek *leftover = bytes_read;
06eeb2ad640ce72d394ac521094bed7681044408ek return (0);
06eeb2ad640ce72d394ac521094bed7681044408ek}
06eeb2ad640ce72d394ac521094bed7681044408ek
06eeb2ad640ce72d394ac521094bed7681044408ek#define HIS_BUF_LEN (128*1024)
06eeb2ad640ce72d394ac521094bed7681044408ek
06eeb2ad640ce72d394ac521094bed7681044408ek/*
06eeb2ad640ce72d394ac521094bed7681044408ek * Retrieve the command history of a pool.
06eeb2ad640ce72d394ac521094bed7681044408ek */
06eeb2ad640ce72d394ac521094bed7681044408ekint
06eeb2ad640ce72d394ac521094bed7681044408ekzpool_get_history(zpool_handle_t *zhp, nvlist_t **nvhisp)
06eeb2ad640ce72d394ac521094bed7681044408ek{
06eeb2ad640ce72d394ac521094bed7681044408ek char buf[HIS_BUF_LEN];
06eeb2ad640ce72d394ac521094bed7681044408ek uint64_t off = 0;
06eeb2ad640ce72d394ac521094bed7681044408ek nvlist_t **records = NULL;
06eeb2ad640ce72d394ac521094bed7681044408ek uint_t numrecords = 0;
06eeb2ad640ce72d394ac521094bed7681044408ek int err, i;
06eeb2ad640ce72d394ac521094bed7681044408ek
06eeb2ad640ce72d394ac521094bed7681044408ek do {
06eeb2ad640ce72d394ac521094bed7681044408ek uint64_t bytes_read = sizeof (buf);
06eeb2ad640ce72d394ac521094bed7681044408ek uint64_t leftover;
06eeb2ad640ce72d394ac521094bed7681044408ek
06eeb2ad640ce72d394ac521094bed7681044408ek if ((err = get_history(zhp, buf, &off, &bytes_read)) != 0)
06eeb2ad640ce72d394ac521094bed7681044408ek break;
06eeb2ad640ce72d394ac521094bed7681044408ek
06eeb2ad640ce72d394ac521094bed7681044408ek /* if nothing else was read in, we're at EOF, just return */
06eeb2ad640ce72d394ac521094bed7681044408ek if (!bytes_read)
06eeb2ad640ce72d394ac521094bed7681044408ek break;
06eeb2ad640ce72d394ac521094bed7681044408ek
06eeb2ad640ce72d394ac521094bed7681044408ek if ((err = zpool_history_unpack(buf, bytes_read,
06eeb2ad640ce72d394ac521094bed7681044408ek &leftover, &records, &numrecords)) != 0)
06eeb2ad640ce72d394ac521094bed7681044408ek break;
06eeb2ad640ce72d394ac521094bed7681044408ek off -= leftover;
06eeb2ad640ce72d394ac521094bed7681044408ek
06eeb2ad640ce72d394ac521094bed7681044408ek /* CONSTCOND */
06eeb2ad640ce72d394ac521094bed7681044408ek } while (1);
06eeb2ad640ce72d394ac521094bed7681044408ek
06eeb2ad640ce72d394ac521094bed7681044408ek if (!err) {
06eeb2ad640ce72d394ac521094bed7681044408ek verify(nvlist_alloc(nvhisp, NV_UNIQUE_NAME, 0) == 0);
06eeb2ad640ce72d394ac521094bed7681044408ek verify(nvlist_add_nvlist_array(*nvhisp, ZPOOL_HIST_RECORD,
06eeb2ad640ce72d394ac521094bed7681044408ek records, numrecords) == 0);
06eeb2ad640ce72d394ac521094bed7681044408ek }
06eeb2ad640ce72d394ac521094bed7681044408ek for (i = 0; i < numrecords; i++)
06eeb2ad640ce72d394ac521094bed7681044408ek nvlist_free(records[i]);
06eeb2ad640ce72d394ac521094bed7681044408ek free(records);
06eeb2ad640ce72d394ac521094bed7681044408ek
06eeb2ad640ce72d394ac521094bed7681044408ek return (err);
06eeb2ad640ce72d394ac521094bed7681044408ek}
55434c770c89aa1b84474f2559a106803511aba0ek
55434c770c89aa1b84474f2559a106803511aba0ekvoid
55434c770c89aa1b84474f2559a106803511aba0ekzpool_obj_to_path(zpool_handle_t *zhp, uint64_t dsobj, uint64_t obj,
55434c770c89aa1b84474f2559a106803511aba0ek char *pathname, size_t len)
55434c770c89aa1b84474f2559a106803511aba0ek{
55434c770c89aa1b84474f2559a106803511aba0ek zfs_cmd_t zc = { 0 };
55434c770c89aa1b84474f2559a106803511aba0ek boolean_t mounted = B_FALSE;
55434c770c89aa1b84474f2559a106803511aba0ek char *mntpnt = NULL;
55434c770c89aa1b84474f2559a106803511aba0ek char dsname[MAXNAMELEN];
55434c770c89aa1b84474f2559a106803511aba0ek
55434c770c89aa1b84474f2559a106803511aba0ek if (dsobj == 0) {
55434c770c89aa1b84474f2559a106803511aba0ek /* special case for the MOS */
55434c770c89aa1b84474f2559a106803511aba0ek (void) snprintf(pathname, len, "<metadata>:<0x%llx>", obj);
55434c770c89aa1b84474f2559a106803511aba0ek return;
55434c770c89aa1b84474f2559a106803511aba0ek }
55434c770c89aa1b84474f2559a106803511aba0ek
55434c770c89aa1b84474f2559a106803511aba0ek /* get the dataset's name */
55434c770c89aa1b84474f2559a106803511aba0ek (void) strlcpy(zc.zc_name, zhp->zpool_name, sizeof (zc.zc_name));
55434c770c89aa1b84474f2559a106803511aba0ek zc.zc_obj = dsobj;
55434c770c89aa1b84474f2559a106803511aba0ek if (ioctl(zhp->zpool_hdl->libzfs_fd,
55434c770c89aa1b84474f2559a106803511aba0ek ZFS_IOC_DSOBJ_TO_DSNAME, &zc) != 0) {
55434c770c89aa1b84474f2559a106803511aba0ek /* just write out a path of two object numbers */
55434c770c89aa1b84474f2559a106803511aba0ek (void) snprintf(pathname, len, "<0x%llx>:<0x%llx>",
55434c770c89aa1b84474f2559a106803511aba0ek dsobj, obj);
55434c770c89aa1b84474f2559a106803511aba0ek return;
55434c770c89aa1b84474f2559a106803511aba0ek }
55434c770c89aa1b84474f2559a106803511aba0ek (void) strlcpy(dsname, zc.zc_value, sizeof (dsname));
55434c770c89aa1b84474f2559a106803511aba0ek
55434c770c89aa1b84474f2559a106803511aba0ek /* find out if the dataset is mounted */
55434c770c89aa1b84474f2559a106803511aba0ek mounted = is_mounted(zhp->zpool_hdl, dsname, &mntpnt);
55434c770c89aa1b84474f2559a106803511aba0ek
55434c770c89aa1b84474f2559a106803511aba0ek /* get the corrupted object's path */
55434c770c89aa1b84474f2559a106803511aba0ek (void) strlcpy(zc.zc_name, dsname, sizeof (zc.zc_name));
55434c770c89aa1b84474f2559a106803511aba0ek zc.zc_obj = obj;
55434c770c89aa1b84474f2559a106803511aba0ek if (ioctl(zhp->zpool_hdl->libzfs_fd, ZFS_IOC_OBJ_TO_PATH,
55434c770c89aa1b84474f2559a106803511aba0ek &zc) == 0) {
55434c770c89aa1b84474f2559a106803511aba0ek if (mounted) {
55434c770c89aa1b84474f2559a106803511aba0ek (void) snprintf(pathname, len, "%s%s", mntpnt,
55434c770c89aa1b84474f2559a106803511aba0ek zc.zc_value);
55434c770c89aa1b84474f2559a106803511aba0ek } else {
55434c770c89aa1b84474f2559a106803511aba0ek (void) snprintf(pathname, len, "%s:%s",
55434c770c89aa1b84474f2559a106803511aba0ek dsname, zc.zc_value);
55434c770c89aa1b84474f2559a106803511aba0ek }
55434c770c89aa1b84474f2559a106803511aba0ek } else {
55434c770c89aa1b84474f2559a106803511aba0ek (void) snprintf(pathname, len, "%s:<0x%llx>", dsname, obj);
55434c770c89aa1b84474f2559a106803511aba0ek }
55434c770c89aa1b84474f2559a106803511aba0ek free(mntpnt);
55434c770c89aa1b84474f2559a106803511aba0ek}
b1b8ab34de515a5e83206da22c3d7e563241b021lling
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor#define RDISK_ROOT "/dev/rdsk"
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor#define BACKUP_SLICE "s2"
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor/*
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor * Don't start the slice at the default block of 34; many storage
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor * devices will use a stripe width of 128k, so start there instead.
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor */
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor#define NEW_START_BLOCK 256
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor/*
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor * determine where a partition starts on a disk in the current
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor * configuration
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor */
8488aeb5df27784d479c16cde06a9e25cd9a1152taylorstatic diskaddr_t
8488aeb5df27784d479c16cde06a9e25cd9a1152taylorfind_start_block(nvlist_t *config)
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor{
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor nvlist_t **child;
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor uint_t c, children;
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor char *path;
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor diskaddr_t sb = MAXOFFSET_T;
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor int fd;
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor char diskname[MAXPATHLEN];
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor uint64_t wholedisk;
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor if (nvlist_lookup_nvlist_array(config,
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor ZPOOL_CONFIG_CHILDREN, &child, &children) != 0) {
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor if (nvlist_lookup_uint64(config,
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor ZPOOL_CONFIG_WHOLE_DISK,
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor &wholedisk) != 0 || !wholedisk) {
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor return (MAXOFFSET_T);
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor }
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor if (nvlist_lookup_string(config,
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor ZPOOL_CONFIG_PATH, &path) != 0) {
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor return (MAXOFFSET_T);
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor }
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor (void) snprintf(diskname, sizeof (diskname), "%s%s",
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor RDISK_ROOT, strrchr(path, '/'));
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor if ((fd = open(diskname, O_RDONLY|O_NDELAY)) >= 0) {
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor struct dk_gpt *vtoc;
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor if (efi_alloc_and_read(fd, &vtoc) >= 0) {
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor sb = vtoc->efi_parts[0].p_start;
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor efi_free(vtoc);
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor }
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor (void) close(fd);
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor }
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor return (sb);
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor }
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor for (c = 0; c < children; c++) {
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor sb = find_start_block(child[c]);
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor if (sb != MAXOFFSET_T) {
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor return (sb);
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor }
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor }
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor return (MAXOFFSET_T);
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor}
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor/*
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor * Label an individual disk. The name provided is the short name,
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor * stripped of any leading /dev path.
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor */
8488aeb5df27784d479c16cde06a9e25cd9a1152taylorint
8488aeb5df27784d479c16cde06a9e25cd9a1152taylorzpool_label_disk(libzfs_handle_t *hdl, zpool_handle_t *zhp, char *name)
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor{
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor char path[MAXPATHLEN];
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor struct dk_gpt *vtoc;
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor int fd;
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor size_t resv = EFI_MIN_RESV_SIZE;
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor uint64_t slice_size;
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor diskaddr_t start_block;
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor char errbuf[1024];
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor if (zhp) {
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor nvlist_t *nvroot;
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor verify(nvlist_lookup_nvlist(zhp->zpool_config,
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor ZPOOL_CONFIG_VDEV_TREE, &nvroot) == 0);
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor if (zhp->zpool_start_block == 0)
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor start_block = find_start_block(nvroot);
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor else
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor start_block = zhp->zpool_start_block;
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor zhp->zpool_start_block = start_block;
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor } else {
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor /* new pool */
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor start_block = NEW_START_BLOCK;
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor }
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor (void) snprintf(path, sizeof (path), "%s/%s%s", RDISK_ROOT, name,
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor BACKUP_SLICE);
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor if ((fd = open(path, O_RDWR | O_NDELAY)) < 0) {
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor /*
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor * This shouldn't happen. We've long since verified that this
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor * is a valid device.
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor */
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor zfs_error_aux(hdl, dgettext(TEXT_DOMAIN, "cannot "
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor "label '%s': unable to open device"), name);
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor return (zfs_error(hdl, EZFS_OPENFAILED, errbuf));
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor }
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor if (efi_alloc_and_init(fd, EFI_NUMPAR, &vtoc) != 0) {
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor /*
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor * The only way this can fail is if we run out of memory, or we
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor * were unable to read the disk's capacity
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor */
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor if (errno == ENOMEM)
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor (void) no_memory(hdl);
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor (void) close(fd);
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor zfs_error_aux(hdl, dgettext(TEXT_DOMAIN, "cannot "
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor "label '%s': unable to read disk capacity"), name);
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor return (zfs_error(hdl, EZFS_NOCAP, errbuf));
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor }
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor slice_size = vtoc->efi_last_u_lba + 1;
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor slice_size -= EFI_MIN_RESV_SIZE;
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor if (start_block == MAXOFFSET_T)
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor start_block = NEW_START_BLOCK;
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor slice_size -= start_block;
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor vtoc->efi_parts[0].p_start = start_block;
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor vtoc->efi_parts[0].p_size = slice_size;
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor /*
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor * Why we use V_USR: V_BACKUP confuses users, and is considered
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor * disposable by some EFI utilities (since EFI doesn't have a backup
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor * slice). V_UNASSIGNED is supposed to be used only for zero size
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor * partitions, and efi_write() will fail if we use it. V_ROOT, V_BOOT,
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor * etc. were all pretty specific. V_USR is as close to reality as we
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor * can get, in the absence of V_OTHER.
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor */
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor vtoc->efi_parts[0].p_tag = V_USR;
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor (void) strcpy(vtoc->efi_parts[0].p_name, "zfs");
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor vtoc->efi_parts[8].p_start = slice_size + start_block;
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor vtoc->efi_parts[8].p_size = resv;
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor vtoc->efi_parts[8].p_tag = V_RESERVED;
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor if (efi_write(fd, vtoc) != 0) {
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor /*
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor * Some block drivers (like pcata) may not support EFI
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor * GPT labels. Print out a helpful error message dir-
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor * ecting the user to manually label the disk and give
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor * a specific slice.
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor */
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor (void) close(fd);
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor efi_free(vtoc);
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor zfs_error_aux(hdl, dgettext(TEXT_DOMAIN,
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor "cannot label '%s': try using fdisk(1M) and then "
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor "provide a specific slice"), name);
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor return (zfs_error(hdl, EZFS_LABELFAILED, errbuf));
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor }
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor (void) close(fd);
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor efi_free(vtoc);
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor return (0);
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor}
8488aeb5df27784d479c16cde06a9e25cd9a1152taylor
b1b8ab34de515a5e83206da22c3d7e563241b021llingint
b1b8ab34de515a5e83206da22c3d7e563241b021llingzpool_set_prop(zpool_handle_t *zhp, const char *propname, const char *propval)
b1b8ab34de515a5e83206da22c3d7e563241b021lling{
b1b8ab34de515a5e83206da22c3d7e563241b021lling zfs_cmd_t zc = { 0 };
b1b8ab34de515a5e83206da22c3d7e563241b021lling int ret = -1;
b1b8ab34de515a5e83206da22c3d7e563241b021lling char errbuf[1024];
b1b8ab34de515a5e83206da22c3d7e563241b021lling nvlist_t *nvl = NULL;
b1b8ab34de515a5e83206da22c3d7e563241b021lling nvlist_t *realprops;
b1b8ab34de515a5e83206da22c3d7e563241b021lling
b1b8ab34de515a5e83206da22c3d7e563241b021lling (void) snprintf(errbuf, sizeof (errbuf),
b1b8ab34de515a5e83206da22c3d7e563241b021lling dgettext(TEXT_DOMAIN, "cannot set property for '%s'"),
b1b8ab34de515a5e83206da22c3d7e563241b021lling zhp->zpool_name);
b1b8ab34de515a5e83206da22c3d7e563241b021lling
e7437265dc2a4920c197ed4337665539d358b22cahrens if (zpool_get_version(zhp) < SPA_VERSION_BOOTFS) {
b1b8ab34de515a5e83206da22c3d7e563241b021lling zfs_error_aux(zhp->zpool_hdl,
b1b8ab34de515a5e83206da22c3d7e563241b021lling dgettext(TEXT_DOMAIN, "pool must be "
b1b8ab34de515a5e83206da22c3d7e563241b021lling "upgraded to support pool properties"));
b1b8ab34de515a5e83206da22c3d7e563241b021lling return (zfs_error(zhp->zpool_hdl, EZFS_BADVERSION, errbuf));
b1b8ab34de515a5e83206da22c3d7e563241b021lling }
b1b8ab34de515a5e83206da22c3d7e563241b021lling
b1b8ab34de515a5e83206da22c3d7e563241b021lling if (zhp->zpool_props == NULL && zpool_get_all_props(zhp))
b1b8ab34de515a5e83206da22c3d7e563241b021lling return (zfs_error(zhp->zpool_hdl, EZFS_POOLPROPS, errbuf));
b1b8ab34de515a5e83206da22c3d7e563241b021lling
b1b8ab34de515a5e83206da22c3d7e563241b021lling if (nvlist_alloc(&nvl, NV_UNIQUE_NAME, 0) != 0 ||
b1b8ab34de515a5e83206da22c3d7e563241b021lling nvlist_add_string(nvl, propname, propval) != 0) {
b1b8ab34de515a5e83206da22c3d7e563241b021lling return (no_memory(zhp->zpool_hdl));
b1b8ab34de515a5e83206da22c3d7e563241b021lling }
b1b8ab34de515a5e83206da22c3d7e563241b021lling
b1b8ab34de515a5e83206da22c3d7e563241b021lling if ((realprops = zfs_validate_properties(zhp->zpool_hdl, ZFS_TYPE_POOL,
b1b8ab34de515a5e83206da22c3d7e563241b021lling zhp->zpool_name, nvl, 0, NULL, errbuf)) == NULL) {
b1b8ab34de515a5e83206da22c3d7e563241b021lling nvlist_free(nvl);
b1b8ab34de515a5e83206da22c3d7e563241b021lling return (-1);
b1b8ab34de515a5e83206da22c3d7e563241b021lling }
b1b8ab34de515a5e83206da22c3d7e563241b021lling
b1b8ab34de515a5e83206da22c3d7e563241b021lling nvlist_free(nvl);
b1b8ab34de515a5e83206da22c3d7e563241b021lling nvl = realprops;
b1b8ab34de515a5e83206da22c3d7e563241b021lling
b1b8ab34de515a5e83206da22c3d7e563241b021lling /*
b1b8ab34de515a5e83206da22c3d7e563241b021lling * Execute the corresponding ioctl() to set this property.
b1b8ab34de515a5e83206da22c3d7e563241b021lling */
b1b8ab34de515a5e83206da22c3d7e563241b021lling (void) strlcpy(zc.zc_name, zhp->zpool_name, sizeof (zc.zc_name));
b1b8ab34de515a5e83206da22c3d7e563241b021lling
b1b8ab34de515a5e83206da22c3d7e563241b021lling if (zcmd_write_src_nvlist(zhp->zpool_hdl, &zc, nvl, NULL) != 0)
b1b8ab34de515a5e83206da22c3d7e563241b021lling return (-1);
b1b8ab34de515a5e83206da22c3d7e563241b021lling
ecd6cf800b63704be73fb264c3f5b6e0dafc068dmarks ret = zfs_ioctl(zhp->zpool_hdl, ZFS_IOC_POOL_SET_PROPS, &zc);
b1b8ab34de515a5e83206da22c3d7e563241b021lling zcmd_free_nvlists(&zc);
b1b8ab34de515a5e83206da22c3d7e563241b021lling
b1b8ab34de515a5e83206da22c3d7e563241b021lling if (ret)
b1b8ab34de515a5e83206da22c3d7e563241b021lling (void) zpool_standard_error(zhp->zpool_hdl, errno, errbuf);
b1b8ab34de515a5e83206da22c3d7e563241b021lling
b1b8ab34de515a5e83206da22c3d7e563241b021lling return (ret);
b1b8ab34de515a5e83206da22c3d7e563241b021lling}
b1b8ab34de515a5e83206da22c3d7e563241b021lling
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrockuint64_t
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrockzpool_get_prop_int(zpool_handle_t *zhp, zpool_prop_t prop)
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock{
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock uint64_t value;
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock nvlist_t *nvp;
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock
e7437265dc2a4920c197ed4337665539d358b22cahrens if (zpool_get_version(zhp) < SPA_VERSION_BOOTFS)
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock return (0);
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock if (zhp->zpool_props == NULL && zpool_get_all_props(zhp))
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock return (zpool_prop_default_numeric(prop));
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock switch (prop) {
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock case ZPOOL_PROP_AUTOREPLACE:
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock if (nvlist_lookup_nvlist(zhp->zpool_props,
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock zpool_prop_to_name(prop), &nvp) != 0) {
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock value = zpool_prop_default_numeric(prop);
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock } else {
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock VERIFY(nvlist_lookup_uint64(nvp, ZFS_PROP_VALUE,
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock &value) == 0);
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock }
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock return (value);
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock break;
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock default:
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock assert(0);
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock }
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock return (0);
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock}
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock
b1b8ab34de515a5e83206da22c3d7e563241b021llingint
b1b8ab34de515a5e83206da22c3d7e563241b021llingzpool_get_prop(zpool_handle_t *zhp, zpool_prop_t prop, char *propbuf,
b1b8ab34de515a5e83206da22c3d7e563241b021lling size_t proplen, zfs_source_t *srctype)
b1b8ab34de515a5e83206da22c3d7e563241b021lling{
b1b8ab34de515a5e83206da22c3d7e563241b021lling uint64_t value;
b1b8ab34de515a5e83206da22c3d7e563241b021lling char msg[1024], *strvalue;
b1b8ab34de515a5e83206da22c3d7e563241b021lling nvlist_t *nvp;
b1b8ab34de515a5e83206da22c3d7e563241b021lling zfs_source_t src = ZFS_SRC_NONE;
b1b8ab34de515a5e83206da22c3d7e563241b021lling
b1b8ab34de515a5e83206da22c3d7e563241b021lling (void) snprintf(msg, sizeof (msg), dgettext(TEXT_DOMAIN,
b1b8ab34de515a5e83206da22c3d7e563241b021lling "cannot get property '%s'"), zpool_prop_to_name(prop));
b1b8ab34de515a5e83206da22c3d7e563241b021lling
e7437265dc2a4920c197ed4337665539d358b22cahrens if (zpool_get_version(zhp) < SPA_VERSION_BOOTFS) {
b1b8ab34de515a5e83206da22c3d7e563241b021lling zfs_error_aux(zhp->zpool_hdl,
b1b8ab34de515a5e83206da22c3d7e563241b021lling dgettext(TEXT_DOMAIN, "pool must be "
b1b8ab34de515a5e83206da22c3d7e563241b021lling "upgraded to support pool properties"));
b1b8ab34de515a5e83206da22c3d7e563241b021lling return (zfs_error(zhp->zpool_hdl, EZFS_BADVERSION, msg));
b1b8ab34de515a5e83206da22c3d7e563241b021lling }
b1b8ab34de515a5e83206da22c3d7e563241b021lling
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock if (zhp->zpool_props == NULL && zpool_get_all_props(zhp) &&
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock prop != ZPOOL_PROP_NAME)
b1b8ab34de515a5e83206da22c3d7e563241b021lling return (zfs_error(zhp->zpool_hdl, EZFS_POOLPROPS, msg));
b1b8ab34de515a5e83206da22c3d7e563241b021lling
b1b8ab34de515a5e83206da22c3d7e563241b021lling switch (prop) {
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock case ZPOOL_PROP_NAME:
b1b8ab34de515a5e83206da22c3d7e563241b021lling (void) strlcpy(propbuf, zhp->zpool_name, proplen);
b1b8ab34de515a5e83206da22c3d7e563241b021lling break;
b1b8ab34de515a5e83206da22c3d7e563241b021lling
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock case ZPOOL_PROP_BOOTFS:
b1b8ab34de515a5e83206da22c3d7e563241b021lling if (nvlist_lookup_nvlist(zhp->zpool_props,
b1b8ab34de515a5e83206da22c3d7e563241b021lling zpool_prop_to_name(prop), &nvp) != 0) {
b1b8ab34de515a5e83206da22c3d7e563241b021lling strvalue = (char *)zfs_prop_default_string(prop);
b1b8ab34de515a5e83206da22c3d7e563241b021lling if (strvalue == NULL)
b1b8ab34de515a5e83206da22c3d7e563241b021lling strvalue = "-";
b1b8ab34de515a5e83206da22c3d7e563241b021lling src = ZFS_SRC_DEFAULT;
b1b8ab34de515a5e83206da22c3d7e563241b021lling } else {
b1b8ab34de515a5e83206da22c3d7e563241b021lling VERIFY(nvlist_lookup_uint64(nvp,
b1b8ab34de515a5e83206da22c3d7e563241b021lling ZFS_PROP_SOURCE, &value) == 0);
b1b8ab34de515a5e83206da22c3d7e563241b021lling src = value;
b1b8ab34de515a5e83206da22c3d7e563241b021lling VERIFY(nvlist_lookup_string(nvp, ZFS_PROP_VALUE,
b1b8ab34de515a5e83206da22c3d7e563241b021lling &strvalue) == 0);
b1b8ab34de515a5e83206da22c3d7e563241b021lling if (strlen(strvalue) >= proplen)
b1b8ab34de515a5e83206da22c3d7e563241b021lling return (-1);
b1b8ab34de515a5e83206da22c3d7e563241b021lling }
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock (void) strlcpy(propbuf, strvalue, proplen);
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock break;
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock
ecd6cf800b63704be73fb264c3f5b6e0dafc068dmarks case ZPOOL_PROP_DELEGATION:
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock case ZPOOL_PROP_AUTOREPLACE:
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock if (nvlist_lookup_nvlist(zhp->zpool_props,
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock zpool_prop_to_name(prop), &nvp) != 0) {
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock value = zpool_prop_default_numeric(prop);
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock src = ZFS_SRC_DEFAULT;
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock } else {
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock VERIFY(nvlist_lookup_uint64(nvp,
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock ZFS_PROP_SOURCE, &value) == 0);
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock src = value;
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock VERIFY(nvlist_lookup_uint64(nvp, ZFS_PROP_VALUE,
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock &value) == 0);
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock }
3d7072f8bd27709dba14f6fe336f149d25d9e207eschrock (void) strlcpy(propbuf, value ? "on" : "off", proplen);
b1b8ab34de515a5e83206da22c3d7e563241b021lling break;
b1b8ab34de515a5e83206da22c3d7e563241b021lling
b1b8ab34de515a5e83206da22c3d7e563241b021lling default:
b1b8ab34de515a5e83206da22c3d7e563241b021lling return (-1);
b1b8ab34de515a5e83206da22c3d7e563241b021lling }
b1b8ab34de515a5e83206da22c3d7e563241b021lling if (srctype)
b1b8ab34de515a5e83206da22c3d7e563241b021lling *srctype = src;
b1b8ab34de515a5e83206da22c3d7e563241b021lling return (0);
b1b8ab34de515a5e83206da22c3d7e563241b021lling}
b1b8ab34de515a5e83206da22c3d7e563241b021lling
b1b8ab34de515a5e83206da22c3d7e563241b021llingint
b1b8ab34de515a5e83206da22c3d7e563241b021llingzpool_get_proplist(libzfs_handle_t *hdl, char *fields, zpool_proplist_t **listp)
b1b8ab34de515a5e83206da22c3d7e563241b021lling{
b1b8ab34de515a5e83206da22c3d7e563241b021lling return (zfs_get_proplist_common(hdl, fields, listp, ZFS_TYPE_POOL));
b1b8ab34de515a5e83206da22c3d7e563241b021lling}
b1b8ab34de515a5e83206da22c3d7e563241b021lling
b1b8ab34de515a5e83206da22c3d7e563241b021lling
b1b8ab34de515a5e83206da22c3d7e563241b021llingint
b1b8ab34de515a5e83206da22c3d7e563241b021llingzpool_expand_proplist(zpool_handle_t *zhp, zpool_proplist_t **plp)
b1b8ab34de515a5e83206da22c3d7e563241b021lling{
b1b8ab34de515a5e83206da22c3d7e563241b021lling libzfs_handle_t *hdl = zhp->zpool_hdl;
b1b8ab34de515a5e83206da22c3d7e563241b021lling zpool_proplist_t *entry;
b1b8ab34de515a5e83206da22c3d7e563241b021lling char buf[ZFS_MAXPROPLEN];
b1b8ab34de515a5e83206da22c3d7e563241b021lling
b1b8ab34de515a5e83206da22c3d7e563241b021lling if (zfs_expand_proplist_common(hdl, plp, ZFS_TYPE_POOL) != 0)
b1b8ab34de515a5e83206da22c3d7e563241b021lling return (-1);
b1b8ab34de515a5e83206da22c3d7e563241b021lling
b1b8ab34de515a5e83206da22c3d7e563241b021lling for (entry = *plp; entry != NULL; entry = entry->pl_next) {
b1b8ab34de515a5e83206da22c3d7e563241b021lling
b1b8ab34de515a5e83206da22c3d7e563241b021lling if (entry->pl_fixed)
b1b8ab34de515a5e83206da22c3d7e563241b021lling continue;
b1b8ab34de515a5e83206da22c3d7e563241b021lling
b1b8ab34de515a5e83206da22c3d7e563241b021lling if (entry->pl_prop != ZFS_PROP_INVAL &&
b1b8ab34de515a5e83206da22c3d7e563241b021lling zpool_get_prop(zhp, entry->pl_prop, buf, sizeof (buf),
b1b8ab34de515a5e83206da22c3d7e563241b021lling NULL) == 0) {
b1b8ab34de515a5e83206da22c3d7e563241b021lling if (strlen(buf) > entry->pl_width)
b1b8ab34de515a5e83206da22c3d7e563241b021lling entry->pl_width = strlen(buf);
b1b8ab34de515a5e83206da22c3d7e563241b021lling }
b1b8ab34de515a5e83206da22c3d7e563241b021lling }
b1b8ab34de515a5e83206da22c3d7e563241b021lling
b1b8ab34de515a5e83206da22c3d7e563241b021lling return (0);
b1b8ab34de515a5e83206da22c3d7e563241b021lling}