mirror of
https://git.proxmox.com/git/mirror_zfs.git
synced 2025-01-12 19:20:28 +03:00
OpenZFS 7254 - ztest failed assertion in ztest_dataset_dirobj_verify: dirobjs + 1 == usedobjs
Authored by: Paul Dagnelie <pcd@delphix.com> Reviewed by: George Wilson <george.wilson@delphix.com> Reviewed by: Prakash Surya <prakash.surya@delphix.com> Reviewed by: Matthew Ahrens <mahrens@delphix.com> Reviewed by: Steve Gonczi <steve.gonczi@delphix.com> Approved by: Robert Mustacchi <rm@joyent.com> Reviewed-by: Brian Behlendorf <behlendorf1@llnl.gov> Ported-by: George Melikov <mail@gmelikov.ru> OpenZFS-issue: https://www.illumos.org/issues/7254 OpenZFS-commit: https://github.com/openzfs/openzfs/commit/c166b69 Closes #5670
This commit is contained in:
parent
687e612f9a
commit
cc9bb3e58e
@ -20,8 +20,8 @@
|
|||||||
*/
|
*/
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2005, 2010, Oracle and/or its affiliates. All rights reserved.
|
* Copyright (c) 2005, 2010, Oracle and/or its affiliates. All rights reserved.
|
||||||
|
* Copyright (c) 2012, 2015 by Delphix. All rights reserved.
|
||||||
* Copyright (c) 2013 by Saso Kiselkov. All rights reserved.
|
* Copyright (c) 2013 by Saso Kiselkov. All rights reserved.
|
||||||
* Copyright (c) 2012, 2014 by Delphix. All rights reserved.
|
|
||||||
* Copyright (c) 2014 Spectra Logic Corporation, All rights reserved.
|
* Copyright (c) 2014 Spectra Logic Corporation, All rights reserved.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
@ -104,9 +104,14 @@ struct objset {
|
|||||||
zfs_redundant_metadata_type_t os_redundant_metadata;
|
zfs_redundant_metadata_type_t os_redundant_metadata;
|
||||||
int os_recordsize;
|
int os_recordsize;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Pointer is constant; the blkptr it points to is protected by
|
||||||
|
* os_dsl_dataset->ds_bp_rwlock
|
||||||
|
*/
|
||||||
|
blkptr_t *os_rootbp;
|
||||||
|
|
||||||
/* no lock needed: */
|
/* no lock needed: */
|
||||||
struct dmu_tx *os_synctx; /* XXX sketchy */
|
struct dmu_tx *os_synctx; /* XXX sketchy */
|
||||||
blkptr_t *os_rootbp;
|
|
||||||
zil_header_t os_zil_header;
|
zil_header_t os_zil_header;
|
||||||
list_t os_synced_dnodes;
|
list_t os_synced_dnodes;
|
||||||
uint64_t os_flags;
|
uint64_t os_flags;
|
||||||
|
@ -38,6 +38,7 @@
|
|||||||
#include <sys/zfs_context.h>
|
#include <sys/zfs_context.h>
|
||||||
#include <sys/dsl_deadlist.h>
|
#include <sys/dsl_deadlist.h>
|
||||||
#include <sys/refcount.h>
|
#include <sys/refcount.h>
|
||||||
|
#include <sys/rrwlock.h>
|
||||||
#include <zfeature_common.h>
|
#include <zfeature_common.h>
|
||||||
|
|
||||||
#ifdef __cplusplus
|
#ifdef __cplusplus
|
||||||
@ -149,6 +150,7 @@ typedef struct dsl_dataset_phys {
|
|||||||
|
|
||||||
typedef struct dsl_dataset {
|
typedef struct dsl_dataset {
|
||||||
dmu_buf_user_t ds_dbu;
|
dmu_buf_user_t ds_dbu;
|
||||||
|
rrwlock_t ds_bp_rwlock; /* Protects ds_phys->ds_bp */
|
||||||
|
|
||||||
/* Immutable: */
|
/* Immutable: */
|
||||||
struct dsl_dir *ds_dir;
|
struct dsl_dir *ds_dir;
|
||||||
|
@ -1583,10 +1583,18 @@ dbuf_dirty(dmu_buf_impl_t *db, dmu_tx_t *tx)
|
|||||||
* objects may be dirtied in syncing context, but only if they
|
* objects may be dirtied in syncing context, but only if they
|
||||||
* were already pre-dirtied in open context.
|
* were already pre-dirtied in open context.
|
||||||
*/
|
*/
|
||||||
|
#ifdef DEBUG
|
||||||
|
if (dn->dn_objset->os_dsl_dataset != NULL) {
|
||||||
|
rrw_enter(&dn->dn_objset->os_dsl_dataset->ds_bp_rwlock,
|
||||||
|
RW_READER, FTAG);
|
||||||
|
}
|
||||||
ASSERT(!dmu_tx_is_syncing(tx) ||
|
ASSERT(!dmu_tx_is_syncing(tx) ||
|
||||||
BP_IS_HOLE(dn->dn_objset->os_rootbp) ||
|
BP_IS_HOLE(dn->dn_objset->os_rootbp) ||
|
||||||
DMU_OBJECT_IS_SPECIAL(dn->dn_object) ||
|
DMU_OBJECT_IS_SPECIAL(dn->dn_object) ||
|
||||||
dn->dn_objset->os_dsl_dataset == NULL);
|
dn->dn_objset->os_dsl_dataset == NULL);
|
||||||
|
if (dn->dn_objset->os_dsl_dataset != NULL)
|
||||||
|
rrw_exit(&dn->dn_objset->os_dsl_dataset->ds_bp_rwlock, FTAG);
|
||||||
|
#endif
|
||||||
/*
|
/*
|
||||||
* We make this assert for private objects as well, but after we
|
* We make this assert for private objects as well, but after we
|
||||||
* check if we're already dirty. They are allowed to re-dirty
|
* check if we're already dirty. They are allowed to re-dirty
|
||||||
@ -1611,12 +1619,21 @@ dbuf_dirty(dmu_buf_impl_t *db, dmu_tx_t *tx)
|
|||||||
* Don't set dirtyctx to SYNC if we're just modifying this as we
|
* Don't set dirtyctx to SYNC if we're just modifying this as we
|
||||||
* initialize the objset.
|
* initialize the objset.
|
||||||
*/
|
*/
|
||||||
if (dn->dn_dirtyctx == DN_UNDIRTIED &&
|
if (dn->dn_dirtyctx == DN_UNDIRTIED) {
|
||||||
!BP_IS_HOLE(dn->dn_objset->os_rootbp)) {
|
if (dn->dn_objset->os_dsl_dataset != NULL) {
|
||||||
dn->dn_dirtyctx =
|
rrw_enter(&dn->dn_objset->os_dsl_dataset->ds_bp_rwlock,
|
||||||
(dmu_tx_is_syncing(tx) ? DN_DIRTY_SYNC : DN_DIRTY_OPEN);
|
RW_READER, FTAG);
|
||||||
ASSERT(dn->dn_dirtyctx_firstset == NULL);
|
}
|
||||||
dn->dn_dirtyctx_firstset = kmem_alloc(1, KM_SLEEP);
|
if (!BP_IS_HOLE(dn->dn_objset->os_rootbp)) {
|
||||||
|
dn->dn_dirtyctx = (dmu_tx_is_syncing(tx) ?
|
||||||
|
DN_DIRTY_SYNC : DN_DIRTY_OPEN);
|
||||||
|
ASSERT(dn->dn_dirtyctx_firstset == NULL);
|
||||||
|
dn->dn_dirtyctx_firstset = kmem_alloc(1, KM_SLEEP);
|
||||||
|
}
|
||||||
|
if (dn->dn_objset->os_dsl_dataset != NULL) {
|
||||||
|
rrw_exit(&dn->dn_objset->os_dsl_dataset->ds_bp_rwlock,
|
||||||
|
FTAG);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
mutex_exit(&dn->dn_mtx);
|
mutex_exit(&dn->dn_mtx);
|
||||||
|
|
||||||
@ -1661,8 +1678,14 @@ dbuf_dirty(dmu_buf_impl_t *db, dmu_tx_t *tx)
|
|||||||
* this assertion only if we're not already dirty.
|
* this assertion only if we're not already dirty.
|
||||||
*/
|
*/
|
||||||
os = dn->dn_objset;
|
os = dn->dn_objset;
|
||||||
|
#ifdef DEBUG
|
||||||
|
if (dn->dn_objset->os_dsl_dataset != NULL)
|
||||||
|
rrw_enter(&os->os_dsl_dataset->ds_bp_rwlock, RW_READER, FTAG);
|
||||||
ASSERT(!dmu_tx_is_syncing(tx) || DMU_OBJECT_IS_SPECIAL(dn->dn_object) ||
|
ASSERT(!dmu_tx_is_syncing(tx) || DMU_OBJECT_IS_SPECIAL(dn->dn_object) ||
|
||||||
os->os_dsl_dataset == NULL || BP_IS_HOLE(os->os_rootbp));
|
os->os_dsl_dataset == NULL || BP_IS_HOLE(os->os_rootbp));
|
||||||
|
if (dn->dn_objset->os_dsl_dataset != NULL)
|
||||||
|
rrw_exit(&os->os_dsl_dataset->ds_bp_rwlock, FTAG);
|
||||||
|
#endif
|
||||||
ASSERT(db->db.db_size != 0);
|
ASSERT(db->db.db_size != 0);
|
||||||
|
|
||||||
dprintf_dbuf(db, "size=%llx\n", (u_longlong_t)db->db.db_size);
|
dprintf_dbuf(db, "size=%llx\n", (u_longlong_t)db->db.db_size);
|
||||||
|
@ -545,8 +545,10 @@ dmu_objset_from_ds(dsl_dataset_t *ds, objset_t **osp)
|
|||||||
mutex_enter(&ds->ds_opening_lock);
|
mutex_enter(&ds->ds_opening_lock);
|
||||||
if (ds->ds_objset == NULL) {
|
if (ds->ds_objset == NULL) {
|
||||||
objset_t *os;
|
objset_t *os;
|
||||||
|
rrw_enter(&ds->ds_bp_rwlock, RW_READER, FTAG);
|
||||||
err = dmu_objset_open_impl(dsl_dataset_get_spa(ds),
|
err = dmu_objset_open_impl(dsl_dataset_get_spa(ds),
|
||||||
ds, dsl_dataset_get_blkptr(ds), &os);
|
ds, dsl_dataset_get_blkptr(ds), &os);
|
||||||
|
rrw_exit(&ds->ds_bp_rwlock, FTAG);
|
||||||
|
|
||||||
if (err == 0) {
|
if (err == 0) {
|
||||||
mutex_enter(&ds->ds_lock);
|
mutex_enter(&ds->ds_lock);
|
||||||
@ -945,9 +947,11 @@ dmu_objset_create_sync(void *arg, dmu_tx_t *tx)
|
|||||||
doca->doca_cred, tx);
|
doca->doca_cred, tx);
|
||||||
|
|
||||||
VERIFY0(dsl_dataset_hold_obj(pdd->dd_pool, obj, FTAG, &ds));
|
VERIFY0(dsl_dataset_hold_obj(pdd->dd_pool, obj, FTAG, &ds));
|
||||||
|
rrw_enter(&ds->ds_bp_rwlock, RW_READER, FTAG);
|
||||||
bp = dsl_dataset_get_blkptr(ds);
|
bp = dsl_dataset_get_blkptr(ds);
|
||||||
os = dmu_objset_create_impl(pdd->dd_pool->dp_spa,
|
os = dmu_objset_create_impl(pdd->dd_pool->dp_spa,
|
||||||
ds, bp, doca->doca_type, tx);
|
ds, bp, doca->doca_type, tx);
|
||||||
|
rrw_exit(&ds->ds_bp_rwlock, FTAG);
|
||||||
|
|
||||||
if (doca->doca_userfunc != NULL) {
|
if (doca->doca_userfunc != NULL) {
|
||||||
doca->doca_userfunc(os, doca->doca_userarg,
|
doca->doca_userfunc(os, doca->doca_userarg,
|
||||||
@ -1179,7 +1183,6 @@ dmu_objset_write_ready(zio_t *zio, arc_buf_t *abuf, void *arg)
|
|||||||
dnode_phys_t *dnp = &os->os_phys->os_meta_dnode;
|
dnode_phys_t *dnp = &os->os_phys->os_meta_dnode;
|
||||||
|
|
||||||
ASSERT(!BP_IS_EMBEDDED(bp));
|
ASSERT(!BP_IS_EMBEDDED(bp));
|
||||||
ASSERT3P(bp, ==, os->os_rootbp);
|
|
||||||
ASSERT3U(BP_GET_TYPE(bp), ==, DMU_OT_OBJSET);
|
ASSERT3U(BP_GET_TYPE(bp), ==, DMU_OT_OBJSET);
|
||||||
ASSERT0(BP_GET_LEVEL(bp));
|
ASSERT0(BP_GET_LEVEL(bp));
|
||||||
|
|
||||||
@ -1192,6 +1195,11 @@ dmu_objset_write_ready(zio_t *zio, arc_buf_t *abuf, void *arg)
|
|||||||
bp->blk_fill = 0;
|
bp->blk_fill = 0;
|
||||||
for (i = 0; i < dnp->dn_nblkptr; i++)
|
for (i = 0; i < dnp->dn_nblkptr; i++)
|
||||||
bp->blk_fill += BP_GET_FILL(&dnp->dn_blkptr[i]);
|
bp->blk_fill += BP_GET_FILL(&dnp->dn_blkptr[i]);
|
||||||
|
if (os->os_dsl_dataset != NULL)
|
||||||
|
rrw_enter(&os->os_dsl_dataset->ds_bp_rwlock, RW_WRITER, FTAG);
|
||||||
|
*os->os_rootbp = *bp;
|
||||||
|
if (os->os_dsl_dataset != NULL)
|
||||||
|
rrw_exit(&os->os_dsl_dataset->ds_bp_rwlock, FTAG);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* ARGSUSED */
|
/* ARGSUSED */
|
||||||
@ -1211,6 +1219,7 @@ dmu_objset_write_done(zio_t *zio, arc_buf_t *abuf, void *arg)
|
|||||||
(void) dsl_dataset_block_kill(ds, bp_orig, tx, B_TRUE);
|
(void) dsl_dataset_block_kill(ds, bp_orig, tx, B_TRUE);
|
||||||
dsl_dataset_block_born(ds, bp, tx);
|
dsl_dataset_block_born(ds, bp, tx);
|
||||||
}
|
}
|
||||||
|
kmem_free(bp, sizeof (*bp));
|
||||||
}
|
}
|
||||||
|
|
||||||
/* called from dsl */
|
/* called from dsl */
|
||||||
@ -1224,6 +1233,8 @@ dmu_objset_sync(objset_t *os, zio_t *pio, dmu_tx_t *tx)
|
|||||||
list_t *list;
|
list_t *list;
|
||||||
list_t *newlist = NULL;
|
list_t *newlist = NULL;
|
||||||
dbuf_dirty_record_t *dr;
|
dbuf_dirty_record_t *dr;
|
||||||
|
blkptr_t *blkptr_copy = kmem_alloc(sizeof (*os->os_rootbp), KM_SLEEP);
|
||||||
|
*blkptr_copy = *os->os_rootbp;
|
||||||
|
|
||||||
dprintf_ds(os->os_dsl_dataset, "txg=%llu\n", tx->tx_txg);
|
dprintf_ds(os->os_dsl_dataset, "txg=%llu\n", tx->tx_txg);
|
||||||
|
|
||||||
@ -1251,7 +1262,7 @@ dmu_objset_sync(objset_t *os, zio_t *pio, dmu_tx_t *tx)
|
|||||||
dmu_write_policy(os, NULL, 0, 0, ZIO_COMPRESS_INHERIT, &zp);
|
dmu_write_policy(os, NULL, 0, 0, ZIO_COMPRESS_INHERIT, &zp);
|
||||||
|
|
||||||
zio = arc_write(pio, os->os_spa, tx->tx_txg,
|
zio = arc_write(pio, os->os_spa, tx->tx_txg,
|
||||||
os->os_rootbp, os->os_phys_buf, DMU_OS_IS_L2CACHEABLE(os),
|
blkptr_copy, os->os_phys_buf, DMU_OS_IS_L2CACHEABLE(os),
|
||||||
&zp, dmu_objset_write_ready, NULL, NULL, dmu_objset_write_done,
|
&zp, dmu_objset_write_ready, NULL, NULL, dmu_objset_write_done,
|
||||||
os, ZIO_PRIORITY_ASYNC_WRITE, ZIO_FLAG_MUSTSUCCEED, &zb);
|
os, ZIO_PRIORITY_ASYNC_WRITE, ZIO_FLAG_MUSTSUCCEED, &zb);
|
||||||
|
|
||||||
|
@ -1613,10 +1613,12 @@ dmu_recv_begin_sync(void *arg, dmu_tx_t *tx)
|
|||||||
* If we actually created a non-clone, we need to create the
|
* If we actually created a non-clone, we need to create the
|
||||||
* objset in our new dataset.
|
* objset in our new dataset.
|
||||||
*/
|
*/
|
||||||
|
rrw_enter(&newds->ds_bp_rwlock, RW_READER, FTAG);
|
||||||
if (BP_IS_HOLE(dsl_dataset_get_blkptr(newds))) {
|
if (BP_IS_HOLE(dsl_dataset_get_blkptr(newds))) {
|
||||||
(void) dmu_objset_create_impl(dp->dp_spa,
|
(void) dmu_objset_create_impl(dp->dp_spa,
|
||||||
newds, dsl_dataset_get_blkptr(newds), drrb->drr_type, tx);
|
newds, dsl_dataset_get_blkptr(newds), drrb->drr_type, tx);
|
||||||
}
|
}
|
||||||
|
rrw_exit(&newds->ds_bp_rwlock, FTAG);
|
||||||
|
|
||||||
drba->drba_cookie->drc_ds = newds;
|
drba->drba_cookie->drc_ds = newds;
|
||||||
|
|
||||||
@ -1759,7 +1761,9 @@ dmu_recv_resume_begin_sync(void *arg, dmu_tx_t *tx)
|
|||||||
dmu_buf_will_dirty(ds->ds_dbuf, tx);
|
dmu_buf_will_dirty(ds->ds_dbuf, tx);
|
||||||
dsl_dataset_phys(ds)->ds_flags |= DS_FLAG_INCONSISTENT;
|
dsl_dataset_phys(ds)->ds_flags |= DS_FLAG_INCONSISTENT;
|
||||||
|
|
||||||
|
rrw_enter(&ds->ds_bp_rwlock, RW_READER, FTAG);
|
||||||
ASSERT(!BP_IS_HOLE(dsl_dataset_get_blkptr(ds)));
|
ASSERT(!BP_IS_HOLE(dsl_dataset_get_blkptr(ds)));
|
||||||
|
rrw_exit(&ds->ds_bp_rwlock, FTAG);
|
||||||
|
|
||||||
drba->drba_cookie->drc_ds = ds;
|
drba->drba_cookie->drc_ds = ds;
|
||||||
|
|
||||||
|
@ -319,6 +319,7 @@ dsl_dataset_evict_async(void *dbu)
|
|||||||
mutex_destroy(&ds->ds_opening_lock);
|
mutex_destroy(&ds->ds_opening_lock);
|
||||||
mutex_destroy(&ds->ds_sendstream_lock);
|
mutex_destroy(&ds->ds_sendstream_lock);
|
||||||
refcount_destroy(&ds->ds_longholds);
|
refcount_destroy(&ds->ds_longholds);
|
||||||
|
rrw_destroy(&ds->ds_bp_rwlock);
|
||||||
|
|
||||||
kmem_free(ds, sizeof (dsl_dataset_t));
|
kmem_free(ds, sizeof (dsl_dataset_t));
|
||||||
}
|
}
|
||||||
@ -455,6 +456,7 @@ dsl_dataset_hold_obj(dsl_pool_t *dp, uint64_t dsobj, void *tag,
|
|||||||
mutex_init(&ds->ds_lock, NULL, MUTEX_DEFAULT, NULL);
|
mutex_init(&ds->ds_lock, NULL, MUTEX_DEFAULT, NULL);
|
||||||
mutex_init(&ds->ds_opening_lock, NULL, MUTEX_DEFAULT, NULL);
|
mutex_init(&ds->ds_opening_lock, NULL, MUTEX_DEFAULT, NULL);
|
||||||
mutex_init(&ds->ds_sendstream_lock, NULL, MUTEX_DEFAULT, NULL);
|
mutex_init(&ds->ds_sendstream_lock, NULL, MUTEX_DEFAULT, NULL);
|
||||||
|
rrw_init(&ds->ds_bp_rwlock, B_FALSE);
|
||||||
refcount_create(&ds->ds_longholds);
|
refcount_create(&ds->ds_longholds);
|
||||||
|
|
||||||
bplist_create(&ds->ds_pending_deadlist);
|
bplist_create(&ds->ds_pending_deadlist);
|
||||||
@ -862,7 +864,9 @@ dsl_dataset_create_sync_dd(dsl_dir_t *dd, dsl_dataset_t *origin,
|
|||||||
dsl_dataset_phys(origin)->ds_compressed_bytes;
|
dsl_dataset_phys(origin)->ds_compressed_bytes;
|
||||||
dsphys->ds_uncompressed_bytes =
|
dsphys->ds_uncompressed_bytes =
|
||||||
dsl_dataset_phys(origin)->ds_uncompressed_bytes;
|
dsl_dataset_phys(origin)->ds_uncompressed_bytes;
|
||||||
|
rrw_enter(&origin->ds_bp_rwlock, RW_READER, FTAG);
|
||||||
dsphys->ds_bp = dsl_dataset_phys(origin)->ds_bp;
|
dsphys->ds_bp = dsl_dataset_phys(origin)->ds_bp;
|
||||||
|
rrw_exit(&origin->ds_bp_rwlock, FTAG);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Inherit flags that describe the dataset's contents
|
* Inherit flags that describe the dataset's contents
|
||||||
@ -1384,7 +1388,9 @@ dsl_dataset_snapshot_sync_impl(dsl_dataset_t *ds, const char *snapname,
|
|||||||
dsphys->ds_uncompressed_bytes =
|
dsphys->ds_uncompressed_bytes =
|
||||||
dsl_dataset_phys(ds)->ds_uncompressed_bytes;
|
dsl_dataset_phys(ds)->ds_uncompressed_bytes;
|
||||||
dsphys->ds_flags = dsl_dataset_phys(ds)->ds_flags;
|
dsphys->ds_flags = dsl_dataset_phys(ds)->ds_flags;
|
||||||
|
rrw_enter(&ds->ds_bp_rwlock, RW_READER, FTAG);
|
||||||
dsphys->ds_bp = dsl_dataset_phys(ds)->ds_bp;
|
dsphys->ds_bp = dsl_dataset_phys(ds)->ds_bp;
|
||||||
|
rrw_exit(&ds->ds_bp_rwlock, FTAG);
|
||||||
dmu_buf_rele(dbuf, FTAG);
|
dmu_buf_rele(dbuf, FTAG);
|
||||||
|
|
||||||
for (f = 0; f < SPA_FEATURES; f++) {
|
for (f = 0; f < SPA_FEATURES; f++) {
|
||||||
@ -1980,18 +1986,25 @@ dsl_dataset_space(dsl_dataset_t *ds,
|
|||||||
else
|
else
|
||||||
*availbytesp = 0;
|
*availbytesp = 0;
|
||||||
}
|
}
|
||||||
|
rrw_enter(&ds->ds_bp_rwlock, RW_READER, FTAG);
|
||||||
*usedobjsp = BP_GET_FILL(&dsl_dataset_phys(ds)->ds_bp);
|
*usedobjsp = BP_GET_FILL(&dsl_dataset_phys(ds)->ds_bp);
|
||||||
|
rrw_exit(&ds->ds_bp_rwlock, FTAG);
|
||||||
*availobjsp = DN_MAX_OBJECT - *usedobjsp;
|
*availobjsp = DN_MAX_OBJECT - *usedobjsp;
|
||||||
}
|
}
|
||||||
|
|
||||||
boolean_t
|
boolean_t
|
||||||
dsl_dataset_modified_since_snap(dsl_dataset_t *ds, dsl_dataset_t *snap)
|
dsl_dataset_modified_since_snap(dsl_dataset_t *ds, dsl_dataset_t *snap)
|
||||||
{
|
{
|
||||||
ASSERT(dsl_pool_config_held(ds->ds_dir->dd_pool));
|
ASSERTV(dsl_pool_t *dp = ds->ds_dir->dd_pool);
|
||||||
|
uint64_t birth;
|
||||||
|
|
||||||
|
ASSERT(dsl_pool_config_held(dp));
|
||||||
if (snap == NULL)
|
if (snap == NULL)
|
||||||
return (B_FALSE);
|
return (B_FALSE);
|
||||||
if (dsl_dataset_phys(ds)->ds_bp.blk_birth >
|
rrw_enter(&ds->ds_bp_rwlock, RW_READER, FTAG);
|
||||||
dsl_dataset_phys(snap)->ds_creation_txg) {
|
birth = dsl_dataset_get_blkptr(ds)->blk_birth;
|
||||||
|
rrw_exit(&ds->ds_bp_rwlock, FTAG);
|
||||||
|
if (birth > dsl_dataset_phys(snap)->ds_creation_txg) {
|
||||||
objset_t *os, *os_snap;
|
objset_t *os, *os_snap;
|
||||||
/*
|
/*
|
||||||
* It may be that only the ZIL differs, because it was
|
* It may be that only the ZIL differs, because it was
|
||||||
@ -2948,6 +2961,7 @@ dsl_dataset_clone_swap_sync_impl(dsl_dataset_t *clone,
|
|||||||
spa_feature_t f;
|
spa_feature_t f;
|
||||||
dsl_pool_t *dp = dmu_tx_pool(tx);
|
dsl_pool_t *dp = dmu_tx_pool(tx);
|
||||||
int64_t unused_refres_delta;
|
int64_t unused_refres_delta;
|
||||||
|
blkptr_t tmp;
|
||||||
|
|
||||||
ASSERT(clone->ds_reserved == 0);
|
ASSERT(clone->ds_reserved == 0);
|
||||||
/*
|
/*
|
||||||
@ -3030,11 +3044,14 @@ dsl_dataset_clone_swap_sync_impl(dsl_dataset_t *clone,
|
|||||||
|
|
||||||
/* swap blkptrs */
|
/* swap blkptrs */
|
||||||
{
|
{
|
||||||
blkptr_t tmp;
|
rrw_enter(&clone->ds_bp_rwlock, RW_WRITER, FTAG);
|
||||||
|
rrw_enter(&origin_head->ds_bp_rwlock, RW_WRITER, FTAG);
|
||||||
tmp = dsl_dataset_phys(origin_head)->ds_bp;
|
tmp = dsl_dataset_phys(origin_head)->ds_bp;
|
||||||
dsl_dataset_phys(origin_head)->ds_bp =
|
dsl_dataset_phys(origin_head)->ds_bp =
|
||||||
dsl_dataset_phys(clone)->ds_bp;
|
dsl_dataset_phys(clone)->ds_bp;
|
||||||
dsl_dataset_phys(clone)->ds_bp = tmp;
|
dsl_dataset_phys(clone)->ds_bp = tmp;
|
||||||
|
rrw_exit(&origin_head->ds_bp_rwlock, FTAG);
|
||||||
|
rrw_exit(&clone->ds_bp_rwlock, FTAG);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* set dd_*_bytes */
|
/* set dd_*_bytes */
|
||||||
|
@ -255,7 +255,9 @@ dsl_destroy_snapshot_sync_impl(dsl_dataset_t *ds, boolean_t defer, dmu_tx_t *tx)
|
|||||||
|
|
||||||
|
|
||||||
ASSERT(RRW_WRITE_HELD(&dp->dp_config_rwlock));
|
ASSERT(RRW_WRITE_HELD(&dp->dp_config_rwlock));
|
||||||
|
rrw_enter(&ds->ds_bp_rwlock, RW_READER, FTAG);
|
||||||
ASSERT3U(dsl_dataset_phys(ds)->ds_bp.blk_birth, <=, tx->tx_txg);
|
ASSERT3U(dsl_dataset_phys(ds)->ds_bp.blk_birth, <=, tx->tx_txg);
|
||||||
|
rrw_exit(&ds->ds_bp_rwlock, FTAG);
|
||||||
ASSERT(refcount_is_zero(&ds->ds_longholds));
|
ASSERT(refcount_is_zero(&ds->ds_longholds));
|
||||||
|
|
||||||
if (defer &&
|
if (defer &&
|
||||||
@ -728,7 +730,9 @@ dsl_destroy_head_sync_impl(dsl_dataset_t *ds, dmu_tx_t *tx)
|
|||||||
ASSERT3U(dsl_dataset_phys(ds)->ds_num_children, <=, 1);
|
ASSERT3U(dsl_dataset_phys(ds)->ds_num_children, <=, 1);
|
||||||
ASSERT(ds->ds_prev == NULL ||
|
ASSERT(ds->ds_prev == NULL ||
|
||||||
dsl_dataset_phys(ds->ds_prev)->ds_next_snap_obj != ds->ds_object);
|
dsl_dataset_phys(ds->ds_prev)->ds_next_snap_obj != ds->ds_object);
|
||||||
|
rrw_enter(&ds->ds_bp_rwlock, RW_READER, FTAG);
|
||||||
ASSERT3U(dsl_dataset_phys(ds)->ds_bp.blk_birth, <=, tx->tx_txg);
|
ASSERT3U(dsl_dataset_phys(ds)->ds_bp.blk_birth, <=, tx->tx_txg);
|
||||||
|
rrw_exit(&ds->ds_bp_rwlock, FTAG);
|
||||||
ASSERT(RRW_WRITE_HELD(&dp->dp_config_rwlock));
|
ASSERT(RRW_WRITE_HELD(&dp->dp_config_rwlock));
|
||||||
|
|
||||||
/* We need to log before removing it from the namespace. */
|
/* We need to log before removing it from the namespace. */
|
||||||
@ -819,10 +823,12 @@ dsl_destroy_head_sync_impl(dsl_dataset_t *ds, dmu_tx_t *tx)
|
|||||||
ASSERT(!DS_UNIQUE_IS_ACCURATE(ds) ||
|
ASSERT(!DS_UNIQUE_IS_ACCURATE(ds) ||
|
||||||
dsl_dataset_phys(ds)->ds_unique_bytes == used);
|
dsl_dataset_phys(ds)->ds_unique_bytes == used);
|
||||||
|
|
||||||
|
rrw_enter(&ds->ds_bp_rwlock, RW_READER, FTAG);
|
||||||
bptree_add(mos, dp->dp_bptree_obj,
|
bptree_add(mos, dp->dp_bptree_obj,
|
||||||
&dsl_dataset_phys(ds)->ds_bp,
|
&dsl_dataset_phys(ds)->ds_bp,
|
||||||
dsl_dataset_phys(ds)->ds_prev_snap_txg,
|
dsl_dataset_phys(ds)->ds_prev_snap_txg,
|
||||||
used, comp, uncomp, tx);
|
used, comp, uncomp, tx);
|
||||||
|
rrw_exit(&ds->ds_bp_rwlock, FTAG);
|
||||||
dsl_dir_diduse_space(ds->ds_dir, DD_USED_HEAD,
|
dsl_dir_diduse_space(ds->ds_dir, DD_USED_HEAD,
|
||||||
-used, -comp, -uncomp, tx);
|
-used, -comp, -uncomp, tx);
|
||||||
dsl_dir_diduse_space(dp->dp_free_dir, DD_USED_HEAD,
|
dsl_dir_diduse_space(dp->dp_free_dir, DD_USED_HEAD,
|
||||||
|
@ -20,7 +20,7 @@
|
|||||||
*/
|
*/
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2005, 2010, Oracle and/or its affiliates. All rights reserved.
|
* Copyright (c) 2005, 2010, Oracle and/or its affiliates. All rights reserved.
|
||||||
* Copyright (c) 2011, 2014 by Delphix. All rights reserved.
|
* Copyright (c) 2011, 2015 by Delphix. All rights reserved.
|
||||||
* Copyright (c) 2013 Steven Hartland. All rights reserved.
|
* Copyright (c) 2013 Steven Hartland. All rights reserved.
|
||||||
* Copyright (c) 2014 Spectra Logic Corporation, All rights reserved.
|
* Copyright (c) 2014 Spectra Logic Corporation, All rights reserved.
|
||||||
*/
|
*/
|
||||||
@ -402,8 +402,10 @@ dsl_pool_create(spa_t *spa, nvlist_t *zplprops, uint64_t txg)
|
|||||||
|
|
||||||
/* create the root objset */
|
/* create the root objset */
|
||||||
VERIFY0(dsl_dataset_hold_obj(dp, obj, FTAG, &ds));
|
VERIFY0(dsl_dataset_hold_obj(dp, obj, FTAG, &ds));
|
||||||
|
rrw_enter(&ds->ds_bp_rwlock, RW_READER, FTAG);
|
||||||
VERIFY(NULL != (os = dmu_objset_create_impl(dp->dp_spa, ds,
|
VERIFY(NULL != (os = dmu_objset_create_impl(dp->dp_spa, ds,
|
||||||
dsl_dataset_get_blkptr(ds), DMU_OST_ZFS, tx)));
|
dsl_dataset_get_blkptr(ds), DMU_OST_ZFS, tx)));
|
||||||
|
rrw_exit(&ds->ds_bp_rwlock, FTAG);
|
||||||
#ifdef _KERNEL
|
#ifdef _KERNEL
|
||||||
zfs_create_fs(os, kcred, zplprops, tx);
|
zfs_create_fs(os, kcred, zplprops, tx);
|
||||||
#endif
|
#endif
|
||||||
@ -718,7 +720,9 @@ upgrade_clones_cb(dsl_pool_t *dp, dsl_dataset_t *hds, void *arg)
|
|||||||
* The $ORIGIN can't have any data, or the accounting
|
* The $ORIGIN can't have any data, or the accounting
|
||||||
* will be wrong.
|
* will be wrong.
|
||||||
*/
|
*/
|
||||||
|
rrw_enter(&ds->ds_bp_rwlock, RW_READER, FTAG);
|
||||||
ASSERT0(dsl_dataset_phys(prev)->ds_bp.blk_birth);
|
ASSERT0(dsl_dataset_phys(prev)->ds_bp.blk_birth);
|
||||||
|
rrw_exit(&ds->ds_bp_rwlock, FTAG);
|
||||||
|
|
||||||
/* The origin doesn't get attached to itself */
|
/* The origin doesn't get attached to itself */
|
||||||
if (ds->ds_object == prev->ds_object) {
|
if (ds->ds_object == prev->ds_object) {
|
||||||
|
@ -1142,7 +1142,9 @@ dsl_scan_visitds(dsl_scan_t *scn, uint64_t dsobj, dmu_tx_t *tx)
|
|||||||
* Iterate over the bps in this ds.
|
* Iterate over the bps in this ds.
|
||||||
*/
|
*/
|
||||||
dmu_buf_will_dirty(ds->ds_dbuf, tx);
|
dmu_buf_will_dirty(ds->ds_dbuf, tx);
|
||||||
|
rrw_enter(&ds->ds_bp_rwlock, RW_READER, FTAG);
|
||||||
dsl_scan_visit_rootbp(scn, ds, &dsl_dataset_phys(ds)->ds_bp, tx);
|
dsl_scan_visit_rootbp(scn, ds, &dsl_dataset_phys(ds)->ds_bp, tx);
|
||||||
|
rrw_exit(&ds->ds_bp_rwlock, FTAG);
|
||||||
|
|
||||||
dsname = kmem_alloc(ZFS_MAX_DATASET_NAME_LEN, KM_SLEEP);
|
dsname = kmem_alloc(ZFS_MAX_DATASET_NAME_LEN, KM_SLEEP);
|
||||||
dsl_dataset_name(ds, dsname);
|
dsl_dataset_name(ds, dsname);
|
||||||
|
Loading…
Reference in New Issue
Block a user