2016-08-31 21:46:58 +00:00
|
|
|
/*
|
|
|
|
* CDDL HEADER START
|
|
|
|
*
|
|
|
|
* The contents of this file are subject to the terms of the
|
|
|
|
* Common Development and Distribution License (the "License").
|
|
|
|
* You may not use this file except in compliance with the License.
|
|
|
|
*
|
|
|
|
* You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
|
|
|
|
* or http://www.opensolaris.org/os/licensing.
|
|
|
|
* See the License for the specific language governing permissions
|
|
|
|
* and limitations under the License.
|
|
|
|
*
|
|
|
|
* When distributing Covered Code, include this CDDL HEADER in each
|
|
|
|
* file and include the License file at usr/src/OPENSOLARIS.LICENSE.
|
|
|
|
* If applicable, add the following below this CDDL HEADER, with the
|
|
|
|
* fields enclosed by brackets "[]" replaced with your own identifying
|
|
|
|
* information: Portions Copyright [yyyy] [name of copyright owner]
|
|
|
|
*
|
|
|
|
* CDDL HEADER END
|
|
|
|
*/
|
|
|
|
/*
|
|
|
|
* Copyright (c) 2007, 2010, Oracle and/or its affiliates. All rights reserved.
|
|
|
|
* Copyright (c) 2012 by Delphix. All rights reserved.
|
|
|
|
* Copyright 2014 Nexenta Systems, Inc. All rights reserved.
|
2017-03-02 16:47:26 +00:00
|
|
|
* Copyright (c) 2016, 2017, Intel Corporation.
|
2017-10-26 19:26:09 +00:00
|
|
|
* Copyright (c) 2017 Open-E, Inc. All Rights Reserved.
|
2016-08-31 21:46:58 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
|
|
|
* ZFS syseventd module.
|
|
|
|
*
|
|
|
|
* file origin: openzfs/usr/src/cmd/syseventd/modules/zfs_mod/zfs_mod.c
|
|
|
|
*
|
|
|
|
* The purpose of this module is to identify when devices are added to the
|
|
|
|
* system, and appropriately online or replace the affected vdevs.
|
|
|
|
*
|
|
|
|
* When a device is added to the system:
|
|
|
|
*
|
|
|
|
* 1. Search for any vdevs whose devid matches that of the newly added
|
|
|
|
* device.
|
|
|
|
*
|
|
|
|
* 2. If no vdevs are found, then search for any vdevs whose udev path
|
|
|
|
* matches that of the new device.
|
|
|
|
*
|
|
|
|
* 3. If no vdevs match by either method, then ignore the event.
|
|
|
|
*
|
|
|
|
* 4. Attempt to online the device with a flag to indicate that it should
|
|
|
|
* be unspared when resilvering completes. If this succeeds, then the
|
|
|
|
* same device was inserted and we should continue normally.
|
|
|
|
*
|
|
|
|
* 5. If the pool does not have the 'autoreplace' property set, attempt to
|
|
|
|
* online the device again without the unspare flag, which will
|
|
|
|
* generate a FMA fault.
|
|
|
|
*
|
|
|
|
* 6. If the pool has the 'autoreplace' property set, and the matching vdev
|
|
|
|
* is a whole disk, then label the new disk and attempt a 'zpool
|
|
|
|
* replace'.
|
|
|
|
*
|
|
|
|
* The module responds to EC_DEV_ADD events. The special ESC_ZFS_VDEV_CHECK
|
|
|
|
* event indicates that a device failed to open during pool load, but the
|
|
|
|
* autoreplace property was set. In this case, we deferred the associated
|
|
|
|
* FMA fault until our module had a chance to process the autoreplace logic.
|
|
|
|
* If the device could not be replaced, then the second online attempt will
|
|
|
|
* trigger the FMA fault that we skipped earlier.
|
|
|
|
*
|
|
|
|
* ZFS on Linux porting notes:
|
|
|
|
* Linux udev provides a disk insert for both the disk and the partition
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <ctype.h>
|
|
|
|
#include <devid.h>
|
|
|
|
#include <fcntl.h>
|
|
|
|
#include <libnvpair.h>
|
|
|
|
#include <libzfs.h>
|
2018-11-05 19:22:33 +00:00
|
|
|
#include <libzutil.h>
|
2016-08-31 21:46:58 +00:00
|
|
|
#include <limits.h>
|
|
|
|
#include <stddef.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <syslog.h>
|
|
|
|
#include <sys/list.h>
|
|
|
|
#include <sys/sunddi.h>
|
|
|
|
#include <sys/sysevent/eventdefs.h>
|
|
|
|
#include <sys/sysevent/dev.h>
|
2017-12-09 00:58:41 +00:00
|
|
|
#include <thread_pool.h>
|
2016-08-31 21:46:58 +00:00
|
|
|
#include <pthread.h>
|
|
|
|
#include <unistd.h>
|
2018-02-16 01:53:18 +00:00
|
|
|
#include <errno.h>
|
2016-08-31 21:46:58 +00:00
|
|
|
#include "zfs_agents.h"
|
|
|
|
#include "../zed_log.h"
|
|
|
|
|
|
|
|
#define DEV_BYID_PATH "/dev/disk/by-id/"
|
|
|
|
#define DEV_BYPATH_PATH "/dev/disk/by-path/"
|
2017-03-02 16:47:26 +00:00
|
|
|
#define DEV_BYVDEV_PATH "/dev/disk/by-vdev/"
|
2016-08-31 21:46:58 +00:00
|
|
|
|
|
|
|
typedef void (*zfs_process_func_t)(zpool_handle_t *, nvlist_t *, boolean_t);
|
|
|
|
|
|
|
|
libzfs_handle_t *g_zfshdl;
|
|
|
|
list_t g_pool_list; /* list of unavailable pools at initialization */
|
|
|
|
list_t g_device_list; /* list of disks with asynchronous label request */
|
2017-12-09 00:58:41 +00:00
|
|
|
tpool_t *g_tpool;
|
2016-08-31 21:46:58 +00:00
|
|
|
boolean_t g_enumeration_done;
|
2017-12-09 00:58:41 +00:00
|
|
|
pthread_t g_zfs_tid; /* zfs_enum_pools() thread */
|
2016-08-31 21:46:58 +00:00
|
|
|
|
|
|
|
typedef struct unavailpool {
|
|
|
|
zpool_handle_t *uap_zhp;
|
|
|
|
list_node_t uap_node;
|
|
|
|
} unavailpool_t;
|
|
|
|
|
|
|
|
typedef struct pendingdev {
|
|
|
|
char pd_physpath[128];
|
|
|
|
list_node_t pd_node;
|
|
|
|
} pendingdev_t;
|
|
|
|
|
|
|
|
static int
|
|
|
|
zfs_toplevel_state(zpool_handle_t *zhp)
|
|
|
|
{
|
|
|
|
nvlist_t *nvroot;
|
|
|
|
vdev_stat_t *vs;
|
|
|
|
unsigned int c;
|
|
|
|
|
|
|
|
verify(nvlist_lookup_nvlist(zpool_get_config(zhp, NULL),
|
|
|
|
ZPOOL_CONFIG_VDEV_TREE, &nvroot) == 0);
|
|
|
|
verify(nvlist_lookup_uint64_array(nvroot, ZPOOL_CONFIG_VDEV_STATS,
|
|
|
|
(uint64_t **)&vs, &c) == 0);
|
|
|
|
return (vs->vs_state);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
zfs_unavail_pool(zpool_handle_t *zhp, void *data)
|
|
|
|
{
|
|
|
|
zed_log_msg(LOG_INFO, "zfs_unavail_pool: examining '%s' (state %d)",
|
|
|
|
zpool_get_name(zhp), (int)zfs_toplevel_state(zhp));
|
|
|
|
|
|
|
|
if (zfs_toplevel_state(zhp) < VDEV_STATE_DEGRADED) {
|
|
|
|
unavailpool_t *uap;
|
|
|
|
uap = malloc(sizeof (unavailpool_t));
|
|
|
|
uap->uap_zhp = zhp;
|
|
|
|
list_insert_tail((list_t *)data, uap);
|
|
|
|
} else {
|
|
|
|
zpool_close(zhp);
|
|
|
|
}
|
|
|
|
return (0);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Two stage replace on Linux
|
|
|
|
* since we get disk notifications
|
|
|
|
* we can wait for partitioned disk slice to show up!
|
|
|
|
*
|
|
|
|
* First stage tags the disk, initiates async partitioning, and returns
|
|
|
|
* Second stage finds the tag and proceeds to ZFS labeling/replace
|
|
|
|
*
|
|
|
|
* disk-add --> label-disk + tag-disk --> partition-add --> zpool_vdev_attach
|
|
|
|
*
|
|
|
|
* 1. physical match with no fs, no partition
|
|
|
|
* tag it top, partition disk
|
|
|
|
*
|
|
|
|
* 2. physical match again, see partion and tag
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
|
|
|
* The device associated with the given vdev (either by devid or physical path)
|
|
|
|
* has been added to the system. If 'isdisk' is set, then we only attempt a
|
|
|
|
* replacement if it's a whole disk. This also implies that we should label the
|
|
|
|
* disk first.
|
|
|
|
*
|
|
|
|
* First, we attempt to online the device (making sure to undo any spare
|
|
|
|
* operation when finished). If this succeeds, then we're done. If it fails,
|
|
|
|
* and the new state is VDEV_CANT_OPEN, it indicates that the device was opened,
|
|
|
|
* but that the label was not what we expected. If the 'autoreplace' property
|
2016-11-07 23:01:38 +00:00
|
|
|
* is enabled, then we relabel the disk (if specified), and attempt a 'zpool
|
2016-08-31 21:46:58 +00:00
|
|
|
* replace'. If the online is successful, but the new state is something else
|
|
|
|
* (REMOVED or FAULTED), it indicates that we're out of sync or in some sort of
|
|
|
|
* race, and we should avoid attempting to relabel the disk.
|
|
|
|
*
|
|
|
|
* Also can arrive here from a ESC_ZFS_VDEV_CHECK event
|
|
|
|
*/
|
|
|
|
static void
|
|
|
|
zfs_process_add(zpool_handle_t *zhp, nvlist_t *vdev, boolean_t labeled)
|
|
|
|
{
|
|
|
|
char *path;
|
|
|
|
vdev_state_t newstate;
|
|
|
|
nvlist_t *nvroot, *newvd;
|
|
|
|
pendingdev_t *device;
|
|
|
|
uint64_t wholedisk = 0ULL;
|
|
|
|
uint64_t offline = 0ULL;
|
|
|
|
uint64_t guid = 0ULL;
|
2016-10-24 17:45:59 +00:00
|
|
|
char *physpath = NULL, *new_devid = NULL, *enc_sysfs_path = NULL;
|
2016-08-31 21:46:58 +00:00
|
|
|
char rawpath[PATH_MAX], fullpath[PATH_MAX];
|
|
|
|
char devpath[PATH_MAX];
|
|
|
|
int ret;
|
2016-10-19 19:55:59 +00:00
|
|
|
int is_dm = 0;
|
2017-03-02 16:47:26 +00:00
|
|
|
int is_sd = 0;
|
2016-10-19 19:55:59 +00:00
|
|
|
uint_t c;
|
|
|
|
vdev_stat_t *vs;
|
2016-08-31 21:46:58 +00:00
|
|
|
|
|
|
|
if (nvlist_lookup_string(vdev, ZPOOL_CONFIG_PATH, &path) != 0)
|
|
|
|
return;
|
|
|
|
|
2016-10-19 19:55:59 +00:00
|
|
|
/* Skip healthy disks */
|
|
|
|
verify(nvlist_lookup_uint64_array(vdev, ZPOOL_CONFIG_VDEV_STATS,
|
|
|
|
(uint64_t **)&vs, &c) == 0);
|
|
|
|
if (vs->vs_state == VDEV_STATE_HEALTHY) {
|
|
|
|
zed_log_msg(LOG_INFO, "%s: %s is already healthy, skip it.",
|
|
|
|
__func__, path);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2016-08-31 21:46:58 +00:00
|
|
|
(void) nvlist_lookup_string(vdev, ZPOOL_CONFIG_PHYS_PATH, &physpath);
|
2016-10-24 17:45:59 +00:00
|
|
|
(void) nvlist_lookup_string(vdev, ZPOOL_CONFIG_VDEV_ENC_SYSFS_PATH,
|
|
|
|
&enc_sysfs_path);
|
2016-08-31 21:46:58 +00:00
|
|
|
(void) nvlist_lookup_uint64(vdev, ZPOOL_CONFIG_WHOLE_DISK, &wholedisk);
|
|
|
|
(void) nvlist_lookup_uint64(vdev, ZPOOL_CONFIG_OFFLINE, &offline);
|
|
|
|
(void) nvlist_lookup_uint64(vdev, ZPOOL_CONFIG_GUID, &guid);
|
|
|
|
|
|
|
|
if (offline)
|
|
|
|
return; /* don't intervene if it was taken offline */
|
|
|
|
|
2016-10-24 17:45:59 +00:00
|
|
|
is_dm = zfs_dev_is_dm(path);
|
2016-10-19 19:55:59 +00:00
|
|
|
zed_log_msg(LOG_INFO, "zfs_process_add: pool '%s' vdev '%s', phys '%s'"
|
|
|
|
" wholedisk %d, dm %d (%llu)", zpool_get_name(zhp), path,
|
|
|
|
physpath ? physpath : "NULL", wholedisk, is_dm,
|
|
|
|
(long long unsigned int)guid);
|
2016-08-31 21:46:58 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* The VDEV guid is preferred for identification (gets passed in path)
|
|
|
|
*/
|
|
|
|
if (guid != 0) {
|
|
|
|
(void) snprintf(fullpath, sizeof (fullpath), "%llu",
|
|
|
|
(long long unsigned int)guid);
|
|
|
|
} else {
|
|
|
|
/*
|
|
|
|
* otherwise use path sans partition suffix for whole disks
|
|
|
|
*/
|
|
|
|
(void) strlcpy(fullpath, path, sizeof (fullpath));
|
|
|
|
if (wholedisk) {
|
2016-10-19 19:55:59 +00:00
|
|
|
char *spath = zfs_strip_partition(fullpath);
|
|
|
|
if (!spath) {
|
|
|
|
zed_log_msg(LOG_INFO, "%s: Can't alloc",
|
|
|
|
__func__);
|
|
|
|
return;
|
|
|
|
}
|
2016-08-31 21:46:58 +00:00
|
|
|
|
|
|
|
(void) strlcpy(fullpath, spath, sizeof (fullpath));
|
|
|
|
free(spath);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Attempt to online the device.
|
|
|
|
*/
|
|
|
|
if (zpool_vdev_online(zhp, fullpath,
|
|
|
|
ZFS_ONLINE_CHECKREMOVE | ZFS_ONLINE_UNSPARE, &newstate) == 0 &&
|
|
|
|
(newstate == VDEV_STATE_HEALTHY ||
|
|
|
|
newstate == VDEV_STATE_DEGRADED)) {
|
|
|
|
zed_log_msg(LOG_INFO, " zpool_vdev_online: vdev %s is %s",
|
|
|
|
fullpath, (newstate == VDEV_STATE_HEALTHY) ?
|
|
|
|
"HEALTHY" : "DEGRADED");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2017-03-02 16:47:26 +00:00
|
|
|
/*
|
|
|
|
* vdev_id alias rule for using scsi_debug devices (FMA automated
|
|
|
|
* testing)
|
|
|
|
*/
|
2017-04-06 20:18:22 +00:00
|
|
|
if (physpath != NULL && strcmp("scsidebug", physpath) == 0)
|
2017-03-02 16:47:26 +00:00
|
|
|
is_sd = 1;
|
|
|
|
|
2016-08-31 21:46:58 +00:00
|
|
|
/*
|
2016-11-07 23:01:38 +00:00
|
|
|
* If the pool doesn't have the autoreplace property set, then use
|
|
|
|
* vdev online to trigger a FMA fault by posting an ereport.
|
2016-08-31 21:46:58 +00:00
|
|
|
*/
|
2016-11-07 23:01:38 +00:00
|
|
|
if (!zpool_get_prop_int(zhp, ZPOOL_PROP_AUTOREPLACE, NULL) ||
|
|
|
|
!(wholedisk || is_dm) || (physpath == NULL)) {
|
2016-08-31 21:46:58 +00:00
|
|
|
(void) zpool_vdev_online(zhp, fullpath, ZFS_ONLINE_FORCEFAULT,
|
|
|
|
&newstate);
|
2016-11-07 23:01:38 +00:00
|
|
|
zed_log_msg(LOG_INFO, "Pool's autoreplace is not enabled or "
|
|
|
|
"not a whole disk for '%s'", fullpath);
|
2016-08-31 21:46:58 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
2017-03-02 16:47:26 +00:00
|
|
|
* Convert physical path into its current device node. Rawpath
|
|
|
|
* needs to be /dev/disk/by-vdev for a scsi_debug device since
|
|
|
|
* /dev/disk/by-path will not be present.
|
2016-08-31 21:46:58 +00:00
|
|
|
*/
|
2017-03-02 16:47:26 +00:00
|
|
|
(void) snprintf(rawpath, sizeof (rawpath), "%s%s",
|
|
|
|
is_sd ? DEV_BYVDEV_PATH : DEV_BYPATH_PATH, physpath);
|
|
|
|
|
2016-10-19 19:55:59 +00:00
|
|
|
if (realpath(rawpath, devpath) == NULL && !is_dm) {
|
2016-08-31 21:46:58 +00:00
|
|
|
zed_log_msg(LOG_INFO, " realpath: %s failed (%s)",
|
|
|
|
rawpath, strerror(errno));
|
|
|
|
|
|
|
|
(void) zpool_vdev_online(zhp, fullpath, ZFS_ONLINE_FORCEFAULT,
|
|
|
|
&newstate);
|
|
|
|
|
|
|
|
zed_log_msg(LOG_INFO, " zpool_vdev_online: %s FORCEFAULT (%s)",
|
|
|
|
fullpath, libzfs_error_description(g_zfshdl));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2016-10-19 19:55:59 +00:00
|
|
|
/* Only autoreplace bad disks */
|
|
|
|
if ((vs->vs_state != VDEV_STATE_DEGRADED) &&
|
|
|
|
(vs->vs_state != VDEV_STATE_FAULTED) &&
|
|
|
|
(vs->vs_state != VDEV_STATE_CANT_OPEN)) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
nvlist_lookup_string(vdev, "new_devid", &new_devid);
|
|
|
|
|
|
|
|
if (is_dm) {
|
|
|
|
/* Don't label device mapper or multipath disks. */
|
|
|
|
} else if (!labeled) {
|
|
|
|
/*
|
|
|
|
* we're auto-replacing a raw disk, so label it first
|
|
|
|
*/
|
2016-08-31 21:46:58 +00:00
|
|
|
char *leafname;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* If this is a request to label a whole disk, then attempt to
|
|
|
|
* write out the label. Before we can label the disk, we need
|
|
|
|
* to map the physical string that was matched on to the under
|
|
|
|
* lying device node.
|
|
|
|
*
|
|
|
|
* If any part of this process fails, then do a force online
|
|
|
|
* to trigger a ZFS fault for the device (and any hot spare
|
|
|
|
* replacement).
|
|
|
|
*/
|
|
|
|
leafname = strrchr(devpath, '/') + 1;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* If this is a request to label a whole disk, then attempt to
|
|
|
|
* write out the label.
|
|
|
|
*/
|
|
|
|
if (zpool_label_disk(g_zfshdl, zhp, leafname) != 0) {
|
|
|
|
zed_log_msg(LOG_INFO, " zpool_label_disk: could not "
|
|
|
|
"label '%s' (%s)", leafname,
|
|
|
|
libzfs_error_description(g_zfshdl));
|
|
|
|
|
|
|
|
(void) zpool_vdev_online(zhp, fullpath,
|
|
|
|
ZFS_ONLINE_FORCEFAULT, &newstate);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* The disk labeling is asynchronous on Linux. Just record
|
|
|
|
* this label request and return as there will be another
|
|
|
|
* disk add event for the partition after the labeling is
|
|
|
|
* completed.
|
|
|
|
*/
|
|
|
|
device = malloc(sizeof (pendingdev_t));
|
|
|
|
(void) strlcpy(device->pd_physpath, physpath,
|
|
|
|
sizeof (device->pd_physpath));
|
|
|
|
list_insert_tail(&g_device_list, device);
|
|
|
|
|
|
|
|
zed_log_msg(LOG_INFO, " zpool_label_disk: async '%s' (%llu)",
|
2016-12-12 18:46:26 +00:00
|
|
|
leafname, (u_longlong_t)guid);
|
2016-08-31 21:46:58 +00:00
|
|
|
|
|
|
|
return; /* resumes at EC_DEV_ADD.ESC_DISK for partition */
|
|
|
|
|
|
|
|
} else /* labeled */ {
|
|
|
|
boolean_t found = B_FALSE;
|
|
|
|
/*
|
|
|
|
* match up with request above to label the disk
|
|
|
|
*/
|
|
|
|
for (device = list_head(&g_device_list); device != NULL;
|
|
|
|
device = list_next(&g_device_list, device)) {
|
|
|
|
if (strcmp(physpath, device->pd_physpath) == 0) {
|
|
|
|
list_remove(&g_device_list, device);
|
|
|
|
free(device);
|
|
|
|
found = B_TRUE;
|
|
|
|
break;
|
|
|
|
}
|
2016-11-07 23:01:38 +00:00
|
|
|
zed_log_msg(LOG_INFO, "zpool_label_disk: %s != %s",
|
|
|
|
physpath, device->pd_physpath);
|
2016-08-31 21:46:58 +00:00
|
|
|
}
|
|
|
|
if (!found) {
|
|
|
|
/* unexpected partition slice encountered */
|
2016-11-07 23:01:38 +00:00
|
|
|
zed_log_msg(LOG_INFO, "labeled disk %s unexpected here",
|
|
|
|
fullpath);
|
2016-08-31 21:46:58 +00:00
|
|
|
(void) zpool_vdev_online(zhp, fullpath,
|
|
|
|
ZFS_ONLINE_FORCEFAULT, &newstate);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
zed_log_msg(LOG_INFO, " zpool_label_disk: resume '%s' (%llu)",
|
2016-12-12 18:46:26 +00:00
|
|
|
physpath, (u_longlong_t)guid);
|
2016-08-31 21:46:58 +00:00
|
|
|
|
|
|
|
(void) snprintf(devpath, sizeof (devpath), "%s%s",
|
|
|
|
DEV_BYID_PATH, new_devid);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Construct the root vdev to pass to zpool_vdev_attach(). While adding
|
|
|
|
* the entire vdev structure is harmless, we construct a reduced set of
|
|
|
|
* path/physpath/wholedisk to keep it simple.
|
|
|
|
*/
|
|
|
|
if (nvlist_alloc(&nvroot, NV_UNIQUE_NAME, 0) != 0) {
|
|
|
|
zed_log_msg(LOG_WARNING, "zfs_mod: nvlist_alloc out of memory");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (nvlist_alloc(&newvd, NV_UNIQUE_NAME, 0) != 0) {
|
|
|
|
zed_log_msg(LOG_WARNING, "zfs_mod: nvlist_alloc out of memory");
|
|
|
|
nvlist_free(nvroot);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (nvlist_add_string(newvd, ZPOOL_CONFIG_TYPE, VDEV_TYPE_DISK) != 0 ||
|
|
|
|
nvlist_add_string(newvd, ZPOOL_CONFIG_PATH, path) != 0 ||
|
|
|
|
nvlist_add_string(newvd, ZPOOL_CONFIG_DEVID, new_devid) != 0 ||
|
|
|
|
(physpath != NULL && nvlist_add_string(newvd,
|
|
|
|
ZPOOL_CONFIG_PHYS_PATH, physpath) != 0) ||
|
2016-11-04 20:34:13 +00:00
|
|
|
(enc_sysfs_path != NULL && nvlist_add_string(newvd,
|
|
|
|
ZPOOL_CONFIG_VDEV_ENC_SYSFS_PATH, enc_sysfs_path) != 0) ||
|
2016-08-31 21:46:58 +00:00
|
|
|
nvlist_add_uint64(newvd, ZPOOL_CONFIG_WHOLE_DISK, wholedisk) != 0 ||
|
|
|
|
nvlist_add_string(nvroot, ZPOOL_CONFIG_TYPE, VDEV_TYPE_ROOT) != 0 ||
|
|
|
|
nvlist_add_nvlist_array(nvroot, ZPOOL_CONFIG_CHILDREN, &newvd,
|
|
|
|
1) != 0) {
|
|
|
|
zed_log_msg(LOG_WARNING, "zfs_mod: unable to add nvlist pairs");
|
|
|
|
nvlist_free(newvd);
|
|
|
|
nvlist_free(nvroot);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
nvlist_free(newvd);
|
|
|
|
|
|
|
|
/*
|
2018-10-29 20:05:14 +00:00
|
|
|
* Wait for udev to verify the links exist, then auto-replace
|
|
|
|
* the leaf disk at same physical location.
|
2016-08-31 21:46:58 +00:00
|
|
|
*/
|
2018-10-29 20:05:14 +00:00
|
|
|
if (zpool_label_disk_wait(path, 3000) != 0) {
|
|
|
|
zed_log_msg(LOG_WARNING, "zfs_mod: expected replacement "
|
|
|
|
"disk %s is missing", path);
|
|
|
|
nvlist_free(nvroot);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2016-08-31 21:46:58 +00:00
|
|
|
ret = zpool_vdev_attach(zhp, fullpath, path, nvroot, B_TRUE);
|
|
|
|
|
|
|
|
zed_log_msg(LOG_INFO, " zpool_vdev_replace: %s with %s (%s)",
|
|
|
|
fullpath, path, (ret == 0) ? "no errors" :
|
|
|
|
libzfs_error_description(g_zfshdl));
|
|
|
|
|
|
|
|
nvlist_free(nvroot);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Utility functions to find a vdev matching given criteria.
|
|
|
|
*/
|
|
|
|
typedef struct dev_data {
|
|
|
|
const char *dd_compare;
|
|
|
|
const char *dd_prop;
|
|
|
|
zfs_process_func_t dd_func;
|
|
|
|
boolean_t dd_found;
|
|
|
|
boolean_t dd_islabeled;
|
|
|
|
uint64_t dd_pool_guid;
|
|
|
|
uint64_t dd_vdev_guid;
|
|
|
|
const char *dd_new_devid;
|
|
|
|
} dev_data_t;
|
|
|
|
|
|
|
|
static void
|
|
|
|
zfs_iter_vdev(zpool_handle_t *zhp, nvlist_t *nvl, void *data)
|
|
|
|
{
|
|
|
|
dev_data_t *dp = data;
|
2016-10-19 19:55:59 +00:00
|
|
|
char *path = NULL;
|
2016-08-31 21:46:58 +00:00
|
|
|
uint_t c, children;
|
|
|
|
nvlist_t **child;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* First iterate over any children.
|
|
|
|
*/
|
|
|
|
if (nvlist_lookup_nvlist_array(nvl, ZPOOL_CONFIG_CHILDREN,
|
|
|
|
&child, &children) == 0) {
|
|
|
|
for (c = 0; c < children; c++)
|
|
|
|
zfs_iter_vdev(zhp, child[c], data);
|
2018-09-18 21:45:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Iterate over any spares and cache devices
|
|
|
|
*/
|
|
|
|
if (nvlist_lookup_nvlist_array(nvl, ZPOOL_CONFIG_SPARES,
|
|
|
|
&child, &children) == 0) {
|
|
|
|
for (c = 0; c < children; c++)
|
|
|
|
zfs_iter_vdev(zhp, child[c], data);
|
|
|
|
}
|
|
|
|
if (nvlist_lookup_nvlist_array(nvl, ZPOOL_CONFIG_L2CACHE,
|
|
|
|
&child, &children) == 0) {
|
|
|
|
for (c = 0; c < children; c++)
|
|
|
|
zfs_iter_vdev(zhp, child[c], data);
|
2016-08-31 21:46:58 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* once a vdev was matched and processed there is nothing left to do */
|
|
|
|
if (dp->dd_found)
|
|
|
|
return;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Match by GUID if available otherwise fallback to devid or physical
|
|
|
|
*/
|
|
|
|
if (dp->dd_vdev_guid != 0) {
|
|
|
|
uint64_t guid;
|
|
|
|
|
|
|
|
if (nvlist_lookup_uint64(nvl, ZPOOL_CONFIG_GUID,
|
|
|
|
&guid) != 0 || guid != dp->dd_vdev_guid) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
zed_log_msg(LOG_INFO, " zfs_iter_vdev: matched on %llu", guid);
|
|
|
|
dp->dd_found = B_TRUE;
|
|
|
|
|
|
|
|
} else if (dp->dd_compare != NULL) {
|
|
|
|
/*
|
|
|
|
* NOTE: On Linux there is an event for partition, so unlike
|
2017-01-03 17:31:18 +00:00
|
|
|
* illumos, substring matching is not required to accommodate
|
2016-08-31 21:46:58 +00:00
|
|
|
* the partition suffix. An exact match will be present in
|
|
|
|
* the dp->dd_compare value.
|
|
|
|
*/
|
|
|
|
if (nvlist_lookup_string(nvl, dp->dd_prop, &path) != 0 ||
|
2016-10-19 19:55:59 +00:00
|
|
|
strcmp(dp->dd_compare, path) != 0)
|
2016-08-31 21:46:58 +00:00
|
|
|
return;
|
2016-10-19 19:55:59 +00:00
|
|
|
|
2016-08-31 21:46:58 +00:00
|
|
|
zed_log_msg(LOG_INFO, " zfs_iter_vdev: matched %s on %s",
|
|
|
|
dp->dd_prop, path);
|
|
|
|
dp->dd_found = B_TRUE;
|
|
|
|
|
|
|
|
/* pass the new devid for use by replacing code */
|
2016-10-19 19:55:59 +00:00
|
|
|
if (dp->dd_new_devid != NULL) {
|
2016-08-31 21:46:58 +00:00
|
|
|
(void) nvlist_add_string(nvl, "new_devid",
|
|
|
|
dp->dd_new_devid);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
(dp->dd_func)(zhp, nvl, dp->dd_islabeled);
|
|
|
|
}
|
|
|
|
|
2017-12-09 00:58:41 +00:00
|
|
|
void
|
2016-08-31 21:46:58 +00:00
|
|
|
zfs_enable_ds(void *arg)
|
|
|
|
{
|
|
|
|
unavailpool_t *pool = (unavailpool_t *)arg;
|
|
|
|
|
|
|
|
(void) zpool_enable_datasets(pool->uap_zhp, NULL, 0);
|
|
|
|
zpool_close(pool->uap_zhp);
|
2017-12-09 00:58:41 +00:00
|
|
|
free(pool);
|
2016-08-31 21:46:58 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
zfs_iter_pool(zpool_handle_t *zhp, void *data)
|
|
|
|
{
|
|
|
|
nvlist_t *config, *nvl;
|
|
|
|
dev_data_t *dp = data;
|
|
|
|
uint64_t pool_guid;
|
|
|
|
unavailpool_t *pool;
|
|
|
|
|
|
|
|
zed_log_msg(LOG_INFO, "zfs_iter_pool: evaluating vdevs on %s (by %s)",
|
|
|
|
zpool_get_name(zhp), dp->dd_vdev_guid ? "GUID" : dp->dd_prop);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* For each vdev in this pool, look for a match to apply dd_func
|
|
|
|
*/
|
|
|
|
if ((config = zpool_get_config(zhp, NULL)) != NULL) {
|
|
|
|
if (dp->dd_pool_guid == 0 ||
|
|
|
|
(nvlist_lookup_uint64(config, ZPOOL_CONFIG_POOL_GUID,
|
|
|
|
&pool_guid) == 0 && pool_guid == dp->dd_pool_guid)) {
|
|
|
|
(void) nvlist_lookup_nvlist(config,
|
|
|
|
ZPOOL_CONFIG_VDEV_TREE, &nvl);
|
|
|
|
zfs_iter_vdev(zhp, nvl, data);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* if this pool was originally unavailable,
|
|
|
|
* then enable its datasets asynchronously
|
|
|
|
*/
|
|
|
|
if (g_enumeration_done) {
|
|
|
|
for (pool = list_head(&g_pool_list); pool != NULL;
|
|
|
|
pool = list_next(&g_pool_list, pool)) {
|
|
|
|
|
|
|
|
if (strcmp(zpool_get_name(zhp),
|
|
|
|
zpool_get_name(pool->uap_zhp)))
|
|
|
|
continue;
|
|
|
|
if (zfs_toplevel_state(zhp) >= VDEV_STATE_DEGRADED) {
|
2017-12-09 00:58:41 +00:00
|
|
|
list_remove(&g_pool_list, pool);
|
|
|
|
(void) tpool_dispatch(g_tpool, zfs_enable_ds,
|
|
|
|
pool);
|
2016-08-31 21:46:58 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
zpool_close(zhp);
|
|
|
|
return (dp->dd_found); /* cease iteration after a match */
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Given a physical device location, iterate over all
|
|
|
|
* (pool, vdev) pairs which correspond to that location.
|
|
|
|
*/
|
|
|
|
static boolean_t
|
|
|
|
devphys_iter(const char *physical, const char *devid, zfs_process_func_t func,
|
|
|
|
boolean_t is_slice)
|
|
|
|
{
|
|
|
|
dev_data_t data = { 0 };
|
|
|
|
|
|
|
|
data.dd_compare = physical;
|
|
|
|
data.dd_func = func;
|
|
|
|
data.dd_prop = ZPOOL_CONFIG_PHYS_PATH;
|
|
|
|
data.dd_found = B_FALSE;
|
|
|
|
data.dd_islabeled = is_slice;
|
|
|
|
data.dd_new_devid = devid; /* used by auto replace code */
|
|
|
|
|
|
|
|
(void) zpool_iter(g_zfshdl, zfs_iter_pool, &data);
|
|
|
|
|
|
|
|
return (data.dd_found);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Given a device identifier, find any vdevs with a matching devid.
|
|
|
|
* On Linux we can match devid directly which is always a whole disk.
|
|
|
|
*/
|
|
|
|
static boolean_t
|
|
|
|
devid_iter(const char *devid, zfs_process_func_t func, boolean_t is_slice)
|
|
|
|
{
|
|
|
|
dev_data_t data = { 0 };
|
|
|
|
|
|
|
|
data.dd_compare = devid;
|
|
|
|
data.dd_func = func;
|
|
|
|
data.dd_prop = ZPOOL_CONFIG_DEVID;
|
|
|
|
data.dd_found = B_FALSE;
|
|
|
|
data.dd_islabeled = is_slice;
|
|
|
|
data.dd_new_devid = devid;
|
|
|
|
|
|
|
|
(void) zpool_iter(g_zfshdl, zfs_iter_pool, &data);
|
|
|
|
|
|
|
|
return (data.dd_found);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Handle a EC_DEV_ADD.ESC_DISK event.
|
|
|
|
*
|
|
|
|
* illumos
|
|
|
|
* Expects: DEV_PHYS_PATH string in schema
|
|
|
|
* Matches: vdev's ZPOOL_CONFIG_PHYS_PATH or ZPOOL_CONFIG_DEVID
|
|
|
|
*
|
|
|
|
* path: '/dev/dsk/c0t1d0s0' (persistent)
|
|
|
|
* devid: 'id1,sd@SATA_____Hitachi_HDS72101______JP2940HZ3H74MC/a'
|
|
|
|
* phys_path: '/pci@0,0/pci103c,1609@11/disk@1,0:a'
|
|
|
|
*
|
|
|
|
* linux
|
|
|
|
* provides: DEV_PHYS_PATH and DEV_IDENTIFIER strings in schema
|
|
|
|
* Matches: vdev's ZPOOL_CONFIG_PHYS_PATH or ZPOOL_CONFIG_DEVID
|
|
|
|
*
|
|
|
|
* path: '/dev/sdc1' (not persistent)
|
|
|
|
* devid: 'ata-SAMSUNG_HD204UI_S2HGJD2Z805891-part1'
|
|
|
|
* phys_path: 'pci-0000:04:00.0-sas-0x4433221106000000-lun-0'
|
|
|
|
*/
|
|
|
|
static int
|
|
|
|
zfs_deliver_add(nvlist_t *nvl, boolean_t is_lofi)
|
|
|
|
{
|
|
|
|
char *devpath = NULL, *devid;
|
|
|
|
boolean_t is_slice;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Expecting a devid string and an optional physical location
|
|
|
|
*/
|
|
|
|
if (nvlist_lookup_string(nvl, DEV_IDENTIFIER, &devid) != 0)
|
|
|
|
return (-1);
|
|
|
|
|
|
|
|
(void) nvlist_lookup_string(nvl, DEV_PHYS_PATH, &devpath);
|
|
|
|
|
|
|
|
is_slice = (nvlist_lookup_boolean(nvl, DEV_IS_PART) == 0);
|
|
|
|
|
2016-10-19 19:55:59 +00:00
|
|
|
zed_log_msg(LOG_INFO, "zfs_deliver_add: adding %s (%s) (is_slice %d)",
|
|
|
|
devid, devpath ? devpath : "NULL", is_slice);
|
|
|
|
|
2016-08-31 21:46:58 +00:00
|
|
|
/*
|
|
|
|
* Iterate over all vdevs looking for a match in the folllowing order:
|
|
|
|
* 1. ZPOOL_CONFIG_DEVID (identifies the unique disk)
|
|
|
|
* 2. ZPOOL_CONFIG_PHYS_PATH (identifies disk physical location).
|
|
|
|
*
|
|
|
|
* For disks, we only want to pay attention to vdevs marked as whole
|
2016-11-07 23:01:38 +00:00
|
|
|
* disks or are a multipath device.
|
2016-08-31 21:46:58 +00:00
|
|
|
*/
|
2016-11-07 23:01:38 +00:00
|
|
|
if (!devid_iter(devid, zfs_process_add, is_slice) && devpath != NULL)
|
|
|
|
(void) devphys_iter(devpath, devid, zfs_process_add, is_slice);
|
2016-08-31 21:46:58 +00:00
|
|
|
|
|
|
|
return (0);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Called when we receive a VDEV_CHECK event, which indicates a device could not
|
|
|
|
* be opened during initial pool open, but the autoreplace property was set on
|
|
|
|
* the pool. In this case, we treat it as if it were an add event.
|
|
|
|
*/
|
|
|
|
static int
|
|
|
|
zfs_deliver_check(nvlist_t *nvl)
|
|
|
|
{
|
|
|
|
dev_data_t data = { 0 };
|
|
|
|
|
|
|
|
if (nvlist_lookup_uint64(nvl, ZFS_EV_POOL_GUID,
|
|
|
|
&data.dd_pool_guid) != 0 ||
|
|
|
|
nvlist_lookup_uint64(nvl, ZFS_EV_VDEV_GUID,
|
|
|
|
&data.dd_vdev_guid) != 0 ||
|
|
|
|
data.dd_vdev_guid == 0)
|
|
|
|
return (0);
|
|
|
|
|
|
|
|
zed_log_msg(LOG_INFO, "zfs_deliver_check: pool '%llu', vdev %llu",
|
|
|
|
data.dd_pool_guid, data.dd_vdev_guid);
|
|
|
|
|
|
|
|
data.dd_func = zfs_process_add;
|
|
|
|
|
|
|
|
(void) zpool_iter(g_zfshdl, zfs_iter_pool, &data);
|
|
|
|
|
|
|
|
return (0);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
zfsdle_vdev_online(zpool_handle_t *zhp, void *data)
|
|
|
|
{
|
|
|
|
char *devname = data;
|
|
|
|
boolean_t avail_spare, l2cache;
|
|
|
|
nvlist_t *tgt;
|
Add support for autoexpand property
While the autoexpand property may seem like a small feature it
depends on a significant amount of system infrastructure. Enough
of that infrastructure is now in place that with a few modifications
for Linux it can be supported.
Auto-expand works as follows; when a block device is modified
(re-sized, closed after being open r/w, etc) a change uevent is
generated for udev. The ZED, which is monitoring udev events,
passes the change event along to zfs_deliver_dle() if the disk
or partition contains a zfs_member as identified by blkid.
From here the device is matched against all imported pool vdevs
using the vdev_guid which was read from the label by blkid. If
a match is found the ZED reopens the pool vdev. This re-opening
is important because it allows the vdev to be briefly closed so
the disk partition table can be re-read. Otherwise, it wouldn't
be possible to report the maximum possible expansion size.
Finally, if the property autoexpand=on a vdev expansion will be
attempted. After performing some sanity checks on the disk to
verify that it is safe to expand, the primary partition (-part1)
will be expanded and the partition table updated. The partition
is then re-opened (again) to detect the updated size which allows
the new capacity to be used.
In order to make all of the above possible the following changes
were required:
* Updated the zpool_expand_001_pos and zpool_expand_003_pos tests.
These tests now create a pool which is layered on a loopback,
scsi_debug, and file vdev. This allows for testing of non-
partitioned block device (loopback), a partition block device
(scsi_debug), and a file which does not receive udev change
events. This provided for better test coverage, and by removing
the layering on ZFS volumes there issues surrounding layering
one pool on another are avoided.
* zpool_find_vdev_by_physpath() updated to accept a vdev guid.
This allows for matching by guid rather than path which is a
more reliable way for the ZED to reference a vdev.
* Fixed zfs_zevent_wait() signal handling which could result
in the ZED spinning when a signal was not handled.
* Removed vdev_disk_rrpart() functionality which can be abandoned
in favor of kernel provided blkdev_reread_part() function.
* Added a rwlock which is held as a writer while a disk is being
reopened. This is important to prevent errors from occurring
for any configuration related IOs which bypass the SCL_ZIO lock.
The zpool_reopen_007_pos.ksh test case was added to verify IO
error are never observed when reopening. This is not expected
to impact IO performance.
Additional fixes which aren't critical but were discovered and
resolved in the course of developing this functionality.
* Added PHYS_PATH="/dev/zvol/dataset" to the vdev configuration for
ZFS volumes. This is as good as a unique physical path, while the
volumes are not used in the test cases anymore for other reasons
this improvement was included.
Reviewed by: Richard Elling <Richard.Elling@RichardElling.com>
Signed-off-by: Sara Hartse <sara.hartse@delphix.com>
Signed-off-by: Brian Behlendorf <behlendorf1@llnl.gov>
Closes #120
Closes #2437
Closes #5771
Closes #7366
Closes #7582
Closes #7629
2018-07-23 22:40:15 +00:00
|
|
|
int error;
|
2016-08-31 21:46:58 +00:00
|
|
|
|
|
|
|
zed_log_msg(LOG_INFO, "zfsdle_vdev_online: searching for '%s' in '%s'",
|
|
|
|
devname, zpool_get_name(zhp));
|
|
|
|
|
|
|
|
if ((tgt = zpool_find_vdev_by_physpath(zhp, devname,
|
|
|
|
&avail_spare, &l2cache, NULL)) != NULL) {
|
|
|
|
char *path, fullpath[MAXPATHLEN];
|
Add support for autoexpand property
While the autoexpand property may seem like a small feature it
depends on a significant amount of system infrastructure. Enough
of that infrastructure is now in place that with a few modifications
for Linux it can be supported.
Auto-expand works as follows; when a block device is modified
(re-sized, closed after being open r/w, etc) a change uevent is
generated for udev. The ZED, which is monitoring udev events,
passes the change event along to zfs_deliver_dle() if the disk
or partition contains a zfs_member as identified by blkid.
From here the device is matched against all imported pool vdevs
using the vdev_guid which was read from the label by blkid. If
a match is found the ZED reopens the pool vdev. This re-opening
is important because it allows the vdev to be briefly closed so
the disk partition table can be re-read. Otherwise, it wouldn't
be possible to report the maximum possible expansion size.
Finally, if the property autoexpand=on a vdev expansion will be
attempted. After performing some sanity checks on the disk to
verify that it is safe to expand, the primary partition (-part1)
will be expanded and the partition table updated. The partition
is then re-opened (again) to detect the updated size which allows
the new capacity to be used.
In order to make all of the above possible the following changes
were required:
* Updated the zpool_expand_001_pos and zpool_expand_003_pos tests.
These tests now create a pool which is layered on a loopback,
scsi_debug, and file vdev. This allows for testing of non-
partitioned block device (loopback), a partition block device
(scsi_debug), and a file which does not receive udev change
events. This provided for better test coverage, and by removing
the layering on ZFS volumes there issues surrounding layering
one pool on another are avoided.
* zpool_find_vdev_by_physpath() updated to accept a vdev guid.
This allows for matching by guid rather than path which is a
more reliable way for the ZED to reference a vdev.
* Fixed zfs_zevent_wait() signal handling which could result
in the ZED spinning when a signal was not handled.
* Removed vdev_disk_rrpart() functionality which can be abandoned
in favor of kernel provided blkdev_reread_part() function.
* Added a rwlock which is held as a writer while a disk is being
reopened. This is important to prevent errors from occurring
for any configuration related IOs which bypass the SCL_ZIO lock.
The zpool_reopen_007_pos.ksh test case was added to verify IO
error are never observed when reopening. This is not expected
to impact IO performance.
Additional fixes which aren't critical but were discovered and
resolved in the course of developing this functionality.
* Added PHYS_PATH="/dev/zvol/dataset" to the vdev configuration for
ZFS volumes. This is as good as a unique physical path, while the
volumes are not used in the test cases anymore for other reasons
this improvement was included.
Reviewed by: Richard Elling <Richard.Elling@RichardElling.com>
Signed-off-by: Sara Hartse <sara.hartse@delphix.com>
Signed-off-by: Brian Behlendorf <behlendorf1@llnl.gov>
Closes #120
Closes #2437
Closes #5771
Closes #7366
Closes #7582
Closes #7629
2018-07-23 22:40:15 +00:00
|
|
|
uint64_t wholedisk;
|
2016-08-31 21:46:58 +00:00
|
|
|
|
Add support for autoexpand property
While the autoexpand property may seem like a small feature it
depends on a significant amount of system infrastructure. Enough
of that infrastructure is now in place that with a few modifications
for Linux it can be supported.
Auto-expand works as follows; when a block device is modified
(re-sized, closed after being open r/w, etc) a change uevent is
generated for udev. The ZED, which is monitoring udev events,
passes the change event along to zfs_deliver_dle() if the disk
or partition contains a zfs_member as identified by blkid.
From here the device is matched against all imported pool vdevs
using the vdev_guid which was read from the label by blkid. If
a match is found the ZED reopens the pool vdev. This re-opening
is important because it allows the vdev to be briefly closed so
the disk partition table can be re-read. Otherwise, it wouldn't
be possible to report the maximum possible expansion size.
Finally, if the property autoexpand=on a vdev expansion will be
attempted. After performing some sanity checks on the disk to
verify that it is safe to expand, the primary partition (-part1)
will be expanded and the partition table updated. The partition
is then re-opened (again) to detect the updated size which allows
the new capacity to be used.
In order to make all of the above possible the following changes
were required:
* Updated the zpool_expand_001_pos and zpool_expand_003_pos tests.
These tests now create a pool which is layered on a loopback,
scsi_debug, and file vdev. This allows for testing of non-
partitioned block device (loopback), a partition block device
(scsi_debug), and a file which does not receive udev change
events. This provided for better test coverage, and by removing
the layering on ZFS volumes there issues surrounding layering
one pool on another are avoided.
* zpool_find_vdev_by_physpath() updated to accept a vdev guid.
This allows for matching by guid rather than path which is a
more reliable way for the ZED to reference a vdev.
* Fixed zfs_zevent_wait() signal handling which could result
in the ZED spinning when a signal was not handled.
* Removed vdev_disk_rrpart() functionality which can be abandoned
in favor of kernel provided blkdev_reread_part() function.
* Added a rwlock which is held as a writer while a disk is being
reopened. This is important to prevent errors from occurring
for any configuration related IOs which bypass the SCL_ZIO lock.
The zpool_reopen_007_pos.ksh test case was added to verify IO
error are never observed when reopening. This is not expected
to impact IO performance.
Additional fixes which aren't critical but were discovered and
resolved in the course of developing this functionality.
* Added PHYS_PATH="/dev/zvol/dataset" to the vdev configuration for
ZFS volumes. This is as good as a unique physical path, while the
volumes are not used in the test cases anymore for other reasons
this improvement was included.
Reviewed by: Richard Elling <Richard.Elling@RichardElling.com>
Signed-off-by: Sara Hartse <sara.hartse@delphix.com>
Signed-off-by: Brian Behlendorf <behlendorf1@llnl.gov>
Closes #120
Closes #2437
Closes #5771
Closes #7366
Closes #7582
Closes #7629
2018-07-23 22:40:15 +00:00
|
|
|
error = nvlist_lookup_string(tgt, ZPOOL_CONFIG_PATH, &path);
|
|
|
|
if (error) {
|
|
|
|
zpool_close(zhp);
|
|
|
|
return (0);
|
|
|
|
}
|
2016-08-31 21:46:58 +00:00
|
|
|
|
Add support for autoexpand property
While the autoexpand property may seem like a small feature it
depends on a significant amount of system infrastructure. Enough
of that infrastructure is now in place that with a few modifications
for Linux it can be supported.
Auto-expand works as follows; when a block device is modified
(re-sized, closed after being open r/w, etc) a change uevent is
generated for udev. The ZED, which is monitoring udev events,
passes the change event along to zfs_deliver_dle() if the disk
or partition contains a zfs_member as identified by blkid.
From here the device is matched against all imported pool vdevs
using the vdev_guid which was read from the label by blkid. If
a match is found the ZED reopens the pool vdev. This re-opening
is important because it allows the vdev to be briefly closed so
the disk partition table can be re-read. Otherwise, it wouldn't
be possible to report the maximum possible expansion size.
Finally, if the property autoexpand=on a vdev expansion will be
attempted. After performing some sanity checks on the disk to
verify that it is safe to expand, the primary partition (-part1)
will be expanded and the partition table updated. The partition
is then re-opened (again) to detect the updated size which allows
the new capacity to be used.
In order to make all of the above possible the following changes
were required:
* Updated the zpool_expand_001_pos and zpool_expand_003_pos tests.
These tests now create a pool which is layered on a loopback,
scsi_debug, and file vdev. This allows for testing of non-
partitioned block device (loopback), a partition block device
(scsi_debug), and a file which does not receive udev change
events. This provided for better test coverage, and by removing
the layering on ZFS volumes there issues surrounding layering
one pool on another are avoided.
* zpool_find_vdev_by_physpath() updated to accept a vdev guid.
This allows for matching by guid rather than path which is a
more reliable way for the ZED to reference a vdev.
* Fixed zfs_zevent_wait() signal handling which could result
in the ZED spinning when a signal was not handled.
* Removed vdev_disk_rrpart() functionality which can be abandoned
in favor of kernel provided blkdev_reread_part() function.
* Added a rwlock which is held as a writer while a disk is being
reopened. This is important to prevent errors from occurring
for any configuration related IOs which bypass the SCL_ZIO lock.
The zpool_reopen_007_pos.ksh test case was added to verify IO
error are never observed when reopening. This is not expected
to impact IO performance.
Additional fixes which aren't critical but were discovered and
resolved in the course of developing this functionality.
* Added PHYS_PATH="/dev/zvol/dataset" to the vdev configuration for
ZFS volumes. This is as good as a unique physical path, while the
volumes are not used in the test cases anymore for other reasons
this improvement was included.
Reviewed by: Richard Elling <Richard.Elling@RichardElling.com>
Signed-off-by: Sara Hartse <sara.hartse@delphix.com>
Signed-off-by: Brian Behlendorf <behlendorf1@llnl.gov>
Closes #120
Closes #2437
Closes #5771
Closes #7366
Closes #7582
Closes #7629
2018-07-23 22:40:15 +00:00
|
|
|
error = nvlist_lookup_uint64(tgt, ZPOOL_CONFIG_WHOLE_DISK,
|
|
|
|
&wholedisk);
|
|
|
|
if (error)
|
|
|
|
wholedisk = 0;
|
2017-10-26 19:26:09 +00:00
|
|
|
|
Add support for autoexpand property
While the autoexpand property may seem like a small feature it
depends on a significant amount of system infrastructure. Enough
of that infrastructure is now in place that with a few modifications
for Linux it can be supported.
Auto-expand works as follows; when a block device is modified
(re-sized, closed after being open r/w, etc) a change uevent is
generated for udev. The ZED, which is monitoring udev events,
passes the change event along to zfs_deliver_dle() if the disk
or partition contains a zfs_member as identified by blkid.
From here the device is matched against all imported pool vdevs
using the vdev_guid which was read from the label by blkid. If
a match is found the ZED reopens the pool vdev. This re-opening
is important because it allows the vdev to be briefly closed so
the disk partition table can be re-read. Otherwise, it wouldn't
be possible to report the maximum possible expansion size.
Finally, if the property autoexpand=on a vdev expansion will be
attempted. After performing some sanity checks on the disk to
verify that it is safe to expand, the primary partition (-part1)
will be expanded and the partition table updated. The partition
is then re-opened (again) to detect the updated size which allows
the new capacity to be used.
In order to make all of the above possible the following changes
were required:
* Updated the zpool_expand_001_pos and zpool_expand_003_pos tests.
These tests now create a pool which is layered on a loopback,
scsi_debug, and file vdev. This allows for testing of non-
partitioned block device (loopback), a partition block device
(scsi_debug), and a file which does not receive udev change
events. This provided for better test coverage, and by removing
the layering on ZFS volumes there issues surrounding layering
one pool on another are avoided.
* zpool_find_vdev_by_physpath() updated to accept a vdev guid.
This allows for matching by guid rather than path which is a
more reliable way for the ZED to reference a vdev.
* Fixed zfs_zevent_wait() signal handling which could result
in the ZED spinning when a signal was not handled.
* Removed vdev_disk_rrpart() functionality which can be abandoned
in favor of kernel provided blkdev_reread_part() function.
* Added a rwlock which is held as a writer while a disk is being
reopened. This is important to prevent errors from occurring
for any configuration related IOs which bypass the SCL_ZIO lock.
The zpool_reopen_007_pos.ksh test case was added to verify IO
error are never observed when reopening. This is not expected
to impact IO performance.
Additional fixes which aren't critical but were discovered and
resolved in the course of developing this functionality.
* Added PHYS_PATH="/dev/zvol/dataset" to the vdev configuration for
ZFS volumes. This is as good as a unique physical path, while the
volumes are not used in the test cases anymore for other reasons
this improvement was included.
Reviewed by: Richard Elling <Richard.Elling@RichardElling.com>
Signed-off-by: Sara Hartse <sara.hartse@delphix.com>
Signed-off-by: Brian Behlendorf <behlendorf1@llnl.gov>
Closes #120
Closes #2437
Closes #5771
Closes #7366
Closes #7582
Closes #7629
2018-07-23 22:40:15 +00:00
|
|
|
if (wholedisk) {
|
|
|
|
path = strrchr(path, '/');
|
|
|
|
if (path != NULL) {
|
|
|
|
path = zfs_strip_partition(path + 1);
|
|
|
|
if (path == NULL) {
|
|
|
|
zpool_close(zhp);
|
|
|
|
return (0);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
zpool_close(zhp);
|
2016-10-19 19:55:59 +00:00
|
|
|
return (0);
|
|
|
|
}
|
2016-08-31 21:46:58 +00:00
|
|
|
|
Add support for autoexpand property
While the autoexpand property may seem like a small feature it
depends on a significant amount of system infrastructure. Enough
of that infrastructure is now in place that with a few modifications
for Linux it can be supported.
Auto-expand works as follows; when a block device is modified
(re-sized, closed after being open r/w, etc) a change uevent is
generated for udev. The ZED, which is monitoring udev events,
passes the change event along to zfs_deliver_dle() if the disk
or partition contains a zfs_member as identified by blkid.
From here the device is matched against all imported pool vdevs
using the vdev_guid which was read from the label by blkid. If
a match is found the ZED reopens the pool vdev. This re-opening
is important because it allows the vdev to be briefly closed so
the disk partition table can be re-read. Otherwise, it wouldn't
be possible to report the maximum possible expansion size.
Finally, if the property autoexpand=on a vdev expansion will be
attempted. After performing some sanity checks on the disk to
verify that it is safe to expand, the primary partition (-part1)
will be expanded and the partition table updated. The partition
is then re-opened (again) to detect the updated size which allows
the new capacity to be used.
In order to make all of the above possible the following changes
were required:
* Updated the zpool_expand_001_pos and zpool_expand_003_pos tests.
These tests now create a pool which is layered on a loopback,
scsi_debug, and file vdev. This allows for testing of non-
partitioned block device (loopback), a partition block device
(scsi_debug), and a file which does not receive udev change
events. This provided for better test coverage, and by removing
the layering on ZFS volumes there issues surrounding layering
one pool on another are avoided.
* zpool_find_vdev_by_physpath() updated to accept a vdev guid.
This allows for matching by guid rather than path which is a
more reliable way for the ZED to reference a vdev.
* Fixed zfs_zevent_wait() signal handling which could result
in the ZED spinning when a signal was not handled.
* Removed vdev_disk_rrpart() functionality which can be abandoned
in favor of kernel provided blkdev_reread_part() function.
* Added a rwlock which is held as a writer while a disk is being
reopened. This is important to prevent errors from occurring
for any configuration related IOs which bypass the SCL_ZIO lock.
The zpool_reopen_007_pos.ksh test case was added to verify IO
error are never observed when reopening. This is not expected
to impact IO performance.
Additional fixes which aren't critical but were discovered and
resolved in the course of developing this functionality.
* Added PHYS_PATH="/dev/zvol/dataset" to the vdev configuration for
ZFS volumes. This is as good as a unique physical path, while the
volumes are not used in the test cases anymore for other reasons
this improvement was included.
Reviewed by: Richard Elling <Richard.Elling@RichardElling.com>
Signed-off-by: Sara Hartse <sara.hartse@delphix.com>
Signed-off-by: Brian Behlendorf <behlendorf1@llnl.gov>
Closes #120
Closes #2437
Closes #5771
Closes #7366
Closes #7582
Closes #7629
2018-07-23 22:40:15 +00:00
|
|
|
(void) strlcpy(fullpath, path, sizeof (fullpath));
|
|
|
|
free(path);
|
2016-08-31 21:46:58 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* We need to reopen the pool associated with this
|
Add support for autoexpand property
While the autoexpand property may seem like a small feature it
depends on a significant amount of system infrastructure. Enough
of that infrastructure is now in place that with a few modifications
for Linux it can be supported.
Auto-expand works as follows; when a block device is modified
(re-sized, closed after being open r/w, etc) a change uevent is
generated for udev. The ZED, which is monitoring udev events,
passes the change event along to zfs_deliver_dle() if the disk
or partition contains a zfs_member as identified by blkid.
From here the device is matched against all imported pool vdevs
using the vdev_guid which was read from the label by blkid. If
a match is found the ZED reopens the pool vdev. This re-opening
is important because it allows the vdev to be briefly closed so
the disk partition table can be re-read. Otherwise, it wouldn't
be possible to report the maximum possible expansion size.
Finally, if the property autoexpand=on a vdev expansion will be
attempted. After performing some sanity checks on the disk to
verify that it is safe to expand, the primary partition (-part1)
will be expanded and the partition table updated. The partition
is then re-opened (again) to detect the updated size which allows
the new capacity to be used.
In order to make all of the above possible the following changes
were required:
* Updated the zpool_expand_001_pos and zpool_expand_003_pos tests.
These tests now create a pool which is layered on a loopback,
scsi_debug, and file vdev. This allows for testing of non-
partitioned block device (loopback), a partition block device
(scsi_debug), and a file which does not receive udev change
events. This provided for better test coverage, and by removing
the layering on ZFS volumes there issues surrounding layering
one pool on another are avoided.
* zpool_find_vdev_by_physpath() updated to accept a vdev guid.
This allows for matching by guid rather than path which is a
more reliable way for the ZED to reference a vdev.
* Fixed zfs_zevent_wait() signal handling which could result
in the ZED spinning when a signal was not handled.
* Removed vdev_disk_rrpart() functionality which can be abandoned
in favor of kernel provided blkdev_reread_part() function.
* Added a rwlock which is held as a writer while a disk is being
reopened. This is important to prevent errors from occurring
for any configuration related IOs which bypass the SCL_ZIO lock.
The zpool_reopen_007_pos.ksh test case was added to verify IO
error are never observed when reopening. This is not expected
to impact IO performance.
Additional fixes which aren't critical but were discovered and
resolved in the course of developing this functionality.
* Added PHYS_PATH="/dev/zvol/dataset" to the vdev configuration for
ZFS volumes. This is as good as a unique physical path, while the
volumes are not used in the test cases anymore for other reasons
this improvement was included.
Reviewed by: Richard Elling <Richard.Elling@RichardElling.com>
Signed-off-by: Sara Hartse <sara.hartse@delphix.com>
Signed-off-by: Brian Behlendorf <behlendorf1@llnl.gov>
Closes #120
Closes #2437
Closes #5771
Closes #7366
Closes #7582
Closes #7629
2018-07-23 22:40:15 +00:00
|
|
|
* device so that the kernel can update the size of
|
|
|
|
* the expanded device. When expanding there is no
|
|
|
|
* need to restart the scrub from the beginning.
|
2016-08-31 21:46:58 +00:00
|
|
|
*/
|
Add support for autoexpand property
While the autoexpand property may seem like a small feature it
depends on a significant amount of system infrastructure. Enough
of that infrastructure is now in place that with a few modifications
for Linux it can be supported.
Auto-expand works as follows; when a block device is modified
(re-sized, closed after being open r/w, etc) a change uevent is
generated for udev. The ZED, which is monitoring udev events,
passes the change event along to zfs_deliver_dle() if the disk
or partition contains a zfs_member as identified by blkid.
From here the device is matched against all imported pool vdevs
using the vdev_guid which was read from the label by blkid. If
a match is found the ZED reopens the pool vdev. This re-opening
is important because it allows the vdev to be briefly closed so
the disk partition table can be re-read. Otherwise, it wouldn't
be possible to report the maximum possible expansion size.
Finally, if the property autoexpand=on a vdev expansion will be
attempted. After performing some sanity checks on the disk to
verify that it is safe to expand, the primary partition (-part1)
will be expanded and the partition table updated. The partition
is then re-opened (again) to detect the updated size which allows
the new capacity to be used.
In order to make all of the above possible the following changes
were required:
* Updated the zpool_expand_001_pos and zpool_expand_003_pos tests.
These tests now create a pool which is layered on a loopback,
scsi_debug, and file vdev. This allows for testing of non-
partitioned block device (loopback), a partition block device
(scsi_debug), and a file which does not receive udev change
events. This provided for better test coverage, and by removing
the layering on ZFS volumes there issues surrounding layering
one pool on another are avoided.
* zpool_find_vdev_by_physpath() updated to accept a vdev guid.
This allows for matching by guid rather than path which is a
more reliable way for the ZED to reference a vdev.
* Fixed zfs_zevent_wait() signal handling which could result
in the ZED spinning when a signal was not handled.
* Removed vdev_disk_rrpart() functionality which can be abandoned
in favor of kernel provided blkdev_reread_part() function.
* Added a rwlock which is held as a writer while a disk is being
reopened. This is important to prevent errors from occurring
for any configuration related IOs which bypass the SCL_ZIO lock.
The zpool_reopen_007_pos.ksh test case was added to verify IO
error are never observed when reopening. This is not expected
to impact IO performance.
Additional fixes which aren't critical but were discovered and
resolved in the course of developing this functionality.
* Added PHYS_PATH="/dev/zvol/dataset" to the vdev configuration for
ZFS volumes. This is as good as a unique physical path, while the
volumes are not used in the test cases anymore for other reasons
this improvement was included.
Reviewed by: Richard Elling <Richard.Elling@RichardElling.com>
Signed-off-by: Sara Hartse <sara.hartse@delphix.com>
Signed-off-by: Brian Behlendorf <behlendorf1@llnl.gov>
Closes #120
Closes #2437
Closes #5771
Closes #7366
Closes #7582
Closes #7629
2018-07-23 22:40:15 +00:00
|
|
|
boolean_t scrub_restart = B_FALSE;
|
2017-10-26 19:26:09 +00:00
|
|
|
(void) zpool_reopen_one(zhp, &scrub_restart);
|
Add support for autoexpand property
While the autoexpand property may seem like a small feature it
depends on a significant amount of system infrastructure. Enough
of that infrastructure is now in place that with a few modifications
for Linux it can be supported.
Auto-expand works as follows; when a block device is modified
(re-sized, closed after being open r/w, etc) a change uevent is
generated for udev. The ZED, which is monitoring udev events,
passes the change event along to zfs_deliver_dle() if the disk
or partition contains a zfs_member as identified by blkid.
From here the device is matched against all imported pool vdevs
using the vdev_guid which was read from the label by blkid. If
a match is found the ZED reopens the pool vdev. This re-opening
is important because it allows the vdev to be briefly closed so
the disk partition table can be re-read. Otherwise, it wouldn't
be possible to report the maximum possible expansion size.
Finally, if the property autoexpand=on a vdev expansion will be
attempted. After performing some sanity checks on the disk to
verify that it is safe to expand, the primary partition (-part1)
will be expanded and the partition table updated. The partition
is then re-opened (again) to detect the updated size which allows
the new capacity to be used.
In order to make all of the above possible the following changes
were required:
* Updated the zpool_expand_001_pos and zpool_expand_003_pos tests.
These tests now create a pool which is layered on a loopback,
scsi_debug, and file vdev. This allows for testing of non-
partitioned block device (loopback), a partition block device
(scsi_debug), and a file which does not receive udev change
events. This provided for better test coverage, and by removing
the layering on ZFS volumes there issues surrounding layering
one pool on another are avoided.
* zpool_find_vdev_by_physpath() updated to accept a vdev guid.
This allows for matching by guid rather than path which is a
more reliable way for the ZED to reference a vdev.
* Fixed zfs_zevent_wait() signal handling which could result
in the ZED spinning when a signal was not handled.
* Removed vdev_disk_rrpart() functionality which can be abandoned
in favor of kernel provided blkdev_reread_part() function.
* Added a rwlock which is held as a writer while a disk is being
reopened. This is important to prevent errors from occurring
for any configuration related IOs which bypass the SCL_ZIO lock.
The zpool_reopen_007_pos.ksh test case was added to verify IO
error are never observed when reopening. This is not expected
to impact IO performance.
Additional fixes which aren't critical but were discovered and
resolved in the course of developing this functionality.
* Added PHYS_PATH="/dev/zvol/dataset" to the vdev configuration for
ZFS volumes. This is as good as a unique physical path, while the
volumes are not used in the test cases anymore for other reasons
this improvement was included.
Reviewed by: Richard Elling <Richard.Elling@RichardElling.com>
Signed-off-by: Sara Hartse <sara.hartse@delphix.com>
Signed-off-by: Brian Behlendorf <behlendorf1@llnl.gov>
Closes #120
Closes #2437
Closes #5771
Closes #7366
Closes #7582
Closes #7629
2018-07-23 22:40:15 +00:00
|
|
|
} else {
|
|
|
|
(void) strlcpy(fullpath, path, sizeof (fullpath));
|
2016-08-31 21:46:58 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (zpool_get_prop_int(zhp, ZPOOL_PROP_AUTOEXPAND, NULL)) {
|
Add support for autoexpand property
While the autoexpand property may seem like a small feature it
depends on a significant amount of system infrastructure. Enough
of that infrastructure is now in place that with a few modifications
for Linux it can be supported.
Auto-expand works as follows; when a block device is modified
(re-sized, closed after being open r/w, etc) a change uevent is
generated for udev. The ZED, which is monitoring udev events,
passes the change event along to zfs_deliver_dle() if the disk
or partition contains a zfs_member as identified by blkid.
From here the device is matched against all imported pool vdevs
using the vdev_guid which was read from the label by blkid. If
a match is found the ZED reopens the pool vdev. This re-opening
is important because it allows the vdev to be briefly closed so
the disk partition table can be re-read. Otherwise, it wouldn't
be possible to report the maximum possible expansion size.
Finally, if the property autoexpand=on a vdev expansion will be
attempted. After performing some sanity checks on the disk to
verify that it is safe to expand, the primary partition (-part1)
will be expanded and the partition table updated. The partition
is then re-opened (again) to detect the updated size which allows
the new capacity to be used.
In order to make all of the above possible the following changes
were required:
* Updated the zpool_expand_001_pos and zpool_expand_003_pos tests.
These tests now create a pool which is layered on a loopback,
scsi_debug, and file vdev. This allows for testing of non-
partitioned block device (loopback), a partition block device
(scsi_debug), and a file which does not receive udev change
events. This provided for better test coverage, and by removing
the layering on ZFS volumes there issues surrounding layering
one pool on another are avoided.
* zpool_find_vdev_by_physpath() updated to accept a vdev guid.
This allows for matching by guid rather than path which is a
more reliable way for the ZED to reference a vdev.
* Fixed zfs_zevent_wait() signal handling which could result
in the ZED spinning when a signal was not handled.
* Removed vdev_disk_rrpart() functionality which can be abandoned
in favor of kernel provided blkdev_reread_part() function.
* Added a rwlock which is held as a writer while a disk is being
reopened. This is important to prevent errors from occurring
for any configuration related IOs which bypass the SCL_ZIO lock.
The zpool_reopen_007_pos.ksh test case was added to verify IO
error are never observed when reopening. This is not expected
to impact IO performance.
Additional fixes which aren't critical but were discovered and
resolved in the course of developing this functionality.
* Added PHYS_PATH="/dev/zvol/dataset" to the vdev configuration for
ZFS volumes. This is as good as a unique physical path, while the
volumes are not used in the test cases anymore for other reasons
this improvement was included.
Reviewed by: Richard Elling <Richard.Elling@RichardElling.com>
Signed-off-by: Sara Hartse <sara.hartse@delphix.com>
Signed-off-by: Brian Behlendorf <behlendorf1@llnl.gov>
Closes #120
Closes #2437
Closes #5771
Closes #7366
Closes #7582
Closes #7629
2018-07-23 22:40:15 +00:00
|
|
|
vdev_state_t newstate;
|
|
|
|
|
|
|
|
if (zpool_get_state(zhp) != POOL_STATE_UNAVAIL) {
|
|
|
|
error = zpool_vdev_online(zhp, fullpath, 0,
|
2016-08-31 21:46:58 +00:00
|
|
|
&newstate);
|
Add support for autoexpand property
While the autoexpand property may seem like a small feature it
depends on a significant amount of system infrastructure. Enough
of that infrastructure is now in place that with a few modifications
for Linux it can be supported.
Auto-expand works as follows; when a block device is modified
(re-sized, closed after being open r/w, etc) a change uevent is
generated for udev. The ZED, which is monitoring udev events,
passes the change event along to zfs_deliver_dle() if the disk
or partition contains a zfs_member as identified by blkid.
From here the device is matched against all imported pool vdevs
using the vdev_guid which was read from the label by blkid. If
a match is found the ZED reopens the pool vdev. This re-opening
is important because it allows the vdev to be briefly closed so
the disk partition table can be re-read. Otherwise, it wouldn't
be possible to report the maximum possible expansion size.
Finally, if the property autoexpand=on a vdev expansion will be
attempted. After performing some sanity checks on the disk to
verify that it is safe to expand, the primary partition (-part1)
will be expanded and the partition table updated. The partition
is then re-opened (again) to detect the updated size which allows
the new capacity to be used.
In order to make all of the above possible the following changes
were required:
* Updated the zpool_expand_001_pos and zpool_expand_003_pos tests.
These tests now create a pool which is layered on a loopback,
scsi_debug, and file vdev. This allows for testing of non-
partitioned block device (loopback), a partition block device
(scsi_debug), and a file which does not receive udev change
events. This provided for better test coverage, and by removing
the layering on ZFS volumes there issues surrounding layering
one pool on another are avoided.
* zpool_find_vdev_by_physpath() updated to accept a vdev guid.
This allows for matching by guid rather than path which is a
more reliable way for the ZED to reference a vdev.
* Fixed zfs_zevent_wait() signal handling which could result
in the ZED spinning when a signal was not handled.
* Removed vdev_disk_rrpart() functionality which can be abandoned
in favor of kernel provided blkdev_reread_part() function.
* Added a rwlock which is held as a writer while a disk is being
reopened. This is important to prevent errors from occurring
for any configuration related IOs which bypass the SCL_ZIO lock.
The zpool_reopen_007_pos.ksh test case was added to verify IO
error are never observed when reopening. This is not expected
to impact IO performance.
Additional fixes which aren't critical but were discovered and
resolved in the course of developing this functionality.
* Added PHYS_PATH="/dev/zvol/dataset" to the vdev configuration for
ZFS volumes. This is as good as a unique physical path, while the
volumes are not used in the test cases anymore for other reasons
this improvement was included.
Reviewed by: Richard Elling <Richard.Elling@RichardElling.com>
Signed-off-by: Sara Hartse <sara.hartse@delphix.com>
Signed-off-by: Brian Behlendorf <behlendorf1@llnl.gov>
Closes #120
Closes #2437
Closes #5771
Closes #7366
Closes #7582
Closes #7629
2018-07-23 22:40:15 +00:00
|
|
|
zed_log_msg(LOG_INFO, "zfsdle_vdev_online: "
|
|
|
|
"setting device '%s' to ONLINE state "
|
|
|
|
"in pool '%s': %d", fullpath,
|
|
|
|
zpool_get_name(zhp), error);
|
|
|
|
}
|
2016-08-31 21:46:58 +00:00
|
|
|
}
|
|
|
|
zpool_close(zhp);
|
|
|
|
return (1);
|
|
|
|
}
|
|
|
|
zpool_close(zhp);
|
|
|
|
return (0);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
Add support for autoexpand property
While the autoexpand property may seem like a small feature it
depends on a significant amount of system infrastructure. Enough
of that infrastructure is now in place that with a few modifications
for Linux it can be supported.
Auto-expand works as follows; when a block device is modified
(re-sized, closed after being open r/w, etc) a change uevent is
generated for udev. The ZED, which is monitoring udev events,
passes the change event along to zfs_deliver_dle() if the disk
or partition contains a zfs_member as identified by blkid.
From here the device is matched against all imported pool vdevs
using the vdev_guid which was read from the label by blkid. If
a match is found the ZED reopens the pool vdev. This re-opening
is important because it allows the vdev to be briefly closed so
the disk partition table can be re-read. Otherwise, it wouldn't
be possible to report the maximum possible expansion size.
Finally, if the property autoexpand=on a vdev expansion will be
attempted. After performing some sanity checks on the disk to
verify that it is safe to expand, the primary partition (-part1)
will be expanded and the partition table updated. The partition
is then re-opened (again) to detect the updated size which allows
the new capacity to be used.
In order to make all of the above possible the following changes
were required:
* Updated the zpool_expand_001_pos and zpool_expand_003_pos tests.
These tests now create a pool which is layered on a loopback,
scsi_debug, and file vdev. This allows for testing of non-
partitioned block device (loopback), a partition block device
(scsi_debug), and a file which does not receive udev change
events. This provided for better test coverage, and by removing
the layering on ZFS volumes there issues surrounding layering
one pool on another are avoided.
* zpool_find_vdev_by_physpath() updated to accept a vdev guid.
This allows for matching by guid rather than path which is a
more reliable way for the ZED to reference a vdev.
* Fixed zfs_zevent_wait() signal handling which could result
in the ZED spinning when a signal was not handled.
* Removed vdev_disk_rrpart() functionality which can be abandoned
in favor of kernel provided blkdev_reread_part() function.
* Added a rwlock which is held as a writer while a disk is being
reopened. This is important to prevent errors from occurring
for any configuration related IOs which bypass the SCL_ZIO lock.
The zpool_reopen_007_pos.ksh test case was added to verify IO
error are never observed when reopening. This is not expected
to impact IO performance.
Additional fixes which aren't critical but were discovered and
resolved in the course of developing this functionality.
* Added PHYS_PATH="/dev/zvol/dataset" to the vdev configuration for
ZFS volumes. This is as good as a unique physical path, while the
volumes are not used in the test cases anymore for other reasons
this improvement was included.
Reviewed by: Richard Elling <Richard.Elling@RichardElling.com>
Signed-off-by: Sara Hartse <sara.hartse@delphix.com>
Signed-off-by: Brian Behlendorf <behlendorf1@llnl.gov>
Closes #120
Closes #2437
Closes #5771
Closes #7366
Closes #7582
Closes #7629
2018-07-23 22:40:15 +00:00
|
|
|
* This function handles the ESC_DEV_DLE device change event. Use the
|
|
|
|
* provided vdev guid when looking up a disk or partition, when the guid
|
|
|
|
* is not present assume the entire disk is owned by ZFS and append the
|
|
|
|
* expected -part1 partition information then lookup by physical path.
|
2016-08-31 21:46:58 +00:00
|
|
|
*/
|
|
|
|
static int
|
|
|
|
zfs_deliver_dle(nvlist_t *nvl)
|
|
|
|
{
|
Add support for autoexpand property
While the autoexpand property may seem like a small feature it
depends on a significant amount of system infrastructure. Enough
of that infrastructure is now in place that with a few modifications
for Linux it can be supported.
Auto-expand works as follows; when a block device is modified
(re-sized, closed after being open r/w, etc) a change uevent is
generated for udev. The ZED, which is monitoring udev events,
passes the change event along to zfs_deliver_dle() if the disk
or partition contains a zfs_member as identified by blkid.
From here the device is matched against all imported pool vdevs
using the vdev_guid which was read from the label by blkid. If
a match is found the ZED reopens the pool vdev. This re-opening
is important because it allows the vdev to be briefly closed so
the disk partition table can be re-read. Otherwise, it wouldn't
be possible to report the maximum possible expansion size.
Finally, if the property autoexpand=on a vdev expansion will be
attempted. After performing some sanity checks on the disk to
verify that it is safe to expand, the primary partition (-part1)
will be expanded and the partition table updated. The partition
is then re-opened (again) to detect the updated size which allows
the new capacity to be used.
In order to make all of the above possible the following changes
were required:
* Updated the zpool_expand_001_pos and zpool_expand_003_pos tests.
These tests now create a pool which is layered on a loopback,
scsi_debug, and file vdev. This allows for testing of non-
partitioned block device (loopback), a partition block device
(scsi_debug), and a file which does not receive udev change
events. This provided for better test coverage, and by removing
the layering on ZFS volumes there issues surrounding layering
one pool on another are avoided.
* zpool_find_vdev_by_physpath() updated to accept a vdev guid.
This allows for matching by guid rather than path which is a
more reliable way for the ZED to reference a vdev.
* Fixed zfs_zevent_wait() signal handling which could result
in the ZED spinning when a signal was not handled.
* Removed vdev_disk_rrpart() functionality which can be abandoned
in favor of kernel provided blkdev_reread_part() function.
* Added a rwlock which is held as a writer while a disk is being
reopened. This is important to prevent errors from occurring
for any configuration related IOs which bypass the SCL_ZIO lock.
The zpool_reopen_007_pos.ksh test case was added to verify IO
error are never observed when reopening. This is not expected
to impact IO performance.
Additional fixes which aren't critical but were discovered and
resolved in the course of developing this functionality.
* Added PHYS_PATH="/dev/zvol/dataset" to the vdev configuration for
ZFS volumes. This is as good as a unique physical path, while the
volumes are not used in the test cases anymore for other reasons
this improvement was included.
Reviewed by: Richard Elling <Richard.Elling@RichardElling.com>
Signed-off-by: Sara Hartse <sara.hartse@delphix.com>
Signed-off-by: Brian Behlendorf <behlendorf1@llnl.gov>
Closes #120
Closes #2437
Closes #5771
Closes #7366
Closes #7582
Closes #7629
2018-07-23 22:40:15 +00:00
|
|
|
char *devname, name[MAXPATHLEN];
|
|
|
|
uint64_t guid;
|
|
|
|
|
|
|
|
if (nvlist_lookup_uint64(nvl, ZFS_EV_VDEV_GUID, &guid) == 0) {
|
|
|
|
sprintf(name, "%llu", (u_longlong_t)guid);
|
|
|
|
} else if (nvlist_lookup_string(nvl, DEV_PHYS_PATH, &devname) == 0) {
|
|
|
|
strlcpy(name, devname, MAXPATHLEN);
|
|
|
|
zfs_append_partition(name, MAXPATHLEN);
|
|
|
|
} else {
|
|
|
|
zed_log_msg(LOG_INFO, "zfs_deliver_dle: no guid or physpath");
|
2016-08-31 21:46:58 +00:00
|
|
|
}
|
|
|
|
|
Add support for autoexpand property
While the autoexpand property may seem like a small feature it
depends on a significant amount of system infrastructure. Enough
of that infrastructure is now in place that with a few modifications
for Linux it can be supported.
Auto-expand works as follows; when a block device is modified
(re-sized, closed after being open r/w, etc) a change uevent is
generated for udev. The ZED, which is monitoring udev events,
passes the change event along to zfs_deliver_dle() if the disk
or partition contains a zfs_member as identified by blkid.
From here the device is matched against all imported pool vdevs
using the vdev_guid which was read from the label by blkid. If
a match is found the ZED reopens the pool vdev. This re-opening
is important because it allows the vdev to be briefly closed so
the disk partition table can be re-read. Otherwise, it wouldn't
be possible to report the maximum possible expansion size.
Finally, if the property autoexpand=on a vdev expansion will be
attempted. After performing some sanity checks on the disk to
verify that it is safe to expand, the primary partition (-part1)
will be expanded and the partition table updated. The partition
is then re-opened (again) to detect the updated size which allows
the new capacity to be used.
In order to make all of the above possible the following changes
were required:
* Updated the zpool_expand_001_pos and zpool_expand_003_pos tests.
These tests now create a pool which is layered on a loopback,
scsi_debug, and file vdev. This allows for testing of non-
partitioned block device (loopback), a partition block device
(scsi_debug), and a file which does not receive udev change
events. This provided for better test coverage, and by removing
the layering on ZFS volumes there issues surrounding layering
one pool on another are avoided.
* zpool_find_vdev_by_physpath() updated to accept a vdev guid.
This allows for matching by guid rather than path which is a
more reliable way for the ZED to reference a vdev.
* Fixed zfs_zevent_wait() signal handling which could result
in the ZED spinning when a signal was not handled.
* Removed vdev_disk_rrpart() functionality which can be abandoned
in favor of kernel provided blkdev_reread_part() function.
* Added a rwlock which is held as a writer while a disk is being
reopened. This is important to prevent errors from occurring
for any configuration related IOs which bypass the SCL_ZIO lock.
The zpool_reopen_007_pos.ksh test case was added to verify IO
error are never observed when reopening. This is not expected
to impact IO performance.
Additional fixes which aren't critical but were discovered and
resolved in the course of developing this functionality.
* Added PHYS_PATH="/dev/zvol/dataset" to the vdev configuration for
ZFS volumes. This is as good as a unique physical path, while the
volumes are not used in the test cases anymore for other reasons
this improvement was included.
Reviewed by: Richard Elling <Richard.Elling@RichardElling.com>
Signed-off-by: Sara Hartse <sara.hartse@delphix.com>
Signed-off-by: Brian Behlendorf <behlendorf1@llnl.gov>
Closes #120
Closes #2437
Closes #5771
Closes #7366
Closes #7582
Closes #7629
2018-07-23 22:40:15 +00:00
|
|
|
if (zpool_iter(g_zfshdl, zfsdle_vdev_online, name) != 1) {
|
Enable remaining tests
Enable most of the remaining test cases which were previously
disabled. The required fixes are as follows:
* cache_001_pos - No changes required.
* cache_010_neg - Updated to use losetup under Linux. Loopback
cache devices are allowed, ZVOLs as cache devices are not.
Disabled until all the builders pass reliably.
* cachefile_001_pos, cachefile_002_pos, cachefile_003_pos,
cachefile_004_pos - Set set_device_dir path in cachefile.cfg,
updated CPATH1 and CPATH2 to reference unique files.
* zfs_clone_005_pos - Wait for udev to create volumes.
* zfs_mount_007_pos - Updated mount options to expected Linux names.
* zfs_mount_009_neg, zfs_mount_all_001_pos - No changes required.
* zfs_unmount_005_pos, zfs_unmount_009_pos, zfs_unmount_all_001_pos -
Updated to expect -f to not unmount busy mount points under Linux.
* rsend_019_pos - Observed to occasionally take a long time on both
32-bit systems and the kmemleak builder.
* zfs_written_property_001_pos - Switched sync(1) to sync_pool.
* devices_001_pos, devices_002_neg - Updated create_dev_file() helper
for Linux.
* exec_002_neg.ksh - Fixed mmap_exec.c to preserve errno. Updated
test case to expect EPERM from Linux as described by mmap(2).
* grow_pool_001_pos - Adding missing setup.ksh and cleanup.ksh
scripts from OpenZFS.
* grow_replicas_001_pos.ksh - Added missing $SLICE_* variables.
* history_004_pos, history_006_neg, history_008_pos - Fixed by
previous commits and were not enabled. No changes required.
* zfs_allow_010_pos - Added missing spaces after assorted zfs
commands in delegate_common.kshlib.
* inuse_* - Illumos dump device tests skipped. Remaining test
cases updated to correctly create required partitions.
* large_files_001_pos - Fixed largest_file.c to accept EINVAL
as well as EFBIG as described in write(2).
* link_count_001 - Added nproc to required commands.
* umountall_001 - Updated to use umount -a.
* online_offline_001_* - Pull in OpenZFS change to file_trunc.c
to make the '-c 0' option run the test in a loop. Included
online_offline.cfg file in all test cases.
* rename_dirs_001_pos - Updated to use the rename_dir test binary,
pkill restricted to exact matches and total runtime reduced.
* slog_013_neg, write_dirs_002_pos - No changes required.
* slog_013_pos.ksh - Updated to use losetup under Linux.
* slog_014_pos.ksh - ZED will not be running, manually degrade
the damaged vdev as expected.
* nopwrite_varying_compression, nopwrite_volume - Forced pool
sync with sync_pool to ensure up to date property values.
* Fixed typos in ZED log messages. Refactored zed_* helper
functions to resolve all-syslog exit=1 errors in zedlog.
* zfs_copies_005_neg, zfs_get_004_pos, zpool_add_004_pos,
zpool_destroy_001_pos, largest_pool_001_pos, clone_001_pos.ksh,
clone_001_pos, - Skip until layering pools on zvols is solid.
* largest_pool_001_pos - Limited to 7eb pool, maximum
supported size in 8eb-1 on Linux.
* zpool_expand_001_pos, zpool_expand_003_neg - Requires
additional support from the ZED, updated skip reason.
* zfs_rollback_001_pos, zfs_rollback_002_pos - Properly cleanup
busy mount points under Linux between test loops.
* privilege_001_pos, privilege_003_pos, rollback_003_pos,
threadsappend_001_pos - Skip with log_unsupported.
* snapshot_016_pos - No changes required.
* snapshot_008_pos - Increased LIMIT from 512K to 2M and added
sync_pool to avoid false positives.
Signed-off-by: Brian Behlendorf <behlendorf1@llnl.gov>
Closes #6128
2017-05-19 00:21:15 +00:00
|
|
|
zed_log_msg(LOG_INFO, "zfs_deliver_dle: device '%s' not "
|
Add support for autoexpand property
While the autoexpand property may seem like a small feature it
depends on a significant amount of system infrastructure. Enough
of that infrastructure is now in place that with a few modifications
for Linux it can be supported.
Auto-expand works as follows; when a block device is modified
(re-sized, closed after being open r/w, etc) a change uevent is
generated for udev. The ZED, which is monitoring udev events,
passes the change event along to zfs_deliver_dle() if the disk
or partition contains a zfs_member as identified by blkid.
From here the device is matched against all imported pool vdevs
using the vdev_guid which was read from the label by blkid. If
a match is found the ZED reopens the pool vdev. This re-opening
is important because it allows the vdev to be briefly closed so
the disk partition table can be re-read. Otherwise, it wouldn't
be possible to report the maximum possible expansion size.
Finally, if the property autoexpand=on a vdev expansion will be
attempted. After performing some sanity checks on the disk to
verify that it is safe to expand, the primary partition (-part1)
will be expanded and the partition table updated. The partition
is then re-opened (again) to detect the updated size which allows
the new capacity to be used.
In order to make all of the above possible the following changes
were required:
* Updated the zpool_expand_001_pos and zpool_expand_003_pos tests.
These tests now create a pool which is layered on a loopback,
scsi_debug, and file vdev. This allows for testing of non-
partitioned block device (loopback), a partition block device
(scsi_debug), and a file which does not receive udev change
events. This provided for better test coverage, and by removing
the layering on ZFS volumes there issues surrounding layering
one pool on another are avoided.
* zpool_find_vdev_by_physpath() updated to accept a vdev guid.
This allows for matching by guid rather than path which is a
more reliable way for the ZED to reference a vdev.
* Fixed zfs_zevent_wait() signal handling which could result
in the ZED spinning when a signal was not handled.
* Removed vdev_disk_rrpart() functionality which can be abandoned
in favor of kernel provided blkdev_reread_part() function.
* Added a rwlock which is held as a writer while a disk is being
reopened. This is important to prevent errors from occurring
for any configuration related IOs which bypass the SCL_ZIO lock.
The zpool_reopen_007_pos.ksh test case was added to verify IO
error are never observed when reopening. This is not expected
to impact IO performance.
Additional fixes which aren't critical but were discovered and
resolved in the course of developing this functionality.
* Added PHYS_PATH="/dev/zvol/dataset" to the vdev configuration for
ZFS volumes. This is as good as a unique physical path, while the
volumes are not used in the test cases anymore for other reasons
this improvement was included.
Reviewed by: Richard Elling <Richard.Elling@RichardElling.com>
Signed-off-by: Sara Hartse <sara.hartse@delphix.com>
Signed-off-by: Brian Behlendorf <behlendorf1@llnl.gov>
Closes #120
Closes #2437
Closes #5771
Closes #7366
Closes #7582
Closes #7629
2018-07-23 22:40:15 +00:00
|
|
|
"found", name);
|
2016-08-31 21:46:58 +00:00
|
|
|
return (1);
|
|
|
|
}
|
Add support for autoexpand property
While the autoexpand property may seem like a small feature it
depends on a significant amount of system infrastructure. Enough
of that infrastructure is now in place that with a few modifications
for Linux it can be supported.
Auto-expand works as follows; when a block device is modified
(re-sized, closed after being open r/w, etc) a change uevent is
generated for udev. The ZED, which is monitoring udev events,
passes the change event along to zfs_deliver_dle() if the disk
or partition contains a zfs_member as identified by blkid.
From here the device is matched against all imported pool vdevs
using the vdev_guid which was read from the label by blkid. If
a match is found the ZED reopens the pool vdev. This re-opening
is important because it allows the vdev to be briefly closed so
the disk partition table can be re-read. Otherwise, it wouldn't
be possible to report the maximum possible expansion size.
Finally, if the property autoexpand=on a vdev expansion will be
attempted. After performing some sanity checks on the disk to
verify that it is safe to expand, the primary partition (-part1)
will be expanded and the partition table updated. The partition
is then re-opened (again) to detect the updated size which allows
the new capacity to be used.
In order to make all of the above possible the following changes
were required:
* Updated the zpool_expand_001_pos and zpool_expand_003_pos tests.
These tests now create a pool which is layered on a loopback,
scsi_debug, and file vdev. This allows for testing of non-
partitioned block device (loopback), a partition block device
(scsi_debug), and a file which does not receive udev change
events. This provided for better test coverage, and by removing
the layering on ZFS volumes there issues surrounding layering
one pool on another are avoided.
* zpool_find_vdev_by_physpath() updated to accept a vdev guid.
This allows for matching by guid rather than path which is a
more reliable way for the ZED to reference a vdev.
* Fixed zfs_zevent_wait() signal handling which could result
in the ZED spinning when a signal was not handled.
* Removed vdev_disk_rrpart() functionality which can be abandoned
in favor of kernel provided blkdev_reread_part() function.
* Added a rwlock which is held as a writer while a disk is being
reopened. This is important to prevent errors from occurring
for any configuration related IOs which bypass the SCL_ZIO lock.
The zpool_reopen_007_pos.ksh test case was added to verify IO
error are never observed when reopening. This is not expected
to impact IO performance.
Additional fixes which aren't critical but were discovered and
resolved in the course of developing this functionality.
* Added PHYS_PATH="/dev/zvol/dataset" to the vdev configuration for
ZFS volumes. This is as good as a unique physical path, while the
volumes are not used in the test cases anymore for other reasons
this improvement was included.
Reviewed by: Richard Elling <Richard.Elling@RichardElling.com>
Signed-off-by: Sara Hartse <sara.hartse@delphix.com>
Signed-off-by: Brian Behlendorf <behlendorf1@llnl.gov>
Closes #120
Closes #2437
Closes #5771
Closes #7366
Closes #7582
Closes #7629
2018-07-23 22:40:15 +00:00
|
|
|
|
2016-08-31 21:46:58 +00:00
|
|
|
return (0);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* syseventd daemon module event handler
|
|
|
|
*
|
|
|
|
* Handles syseventd daemon zfs device related events:
|
|
|
|
*
|
|
|
|
* EC_DEV_ADD.ESC_DISK
|
|
|
|
* EC_DEV_STATUS.ESC_DEV_DLE
|
|
|
|
* EC_ZFS.ESC_ZFS_VDEV_CHECK
|
|
|
|
*
|
|
|
|
* Note: assumes only one thread active at a time (not thread safe)
|
|
|
|
*/
|
|
|
|
static int
|
|
|
|
zfs_slm_deliver_event(const char *class, const char *subclass, nvlist_t *nvl)
|
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
boolean_t is_lofi = B_FALSE, is_check = B_FALSE, is_dle = B_FALSE;
|
|
|
|
|
|
|
|
if (strcmp(class, EC_DEV_ADD) == 0) {
|
|
|
|
/*
|
|
|
|
* We're mainly interested in disk additions, but we also listen
|
|
|
|
* for new loop devices, to allow for simplified testing.
|
|
|
|
*/
|
|
|
|
if (strcmp(subclass, ESC_DISK) == 0)
|
|
|
|
is_lofi = B_FALSE;
|
|
|
|
else if (strcmp(subclass, ESC_LOFI) == 0)
|
|
|
|
is_lofi = B_TRUE;
|
|
|
|
else
|
|
|
|
return (0);
|
|
|
|
|
|
|
|
is_check = B_FALSE;
|
|
|
|
} else if (strcmp(class, EC_ZFS) == 0 &&
|
|
|
|
strcmp(subclass, ESC_ZFS_VDEV_CHECK) == 0) {
|
|
|
|
/*
|
|
|
|
* This event signifies that a device failed to open
|
|
|
|
* during pool load, but the 'autoreplace' property was
|
|
|
|
* set, so we should pretend it's just been added.
|
|
|
|
*/
|
|
|
|
is_check = B_TRUE;
|
|
|
|
} else if (strcmp(class, EC_DEV_STATUS) == 0 &&
|
|
|
|
strcmp(subclass, ESC_DEV_DLE) == 0) {
|
|
|
|
is_dle = B_TRUE;
|
|
|
|
} else {
|
|
|
|
return (0);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (is_dle)
|
|
|
|
ret = zfs_deliver_dle(nvl);
|
|
|
|
else if (is_check)
|
|
|
|
ret = zfs_deliver_check(nvl);
|
|
|
|
else
|
|
|
|
ret = zfs_deliver_add(nvl, is_lofi);
|
|
|
|
|
|
|
|
return (ret);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*ARGSUSED*/
|
|
|
|
static void *
|
|
|
|
zfs_enum_pools(void *arg)
|
|
|
|
{
|
|
|
|
(void) zpool_iter(g_zfshdl, zfs_unavail_pool, (void *)&g_pool_list);
|
|
|
|
/*
|
|
|
|
* Linux - instead of using a thread pool, each list entry
|
|
|
|
* will spawn a thread when an unavailable pool transitions
|
|
|
|
* to available. zfs_slm_fini will wait for these threads.
|
|
|
|
*/
|
|
|
|
g_enumeration_done = B_TRUE;
|
|
|
|
return (NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* called from zed daemon at startup
|
|
|
|
*
|
|
|
|
* sent messages from zevents or udev monitor
|
|
|
|
*
|
|
|
|
* For now, each agent has it's own libzfs instance
|
|
|
|
*/
|
|
|
|
int
|
2016-11-07 23:01:38 +00:00
|
|
|
zfs_slm_init()
|
2016-08-31 21:46:58 +00:00
|
|
|
{
|
2017-12-09 00:58:41 +00:00
|
|
|
if ((g_zfshdl = libzfs_init()) == NULL)
|
2016-08-31 21:46:58 +00:00
|
|
|
return (-1);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* collect a list of unavailable pools (asynchronously,
|
|
|
|
* since this can take a while)
|
|
|
|
*/
|
|
|
|
list_create(&g_pool_list, sizeof (struct unavailpool),
|
|
|
|
offsetof(struct unavailpool, uap_node));
|
|
|
|
|
|
|
|
if (pthread_create(&g_zfs_tid, NULL, zfs_enum_pools, NULL) != 0) {
|
|
|
|
list_destroy(&g_pool_list);
|
2017-12-09 00:58:41 +00:00
|
|
|
libzfs_fini(g_zfshdl);
|
2016-08-31 21:46:58 +00:00
|
|
|
return (-1);
|
|
|
|
}
|
|
|
|
|
|
|
|
list_create(&g_device_list, sizeof (struct pendingdev),
|
|
|
|
offsetof(struct pendingdev, pd_node));
|
|
|
|
|
|
|
|
return (0);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
zfs_slm_fini()
|
|
|
|
{
|
|
|
|
unavailpool_t *pool;
|
|
|
|
pendingdev_t *device;
|
|
|
|
|
|
|
|
/* wait for zfs_enum_pools thread to complete */
|
|
|
|
(void) pthread_join(g_zfs_tid, NULL);
|
2017-12-09 00:58:41 +00:00
|
|
|
/* destroy the thread pool */
|
|
|
|
if (g_tpool != NULL) {
|
|
|
|
tpool_wait(g_tpool);
|
|
|
|
tpool_destroy(g_tpool);
|
|
|
|
}
|
2016-08-31 21:46:58 +00:00
|
|
|
|
|
|
|
while ((pool = (list_head(&g_pool_list))) != NULL) {
|
|
|
|
list_remove(&g_pool_list, pool);
|
2017-12-09 00:58:41 +00:00
|
|
|
zpool_close(pool->uap_zhp);
|
2016-08-31 21:46:58 +00:00
|
|
|
free(pool);
|
|
|
|
}
|
|
|
|
list_destroy(&g_pool_list);
|
|
|
|
|
|
|
|
while ((device = (list_head(&g_device_list))) != NULL) {
|
|
|
|
list_remove(&g_device_list, device);
|
|
|
|
free(device);
|
|
|
|
}
|
|
|
|
list_destroy(&g_device_list);
|
|
|
|
|
2017-12-09 00:58:41 +00:00
|
|
|
libzfs_fini(g_zfshdl);
|
2016-08-31 21:46:58 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
zfs_slm_event(const char *class, const char *subclass, nvlist_t *nvl)
|
|
|
|
{
|
|
|
|
zed_log_msg(LOG_INFO, "zfs_slm_event: %s.%s", class, subclass);
|
|
|
|
(void) zfs_slm_deliver_event(class, subclass, nvl);
|
|
|
|
}
|