summaryrefslogtreecommitdiffstats
path: root/sys/cddl
diff options
context:
space:
mode:
authorpjd <pjd@FreeBSD.org>2009-12-05 14:33:11 +0000
committerpjd <pjd@FreeBSD.org>2009-12-05 14:33:11 +0000
commit701c04d679b8eed0b7ed8984a58734d652752577 (patch)
tree6d0833834ed9ee6af4c611ae73a9cddb17d6dda6 /sys/cddl
parent90bd91d1de8110c18b3a8e2ee001ed1096b6d294 (diff)
downloadFreeBSD-src-701c04d679b8eed0b7ed8984a58734d652752577.zip
FreeBSD-src-701c04d679b8eed0b7ed8984a58734d652752577.tar.gz
Fix deadlock when ZVOLs are present and we are replacing dead component or
calling scrub when pool is in a degraded state. It will try to taste ZVOLs, which will lead to deadlock, as ZVOL will try to acquire the same locks as replace/scrub is holding already. We can't simply skip provider based on their GEOM class, because ZVOL can have providers build on top of it and we need to skip those as well. We do it by asking for ZFS::iszvol attribute. Any ZVOL-based provider will give us positive answer and we have to skip those providers. This way we remove possibility to create ZFS pools on top of ZVOLs, but it is not very useful anyway. I believe deadlock is still possible in some very complex situations like when we have MD provider on top of UFS file on top of ZVOL. When we try to replace dead component in the pool mentioned ZVOL is based on, there might be a deadlock when ZFS will try to taste MD provider. There is no easy way to detect that, but it isn't very common. MFC after: 1 week
Diffstat (limited to 'sys/cddl')
-rw-r--r--sys/cddl/contrib/opensolaris/uts/common/fs/zfs/vdev_geom.c7
-rw-r--r--sys/cddl/contrib/opensolaris/uts/common/fs/zfs/zvol.c5
2 files changed, 10 insertions, 2 deletions
diff --git a/sys/cddl/contrib/opensolaris/uts/common/fs/zfs/vdev_geom.c b/sys/cddl/contrib/opensolaris/uts/common/fs/zfs/vdev_geom.c
index e744001..5cc5f37 100644
--- a/sys/cddl/contrib/opensolaris/uts/common/fs/zfs/vdev_geom.c
+++ b/sys/cddl/contrib/opensolaris/uts/common/fs/zfs/vdev_geom.c
@@ -293,11 +293,16 @@ vdev_geom_read_guid(struct g_consumer *cp)
uint64_t psize;
off_t offset, size;
uint64_t guid;
- int l, len;
+ int error, l, len, iszvol;
g_topology_assert_not();
pp = cp->provider;
+ ZFS_LOG(1, "Reading guid from %s...", pp->name);
+ if (g_getattr("ZFS::iszvol", cp, &iszvol) == 0 && iszvol) {
+ ZFS_LOG(1, "Skipping ZVOL-based provider %s.", pp->name);
+ return (0);
+ }
psize = pp->mediasize;
psize = P2ALIGN(psize, (uint64_t)sizeof(vdev_label_t));
diff --git a/sys/cddl/contrib/opensolaris/uts/common/fs/zfs/zvol.c b/sys/cddl/contrib/opensolaris/uts/common/fs/zfs/zvol.c
index e9b00cb..b07a8c1 100644
--- a/sys/cddl/contrib/opensolaris/uts/common/fs/zfs/zvol.c
+++ b/sys/cddl/contrib/opensolaris/uts/common/fs/zfs/zvol.c
@@ -335,8 +335,11 @@ zvol_start(struct bio *bp)
wakeup_one(&zv->zv_queue);
mtx_unlock(&zv->zv_queue_mtx);
break;
- case BIO_DELETE:
case BIO_GETATTR:
+ if (g_handleattr_int(bp, "ZFS::iszvol", 1))
+ break;
+ /* FALLTHROUGH */
+ case BIO_DELETE:
default:
g_io_deliver(bp, EOPNOTSUPP);
break;
OpenPOWER on IntegriCloud