]> granicus.if.org Git - zfs/commitdiff
OpenZFS 8377 - Panic in bookmark deletion
authorMatthew Ahrens <mahrens@delphix.com>
Fri, 14 Apr 2017 19:52:43 +0000 (12:52 -0700)
committerBrian Behlendorf <behlendorf1@llnl.gov>
Fri, 30 Jun 2017 18:11:01 +0000 (11:11 -0700)
Authored by: Matthew Ahrens <mahrens@delphix.com>
Reviewed by: Paul Dagnelie <pcd@delphix.com>
Reviewed by: Pavel Zakharov <pavel.zakharov@delphix.com>
Reviewed by: George Wilson <george.wilson@delphix.com>
Approved by: Robert Mustacchi <rm@joyent.com>
Reviewed-by: Brian Behlendorf <behlendorf1@llnl.gov>
Reviewed-by: George Melikov <mail@gmelikov.ru>
Ported-by: Giuseppe Di Natale <dinatale2@llnl.gov>
The problem is that when dsl_bookmark_destroy_check() is
executed from open context (the pre-check), it fills in
dbda_success based on the existence of the bookmark. But
the bookmark (or containing filesystem as in this case)
can be destroyed before we get to syncing context. When
we re-run dsl_bookmark_destroy_check() in syncing context,
it will not add the deleted bookmark to dbda_success,
intending for dsl_bookmark_destroy_sync() to not process
it. But because the bookmark is still in dbda_success from
the open-context call, we do try to destroy it.
The fix is that dsl_bookmark_destroy_check() should not
modify dbda_success when called from open context.

OpenZFS-issue: https://www.illumos.org/issues/8377
OpenZFS-commit: https://github.com/openzfs/openzfs/commit/b0b6fe3
Closes #6286

module/zfs/dsl_bookmark.c

index 64f5788c4042b6653c447cd301e766f62e5d05ed..a6ca1ebab3b24163ab08dd5fa3a0fb8a943395f3 100644 (file)
@@ -358,6 +358,9 @@ dsl_bookmark_destroy_check(void *arg, dmu_tx_t *tx)
        int rv = 0;
        nvpair_t *pair;
 
+       ASSERT(nvlist_empty(dbda->dbda_success));
+       ASSERT(nvlist_empty(dbda->dbda_errors));
+
        if (!spa_feature_is_enabled(dp->dp_spa, SPA_FEATURE_BOOKMARKS))
                return (0);
 
@@ -387,7 +390,10 @@ dsl_bookmark_destroy_check(void *arg, dmu_tx_t *tx)
                        }
                }
                if (error == 0) {
-                       fnvlist_add_boolean(dbda->dbda_success, fullname);
+                       if (dmu_tx_is_syncing(tx)) {
+                               fnvlist_add_boolean(dbda->dbda_success,
+                                   fullname);
+                       }
                } else {
                        fnvlist_add_int32(dbda->dbda_errors, fullname, error);
                        rv = error;