Merge commit 'refs/top-bases/linux-configure-branch' into linux-configure-branch

This commit is contained in:
Brian Behlendorf 2010-05-14 12:59:08 -07:00
commit 66c45ef1f4
18 changed files with 889 additions and 421 deletions

View File

@ -7,6 +7,7 @@ DEFAULT_INCLUDES += \
-I${top_srcdir}/lib/libzfs/include \ -I${top_srcdir}/lib/libzfs/include \
-I${top_srcdir}/lib/libnvpair/include \ -I${top_srcdir}/lib/libnvpair/include \
-I${top_srcdir}/module/zcommon/include \ -I${top_srcdir}/module/zcommon/include \
-I${top_srcdir}/module/zfs/include \
-I${top_srcdir}/module/nvpair/include \ -I${top_srcdir}/module/nvpair/include \
-I${top_srcdir}/module/avl/include \ -I${top_srcdir}/module/avl/include \
-I${top_srcdir}/module/unicode/include -I${top_srcdir}/module/unicode/include

View File

@ -42,8 +42,9 @@
#include <pwd.h> #include <pwd.h>
#include <zone.h> #include <zone.h>
#include <sys/fs/zfs.h> #include <sys/fs/zfs.h>
#include <sys/stat.h> #include <sys/stat.h>
#include <sys/fm/util.h>
#include <sys/fm/protocol.h>
#include <libzfs.h> #include <libzfs.h>
@ -76,6 +77,7 @@ static int zpool_do_export(int, char **);
static int zpool_do_upgrade(int, char **); static int zpool_do_upgrade(int, char **);
static int zpool_do_history(int, char **); static int zpool_do_history(int, char **);
static int zpool_do_events(int, char **);
static int zpool_do_get(int, char **); static int zpool_do_get(int, char **);
static int zpool_do_set(int, char **); static int zpool_do_set(int, char **);
@ -118,6 +120,7 @@ typedef enum {
HELP_SCRUB, HELP_SCRUB,
HELP_STATUS, HELP_STATUS,
HELP_UPGRADE, HELP_UPGRADE,
HELP_EVENTS,
HELP_GET, HELP_GET,
HELP_SET HELP_SET
} zpool_help_t; } zpool_help_t;
@ -164,6 +167,8 @@ static zpool_command_t command_table[] = {
{ "upgrade", zpool_do_upgrade, HELP_UPGRADE }, { "upgrade", zpool_do_upgrade, HELP_UPGRADE },
{ NULL }, { NULL },
{ "history", zpool_do_history, HELP_HISTORY }, { "history", zpool_do_history, HELP_HISTORY },
{ "events", zpool_do_events, HELP_EVENTS },
{ NULL },
{ "get", zpool_do_get, HELP_GET }, { "get", zpool_do_get, HELP_GET },
{ "set", zpool_do_set, HELP_SET }, { "set", zpool_do_set, HELP_SET },
}; };
@ -225,6 +230,8 @@ get_usage(zpool_help_t idx) {
return (gettext("\tupgrade\n" return (gettext("\tupgrade\n"
"\tupgrade -v\n" "\tupgrade -v\n"
"\tupgrade [-V version] <-a | pool ...>\n")); "\tupgrade [-V version] <-a | pool ...>\n"));
case HELP_EVENTS:
return (gettext("\tevents [-vfc]\n"));
case HELP_GET: case HELP_GET:
return (gettext("\tget <\"all\" | property[,...]> " return (gettext("\tget <\"all\" | property[,...]> "
"<pool> ...\n")); "<pool> ...\n"));
@ -3768,6 +3775,325 @@ zpool_do_history(int argc, char **argv)
return (ret); return (ret);
} }
typedef struct ev_opts {
int verbose;
int follow;
int clear;
} ev_opts_t;
static void
zpool_do_events_short(nvlist_t *nvl)
{
char ctime_str[26], str[32], *ptr;
int64_t *tv;
uint_t n;
verify(nvlist_lookup_int64_array(nvl, FM_EREPORT_TIME, &tv, &n) == 0);
memset(str, ' ', 32);
(void) ctime_r(&tv[0], ctime_str);
(void) strncpy(str, ctime_str+4, 6); /* 'Jun 30' */
(void) strncpy(str+7, ctime_str+20, 4); /* '1993' */
(void) strncpy(str+12, ctime_str+11, 8); /* '21:49:08' */
(void) sprintf(str+20, ".%09lld", (longlong_t)tv[1]);/* '.123456789' */
(void) printf(gettext("%s "), str);
verify(nvlist_lookup_string(nvl, FM_CLASS, &ptr) == 0);
(void) printf(gettext("%s\n"), ptr);
}
static void
zpool_do_events_nvprint(nvlist_t *nvl, int depth)
{
nvpair_t *nvp;
for (nvp = nvlist_next_nvpair(nvl, NULL);
nvp != NULL; nvp = nvlist_next_nvpair(nvl, nvp)) {
data_type_t type = nvpair_type(nvp);
const char *name = nvpair_name(nvp);
boolean_t b;
uint8_t i8;
uint16_t i16;
uint32_t i32;
uint64_t i64;
char *str;
nvlist_t *cnv;
printf(gettext("%*s%s = "), depth, "", name);
switch (type) {
case DATA_TYPE_BOOLEAN:
printf(gettext("%s"), "1");
break;
case DATA_TYPE_BOOLEAN_VALUE:
(void) nvpair_value_boolean_value(nvp, &b);
printf(gettext("%s"), b ? "1" : "0");
break;
case DATA_TYPE_BYTE:
(void) nvpair_value_byte(nvp, &i8);
printf(gettext("0x%x"), i8);
break;
case DATA_TYPE_INT8:
(void) nvpair_value_int8(nvp, (void *)&i8);
printf(gettext("0x%x"), i8);
break;
case DATA_TYPE_UINT8:
(void) nvpair_value_uint8(nvp, &i8);
printf(gettext("0x%x"), i8);
break;
case DATA_TYPE_INT16:
(void) nvpair_value_int16(nvp, (void *)&i16);
printf(gettext("0x%x"), i16);
break;
case DATA_TYPE_UINT16:
(void) nvpair_value_uint16(nvp, &i16);
printf(gettext("0x%x"), i16);
break;
case DATA_TYPE_INT32:
(void) nvpair_value_int32(nvp, (void *)&i32);
printf(gettext("0x%x"), i32);
break;
case DATA_TYPE_UINT32:
(void) nvpair_value_uint32(nvp, &i32);
printf(gettext("0x%x"), i32);
break;
case DATA_TYPE_INT64:
(void) nvpair_value_int64(nvp, (void *)&i64);
printf(gettext("0x%llx"), (u_longlong_t)i64);
break;
case DATA_TYPE_UINT64:
(void) nvpair_value_uint64(nvp, &i64);
printf(gettext("0x%llx"), (u_longlong_t)i64);
break;
case DATA_TYPE_HRTIME:
(void) nvpair_value_hrtime(nvp, (void *)&i64);
printf(gettext("0x%llx"), (u_longlong_t)i64);
break;
case DATA_TYPE_STRING:
(void) nvpair_value_string(nvp, &str);
printf(gettext("\"%s\""), str ? str : "<NULL>");
break;
case DATA_TYPE_NVLIST:
printf(gettext("(embedded nvlist)\n"));
(void) nvpair_value_nvlist(nvp, &cnv);
zpool_do_events_nvprint(cnv, depth + 8);
printf(gettext("%*s(end %s)\n"), depth, "", name);
break;
case DATA_TYPE_NVLIST_ARRAY: {
nvlist_t **val;
uint_t i, nelem;
(void) nvpair_value_nvlist_array(nvp, &val, &nelem);
printf(gettext("(%d embedded nvlists)\n"), nelem);
for (i = 0; i < nelem; i++) {
printf(gettext("%*s%s[%d] = %s\n"),
depth, "", name, i, "(embedded nvlist)");
zpool_do_events_nvprint(val[i], depth + 8);
printf(gettext("%*s(end %s[%i])\n"),
depth, "", name, i);
}
printf(gettext("%*s(end %s)\n"), depth, "", name);
}
break;
case DATA_TYPE_INT8_ARRAY: {
int8_t *val;
uint_t i, nelem;
(void) nvpair_value_int8_array(nvp, &val, &nelem);
for (i = 0; i < nelem; i++)
printf(gettext("0x%x "), val[i]);
break;
}
case DATA_TYPE_UINT8_ARRAY: {
uint8_t *val;
uint_t i, nelem;
(void) nvpair_value_uint8_array(nvp, &val, &nelem);
for (i = 0; i < nelem; i++)
printf(gettext("0x%x "), val[i]);
break;
}
case DATA_TYPE_INT16_ARRAY: {
int16_t *val;
uint_t i, nelem;
(void) nvpair_value_int16_array(nvp, &val, &nelem);
for (i = 0; i < nelem; i++)
printf(gettext("0x%x "), val[i]);
break;
}
case DATA_TYPE_UINT16_ARRAY: {
uint16_t *val;
uint_t i, nelem;
(void) nvpair_value_uint16_array(nvp, &val, &nelem);
for (i = 0; i < nelem; i++)
printf(gettext("0x%x "), val[i]);
break;
}
case DATA_TYPE_INT32_ARRAY: {
int32_t *val;
uint_t i, nelem;
(void) nvpair_value_int32_array(nvp, &val, &nelem);
for (i = 0; i < nelem; i++)
printf(gettext("0x%x "), val[i]);
break;
}
case DATA_TYPE_UINT32_ARRAY: {
uint32_t *val;
uint_t i, nelem;
(void) nvpair_value_uint32_array(nvp, &val, &nelem);
for (i = 0; i < nelem; i++)
printf(gettext("0x%x "), val[i]);
break;
}
case DATA_TYPE_INT64_ARRAY: {
int64_t *val;
uint_t i, nelem;
(void) nvpair_value_int64_array(nvp, &val, &nelem);
for (i = 0; i < nelem; i++)
printf(gettext("0x%llx "), (u_longlong_t)val[i]);
break;
}
case DATA_TYPE_UINT64_ARRAY: {
uint64_t *val;
uint_t i, nelem;
(void) nvpair_value_uint64_array(nvp, &val, &nelem);
for (i = 0; i < nelem; i++)
printf(gettext("0x%llx "), (u_longlong_t)val[i]);
break;
}
case DATA_TYPE_STRING_ARRAY:
case DATA_TYPE_BOOLEAN_ARRAY:
case DATA_TYPE_BYTE_ARRAY:
case DATA_TYPE_DOUBLE:
case DATA_TYPE_UNKNOWN:
printf(gettext("<unknown>"));
break;
}
printf(gettext("\n"));
}
}
static int
zpool_do_events_next(ev_opts_t *opts)
{
nvlist_t *nvl;
int ret, dropped;
(void) printf(gettext("%-27s %s\n"), "TIME", "CLASS");
while (1) {
ret = zpool_events_next(g_zfs, &nvl, &dropped, !!opts->follow);
if (ret || nvl == NULL)
break;
if (dropped > 0)
(void) printf(gettext("dropped %d events\n"), dropped);
zpool_do_events_short(nvl);
if (opts->verbose) {
zpool_do_events_nvprint(nvl, 8);
printf(gettext("\n"));
}
nvlist_free(nvl);
}
return (ret);
}
static int
zpool_do_events_clear(ev_opts_t *opts)
{
int count, ret;
ret = zpool_events_clear(g_zfs, &count);
if (!ret)
(void) printf(gettext("cleared %d events\n"), count);
return (ret);
}
/*
* zpool events [-vfc]
*
* Displays events logs by ZFS.
*/
int
zpool_do_events(int argc, char **argv)
{
ev_opts_t opts = { 0 };
int ret;
int c;
/* check options */
while ((c = getopt(argc, argv, "vfc")) != -1) {
switch (c) {
case 'v':
opts.verbose = 1;
break;
case 'f':
opts.follow = 1;
break;
case 'c':
opts.clear = 1;
break;
case '?':
(void) fprintf(stderr, gettext("invalid option '%c'\n"),
optopt);
usage(B_FALSE);
}
}
argc -= optind;
argv += optind;
if (opts.clear)
ret = zpool_do_events_clear(&opts);
else
ret = zpool_do_events_next(&opts);
return ret;
}
static int static int
get_callback(zpool_handle_t *zhp, void *data) get_callback(zpool_handle_t *zhp, void *data)
{ {

View File

@ -360,6 +360,8 @@ extern int zpool_get_history(zpool_handle_t *, nvlist_t **);
extern void zpool_set_history_str(const char *subcommand, int argc, extern void zpool_set_history_str(const char *subcommand, int argc,
char **argv, char *history_str); char **argv, char *history_str);
extern int zpool_stage_history(libzfs_handle_t *, const char *); extern int zpool_stage_history(libzfs_handle_t *, const char *);
extern int zpool_events_next(libzfs_handle_t *, nvlist_t **, int *, int);
extern int zpool_events_clear(libzfs_handle_t *, int *);
extern void zpool_obj_to_path(zpool_handle_t *, uint64_t, uint64_t, char *, extern void zpool_obj_to_path(zpool_handle_t *, uint64_t, uint64_t, char *,
size_t len); size_t len);
extern int zfs_ioctl(libzfs_handle_t *, int, struct zfs_cmd *); extern int zfs_ioctl(libzfs_handle_t *, int, struct zfs_cmd *);

View File

@ -2989,6 +2989,92 @@ zpool_get_history(zpool_handle_t *zhp, nvlist_t **nvhisp)
return (err); return (err);
} }
/*
* Retrieve the next event. If there is a new event available 'nvp' will
* contain a newly allocated nvlist and 'dropped' will be set to the number
* of missed events since the last call to this function. When 'nvp' is
* set to NULL it indicates no new events are available. In either case
* the function returns 0 and it is up to the caller to free 'nvp'. In
* the case of a fatal error the function will return a non-zero value.
* When the function is called in blocking mode it will not return until
* a new event is available.
*/
int
zpool_events_next(libzfs_handle_t *hdl, nvlist_t **nvp, int *dropped, int block)
{
zfs_cmd_t zc = { "\0", "\0", "\0", 0 };
int error = 0;
*nvp = NULL;
*dropped = 0;
if (!block)
zc.zc_guid = ZEVENT_NONBLOCK;
if (zcmd_alloc_dst_nvlist(hdl, &zc, ZEVENT_SIZE) != 0)
return (-1);
retry:
if (zfs_ioctl(hdl, ZFS_IOC_EVENTS_NEXT, &zc) != 0) {
switch (errno) {
case ESHUTDOWN:
error = zfs_error_fmt(hdl, EZFS_POOLUNAVAIL,
dgettext(TEXT_DOMAIN, "zfs shutdown"));
goto out;
case ENOENT:
/* Blocking error case should not occur */
if (block)
error = zpool_standard_error_fmt(hdl, errno,
dgettext(TEXT_DOMAIN, "cannot get event"));
goto out;
case ENOMEM:
if (zcmd_expand_dst_nvlist(hdl, &zc) != 0) {
error = zfs_error_fmt(hdl, EZFS_NOMEM,
dgettext(TEXT_DOMAIN, "cannot get event"));
goto out;
} else {
goto retry;
}
default:
error = zpool_standard_error_fmt(hdl, errno,
dgettext(TEXT_DOMAIN, "cannot get event"));
goto out;
}
}
error = zcmd_read_dst_nvlist(hdl, &zc, nvp);
if (error != 0)
goto out;
*dropped = (int)zc.zc_cookie;
out:
zcmd_free_nvlists(&zc);
return (error);
}
/*
* Clear all events.
*/
int
zpool_events_clear(libzfs_handle_t *hdl, int *count)
{
zfs_cmd_t zc = { "\0", "\0", "\0", 0 };
char msg[1024];
(void) snprintf(msg, sizeof (msg), dgettext(TEXT_DOMAIN,
"cannot clear events"));
if (zfs_ioctl(hdl, ZFS_IOC_EVENTS_CLEAR, &zc) != 0)
return (zpool_standard_error_fmt(hdl, errno, msg));
if (count != NULL)
*count = (int)zc.zc_cookie; /* # of events cleared */
return (0);
}
void void
zpool_obj_to_path(zpool_handle_t *zhp, uint64_t dsobj, uint64_t obj, zpool_obj_to_path(zpool_handle_t *zhp, uint64_t dsobj, uint64_t obj,
char *pathname, size_t len) char *pathname, size_t len)

View File

@ -58,7 +58,6 @@ extern "C" {
#include <atomic.h> #include <atomic.h>
#include <dirent.h> #include <dirent.h>
#include <time.h> #include <time.h>
#include <libsysevent.h>
#include <sys/note.h> #include <sys/note.h>
#include <sys/types.h> #include <sys/types.h>
#include <sys/cred.h> #include <sys/cred.h>
@ -72,8 +71,7 @@ extern "C" {
#include <sys/sdt.h> #include <sys/sdt.h>
#include <sys/kstat.h> #include <sys/kstat.h>
#include <sys/u8_textprep.h> #include <sys/u8_textprep.h>
#include <sys/sysevent/eventdefs.h> #include <sys/fm/fs/zfs.h>
#include <sys/sysevent/dev.h>
/* /*
* Stack * Stack

View File

@ -603,7 +603,9 @@ typedef enum zfs_ioc {
ZFS_IOC_USERSPACE_UPGRADE, ZFS_IOC_USERSPACE_UPGRADE,
ZFS_IOC_HOLD, ZFS_IOC_HOLD,
ZFS_IOC_RELEASE, ZFS_IOC_RELEASE,
ZFS_IOC_GET_HOLDS ZFS_IOC_GET_HOLDS,
ZFS_IOC_EVENTS_NEXT,
ZFS_IOC_EVENTS_CLEAR,
} zfs_ioc_t; } zfs_ioc_t;
/* /*

View File

@ -94,12 +94,12 @@ dsl_pool_scrub_setup_sync(void *arg1, void *arg2, cred_t *cr, dmu_tx_t *tx)
if (vdev_resilver_needed(rvd, if (vdev_resilver_needed(rvd,
&dp->dp_scrub_min_txg, &dp->dp_scrub_max_txg)) { &dp->dp_scrub_min_txg, &dp->dp_scrub_max_txg)) {
spa_event_notify(dp->dp_spa, NULL, spa_event_notify(dp->dp_spa, NULL,
ESC_ZFS_RESILVER_START); FM_EREPORT_ZFS_RESILVER_START);
dp->dp_scrub_max_txg = MIN(dp->dp_scrub_max_txg, dp->dp_scrub_max_txg = MIN(dp->dp_scrub_max_txg,
tx->tx_txg); tx->tx_txg);
} else { } else {
spa_event_notify(dp->dp_spa, NULL, spa_event_notify(dp->dp_spa, NULL,
ESC_ZFS_SCRUB_START); FM_EREPORT_ZFS_SCRUB_START);
} }
/* zero out the scrub stats in all vdev_stat_t's */ /* zero out the scrub stats in all vdev_stat_t's */
@ -219,7 +219,8 @@ dsl_pool_scrub_cancel_sync(void *arg1, void *arg2, cred_t *cr, dmu_tx_t *tx)
*completep ? dp->dp_scrub_max_txg : 0, B_TRUE); *completep ? dp->dp_scrub_max_txg : 0, B_TRUE);
if (*completep) if (*completep)
spa_event_notify(dp->dp_spa, NULL, dp->dp_scrub_min_txg ? spa_event_notify(dp->dp_spa, NULL, dp->dp_scrub_min_txg ?
ESC_ZFS_RESILVER_FINISH : ESC_ZFS_SCRUB_FINISH); FM_EREPORT_ZFS_RESILVER_FINISH :
FM_EREPORT_ZFS_SCRUB_FINISH);
spa_errlog_rotate(dp->dp_spa); spa_errlog_rotate(dp->dp_spa);
/* /*

View File

@ -54,49 +54,43 @@
#include <sys/types.h> #include <sys/types.h>
#include <sys/time.h> #include <sys/time.h>
#include <sys/sysevent.h> #include <sys/list.h>
#include <sys/sysevent_impl.h>
#include <sys/nvpair.h> #include <sys/nvpair.h>
#include <sys/cmn_err.h> #include <sys/cmn_err.h>
#include <sys/cpuvar.h>
#include <sys/sysmacros.h> #include <sys/sysmacros.h>
#include <sys/systm.h>
#include <sys/ddifm.h>
#include <sys/ddifm_impl.h>
#include <sys/spl.h>
#include <sys/dumphdr.h>
#include <sys/compress.h> #include <sys/compress.h>
#include <sys/cpuvar.h>
#include <sys/console.h>
#include <sys/panic.h>
#include <sys/kobj.h>
#include <sys/sunddi.h> #include <sys/sunddi.h>
#include <sys/systeminfo.h> #include <sys/systeminfo.h>
#include <sys/sysevent/eventdefs.h>
#include <sys/fm/util.h> #include <sys/fm/util.h>
#include <sys/fm/protocol.h> #include <sys/fm/protocol.h>
#include <sys/kstat.h>
#include <sys/zfs_context.h>
#ifdef _KERNEL
#include <sys/atomic.h>
#include <sys/condvar.h>
#include <sys/cpuvar.h>
#include <sys/systm.h>
#include <sys/dumphdr.h>
#include <sys/cpuvar.h>
#include <sys/console.h>
#include <sys/kobj.h>
#include <sys/time.h>
int zevent_len_max = 0;
int zevent_cols = 80;
int zevent_console = 0;
static int zevent_len_cur = 0;
static int zevent_waiters = 0;
static int zevent_flags = 0;
static kmutex_t zevent_lock;
static list_t zevent_list;
static kcondvar_t zevent_cv;
#endif /* _KERNEL */
/* /*
* URL and SUNW-MSG-ID value to display for fm_panic(), defined below. These * Common fault management kstats to record event generation failures
* values must be kept in sync with the FMA source code in usr/src/cmd/fm.
*/
static const char *fm_url = "http://www.sun.com/msg";
static const char *fm_msgid = "SUNOS-8000-0G";
static char *volatile fm_panicstr = NULL;
errorq_t *ereport_errorq;
void *ereport_dumpbuf;
size_t ereport_dumplen;
static uint_t ereport_chanlen = ERPT_EVCH_MAX;
static evchan_t *ereport_chan = NULL;
static ulong_t ereport_qlen = 0;
static size_t ereport_size = 0;
static int ereport_cols = 80;
/*
* Common fault management kstats to record ereport generation
* failures
*/ */
struct erpt_kstat { struct erpt_kstat {
@ -113,57 +107,9 @@ static struct erpt_kstat erpt_kstat_data = {
{ "payload-set-failed", KSTAT_DATA_UINT64 } { "payload-set-failed", KSTAT_DATA_UINT64 }
}; };
/*ARGSUSED*/ kstat_t *fm_ksp;
static void
fm_drain(void *private, void *data, errorq_elem_t *eep)
{
nvlist_t *nvl = errorq_elem_nvl(ereport_errorq, eep);
if (!panicstr) #ifdef _KERNEL
(void) fm_ereport_post(nvl, EVCH_TRYHARD);
else
fm_nvprint(nvl);
}
void
fm_init(void)
{
kstat_t *ksp;
(void) sysevent_evc_bind(FM_ERROR_CHAN,
&ereport_chan, EVCH_CREAT | EVCH_HOLD_PEND);
(void) sysevent_evc_control(ereport_chan,
EVCH_SET_CHAN_LEN, &ereport_chanlen);
if (ereport_qlen == 0)
ereport_qlen = ERPT_MAX_ERRS * MAX(max_ncpus, 4);
if (ereport_size == 0)
ereport_size = ERPT_DATA_SZ;
ereport_errorq = errorq_nvcreate("fm_ereport_queue",
(errorq_func_t)fm_drain, NULL, ereport_qlen, ereport_size,
FM_ERR_PIL, ERRORQ_VITAL);
if (ereport_errorq == NULL)
panic("failed to create required ereport error queue");
ereport_dumpbuf = kmem_alloc(ereport_size, KM_SLEEP);
ereport_dumplen = ereport_size;
/* Initialize ereport allocation and generation kstats */
ksp = kstat_create("unix", 0, "fm", "misc", KSTAT_TYPE_NAMED,
sizeof (struct erpt_kstat) / sizeof (kstat_named_t),
KSTAT_FLAG_VIRTUAL);
if (ksp != NULL) {
ksp->ks_data = &erpt_kstat_data;
kstat_install(ksp);
} else {
cmn_err(CE_NOTE, "failed to create fm/misc kstat\n");
}
}
/* /*
* Formatting utility function for fm_nvprintr. We attempt to wrap chunks of * Formatting utility function for fm_nvprintr. We attempt to wrap chunks of
@ -182,7 +128,7 @@ fm_printf(int depth, int c, int cols, const char *format, ...)
va_end(ap); va_end(ap);
if (c + width >= cols) { if (c + width >= cols) {
console_printf("\n\r"); console_printf("\n");
c = 0; c = 0;
if (format[0] != ' ' && depth > 0) { if (format[0] != ' ' && depth > 0) {
console_printf(" "); console_printf(" ");
@ -244,54 +190,54 @@ fm_nvprintr(nvlist_t *nvl, int d, int c, int cols)
case DATA_TYPE_BYTE: case DATA_TYPE_BYTE:
(void) nvpair_value_byte(nvp, &i8); (void) nvpair_value_byte(nvp, &i8);
c = fm_printf(d + 1, c, cols, "%x", i8); c = fm_printf(d + 1, c, cols, "0x%x", i8);
break; break;
case DATA_TYPE_INT8: case DATA_TYPE_INT8:
(void) nvpair_value_int8(nvp, (void *)&i8); (void) nvpair_value_int8(nvp, (void *)&i8);
c = fm_printf(d + 1, c, cols, "%x", i8); c = fm_printf(d + 1, c, cols, "0x%x", i8);
break; break;
case DATA_TYPE_UINT8: case DATA_TYPE_UINT8:
(void) nvpair_value_uint8(nvp, &i8); (void) nvpair_value_uint8(nvp, &i8);
c = fm_printf(d + 1, c, cols, "%x", i8); c = fm_printf(d + 1, c, cols, "0x%x", i8);
break; break;
case DATA_TYPE_INT16: case DATA_TYPE_INT16:
(void) nvpair_value_int16(nvp, (void *)&i16); (void) nvpair_value_int16(nvp, (void *)&i16);
c = fm_printf(d + 1, c, cols, "%x", i16); c = fm_printf(d + 1, c, cols, "0x%x", i16);
break; break;
case DATA_TYPE_UINT16: case DATA_TYPE_UINT16:
(void) nvpair_value_uint16(nvp, &i16); (void) nvpair_value_uint16(nvp, &i16);
c = fm_printf(d + 1, c, cols, "%x", i16); c = fm_printf(d + 1, c, cols, "0x%x", i16);
break; break;
case DATA_TYPE_INT32: case DATA_TYPE_INT32:
(void) nvpair_value_int32(nvp, (void *)&i32); (void) nvpair_value_int32(nvp, (void *)&i32);
c = fm_printf(d + 1, c, cols, "%x", i32); c = fm_printf(d + 1, c, cols, "0x%x", i32);
break; break;
case DATA_TYPE_UINT32: case DATA_TYPE_UINT32:
(void) nvpair_value_uint32(nvp, &i32); (void) nvpair_value_uint32(nvp, &i32);
c = fm_printf(d + 1, c, cols, "%x", i32); c = fm_printf(d + 1, c, cols, "0x%x", i32);
break; break;
case DATA_TYPE_INT64: case DATA_TYPE_INT64:
(void) nvpair_value_int64(nvp, (void *)&i64); (void) nvpair_value_int64(nvp, (void *)&i64);
c = fm_printf(d + 1, c, cols, "%llx", c = fm_printf(d + 1, c, cols, "0x%llx",
(u_longlong_t)i64); (u_longlong_t)i64);
break; break;
case DATA_TYPE_UINT64: case DATA_TYPE_UINT64:
(void) nvpair_value_uint64(nvp, &i64); (void) nvpair_value_uint64(nvp, &i64);
c = fm_printf(d + 1, c, cols, "%llx", c = fm_printf(d + 1, c, cols, "0x%llx",
(u_longlong_t)i64); (u_longlong_t)i64);
break; break;
case DATA_TYPE_HRTIME: case DATA_TYPE_HRTIME:
(void) nvpair_value_hrtime(nvp, (void *)&i64); (void) nvpair_value_hrtime(nvp, (void *)&i64);
c = fm_printf(d + 1, c, cols, "%llx", c = fm_printf(d + 1, c, cols, "0x%llx",
(u_longlong_t)i64); (u_longlong_t)i64);
break; break;
@ -321,19 +267,124 @@ fm_nvprintr(nvlist_t *nvl, int d, int c, int cols)
} }
break; break;
case DATA_TYPE_INT8_ARRAY: {
int8_t *val;
uint_t i, nelem;
c = fm_printf(d + 1, c, cols, "[ ");
(void) nvpair_value_int8_array(nvp, &val, &nelem);
for (i = 0; i < nelem; i++)
c = fm_printf(d + 1, c, cols, "0x%llx ",
(u_longlong_t)val[i]);
c = fm_printf(d + 1, c, cols, "]");
break;
}
case DATA_TYPE_UINT8_ARRAY: {
uint8_t *val;
uint_t i, nelem;
c = fm_printf(d + 1, c, cols, "[ ");
(void) nvpair_value_uint8_array(nvp, &val, &nelem);
for (i = 0; i < nelem; i++)
c = fm_printf(d + 1, c, cols, "0x%llx ",
(u_longlong_t)val[i]);
c = fm_printf(d + 1, c, cols, "]");
break;
}
case DATA_TYPE_INT16_ARRAY: {
int16_t *val;
uint_t i, nelem;
c = fm_printf(d + 1, c, cols, "[ ");
(void) nvpair_value_int16_array(nvp, &val, &nelem);
for (i = 0; i < nelem; i++)
c = fm_printf(d + 1, c, cols, "0x%llx ",
(u_longlong_t)val[i]);
c = fm_printf(d + 1, c, cols, "]");
break;
}
case DATA_TYPE_UINT16_ARRAY: {
uint16_t *val;
uint_t i, nelem;
c = fm_printf(d + 1, c, cols, "[ ");
(void) nvpair_value_uint16_array(nvp, &val, &nelem);
for (i = 0; i < nelem; i++)
c = fm_printf(d + 1, c, cols, "0x%llx ",
(u_longlong_t)val[i]);
c = fm_printf(d + 1, c, cols, "]");
break;
}
case DATA_TYPE_INT32_ARRAY: {
int32_t *val;
uint_t i, nelem;
c = fm_printf(d + 1, c, cols, "[ ");
(void) nvpair_value_int32_array(nvp, &val, &nelem);
for (i = 0; i < nelem; i++)
c = fm_printf(d + 1, c, cols, "0x%llx ",
(u_longlong_t)val[i]);
c = fm_printf(d + 1, c, cols, "]");
break;
}
case DATA_TYPE_UINT32_ARRAY: {
uint32_t *val;
uint_t i, nelem;
c = fm_printf(d + 1, c, cols, "[ ");
(void) nvpair_value_uint32_array(nvp, &val, &nelem);
for (i = 0; i < nelem; i++)
c = fm_printf(d + 1, c, cols, "0x%llx ",
(u_longlong_t)val[i]);
c = fm_printf(d + 1, c, cols, "]");
break;
}
case DATA_TYPE_INT64_ARRAY: {
int64_t *val;
uint_t i, nelem;
c = fm_printf(d + 1, c, cols, "[ ");
(void) nvpair_value_int64_array(nvp, &val, &nelem);
for (i = 0; i < nelem; i++)
c = fm_printf(d + 1, c, cols, "0x%llx ",
(u_longlong_t)val[i]);
c = fm_printf(d + 1, c, cols, "]");
break;
}
case DATA_TYPE_UINT64_ARRAY: {
uint64_t *val;
uint_t i, nelem;
c = fm_printf(d + 1, c, cols, "[ ");
(void) nvpair_value_uint64_array(nvp, &val, &nelem);
for (i = 0; i < nelem; i++)
c = fm_printf(d + 1, c, cols, "0x%llx ",
(u_longlong_t)val[i]);
c = fm_printf(d + 1, c, cols, "]");
break;
}
case DATA_TYPE_STRING_ARRAY:
case DATA_TYPE_BOOLEAN_ARRAY: case DATA_TYPE_BOOLEAN_ARRAY:
case DATA_TYPE_BYTE_ARRAY: case DATA_TYPE_BYTE_ARRAY:
case DATA_TYPE_INT8_ARRAY:
case DATA_TYPE_UINT8_ARRAY:
case DATA_TYPE_INT16_ARRAY:
case DATA_TYPE_UINT16_ARRAY:
case DATA_TYPE_INT32_ARRAY:
case DATA_TYPE_UINT32_ARRAY:
case DATA_TYPE_INT64_ARRAY:
case DATA_TYPE_UINT64_ARRAY:
case DATA_TYPE_STRING_ARRAY:
c = fm_printf(d + 1, c, cols, "[...]"); c = fm_printf(d + 1, c, cols, "[...]");
break; break;
case DATA_TYPE_UNKNOWN: case DATA_TYPE_UNKNOWN:
c = fm_printf(d + 1, c, cols, "<unknown>"); c = fm_printf(d + 1, c, cols, "<unknown>");
break; break;
@ -349,174 +400,210 @@ fm_nvprint(nvlist_t *nvl)
char *class; char *class;
int c = 0; int c = 0;
console_printf("\r"); console_printf("\n");
if (nvlist_lookup_string(nvl, FM_CLASS, &class) == 0) if (nvlist_lookup_string(nvl, FM_CLASS, &class) == 0)
c = fm_printf(0, c, ereport_cols, "%s", class); c = fm_printf(0, c, zevent_cols, "%s", class);
if (fm_nvprintr(nvl, 0, c, ereport_cols) != 0) if (fm_nvprintr(nvl, 0, c, zevent_cols) != 0)
console_printf("\n"); console_printf("\n");
console_printf("\n"); console_printf("\n");
} }
/* static zevent_t *
* Wrapper for panic() that first produces an FMA-style message for admins. fm_event_alloc(void)
* Normally such messages are generated by fmd(1M)'s syslog-msgs agent: this
* is the one exception to that rule and the only error that gets messaged.
* This function is intended for use by subsystems that have detected a fatal
* error and enqueued appropriate ereports and wish to then force a panic.
*/
/*PRINTFLIKE1*/
void
fm_panic(const char *format, ...)
{ {
va_list ap; zevent_t *ev;
(void) casptr((void *)&fm_panicstr, NULL, (void *)format); ev = kmem_zalloc(sizeof(zevent_t), KM_SLEEP);
va_start(ap, format); if (ev == NULL)
vpanic(format, ap); return NULL;
va_end(ap);
list_create(&ev->ev_zpd_list, sizeof(zfs_private_data_t),
offsetof(zfs_private_data_t, zpd_node));
list_link_init(&ev->ev_node);
return ev;
} }
/* static void
* Print any appropriate FMA banner message before the panic message. This fm_event_free(zevent_t *ev)
* function is called by panicsys() and prints the message for fm_panic().
* We print the message here so that it comes after the system is quiesced.
* A one-line summary is recorded in the log only (cmn_err(9F) with "!" prefix).
* The rest of the message is for the console only and not needed in the log,
* so it is printed using console_printf(). We break it up into multiple
* chunks so as to avoid overflowing any small legacy prom_printf() buffers.
*/
void
fm_banner(void)
{ {
timespec_t tod; /* Run provided cleanup callback */
hrtime_t now; ev->ev_cb(ev->ev_nvl);
if (!fm_panicstr) list_destroy(&ev->ev_zpd_list);
return; /* panic was not initiated by fm_panic(); do nothing */ kmem_free(ev, sizeof(zevent_t));
if (panicstr) {
tod = panic_hrestime;
now = panic_hrtime;
} else {
gethrestime(&tod);
now = gethrtime_waitfree();
} }
cmn_err(CE_NOTE, "!SUNW-MSG-ID: %s, " static void
"TYPE: Error, VER: 1, SEVERITY: Major\n", fm_msgid); fm_zevent_drain(zevent_t *ev)
console_printf(
"\n\rSUNW-MSG-ID: %s, TYPE: Error, VER: 1, SEVERITY: Major\n"
"EVENT-TIME: 0x%lx.0x%lx (0x%llx)\n",
fm_msgid, tod.tv_sec, tod.tv_nsec, (u_longlong_t)now);
console_printf(
"PLATFORM: %s, CSN: -, HOSTNAME: %s\n"
"SOURCE: %s, REV: %s %s\n",
platform, utsname.nodename, utsname.sysname,
utsname.release, utsname.version);
console_printf(
"DESC: Errors have been detected that require a reboot to ensure system\n"
"integrity. See %s/%s for more information.\n",
fm_url, fm_msgid);
console_printf(
"AUTO-RESPONSE: Solaris will attempt to save and diagnose the error telemetry\n"
"IMPACT: The system will sync files, save a crash dump if needed, and reboot\n"
"REC-ACTION: Save the error summary below in case telemetry cannot be saved\n");
console_printf("\n");
}
/*
* Utility function to write all of the pending ereports to the dump device.
* This function is called at either normal reboot or panic time, and simply
* iterates over the in-transit messages in the ereport sysevent channel.
*/
void
fm_ereport_dump(void)
{ {
evchanq_t *chq; zfs_private_data_t *zpd;
sysevent_t *sep;
erpt_dump_t ed;
timespec_t tod; ASSERT(MUTEX_HELD(&zevent_lock));
hrtime_t now; list_remove(&zevent_list, ev);
char *buf;
size_t len;
if (panicstr) { /* Remove references to this event in all private file data */
tod = panic_hrestime; while ((zpd = list_head(&ev->ev_zpd_list)) != NULL) {
now = panic_hrtime; list_remove(&ev->ev_zpd_list, zpd);
} else { zpd->zpd_zevent = NULL;
if (ereport_errorq != NULL) zpd->zpd_dropped++;
errorq_drain(ereport_errorq); }
gethrestime(&tod);
now = gethrtime_waitfree(); fm_event_free(ev);
}
void
fm_zevent_drain_all(int *count)
{
zevent_t *ev;
mutex_enter(&zevent_lock);
while ((ev = list_head(&zevent_list)) != NULL)
fm_zevent_drain(ev);
*count = zevent_len_cur;
zevent_len_cur = 0;
mutex_exit(&zevent_lock);
} }
/* /*
* In the panic case, sysevent_evc_walk_init() will return NULL. * New zevents are inserted at the head. If the maximum queue
* length is exceeded a zevent will be drained from the tail.
* As part of this any user space processes which currently have
* a reference to this zevent_t in their private data will have
* this reference set to NULL.
*/ */
if ((chq = sysevent_evc_walk_init(ereport_chan, NULL)) == NULL && static void
!panicstr) fm_zevent_insert(zevent_t *ev)
return; /* event channel isn't initialized yet */ {
mutex_enter(&zevent_lock);
list_insert_head(&zevent_list, ev);
if (zevent_len_cur >= zevent_len_max)
fm_zevent_drain(list_tail(&zevent_list));
else
zevent_len_cur++;
while ((sep = sysevent_evc_walk_step(chq)) != NULL) { mutex_exit(&zevent_lock);
if ((buf = sysevent_evc_event_attr(sep, &len)) == NULL)
break;
ed.ed_magic = ERPT_MAGIC;
ed.ed_chksum = checksum32(buf, len);
ed.ed_size = (uint32_t)len;
ed.ed_pad = 0;
ed.ed_hrt_nsec = SE_TIME(sep);
ed.ed_hrt_base = now;
ed.ed_tod_base.sec = tod.tv_sec;
ed.ed_tod_base.nsec = tod.tv_nsec;
dumpvp_write(&ed, sizeof (ed));
dumpvp_write(buf, len);
}
sysevent_evc_walk_fini(chq);
} }
/* /*
* Post an error report (ereport) to the sysevent error channel. The error * Post a zevent
* channel must be established with a prior call to sysevent_evc_create()
* before publication may occur.
*/ */
void void
fm_ereport_post(nvlist_t *ereport, int evc_flag) fm_zevent_post(nvlist_t *nvl, zevent_cb_t *cb)
{ {
size_t nvl_size = 0; size_t nvl_size = 0;
evchan_t *error_chan; zevent_t *ev;
(void) nvlist_size(ereport, &nvl_size, NV_ENCODE_NATIVE); (void) nvlist_size(nvl, &nvl_size, NV_ENCODE_NATIVE);
if (nvl_size > ERPT_DATA_SZ || nvl_size == 0) { if (nvl_size > ERPT_DATA_SZ || nvl_size == 0) {
atomic_add_64(&erpt_kstat_data.erpt_dropped.value.ui64, 1); atomic_add_64(&erpt_kstat_data.erpt_dropped.value.ui64, 1);
return; return;
} }
if (sysevent_evc_bind(FM_ERROR_CHAN, &error_chan, if (zevent_console)
EVCH_CREAT|EVCH_HOLD_PEND) != 0) { fm_nvprint(nvl);
ev = fm_event_alloc();
if (ev == NULL) {
atomic_add_64(&erpt_kstat_data.erpt_dropped.value.ui64, 1); atomic_add_64(&erpt_kstat_data.erpt_dropped.value.ui64, 1);
return; return;
} }
if (sysevent_evc_publish(error_chan, EC_FM, ESC_FM_ERROR, ev->ev_nvl = nvl;
SUNW_VENDOR, FM_PUB, ereport, evc_flag) != 0) { ev->ev_cb = cb;
atomic_add_64(&erpt_kstat_data.erpt_dropped.value.ui64, 1); fm_zevent_insert(ev);
sysevent_evc_unbind(error_chan); cv_broadcast(&zevent_cv);
return;
} }
sysevent_evc_unbind(error_chan);
/*
* Get the next zevent in the stream. To avoid making an extra copy of the
* nvlist we must call put_nvlist() here safely under the zevent_lock.
*/
int
fm_zevent_next(zfs_private_data_t *zpd, zfs_cmd_t *zc)
{
zevent_t *ev;
int error;
mutex_enter(&zevent_lock);
if (zpd->zpd_zevent == NULL) {
/* New stream start at the beginning/tail */
ev = list_tail(&zevent_list);
if (ev == NULL) {
error = ENOENT;
goto out;
} }
} else {
/* Existing stream continue with the next element and remove
* ourselves from the wait queue for the previous element */
ev = list_prev(&zevent_list, zpd->zpd_zevent);
if (ev == NULL) {
error = ENOENT;
goto out;
}
list_remove(&zpd->zpd_zevent->ev_zpd_list, zpd);
}
zpd->zpd_zevent = ev;
list_insert_head(&ev->ev_zpd_list, zpd);
error = put_nvlist(zc, ev->ev_nvl);
zc->zc_cookie = zpd->zpd_dropped;
zpd->zpd_dropped = 0;
out:
mutex_exit(&zevent_lock);
return error;
}
int
fm_zevent_wait(zfs_private_data_t *zpd)
{
int error = 0;
mutex_enter(&zevent_lock);
if (zevent_flags & ZEVENT_SHUTDOWN) {
error = ESHUTDOWN;
goto out;
}
zevent_waiters++;
cv_wait_interruptible(&zevent_cv, &zevent_lock);
if (issig(JUSTLOOKING))
error = EINTR;
zevent_waiters--;
out:
mutex_exit(&zevent_lock);
return error;
}
void
fm_zevent_init(zfs_private_data_t *zpd)
{
list_link_init(&zpd->zpd_node);
zpd->zpd_zevent = NULL;
zpd->zpd_dropped = 0;
}
void
fm_zevent_fini(zfs_private_data_t *zpd)
{
mutex_enter(&zevent_lock);
if (zpd->zpd_zevent)
list_remove(&zpd->zpd_zevent->ev_zpd_list, zpd);
zpd->zpd_zevent = NULL;
zpd->zpd_dropped = 0;
mutex_exit(&zevent_lock);
}
#endif /* _KERNEL */
/* /*
* Wrapppers for FM nvlist allocators * Wrapppers for FM nvlist allocators
@ -795,6 +882,8 @@ fm_ereport_set(nvlist_t *ereport, int version, const char *erpt_class,
{ {
char ereport_class[FM_MAX_CLASS]; char ereport_class[FM_MAX_CLASS];
const char *name; const char *name;
timestruc_t tv;
int64_t tv_array[2];
va_list ap; va_list ap;
int ret; int ret;
@ -826,6 +915,13 @@ fm_ereport_set(nvlist_t *ereport, int version, const char *erpt_class,
if (ret) if (ret)
atomic_add_64(&erpt_kstat_data.erpt_set_failed.value.ui64, 1); atomic_add_64(&erpt_kstat_data.erpt_set_failed.value.ui64, 1);
gethrestime(&tv);
tv_array[0] = tv.tv_sec;
tv_array[1] = tv.tv_nsec;
if (nvlist_add_int64_array(ereport, FM_EREPORT_TIME, tv_array, 2)) {
atomic_add_64(&erpt_kstat_data.erpt_set_failed.value.ui64, 1);
}
} }
/* /*
@ -1146,7 +1242,7 @@ fm_ena_generate_cpu(uint64_t timestamp, processorid_t cpuid, uchar_t format)
ena = (uint64_t)((format & ENA_FORMAT_MASK) | ena = (uint64_t)((format & ENA_FORMAT_MASK) |
((cpuid << ENA_FMT1_CPUID_SHFT) & ((cpuid << ENA_FMT1_CPUID_SHFT) &
ENA_FMT1_CPUID_MASK) | ENA_FMT1_CPUID_MASK) |
((gethrtime_waitfree() << ENA_FMT1_TIME_SHFT) & ((gethrtime() << ENA_FMT1_TIME_SHFT) &
ENA_FMT1_TIME_MASK)); ENA_FMT1_TIME_MASK));
} }
break; break;
@ -1164,7 +1260,7 @@ fm_ena_generate_cpu(uint64_t timestamp, processorid_t cpuid, uchar_t format)
uint64_t uint64_t
fm_ena_generate(uint64_t timestamp, uchar_t format) fm_ena_generate(uint64_t timestamp, uchar_t format)
{ {
return (fm_ena_generate_cpu(timestamp, CPU->cpu_id, format)); return (fm_ena_generate_cpu(timestamp, getcpuid(), format));
} }
uint64_t uint64_t
@ -1232,35 +1328,67 @@ fm_ena_time_get(uint64_t ena)
return (time); return (time);
} }
/* #ifdef _KERNEL
* Convert a getpcstack() trace to symbolic name+offset, and add the resulting
* string array to a Fault Management ereport as FM_EREPORT_PAYLOAD_NAME_STACK.
*/
void void
fm_payload_stack_add(nvlist_t *payload, const pc_t *stack, int depth) fm_init(void)
{ {
int i; zevent_len_cur = 0;
char *sym; zevent_flags = 0;
ulong_t off;
char *stkpp[FM_STK_DEPTH];
char buf[FM_STK_DEPTH * FM_SYM_SZ];
char *stkp = buf;
for (i = 0; i < depth && i != FM_STK_DEPTH; i++, stkp += FM_SYM_SZ) { if (zevent_len_max == 0)
if ((sym = kobj_getsymname(stack[i], &off)) != NULL) zevent_len_max = ERPT_MAX_ERRS * MAX(max_ncpus, 4);
(void) snprintf(stkp, FM_SYM_SZ, "%s+%lx", sym, off);
else /* Initialize zevent allocation and generation kstats */
(void) snprintf(stkp, FM_SYM_SZ, "%lx", (long)stack[i]); fm_ksp = kstat_create("zfs", 0, "fm", "misc", KSTAT_TYPE_NAMED,
stkpp[i] = stkp; sizeof (struct erpt_kstat) / sizeof (kstat_named_t),
KSTAT_FLAG_VIRTUAL);
if (fm_ksp != NULL) {
fm_ksp->ks_data = &erpt_kstat_data;
kstat_install(fm_ksp);
} else {
cmn_err(CE_NOTE, "failed to create fm/misc kstat\n");
} }
fm_payload_set(payload, FM_EREPORT_PAYLOAD_NAME_STACK, mutex_init(&zevent_lock, NULL, MUTEX_DEFAULT, NULL);
DATA_TYPE_STRING_ARRAY, depth, stkpp, NULL); list_create(&zevent_list, sizeof(zevent_t), offsetof(zevent_t, ev_node));
cv_init(&zevent_cv, NULL, CV_DEFAULT, NULL);
} }
void void
print_msg_hwerr(ctid_t ct_id, proc_t *p) fm_fini(void)
{ {
uprintf("Killed process %d (%s) in contract id %d " int count;
"due to hardware error\n", p->p_pid, p->p_user.u_comm, ct_id);
fm_zevent_drain_all(&count);
cv_broadcast(&zevent_cv);
mutex_enter(&zevent_lock);
zevent_flags |= ZEVENT_SHUTDOWN;
while (zevent_waiters > 0) {
mutex_exit(&zevent_lock);
schedule();
mutex_enter(&zevent_lock);
} }
mutex_exit(&zevent_lock);
cv_destroy(&zevent_cv);
list_destroy(&zevent_list);
mutex_destroy(&zevent_lock);
if (fm_ksp != NULL) {
kstat_delete(fm_ksp);
fm_ksp = NULL;
}
}
module_param(zevent_len_max, int, 0644);
MODULE_PARM_DESC(zevent_len_max, "Maximum event queue length");
module_param(zevent_cols, int, 0644);
MODULE_PARM_DESC(zevent_cols, "Maximum event column width");
module_param(zevent_console, int, 0644);
MODULE_PARM_DESC(zevent_console, "Log events to the console");
#endif /* _KERNEL */

View File

@ -35,7 +35,9 @@ extern "C" {
#define FM_EREPORT_ZFS_CHECKSUM "checksum" #define FM_EREPORT_ZFS_CHECKSUM "checksum"
#define FM_EREPORT_ZFS_IO "io" #define FM_EREPORT_ZFS_IO "io"
#define FM_EREPORT_ZFS_DATA "data" #define FM_EREPORT_ZFS_DATA "data"
#define FM_EREPORT_ZFS_CONFIG_SYNC "config.sync"
#define FM_EREPORT_ZFS_POOL "zpool" #define FM_EREPORT_ZFS_POOL "zpool"
#define FM_EREPORT_ZFS_POOL_DESTROY "zpool.destroy"
#define FM_EREPORT_ZFS_DEVICE_UNKNOWN "vdev.unknown" #define FM_EREPORT_ZFS_DEVICE_UNKNOWN "vdev.unknown"
#define FM_EREPORT_ZFS_DEVICE_OPEN_FAILED "vdev.open_failed" #define FM_EREPORT_ZFS_DEVICE_OPEN_FAILED "vdev.open_failed"
#define FM_EREPORT_ZFS_DEVICE_CORRUPT_DATA "vdev.corrupt_data" #define FM_EREPORT_ZFS_DEVICE_CORRUPT_DATA "vdev.corrupt_data"
@ -43,9 +45,18 @@ extern "C" {
#define FM_EREPORT_ZFS_DEVICE_BAD_GUID_SUM "vdev.bad_guid_sum" #define FM_EREPORT_ZFS_DEVICE_BAD_GUID_SUM "vdev.bad_guid_sum"
#define FM_EREPORT_ZFS_DEVICE_TOO_SMALL "vdev.too_small" #define FM_EREPORT_ZFS_DEVICE_TOO_SMALL "vdev.too_small"
#define FM_EREPORT_ZFS_DEVICE_BAD_LABEL "vdev.bad_label" #define FM_EREPORT_ZFS_DEVICE_BAD_LABEL "vdev.bad_label"
#define FM_EREPORT_ZFS_DEVICE_REMOVE "vdev.remove"
#define FM_EREPORT_ZFS_DEVICE_CLEAR "vdev.clear"
#define FM_EREPORT_ZFS_DEVICE_CHECK "vdev.check"
#define FM_EREPORT_ZFS_DEVICE_SPARE "vdev.spare"
#define FM_EREPORT_ZFS_DEVICE_AUTOEXPAND "vdev.autoexpand"
#define FM_EREPORT_ZFS_IO_FAILURE "io_failure" #define FM_EREPORT_ZFS_IO_FAILURE "io_failure"
#define FM_EREPORT_ZFS_PROBE_FAILURE "probe_failure" #define FM_EREPORT_ZFS_PROBE_FAILURE "probe_failure"
#define FM_EREPORT_ZFS_LOG_REPLAY "log_replay" #define FM_EREPORT_ZFS_LOG_REPLAY "log_replay"
#define FM_EREPORT_ZFS_RESILVER_START "resilver.start"
#define FM_EREPORT_ZFS_RESILVER_FINISH "resilver.finish"
#define FM_EREPORT_ZFS_SCRUB_START "scrub.start"
#define FM_EREPORT_ZFS_SCRUB_FINISH "scrub.finish"
#define FM_EREPORT_PAYLOAD_ZFS_POOL "pool" #define FM_EREPORT_PAYLOAD_ZFS_POOL "pool"
#define FM_EREPORT_PAYLOAD_ZFS_POOL_FAILMODE "pool_failmode" #define FM_EREPORT_PAYLOAD_ZFS_POOL_FAILMODE "pool_failmode"
@ -73,8 +84,8 @@ extern "C" {
#define FM_EREPORT_FAILMODE_CONTINUE "continue" #define FM_EREPORT_FAILMODE_CONTINUE "continue"
#define FM_EREPORT_FAILMODE_PANIC "panic" #define FM_EREPORT_FAILMODE_PANIC "panic"
#define FM_RESOURCE_REMOVED "removed" #define FM_EREPORT_RESOURCE_REMOVED "removed"
#define FM_RESOURCE_AUTOREPLACE "autoreplace" #define FM_EREPORT_RESOURCE_AUTOREPLACE "autoreplace"
#ifdef __cplusplus #ifdef __cplusplus
} }

View File

@ -68,6 +68,7 @@ extern "C" {
/* ereport payload member names */ /* ereport payload member names */
#define FM_EREPORT_DETECTOR "detector" #define FM_EREPORT_DETECTOR "detector"
#define FM_EREPORT_ENA "ena" #define FM_EREPORT_ENA "ena"
#define FM_EREPORT_TIME "time"
/* list.* event payload member names */ /* list.* event payload member names */
#define FM_LIST_EVENT_SIZE "list-sz" #define FM_LIST_EVENT_SIZE "list-sz"
@ -295,15 +296,13 @@ extern "C" {
#define FM_FMRI_ZFS_POOL "pool" #define FM_FMRI_ZFS_POOL "pool"
#define FM_FMRI_ZFS_VDEV "vdev" #define FM_FMRI_ZFS_VDEV "vdev"
extern nv_alloc_t *fm_nva_xcreate(char *, size_t);
extern void fm_nva_xdestroy(nv_alloc_t *);
extern nvlist_t *fm_nvlist_create(nv_alloc_t *);
extern void fm_nvlist_destroy(nvlist_t *, int);
#define FM_NVA_FREE 0 /* free allocator on nvlist_destroy */ #define FM_NVA_FREE 0 /* free allocator on nvlist_destroy */
#define FM_NVA_RETAIN 1 /* keep allocator on nvlist_destroy */ #define FM_NVA_RETAIN 1 /* keep allocator on nvlist_destroy */
extern nv_alloc_t *fm_nva_xcreate(char *, size_t);
extern void fm_nva_xdestroy(nv_alloc_t *);
extern nvlist_t *fm_nvlist_create(nv_alloc_t *);
extern void fm_nvlist_destroy(nvlist_t *, int);
extern void fm_ereport_set(nvlist_t *, int, const char *, uint64_t, extern void fm_ereport_set(nvlist_t *, int, const char *, uint64_t,
const nvlist_t *, ...); const nvlist_t *, ...);
extern void fm_payload_set(nvlist_t *, ...); extern void fm_payload_set(nvlist_t *, ...);
@ -312,15 +311,11 @@ extern void fm_fmri_hc_set(nvlist_t *, int, const nvlist_t *, nvlist_t *,
int, ...); int, ...);
extern void fm_fmri_dev_set(nvlist_t *, int, const nvlist_t *, const char *, extern void fm_fmri_dev_set(nvlist_t *, int, const nvlist_t *, const char *,
const char *); const char *);
extern void fm_fmri_de_set(nvlist_t *, int, const nvlist_t *, const char *);
extern void fm_fmri_cpu_set(nvlist_t *, int, const nvlist_t *, uint32_t, extern void fm_fmri_cpu_set(nvlist_t *, int, const nvlist_t *, uint32_t,
uint8_t *, const char *); uint8_t *, const char *);
extern void fm_fmri_mem_set(nvlist_t *, int, const nvlist_t *, const char *, extern void fm_fmri_mem_set(nvlist_t *, int, const nvlist_t *, const char *,
const char *, uint64_t); const char *, uint64_t);
extern void fm_authority_set(nvlist_t *, int, const char *, const char *,
const char *, const char *);
extern void fm_fmri_zfs_set(nvlist_t *, int, uint64_t, uint64_t); extern void fm_fmri_zfs_set(nvlist_t *, int, uint64_t, uint64_t);
extern uint64_t fm_ena_increment(uint64_t); extern uint64_t fm_ena_increment(uint64_t);
extern uint64_t fm_ena_generate(uint64_t, uchar_t); extern uint64_t fm_ena_generate(uint64_t, uchar_t);
extern uint64_t fm_ena_generate_cpu(uint64_t, processorid_t, uchar_t); extern uint64_t fm_ena_generate_cpu(uint64_t, processorid_t, uchar_t);

View File

@ -34,7 +34,6 @@ extern "C" {
#endif #endif
#include <sys/nvpair.h> #include <sys/nvpair.h>
#include <sys/errorq.h>
/* /*
* Shared user/kernel definitions for class length, error channel name, * Shared user/kernel definitions for class length, error channel name,
@ -74,27 +73,41 @@ typedef struct erpt_dump {
} erpt_dump_t; } erpt_dump_t;
#ifdef _KERNEL #ifdef _KERNEL
#include <sys/systm.h> #include <sys/systm.h>
#include <sys/zfs_ioctl.h>
#define FM_STK_DEPTH 20 /* maximum stack depth */ #define ZEVENT_SHUTDOWN 0x1
#define FM_SYM_SZ 64 /* maximum symbol size */
#define FM_ERR_PIL 2 /* PIL for ereport_errorq drain processing */
#define FM_EREPORT_PAYLOAD_NAME_STACK "stack" typedef void zevent_cb_t(nvlist_t *);
extern errorq_t *ereport_errorq; typedef struct zevent_s {
extern void *ereport_dumpbuf; nvlist_t *ev_nvl; /* protected by the zevent_lock */
extern size_t ereport_dumplen; list_t ev_zpd_list; /* " */
list_node_t ev_node; /* " */
zevent_cb_t *ev_cb; /* " */
} zevent_t;
typedef struct zfs_private_data {
zevent_t *zpd_zevent; /* protected by the zevent_lock */
list_node_t zpd_node; /* " */
uint64_t zpd_dropped; /* " */
} zfs_private_data_t;
extern void fm_init(void); extern void fm_init(void);
extern void fm_fini(void);
extern void fm_nvprint(nvlist_t *); extern void fm_nvprint(nvlist_t *);
extern void fm_panic(const char *, ...); extern void fm_zevent_init(zfs_private_data_t *);
extern void fm_banner(void); extern void fm_zevent_fini(zfs_private_data_t *);
extern void fm_zevent_post(nvlist_t *, zevent_cb_t *);
extern void fm_zevent_drain_all(int *);
extern int fm_zevent_next(zfs_private_data_t *, zfs_cmd_t *);
extern int fm_zevent_wait(zfs_private_data_t *);
extern void fm_ereport_dump(void); #else
extern void fm_ereport_post(nvlist_t *, int);
extern void fm_payload_stack_add(nvlist_t *, const pc_t *, int); static inline void fm_init(void) { }
static inline void fm_fini(void) { }
#endif /* _KERNEL */ #endif /* _KERNEL */

View File

@ -509,8 +509,6 @@ struct zio;
extern void spa_log_error(spa_t *spa, struct zio *zio); extern void spa_log_error(spa_t *spa, struct zio *zio);
extern void zfs_ereport_post(const char *class, spa_t *spa, vdev_t *vd, extern void zfs_ereport_post(const char *class, spa_t *spa, vdev_t *vd,
struct zio *zio, uint64_t stateoroffset, uint64_t length); struct zio *zio, uint64_t stateoroffset, uint64_t length);
extern void zfs_post_remove(spa_t *spa, vdev_t *vd);
extern void zfs_post_autoreplace(spa_t *spa, vdev_t *vd);
extern uint64_t spa_get_errlog_size(spa_t *spa); extern uint64_t spa_get_errlog_size(spa_t *spa);
extern int spa_get_errlog(spa_t *spa, void *uaddr, size_t *count); extern int spa_get_errlog(spa_t *spa, void *uaddr, size_t *count);
extern void spa_errlog_rotate(spa_t *spa); extern void spa_errlog_rotate(spa_t *spa);

View File

@ -58,10 +58,7 @@ extern "C" {
#include <sys/zone.h> #include <sys/zone.h>
#include <sys/uio.h> #include <sys/uio.h>
#include <sys/zfs_debug.h> #include <sys/zfs_debug.h>
#include <sys/sysevent.h> #include <sys/fm/fs/zfs.h>
#include <sys/sysevent/eventdefs.h>
#include <sys/sysevent/dev.h>
#include <sys/fm/util.h>
#ifdef __cplusplus #ifdef __cplusplus
} }

View File

@ -1101,8 +1101,9 @@ spa_check_removed(vdev_t *vd)
spa_check_removed(vd->vdev_child[c]); spa_check_removed(vd->vdev_child[c]);
if (vd->vdev_ops->vdev_op_leaf && vdev_is_dead(vd)) { if (vd->vdev_ops->vdev_op_leaf && vdev_is_dead(vd)) {
zfs_post_autoreplace(vd->vdev_spa, vd); zfs_ereport_post(FM_EREPORT_RESOURCE_AUTOREPLACE,
spa_event_notify(vd->vdev_spa, vd, ESC_ZFS_VDEV_CHECK); vd->vdev_spa, vd, NULL, 0, 0);
spa_event_notify(vd->vdev_spa, vd, FM_EREPORT_ZFS_DEVICE_CHECK);
} }
} }
@ -2851,7 +2852,7 @@ spa_export_common(char *pool, int new_state, nvlist_t **oldconfig,
} }
} }
spa_event_notify(spa, NULL, ESC_ZFS_POOL_DESTROY); spa_event_notify(spa, NULL, FM_EREPORT_ZFS_POOL_DESTROY);
if (spa->spa_state != POOL_STATE_UNINITIALIZED) { if (spa->spa_state != POOL_STATE_UNINITIALIZED) {
spa_unload(spa); spa_unload(spa);
@ -3161,7 +3162,7 @@ spa_vdev_attach(spa_t *spa, uint64_t guid, nvlist_t *nvroot, int replacing)
if (newvd->vdev_isspare) { if (newvd->vdev_isspare) {
spa_spare_activate(newvd); spa_spare_activate(newvd);
spa_event_notify(spa, newvd, ESC_ZFS_VDEV_SPARE); spa_event_notify(spa, newvd, FM_EREPORT_ZFS_DEVICE_SPARE);
} }
oldvdpath = spa_strdup(oldvd->vdev_path); oldvdpath = spa_strdup(oldvd->vdev_path);
@ -3379,7 +3380,7 @@ spa_vdev_detach(spa_t *spa, uint64_t guid, uint64_t pguid, int replace_done)
vd->vdev_detached = B_TRUE; vd->vdev_detached = B_TRUE;
vdev_dirty(tvd, VDD_DTL, vd, txg); vdev_dirty(tvd, VDD_DTL, vd, txg);
spa_event_notify(spa, vd, ESC_ZFS_VDEV_REMOVE); spa_event_notify(spa, vd, FM_EREPORT_ZFS_DEVICE_REMOVE);
error = spa_vdev_exit(spa, vd, txg, 0); error = spa_vdev_exit(spa, vd, txg, 0);
@ -3721,9 +3722,6 @@ spa_async_probe(spa_t *spa, vdev_t *vd)
static void static void
spa_async_autoexpand(spa_t *spa, vdev_t *vd) spa_async_autoexpand(spa_t *spa, vdev_t *vd)
{ {
sysevent_id_t eid;
nvlist_t *attr;
char *physpath;
int c; int c;
if (!spa->spa_autoexpand) if (!spa->spa_autoexpand)
@ -3737,17 +3735,7 @@ spa_async_autoexpand(spa_t *spa, vdev_t *vd)
if (!vd->vdev_ops->vdev_op_leaf || vd->vdev_physpath == NULL) if (!vd->vdev_ops->vdev_op_leaf || vd->vdev_physpath == NULL)
return; return;
physpath = kmem_zalloc(MAXPATHLEN, KM_SLEEP); spa_event_notify(vd->vdev_spa, vd, FM_EREPORT_ZFS_DEVICE_AUTOEXPAND);
(void) snprintf(physpath, MAXPATHLEN, "/devices%s", vd->vdev_physpath);
VERIFY(nvlist_alloc(&attr, NV_UNIQUE_NAME, KM_SLEEP) == 0);
VERIFY(nvlist_add_string(attr, DEV_PHYS_PATH, physpath) == 0);
(void) ddi_log_sysevent(zfs_dip, SUNW_VENDOR, EC_DEV_STATUS,
ESC_DEV_DLE, attr, &eid, DDI_SLEEP);
nvlist_free(attr);
kmem_free(physpath, MAXPATHLEN);
} }
static void static void
@ -4511,8 +4499,7 @@ spa_has_active_shared_spare(spa_t *spa)
} }
/* /*
* Post a sysevent corresponding to the given event. The 'name' must be one of * Post a FM_EREPORT_ZFS_* event from sys/fm/fs/zfs.h. The payload will be
* the event definitions in sys/sysevent/eventdefs.h. The payload will be
* filled in from the spa and (optionally) the vdev. This doesn't do anything * filled in from the spa and (optionally) the vdev. This doesn't do anything
* in the userland libzpool, as we don't want consumers to misinterpret ztest * in the userland libzpool, as we don't want consumers to misinterpret ztest
* or zdb as real changes. * or zdb as real changes.
@ -4521,50 +4508,7 @@ void
spa_event_notify(spa_t *spa, vdev_t *vd, const char *name) spa_event_notify(spa_t *spa, vdev_t *vd, const char *name)
{ {
#ifdef _KERNEL #ifdef _KERNEL
sysevent_t *ev; zfs_ereport_post(name, spa, vd, NULL, 0, 0);
sysevent_attr_list_t *attr = NULL;
sysevent_value_t value;
sysevent_id_t eid;
ev = sysevent_alloc(EC_ZFS, (char *)name, SUNW_KERN_PUB "zfs",
SE_SLEEP);
value.value_type = SE_DATA_TYPE_STRING;
value.value.sv_string = spa_name(spa);
if (sysevent_add_attr(&attr, ZFS_EV_POOL_NAME, &value, SE_SLEEP) != 0)
goto done;
value.value_type = SE_DATA_TYPE_UINT64;
value.value.sv_uint64 = spa_guid(spa);
if (sysevent_add_attr(&attr, ZFS_EV_POOL_GUID, &value, SE_SLEEP) != 0)
goto done;
if (vd) {
value.value_type = SE_DATA_TYPE_UINT64;
value.value.sv_uint64 = vd->vdev_guid;
if (sysevent_add_attr(&attr, ZFS_EV_VDEV_GUID, &value,
SE_SLEEP) != 0)
goto done;
if (vd->vdev_path) {
value.value_type = SE_DATA_TYPE_STRING;
value.value.sv_string = vd->vdev_path;
if (sysevent_add_attr(&attr, ZFS_EV_VDEV_PATH,
&value, SE_SLEEP) != 0)
goto done;
}
}
if (sysevent_attach_attributes(ev, attr) != 0)
goto done;
attr = NULL;
(void) log_sysevent(ev, SE_SLEEP, &eid);
done:
if (attr)
sysevent_free_attr(attr);
sysevent_free(ev);
#endif #endif
} }

View File

@ -267,7 +267,7 @@ spa_config_sync(spa_t *target, boolean_t removing, boolean_t postsysevent)
spa_config_generation++; spa_config_generation++;
if (postsysevent) if (postsysevent)
spa_event_notify(target, NULL, ESC_ZFS_CONFIG_SYNC); spa_event_notify(target, NULL, FM_EREPORT_ZFS_CONFIG_SYNC);
} }
/* /*

View File

@ -41,6 +41,7 @@
#include <sys/dsl_pool.h> #include <sys/dsl_pool.h>
#include <sys/dsl_dir.h> #include <sys/dsl_dir.h>
#include <sys/dsl_prop.h> #include <sys/dsl_prop.h>
#include <sys/fm/util.h>
#include <sys/fs/zfs.h> #include <sys/fs/zfs.h>
#include <sys/metaslab_impl.h> #include <sys/metaslab_impl.h>
#include <sys/sunddi.h> #include <sys/sunddi.h>
@ -1373,6 +1374,7 @@ spa_init(int mode)
spa_mode_global = mode; spa_mode_global = mode;
fm_init();
refcount_init(); refcount_init();
unique_init(); unique_init();
zio_init(); zio_init();
@ -1398,6 +1400,7 @@ spa_fini(void)
zio_fini(); zio_fini();
unique_fini(); unique_fini();
refcount_fini(); refcount_fini();
fm_fini();
avl_destroy(&spa_namespace_avl); avl_destroy(&spa_namespace_avl);
avl_destroy(&spa_spare_avl); avl_destroy(&spa_spare_avl);

View File

@ -2097,7 +2097,7 @@ vdev_clear(spa_t *spa, vdev_t *vd)
if (vd->vdev_aux == NULL && !vdev_is_dead(vd)) if (vd->vdev_aux == NULL && !vdev_is_dead(vd))
spa_async_request(spa, SPA_ASYNC_RESILVER); spa_async_request(spa, SPA_ASYNC_RESILVER);
spa_event_notify(spa, vd, ESC_ZFS_VDEV_CLEAR); spa_event_notify(spa, vd, FM_EREPORT_ZFS_DEVICE_CLEAR);
} }
} }
@ -2634,7 +2634,8 @@ vdev_set_state(vdev_t *vd, boolean_t isopen, vdev_state_t state, vdev_aux_t aux)
* Indicate to the ZFS DE that this device has been removed, and * Indicate to the ZFS DE that this device has been removed, and
* any recent errors should be ignored. * any recent errors should be ignored.
*/ */
zfs_post_remove(spa, vd); zfs_ereport_post(FM_EREPORT_RESOURCE_REMOVED,
spa, vd, NULL, 0, 0);
vd->vdev_removed = B_TRUE; vd->vdev_removed = B_TRUE;
} else if (state == VDEV_STATE_CANT_OPEN) { } else if (state == VDEV_STATE_CANT_OPEN) {
/* /*

View File

@ -88,6 +88,14 @@
* doesn't actually correspond to any particular device or piece of data, * doesn't actually correspond to any particular device or piece of data,
* and the caller will always retry without caching or queueing anyway). * and the caller will always retry without caching or queueing anyway).
*/ */
#ifdef _KERNEL
static void
zfs_ereport_post_cb(nvlist_t *nvl)
{
fm_nvlist_destroy(nvl, FM_NVA_FREE);
}
#endif /* _KERNEL */
void void
zfs_ereport_post(const char *subclass, spa_t *spa, vdev_t *vd, zio_t *zio, zfs_ereport_post(const char *subclass, spa_t *spa, vdev_t *vd, zio_t *zio,
uint64_t stateoroffset, uint64_t size) uint64_t stateoroffset, uint64_t size)
@ -205,6 +213,7 @@ zfs_ereport_post(const char *subclass, spa_t *spa, vdev_t *vd, zio_t *zio,
vd != NULL ? vd->vdev_guid : 0); vd != NULL ? vd->vdev_guid : 0);
fm_ereport_set(ereport, FM_EREPORT_VERSION, class, ena, detector, NULL); fm_ereport_set(ereport, FM_EREPORT_VERSION, class, ena, detector, NULL);
fm_nvlist_destroy(detector, FM_NVA_FREE);
/* /*
* Construct the per-ereport payload, depending on which parameters are * Construct the per-ereport payload, depending on which parameters are
@ -324,58 +333,11 @@ zfs_ereport_post(const char *subclass, spa_t *spa, vdev_t *vd, zio_t *zio,
} }
mutex_exit(&spa->spa_errlist_lock); mutex_exit(&spa->spa_errlist_lock);
fm_ereport_post(ereport, EVCH_SLEEP); /* Cleanup must be handled by the passed callback function */
fm_zevent_post(ereport, zfs_ereport_post_cb);
fm_nvlist_destroy(ereport, FM_NVA_FREE); #endif /* _KERNEL */
fm_nvlist_destroy(detector, FM_NVA_FREE);
#endif
} }
static void #if defined(_KERNEL) && defined(HAVE_SPL)
zfs_post_common(spa_t *spa, vdev_t *vd, const char *name) EXPORT_SYMBOL(zfs_ereport_post);
{ #endif /* _KERNEL */
#ifdef _KERNEL
nvlist_t *resource;
char class[64];
if ((resource = fm_nvlist_create(NULL)) == NULL)
return;
(void) snprintf(class, sizeof (class), "%s.%s.%s", FM_RSRC_RESOURCE,
ZFS_ERROR_CLASS, name);
VERIFY(nvlist_add_uint8(resource, FM_VERSION, FM_RSRC_VERSION) == 0);
VERIFY(nvlist_add_string(resource, FM_CLASS, class) == 0);
VERIFY(nvlist_add_uint64(resource,
FM_EREPORT_PAYLOAD_ZFS_POOL_GUID, spa_guid(spa)) == 0);
if (vd)
VERIFY(nvlist_add_uint64(resource,
FM_EREPORT_PAYLOAD_ZFS_VDEV_GUID, vd->vdev_guid) == 0);
fm_ereport_post(resource, EVCH_SLEEP);
fm_nvlist_destroy(resource, FM_NVA_FREE);
#endif
}
/*
* The 'resource.fs.zfs.removed' event is an internal signal that the given vdev
* has been removed from the system. This will cause the DE to ignore any
* recent I/O errors, inferring that they are due to the asynchronous device
* removal.
*/
void
zfs_post_remove(spa_t *spa, vdev_t *vd)
{
zfs_post_common(spa, vd, FM_RESOURCE_REMOVED);
}
/*
* The 'resource.fs.zfs.autoreplace' event is an internal signal that the pool
* has the 'autoreplace' property set, and therefore any broken vdevs will be
* handled by higher level logic, and no vdev fault should be generated.
*/
void
zfs_post_autoreplace(spa_t *spa, vdev_t *vd)
{
zfs_post_common(spa, vd, FM_RESOURCE_AUTOREPLACE);
}