mirror of
https://git.proxmox.com/git/mirror_zfs.git
synced 2025-01-13 11:40:25 +03:00
Illumos #1951: leaking a vdev when removing an l2cache device
1952 memory leak when adding a file-based l2arc device 1954 leak in ZFS from metaslab_group_create and zfs_ereport_checksum Reviewed by: Adam Leventhal <ahl@delphix.com> Reviewed by: Matt Ahrens <matt@delphix.com> Reviewed by: Eric Schrock <eric.schrock@delphix.com> Reviewed by: Bill Pijewski <wdp@joyent.com> Reviewed by: Dan McDonald <danmcd@nexenta.com> Approved by: Eric Schrock <eric.schrock@delphix.com> References to Illumos issues: https://www.illumos.org/issues/1951 https://www.illumos.org/issues/1952 https://www.illumos.org/issues/1954 Ported-by: Richard Yao <ryao@cs.stonybrook.edu> Signed-off-by: Brian Behlendorf <behlendorf1@llnl.gov> Closes #650
This commit is contained in:
parent
b129c6590e
commit
5ffb9d1d05
@ -261,6 +261,7 @@ typedef struct vdev_label {
|
|||||||
#define VDEV_ALLOC_L2CACHE 3
|
#define VDEV_ALLOC_L2CACHE 3
|
||||||
#define VDEV_ALLOC_ROOTPOOL 4
|
#define VDEV_ALLOC_ROOTPOOL 4
|
||||||
#define VDEV_ALLOC_SPLIT 5
|
#define VDEV_ALLOC_SPLIT 5
|
||||||
|
#define VDEV_ALLOC_ATTACH 6
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Allocate or free a vdev
|
* Allocate or free a vdev
|
||||||
|
@ -1007,8 +1007,10 @@ spa_unload(spa_t *spa)
|
|||||||
}
|
}
|
||||||
spa->spa_spares.sav_count = 0;
|
spa->spa_spares.sav_count = 0;
|
||||||
|
|
||||||
for (i = 0; i < spa->spa_l2cache.sav_count; i++)
|
for (i = 0; i < spa->spa_l2cache.sav_count; i++) {
|
||||||
|
vdev_clear_stats(spa->spa_l2cache.sav_vdevs[i]);
|
||||||
vdev_free(spa->spa_l2cache.sav_vdevs[i]);
|
vdev_free(spa->spa_l2cache.sav_vdevs[i]);
|
||||||
|
}
|
||||||
if (spa->spa_l2cache.sav_vdevs) {
|
if (spa->spa_l2cache.sav_vdevs) {
|
||||||
kmem_free(spa->spa_l2cache.sav_vdevs,
|
kmem_free(spa->spa_l2cache.sav_vdevs,
|
||||||
spa->spa_l2cache.sav_count * sizeof (void *));
|
spa->spa_l2cache.sav_count * sizeof (void *));
|
||||||
@ -1231,11 +1233,13 @@ spa_load_l2cache(spa_t *spa)
|
|||||||
|
|
||||||
vd = oldvdevs[i];
|
vd = oldvdevs[i];
|
||||||
if (vd != NULL) {
|
if (vd != NULL) {
|
||||||
|
ASSERT(vd->vdev_isl2cache);
|
||||||
|
|
||||||
if (spa_l2cache_exists(vd->vdev_guid, &pool) &&
|
if (spa_l2cache_exists(vd->vdev_guid, &pool) &&
|
||||||
pool != 0ULL && l2arc_vdev_present(vd))
|
pool != 0ULL && l2arc_vdev_present(vd))
|
||||||
l2arc_remove_vdev(vd);
|
l2arc_remove_vdev(vd);
|
||||||
(void) vdev_close(vd);
|
vdev_clear_stats(vd);
|
||||||
spa_l2cache_remove(vd);
|
vdev_free(vd);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -2743,6 +2747,7 @@ spa_validate_aux_devs(spa_t *spa, nvlist_t *nvroot, uint64_t crtxg, int mode,
|
|||||||
if ((strcmp(config, ZPOOL_CONFIG_L2CACHE) == 0) &&
|
if ((strcmp(config, ZPOOL_CONFIG_L2CACHE) == 0) &&
|
||||||
strcmp(vd->vdev_ops->vdev_op_type, VDEV_TYPE_DISK) != 0) {
|
strcmp(vd->vdev_ops->vdev_op_type, VDEV_TYPE_DISK) != 0) {
|
||||||
error = ENOTBLK;
|
error = ENOTBLK;
|
||||||
|
vdev_free(vd);
|
||||||
goto out;
|
goto out;
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
@ -2852,10 +2857,6 @@ spa_l2cache_drop(spa_t *spa)
|
|||||||
if (spa_l2cache_exists(vd->vdev_guid, &pool) &&
|
if (spa_l2cache_exists(vd->vdev_guid, &pool) &&
|
||||||
pool != 0ULL && l2arc_vdev_present(vd))
|
pool != 0ULL && l2arc_vdev_present(vd))
|
||||||
l2arc_remove_vdev(vd);
|
l2arc_remove_vdev(vd);
|
||||||
if (vd->vdev_isl2cache)
|
|
||||||
spa_l2cache_remove(vd);
|
|
||||||
vdev_clear_stats(vd);
|
|
||||||
(void) vdev_close(vd);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -3851,7 +3852,7 @@ spa_vdev_attach(spa_t *spa, uint64_t guid, nvlist_t *nvroot, int replacing)
|
|||||||
pvd = oldvd->vdev_parent;
|
pvd = oldvd->vdev_parent;
|
||||||
|
|
||||||
if ((error = spa_config_parse(spa, &newrootvd, nvroot, NULL, 0,
|
if ((error = spa_config_parse(spa, &newrootvd, nvroot, NULL, 0,
|
||||||
VDEV_ALLOC_ADD)) != 0)
|
VDEV_ALLOC_ATTACH)) != 0)
|
||||||
return (spa_vdev_exit(spa, NULL, txg, EINVAL));
|
return (spa_vdev_exit(spa, NULL, txg, EINVAL));
|
||||||
|
|
||||||
if (newrootvd->vdev_children != 1)
|
if (newrootvd->vdev_children != 1)
|
||||||
|
@ -492,7 +492,7 @@ vdev_alloc(spa_t *spa, vdev_t **vdp, nvlist_t *nv, vdev_t *parent, uint_t id,
|
|||||||
&vd->vdev_removing);
|
&vd->vdev_removing);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (parent && !parent->vdev_parent) {
|
if (parent && !parent->vdev_parent && alloctype != VDEV_ALLOC_ATTACH) {
|
||||||
ASSERT(alloctype == VDEV_ALLOC_LOAD ||
|
ASSERT(alloctype == VDEV_ALLOC_LOAD ||
|
||||||
alloctype == VDEV_ALLOC_ADD ||
|
alloctype == VDEV_ALLOC_ADD ||
|
||||||
alloctype == VDEV_ALLOC_SPLIT ||
|
alloctype == VDEV_ALLOC_SPLIT ||
|
||||||
@ -669,6 +669,8 @@ vdev_top_transfer(vdev_t *svd, vdev_t *tvd)
|
|||||||
svd->vdev_ms_shift = 0;
|
svd->vdev_ms_shift = 0;
|
||||||
svd->vdev_ms_count = 0;
|
svd->vdev_ms_count = 0;
|
||||||
|
|
||||||
|
if (tvd->vdev_mg)
|
||||||
|
ASSERT3P(tvd->vdev_mg, ==, svd->vdev_mg);
|
||||||
tvd->vdev_mg = svd->vdev_mg;
|
tvd->vdev_mg = svd->vdev_mg;
|
||||||
tvd->vdev_ms = svd->vdev_ms;
|
tvd->vdev_ms = svd->vdev_ms;
|
||||||
|
|
||||||
|
@ -23,6 +23,10 @@
|
|||||||
* Use is subject to license terms.
|
* Use is subject to license terms.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Copyright (c) 2012 by Delphix. All rights reserved.
|
||||||
|
*/
|
||||||
|
|
||||||
#include <sys/spa.h>
|
#include <sys/spa.h>
|
||||||
#include <sys/spa_impl.h>
|
#include <sys/spa_impl.h>
|
||||||
#include <sys/vdev.h>
|
#include <sys/vdev.h>
|
||||||
@ -706,6 +710,10 @@ zfs_ereport_start_checksum(spa_t *spa, vdev_t *vd,
|
|||||||
|
|
||||||
if (report->zcr_ereport == NULL) {
|
if (report->zcr_ereport == NULL) {
|
||||||
report->zcr_free(report->zcr_cbdata, report->zcr_cbinfo);
|
report->zcr_free(report->zcr_cbdata, report->zcr_cbinfo);
|
||||||
|
if (report->zcr_ckinfo != NULL) {
|
||||||
|
kmem_free(report->zcr_ckinfo,
|
||||||
|
sizeof (*report->zcr_ckinfo));
|
||||||
|
}
|
||||||
kmem_free(report, sizeof (*report));
|
kmem_free(report, sizeof (*report));
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user