Add -p switch to "zpool get"
[zfs.git] / module / zfs / zfs_acl.c
index 45ec206..8ab5abe 100644 (file)
@@ -345,7 +345,7 @@ zfs_external_acl(znode_t *zp)
         * changed.
         */
 
-       if ((error = sa_lookup(zp->z_sa_hdl, SA_ZPL_ZNODE_ACL(zp->z_zfsvfs),
+       if ((error = sa_lookup(zp->z_sa_hdl, SA_ZPL_ZNODE_ACL(ZTOZSB(zp)),
            &acl_phys, sizeof (acl_phys))) == 0)
                return (acl_phys.z_acl_extern_obj);
        else {
@@ -368,23 +368,23 @@ static int
 zfs_acl_znode_info(znode_t *zp, int *aclsize, int *aclcount,
     zfs_acl_phys_t *aclphys)
 {
-       zfsvfs_t *zfsvfs = zp->z_zfsvfs;
+       zfs_sb_t *zsb = ZTOZSB(zp);
        uint64_t acl_count;
        int size;
        int error;
 
        ASSERT(MUTEX_HELD(&zp->z_acl_lock));
        if (zp->z_is_sa) {
-               if ((error = sa_size(zp->z_sa_hdl, SA_ZPL_DACL_ACES(zfsvfs),
+               if ((error = sa_size(zp->z_sa_hdl, SA_ZPL_DACL_ACES(zsb),
                    &size)) != 0)
                        return (error);
                *aclsize = size;
-               if ((error = sa_lookup(zp->z_sa_hdl, SA_ZPL_DACL_COUNT(zfsvfs),
+               if ((error = sa_lookup(zp->z_sa_hdl, SA_ZPL_DACL_COUNT(zsb),
                    &acl_count, sizeof (acl_count))) != 0)
                        return (error);
                *aclcount = acl_count;
        } else {
-               if ((error = sa_lookup(zp->z_sa_hdl, SA_ZPL_ZNODE_ACL(zfsvfs),
+               if ((error = sa_lookup(zp->z_sa_hdl, SA_ZPL_ZNODE_ACL(zsb),
                    aclphys, sizeof (*aclphys))) != 0)
                        return (error);
 
@@ -418,7 +418,7 @@ zfs_znode_acl_version(znode_t *zp)
                 * changed.
                 */
                if ((error = sa_lookup(zp->z_sa_hdl,
-                   SA_ZPL_ZNODE_ACL(zp->z_zfsvfs),
+                   SA_ZPL_ZNODE_ACL(ZTOZSB(zp)),
                    &acl_phys, sizeof (acl_phys))) == 0)
                        return (acl_phys.z_acl_version);
                else {
@@ -444,7 +444,7 @@ zfs_acl_version(int version)
 static int
 zfs_acl_version_zp(znode_t *zp)
 {
-       return (zfs_acl_version(zp->z_zfsvfs->z_version));
+       return (zfs_acl_version(ZTOZSB(zp)->z_version));
 }
 
 zfs_acl_t *
@@ -452,14 +452,14 @@ zfs_acl_alloc(int vers)
 {
        zfs_acl_t *aclp;
 
-       aclp = kmem_zalloc(sizeof (zfs_acl_t), KM_SLEEP);
+       aclp = kmem_zalloc(sizeof (zfs_acl_t), KM_PUSHPAGE);
        list_create(&aclp->z_acl, sizeof (zfs_acl_node_t),
            offsetof(zfs_acl_node_t, z_next));
        aclp->z_version = vers;
        if (vers == ZFS_ACL_VERSION_FUID)
-               aclp->z_ops = zfs_acl_fuid_ops;
+               aclp->z_ops = &zfs_acl_fuid_ops;
        else
-               aclp->z_ops = zfs_acl_v0_ops;
+               aclp->z_ops = &zfs_acl_v0_ops;
        return (aclp);
 }
 
@@ -468,9 +468,9 @@ zfs_acl_node_alloc(size_t bytes)
 {
        zfs_acl_node_t *aclnode;
 
-       aclnode = kmem_zalloc(sizeof (zfs_acl_node_t), KM_SLEEP);
+       aclnode = kmem_zalloc(sizeof (zfs_acl_node_t), KM_PUSHPAGE);
        if (bytes) {
-               aclnode->z_acldata = kmem_alloc(bytes, KM_SLEEP);
+               aclnode->z_acldata = kmem_alloc(bytes, KM_PUSHPAGE);
                aclnode->z_allocdata = aclnode->z_acldata;
                aclnode->z_allocsize = bytes;
                aclnode->z_size = bytes;
@@ -531,7 +531,7 @@ zfs_acl_valid_ace_type(uint_t type, uint_t flags)
 }
 
 static boolean_t
-zfs_ace_valid(vtype_t obj_type, zfs_acl_t *aclp, uint16_t type, uint16_t iflags)
+zfs_ace_valid(umode_t obj_mode, zfs_acl_t *aclp, uint16_t type, uint16_t iflags)
 {
        /*
         * first check type of entry
@@ -554,7 +554,7 @@ zfs_ace_valid(vtype_t obj_type, zfs_acl_t *aclp, uint16_t type, uint16_t iflags)
         * next check inheritance level flags
         */
 
-       if (obj_type == VDIR &&
+       if (S_ISDIR(obj_mode) &&
            (iflags & (ACE_FILE_INHERIT_ACE|ACE_DIRECTORY_INHERIT_ACE)))
                aclp->z_hints |= ZFS_INHERIT_ACE;
 
@@ -609,17 +609,17 @@ zfs_acl_next_ace(zfs_acl_t *aclp, void *start, uint64_t *who,
                /*
                 * Make sure we don't overstep our bounds
                 */
-               ace_size = aclp->z_ops.ace_size(acep);
+               ace_size = aclp->z_ops->ace_size(acep);
 
                if (((caddr_t)acep + ace_size) >
                    ((caddr_t)aclnode->z_acldata + aclnode->z_size)) {
                        return (NULL);
                }
 
-               *iflags = aclp->z_ops.ace_flags_get(acep);
-               *type = aclp->z_ops.ace_type_get(acep);
-               *access_mask = aclp->z_ops.ace_mask_get(acep);
-               *who = aclp->z_ops.ace_who_get(acep);
+               *iflags = aclp->z_ops->ace_flags_get(acep);
+               *type = aclp->z_ops->ace_type_get(acep);
+               *access_mask = aclp->z_ops->ace_mask_get(acep);
+               *who = aclp->z_ops->ace_who_get(acep);
                aclp->z_next_ace = (caddr_t)aclp->z_next_ace + ace_size;
                aclnode->z_ace_idx++;
 
@@ -648,7 +648,7 @@ zfs_ace_walk(void *datap, uint64_t cookie, int aclcnt,
  * ACE FUIDs will be created later.
  */
 int
-zfs_copy_ace_2_fuid(zfsvfs_t *zfsvfs, vtype_t obj_type, zfs_acl_t *aclp,
+zfs_copy_ace_2_fuid(zfs_sb_t *zsb, umode_t obj_mode, zfs_acl_t *aclp,
     void *datap, zfs_ace_t *z_acl, uint64_t aclcnt, size_t *size,
     zfs_fuid_info_t **fuidp, cred_t *cr)
 {
@@ -666,7 +666,7 @@ zfs_copy_ace_2_fuid(zfsvfs_t *zfsvfs, vtype_t obj_type, zfs_acl_t *aclp,
                entry_type = aceptr->z_hdr.z_flags & ACE_TYPE_FLAGS;
                if (entry_type != ACE_OWNER && entry_type != OWNING_GROUP &&
                    entry_type != ACE_EVERYONE) {
-                       aceptr->z_fuid = zfs_fuid_create(zfsvfs, acep->a_who,
+                       aceptr->z_fuid = zfs_fuid_create(zsb, acep->a_who,
                            cr, (entry_type == 0) ?
                            ZFS_ACE_USER : ZFS_ACE_GROUP, fuidp);
                }
@@ -674,7 +674,7 @@ zfs_copy_ace_2_fuid(zfsvfs_t *zfsvfs, vtype_t obj_type, zfs_acl_t *aclp,
                /*
                 * Make sure ACE is valid
                 */
-               if (zfs_ace_valid(obj_type, aclp, aceptr->z_hdr.z_type,
+               if (zfs_ace_valid(obj_mode, aclp, aceptr->z_hdr.z_type,
                    aceptr->z_hdr.z_flags) != B_TRUE)
                        return (EINVAL);
 
@@ -698,7 +698,7 @@ zfs_copy_ace_2_fuid(zfsvfs_t *zfsvfs, vtype_t obj_type, zfs_acl_t *aclp,
                }
 
                aceptr = (zfs_ace_t *)((caddr_t)aceptr +
-                   aclp->z_ops.ace_size(aceptr));
+                   aclp->z_ops->ace_size(aceptr));
        }
 
        *size = (caddr_t)aceptr - (caddr_t)z_acl;
@@ -710,7 +710,7 @@ zfs_copy_ace_2_fuid(zfsvfs_t *zfsvfs, vtype_t obj_type, zfs_acl_t *aclp,
  * Copy ZFS ACEs to fixed size ace_t layout
  */
 static void
-zfs_copy_fuid_2_ace(zfsvfs_t *zfsvfs, zfs_acl_t *aclp, cred_t *cr,
+zfs_copy_fuid_2_ace(zfs_sb_t *zsb, zfs_acl_t *aclp, cred_t *cr,
     void *datap, int filter)
 {
        uint64_t who;
@@ -753,7 +753,7 @@ zfs_copy_fuid_2_ace(zfsvfs_t *zfsvfs, zfs_acl_t *aclp, cred_t *cr,
                if ((entry_type != ACE_OWNER &&
                    entry_type != OWNING_GROUP &&
                    entry_type != ACE_EVERYONE)) {
-                       acep->a_who = zfs_fuid_map_id(zfsvfs, who,
+                       acep->a_who = zfs_fuid_map_id(zsb, who,
                            cr, (entry_type & ACE_IDENTIFIER_GROUP) ?
                            ZFS_ACE_GROUP : ZFS_ACE_USER);
                } else {
@@ -767,7 +767,7 @@ zfs_copy_fuid_2_ace(zfsvfs_t *zfsvfs, zfs_acl_t *aclp, cred_t *cr,
 }
 
 static int
-zfs_copy_ace_2_oldace(vtype_t obj_type, zfs_acl_t *aclp, ace_t *acep,
+zfs_copy_ace_2_oldace(umode_t obj_mode, zfs_acl_t *aclp, ace_t *acep,
     zfs_oldace_t *z_acl, int aclcnt, size_t *size)
 {
        int i;
@@ -781,7 +781,7 @@ zfs_copy_ace_2_oldace(vtype_t obj_type, zfs_acl_t *aclp, ace_t *acep,
                /*
                 * Make sure ACE is valid
                 */
-               if (zfs_ace_valid(obj_type, aclp, aceptr->z_type,
+               if (zfs_ace_valid(obj_mode, aclp, aceptr->z_type,
                    aceptr->z_flags) != B_TRUE)
                        return (EINVAL);
        }
@@ -824,9 +824,9 @@ zfs_acl_xform(znode_t *zp, zfs_acl_t *aclp, cred_t *cr)
 
        newaclnode = zfs_acl_node_alloc(aclp->z_acl_count *
            sizeof (zfs_object_ace_t));
-       aclp->z_ops = zfs_acl_fuid_ops;
-       VERIFY(zfs_copy_ace_2_fuid(zp->z_zfsvfs, ZTOV(zp)->v_type, aclp,
-           oldaclp, newaclnode->z_acldata, aclp->z_acl_count,
+       aclp->z_ops = &zfs_acl_fuid_ops;
+       VERIFY(zfs_copy_ace_2_fuid(ZTOZSB(zp), ZTOI(zp)->i_mode,
+           aclp, oldaclp, newaclnode->z_acldata, aclp->z_acl_count,
            &newaclnode->z_size, NULL, cr) == 0);
        newaclnode->z_ace_count = aclp->z_acl_count;
        aclp->z_version = ZFS_ACL_VERSION;
@@ -868,12 +868,12 @@ zfs_set_ace(zfs_acl_t *aclp, void *acep, uint32_t access_mask,
 {
        uint16_t type = entry_type & ACE_TYPE_FLAGS;
 
-       aclp->z_ops.ace_mask_set(acep, access_mask);
-       aclp->z_ops.ace_type_set(acep, access_type);
-       aclp->z_ops.ace_flags_set(acep, entry_type);
+       aclp->z_ops->ace_mask_set(acep, access_mask);
+       aclp->z_ops->ace_type_set(acep, access_type);
+       aclp->z_ops->ace_flags_set(acep, entry_type);
        if ((type != ACE_OWNER && type != OWNING_GROUP &&
            type != ACE_EVERYONE))
-               aclp->z_ops.ace_who_set(acep, fuid);
+               aclp->z_ops->ace_who_set(acep, fuid);
 }
 
 /*
@@ -1056,8 +1056,8 @@ zfs_acl_node_read(znode_t *zp, boolean_t have_lock, zfs_acl_t **aclpp,
     boolean_t will_modify)
 {
        zfs_acl_t       *aclp;
-       int             aclsize;
-       int             acl_count;
+       int             aclsize = 0;
+       int             acl_count = 0;
        zfs_acl_node_t  *aclnode;
        zfs_acl_phys_t  znode_acl;
        int             version;
@@ -1100,7 +1100,7 @@ zfs_acl_node_read(znode_t *zp, boolean_t have_lock, zfs_acl_t **aclpp,
 
        if (!zp->z_is_sa) {
                if (znode_acl.z_acl_extern_obj) {
-                       error = dmu_read(zp->z_zfsvfs->z_os,
+                       error = dmu_read(ZTOZSB(zp)->z_os,
                            znode_acl.z_acl_extern_obj, 0, aclnode->z_size,
                            aclnode->z_acldata, DMU_READ_PREFETCH);
                } else {
@@ -1108,7 +1108,7 @@ zfs_acl_node_read(znode_t *zp, boolean_t have_lock, zfs_acl_t **aclpp,
                            aclnode->z_size);
                }
        } else {
-               error = sa_lookup(zp->z_sa_hdl, SA_ZPL_DACL_ACES(zp->z_zfsvfs),
+               error = sa_lookup(zp->z_sa_hdl, SA_ZPL_DACL_ACES(ZTOZSB(zp)),
                    aclnode->z_acldata, aclnode->z_size);
        }
 
@@ -1295,7 +1295,7 @@ int
 zfs_aclset_common(znode_t *zp, zfs_acl_t *aclp, cred_t *cr, dmu_tx_t *tx)
 {
        int                     error;
-       zfsvfs_t                *zfsvfs = zp->z_zfsvfs;
+       zfs_sb_t                *zsb = ZTOZSB(zp);
        dmu_object_type_t       otype;
        zfs_acl_locator_cb_t    locate = { 0 };
        uint64_t                mode;
@@ -1309,11 +1309,11 @@ zfs_aclset_common(znode_t *zp, zfs_acl_t *aclp, cred_t *cr, dmu_tx_t *tx)
            zp->z_uid, zp->z_gid);
 
        zp->z_mode = mode;
-       SA_ADD_BULK_ATTR(bulk, count, SA_ZPL_MODE(zfsvfs), NULL,
+       SA_ADD_BULK_ATTR(bulk, count, SA_ZPL_MODE(zsb), NULL,
            &mode, sizeof (mode));
-       SA_ADD_BULK_ATTR(bulk, count, SA_ZPL_FLAGS(zfsvfs), NULL,
+       SA_ADD_BULK_ATTR(bulk, count, SA_ZPL_FLAGS(zsb), NULL,
            &zp->z_pflags, sizeof (zp->z_pflags));
-       SA_ADD_BULK_ATTR(bulk, count, SA_ZPL_CTIME(zfsvfs), NULL,
+       SA_ADD_BULK_ATTR(bulk, count, SA_ZPL_CTIME(zsb), NULL,
            &ctime, sizeof (ctime));
 
        if (zp->z_acl_cached) {
@@ -1324,11 +1324,11 @@ zfs_aclset_common(znode_t *zp, zfs_acl_t *aclp, cred_t *cr, dmu_tx_t *tx)
        /*
         * Upgrade needed?
         */
-       if (!zfsvfs->z_use_fuids) {
+       if (!zsb->z_use_fuids) {
                otype = DMU_OT_OLDACL;
        } else {
                if ((aclp->z_version == ZFS_ACL_VERSION_INITIAL) &&
-                   (zfsvfs->z_version >= ZPL_VERSION_FUID))
+                   (zsb->z_version >= ZPL_VERSION_FUID))
                        zfs_acl_xform(zp, aclp, cr);
                ASSERT(aclp->z_version >= ZFS_ACL_VERSION_FUID);
                otype = DMU_OT_ACL;
@@ -1341,9 +1341,9 @@ zfs_aclset_common(znode_t *zp, zfs_acl_t *aclp, cred_t *cr, dmu_tx_t *tx)
 
        if (zp->z_is_sa) { /* the easy case, just update the ACL attribute */
                locate.cb_aclp = aclp;
-               SA_ADD_BULK_ATTR(bulk, count, SA_ZPL_DACL_ACES(zfsvfs),
+               SA_ADD_BULK_ATTR(bulk, count, SA_ZPL_DACL_ACES(zsb),
                    zfs_acl_data_locator, &locate, aclp->z_acl_bytes);
-               SA_ADD_BULK_ATTR(bulk, count, SA_ZPL_DACL_COUNT(zfsvfs),
+               SA_ADD_BULK_ATTR(bulk, count, SA_ZPL_DACL_COUNT(zsb),
                    NULL, &aclp->z_acl_count, sizeof (uint64_t));
        } else { /* Painful legacy way */
                zfs_acl_node_t *aclnode;
@@ -1351,7 +1351,7 @@ zfs_aclset_common(znode_t *zp, zfs_acl_t *aclp, cred_t *cr, dmu_tx_t *tx)
                zfs_acl_phys_t acl_phys;
                uint64_t aoid;
 
-               if ((error = sa_lookup(zp->z_sa_hdl, SA_ZPL_ZNODE_ACL(zfsvfs),
+               if ((error = sa_lookup(zp->z_sa_hdl, SA_ZPL_ZNODE_ACL(zsb),
                    &acl_phys, sizeof (acl_phys))) != 0)
                        return (error);
 
@@ -1365,20 +1365,20 @@ zfs_aclset_common(znode_t *zp, zfs_acl_t *aclp, cred_t *cr, dmu_tx_t *tx)
                         */
                        if (aoid &&
                            aclp->z_version != acl_phys.z_acl_version) {
-                               error = dmu_object_free(zfsvfs->z_os, aoid, tx);
+                               error = dmu_object_free(zsb->z_os, aoid, tx);
                                if (error)
                                        return (error);
                                aoid = 0;
                        }
                        if (aoid == 0) {
-                               aoid = dmu_object_alloc(zfsvfs->z_os,
+                               aoid = dmu_object_alloc(zsb->z_os,
                                    otype, aclp->z_acl_bytes,
                                    otype == DMU_OT_ACL ?
                                    DMU_OT_SYSACL : DMU_OT_NONE,
                                    otype == DMU_OT_ACL ?
                                    DN_MAX_BONUSLEN : 0, tx);
                        } else {
-                               (void) dmu_object_set_blocksize(zfsvfs->z_os,
+                               (void) dmu_object_set_blocksize(zsb->z_os,
                                    aoid, aclp->z_acl_bytes, 0, tx);
                        }
                        acl_phys.z_acl_extern_obj = aoid;
@@ -1386,7 +1386,7 @@ zfs_aclset_common(znode_t *zp, zfs_acl_t *aclp, cred_t *cr, dmu_tx_t *tx)
                            aclnode = list_next(&aclp->z_acl, aclnode)) {
                                if (aclnode->z_ace_count == 0)
                                        continue;
-                               dmu_write(zfsvfs->z_os, aoid, off,
+                               dmu_write(zsb->z_os, aoid, off,
                                    aclnode->z_size, aclnode->z_acldata, tx);
                                off += aclnode->z_size;
                        }
@@ -1396,7 +1396,7 @@ zfs_aclset_common(znode_t *zp, zfs_acl_t *aclp, cred_t *cr, dmu_tx_t *tx)
                         * Migrating back embedded?
                         */
                        if (acl_phys.z_acl_extern_obj) {
-                               error = dmu_object_free(zfsvfs->z_os,
+                               error = dmu_object_free(zsb->z_os,
                                    acl_phys.z_acl_extern_obj, tx);
                                if (error)
                                        return (error);
@@ -1425,7 +1425,7 @@ zfs_aclset_common(znode_t *zp, zfs_acl_t *aclp, cred_t *cr, dmu_tx_t *tx)
                }
                acl_phys.z_acl_version = aclp->z_version;
 
-               SA_ADD_BULK_ATTR(bulk, count, SA_ZPL_ZNODE_ACL(zfsvfs), NULL,
+               SA_ADD_BULK_ATTR(bulk, count, SA_ZPL_ZNODE_ACL(zsb), NULL,
                    &acl_phys, sizeof (acl_phys));
        }
 
@@ -1444,19 +1444,19 @@ zfs_aclset_common(znode_t *zp, zfs_acl_t *aclp, cred_t *cr, dmu_tx_t *tx)
 }
 
 static void
-zfs_acl_chmod(zfsvfs_t *zfsvfs, uint64_t mode, zfs_acl_t *aclp)
+zfs_acl_chmod(zfs_sb_t *zsb, uint64_t mode, zfs_acl_t *aclp)
 {
        void            *acep = NULL;
        uint64_t        who;
        int             new_count, new_bytes;
        int             ace_size;
-       int             entry_type;
+       int             entry_type;
        uint16_t        iflags, type;
        uint32_t        access_mask;
        zfs_acl_node_t  *newnode;
-       size_t          abstract_size = aclp->z_ops.ace_abstract_size();
-       void            *zacep;
-       uint32_t        owner, group, everyone;
+       size_t          abstract_size = aclp->z_ops->ace_abstract_size();
+       void            *zacep;
+       uint32_t        owner, group, everyone;
        uint32_t        deny1, deny2, allow0;
 
        new_count = new_bytes = 0;
@@ -1516,7 +1516,7 @@ zfs_acl_chmod(zfsvfs_t *zfsvfs, uint64_t mode, zfs_acl_t *aclp)
                         * Limit permissions to be no greater than
                         * group permissions
                         */
-                       if (zfsvfs->z_acl_inherit == ZFS_ACL_RESTRICTED) {
+                       if (zsb->z_acl_inherit == ZFS_ACL_RESTRICTED) {
                                if (!(mode & S_IRGRP))
                                        access_mask &= ~ACE_READ_DATA;
                                if (!(mode & S_IWGRP))
@@ -1530,7 +1530,7 @@ zfs_acl_chmod(zfsvfs_t *zfsvfs, uint64_t mode, zfs_acl_t *aclp)
                        }
                }
                zfs_set_ace(aclp, zacep, access_mask, type, who, iflags);
-               ace_size = aclp->z_ops.ace_size(acep);
+               ace_size = aclp->z_ops->ace_size(acep);
                zacep = (void *)((uintptr_t)zacep + ace_size);
                new_count++;
                new_bytes += ace_size;
@@ -1558,7 +1558,7 @@ zfs_acl_chmod_setattr(znode_t *zp, zfs_acl_t **aclp, uint64_t mode)
        mutex_enter(&zp->z_lock);
        *aclp = zfs_acl_alloc(zfs_acl_version_zp(zp));
        (*aclp)->z_hints = zp->z_pflags & V4_ACL_WIDE_FLAGS;
-       zfs_acl_chmod(zp->z_zfsvfs, mode, *aclp);
+       zfs_acl_chmod(ZTOZSB(zp), mode, *aclp);
        mutex_exit(&zp->z_lock);
        mutex_exit(&zp->z_acl_lock);
        ASSERT(*aclp);
@@ -1568,14 +1568,14 @@ zfs_acl_chmod_setattr(znode_t *zp, zfs_acl_t **aclp, uint64_t mode)
  * strip off write_owner and write_acl
  */
 static void
-zfs_restricted_update(zfsvfs_t *zfsvfs, zfs_acl_t *aclp, void *acep)
+zfs_restricted_update(zfs_sb_t *zsb, zfs_acl_t *aclp, void *acep)
 {
-       uint32_t mask = aclp->z_ops.ace_mask_get(acep);
+       uint32_t mask = aclp->z_ops->ace_mask_get(acep);
 
-       if ((zfsvfs->z_acl_inherit == ZFS_ACL_RESTRICTED) &&
-           (aclp->z_ops.ace_type_get(acep) == ALLOW)) {
+       if ((zsb->z_acl_inherit == ZFS_ACL_RESTRICTED) &&
+           (aclp->z_ops->ace_type_get(acep) == ALLOW)) {
                mask &= ~RESTRICTED_CLEAR;
-               aclp->z_ops.ace_mask_set(acep, mask);
+               aclp->z_ops->ace_mask_set(acep, mask);
        }
 }
 
@@ -1583,14 +1583,14 @@ zfs_restricted_update(zfsvfs_t *zfsvfs, zfs_acl_t *aclp, void *acep)
  * Should ACE be inherited?
  */
 static int
-zfs_ace_can_use(vtype_t vtype, uint16_t acep_flags)
+zfs_ace_can_use(umode_t obj_mode, uint16_t acep_flags)
 {
        int     iflags = (acep_flags & 0xf);
 
-       if ((vtype == VDIR) && (iflags & ACE_DIRECTORY_INHERIT_ACE))
+       if (S_ISDIR(obj_mode) && (iflags & ACE_DIRECTORY_INHERIT_ACE))
                return (1);
        else if (iflags & ACE_FILE_INHERIT_ACE)
-               return (!((vtype == VDIR) &&
+               return (!(S_ISDIR(obj_mode) &&
                    (iflags & ACE_NO_PROPAGATE_INHERIT_ACE)));
        return (0);
 }
@@ -1599,7 +1599,7 @@ zfs_ace_can_use(vtype_t vtype, uint16_t acep_flags)
  * inherit inheritable ACEs from parent
  */
 static zfs_acl_t *
-zfs_acl_inherit(zfsvfs_t *zfsvfs, vtype_t vtype, zfs_acl_t *paclp,
+zfs_acl_inherit(zfs_sb_t *zsb, umode_t obj_mode, zfs_acl_t *paclp,
     uint64_t mode, boolean_t *need_chmod)
 {
        void            *pacep;
@@ -1612,21 +1612,21 @@ zfs_acl_inherit(zfsvfs_t *zfsvfs, vtype_t vtype, zfs_acl_t *paclp,
        size_t          ace_size;
        void            *data1, *data2;
        size_t          data1sz, data2sz;
-       boolean_t       vdir = vtype == VDIR;
-       boolean_t       vreg = vtype == VREG;
+       boolean_t       vdir = S_ISDIR(obj_mode);
+       boolean_t       vreg = S_ISREG(obj_mode);
        boolean_t       passthrough, passthrough_x, noallow;
 
        passthrough_x =
-           zfsvfs->z_acl_inherit == ZFS_ACL_PASSTHROUGH_X;
+           zsb->z_acl_inherit == ZFS_ACL_PASSTHROUGH_X;
        passthrough = passthrough_x ||
-           zfsvfs->z_acl_inherit == ZFS_ACL_PASSTHROUGH;
+           zsb->z_acl_inherit == ZFS_ACL_PASSTHROUGH;
        noallow =
-           zfsvfs->z_acl_inherit == ZFS_ACL_NOALLOW;
+           zsb->z_acl_inherit == ZFS_ACL_NOALLOW;
 
        *need_chmod = B_TRUE;
        pacep = NULL;
        aclp = zfs_acl_alloc(paclp->z_version);
-       if (zfsvfs->z_acl_inherit == ZFS_ACL_DISCARD || vtype == VLNK)
+       if (zsb->z_acl_inherit == ZFS_ACL_DISCARD || S_ISLNK(obj_mode))
                return (aclp);
        while ((pacep = zfs_acl_next_ace(paclp, pacep, &who,
            &access_mask, &iflags, &type))) {
@@ -1640,9 +1640,9 @@ zfs_acl_inherit(zfsvfs_t *zfsvfs, vtype_t vtype, zfs_acl_t *paclp,
                if (noallow && type == ALLOW)
                        continue;
 
-               ace_size = aclp->z_ops.ace_size(pacep);
+               ace_size = aclp->z_ops->ace_size(pacep);
 
-               if (!zfs_ace_can_use(vtype, iflags))
+               if (!zfs_ace_can_use(obj_mode, iflags))
                        continue;
 
                /*
@@ -1672,8 +1672,8 @@ zfs_acl_inherit(zfsvfs_t *zfsvfs, vtype_t vtype, zfs_acl_t *paclp,
                /*
                 * Copy special opaque data if any
                 */
-               if ((data1sz = paclp->z_ops.ace_data(pacep, &data1)) != 0) {
-                       VERIFY((data2sz = aclp->z_ops.ace_data(acep,
+               if ((data1sz = paclp->z_ops->ace_data(pacep, &data1)) != 0) {
+                       VERIFY((data2sz = aclp->z_ops->ace_data(acep,
                            &data2)) == data1sz);
                        bcopy(data1, data2, data2sz);
                }
@@ -1681,16 +1681,16 @@ zfs_acl_inherit(zfsvfs_t *zfsvfs, vtype_t vtype, zfs_acl_t *paclp,
                aclp->z_acl_count++;
                aclnode->z_ace_count++;
                aclp->z_acl_bytes += aclnode->z_size;
-               newflags = aclp->z_ops.ace_flags_get(acep);
+               newflags = aclp->z_ops->ace_flags_get(acep);
 
                if (vdir)
                        aclp->z_hints |= ZFS_INHERIT_ACE;
 
                if ((iflags & ACE_NO_PROPAGATE_INHERIT_ACE) || !vdir) {
                        newflags &= ~ALL_INHERIT;
-                       aclp->z_ops.ace_flags_set(acep,
+                       aclp->z_ops->ace_flags_set(acep,
                            newflags|ACE_INHERITED_ACE);
-                       zfs_restricted_update(zfsvfs, aclp, acep);
+                       zfs_restricted_update(zsb, aclp, acep);
                        continue;
                }
 
@@ -1703,11 +1703,11 @@ zfs_acl_inherit(zfsvfs_t *zfsvfs, vtype_t vtype, zfs_acl_t *paclp,
                if ((iflags & (ACE_FILE_INHERIT_ACE |
                    ACE_DIRECTORY_INHERIT_ACE)) == ACE_FILE_INHERIT_ACE) {
                        newflags |= ACE_INHERIT_ONLY_ACE;
-                       aclp->z_ops.ace_flags_set(acep,
+                       aclp->z_ops->ace_flags_set(acep,
                            newflags|ACE_INHERITED_ACE);
                } else {
                        newflags &= ~ACE_INHERIT_ONLY_ACE;
-                       aclp->z_ops.ace_flags_set(acep,
+                       aclp->z_ops->ace_flags_set(acep,
                            newflags|ACE_INHERITED_ACE);
                }
        }
@@ -1723,7 +1723,7 @@ zfs_acl_ids_create(znode_t *dzp, int flag, vattr_t *vap, cred_t *cr,
     vsecattr_t *vsecp, zfs_acl_ids_t *acl_ids)
 {
        int             error;
-       zfsvfs_t        *zfsvfs = dzp->z_zfsvfs;
+       zfs_sb_t        *zsb = ZTOZSB(dzp);
        zfs_acl_t       *paclp;
 #ifdef HAVE_KSID
        gid_t           gid;
@@ -1732,11 +1732,11 @@ zfs_acl_ids_create(znode_t *dzp, int flag, vattr_t *vap, cred_t *cr,
        boolean_t       inherited = B_FALSE;
 
        bzero(acl_ids, sizeof (zfs_acl_ids_t));
-       acl_ids->z_mode = MAKEIMODE(vap->va_type, vap->va_mode);
+       acl_ids->z_mode = vap->va_mode;
 
        if (vsecp)
-               if ((error = zfs_vsec_2_aclp(zfsvfs, vap->va_type, vsecp, cr,
-                   &acl_ids->z_fuidp, &acl_ids->z_aclp)) != 0)
+               if ((error = zfs_vsec_2_aclp(zsb, vap->va_mode, vsecp,
+                   cr, &acl_ids->z_fuidp, &acl_ids->z_aclp)) != 0)
                        return (error);
 
        acl_ids->z_fuid = vap->va_uid;
@@ -1745,21 +1745,19 @@ zfs_acl_ids_create(znode_t *dzp, int flag, vattr_t *vap, cred_t *cr,
        /*
         * Determine uid and gid.
         */
-       if ((flag & IS_ROOT_NODE) || zfsvfs->z_replay ||
-           ((flag & IS_XATTR) && (vap->va_type == VDIR))) {
-               acl_ids->z_fuid = zfs_fuid_create(zfsvfs,
-                   (uint64_t)vap->va_uid, cr,
-                   ZFS_OWNER, &acl_ids->z_fuidp);
-               acl_ids->z_fgid = zfs_fuid_create(zfsvfs,
-                   (uint64_t)vap->va_gid, cr,
-                   ZFS_GROUP, &acl_ids->z_fuidp);
+       if ((flag & IS_ROOT_NODE) || zsb->z_replay ||
+           ((flag & IS_XATTR) && (S_ISDIR(vap->va_mode)))) {
+               acl_ids->z_fuid = zfs_fuid_create(zsb, (uint64_t)vap->va_uid,
+                   cr, ZFS_OWNER, &acl_ids->z_fuidp);
+               acl_ids->z_fgid = zfs_fuid_create(zsb, (uint64_t)vap->va_gid,
+                   cr, ZFS_GROUP, &acl_ids->z_fuidp);
                gid = vap->va_gid;
        } else {
-               acl_ids->z_fuid = zfs_fuid_create_cred(zfsvfs, ZFS_OWNER,
+               acl_ids->z_fuid = zfs_fuid_create_cred(zsb, ZFS_OWNER,
                    cr, &acl_ids->z_fuidp);
                acl_ids->z_fgid = 0;
                if (vap->va_mask & AT_GID)  {
-                       acl_ids->z_fgid = zfs_fuid_create(zfsvfs,
+                       acl_ids->z_fgid = zfs_fuid_create(zsb,
                            (uint64_t)vap->va_gid,
                            cr, ZFS_GROUP, &acl_ids->z_fuidp);
                        gid = vap->va_gid;
@@ -1774,13 +1772,13 @@ zfs_acl_ids_create(znode_t *dzp, int flag, vattr_t *vap, cred_t *cr,
                                uint32_t        rid;
 
                                acl_ids->z_fgid = dzp->z_gid;
-                               gid = zfs_fuid_map_id(zfsvfs, acl_ids->z_fgid,
+                               gid = zfs_fuid_map_id(zsb, acl_ids->z_fgid,
                                    cr, ZFS_GROUP);
 
-                               if (zfsvfs->z_use_fuids &&
+                               if (zsb->z_use_fuids &&
                                    IS_EPHEMERAL(acl_ids->z_fgid)) {
                                        domain = zfs_fuid_idx_domain(
-                                           &zfsvfs->z_fuid_idx,
+                                           &zsb->z_fuid_idx,
                                            FUID_INDEX(acl_ids->z_fgid));
                                        rid = FUID_RID(acl_ids->z_fgid);
                                        zfs_fuid_node_add(&acl_ids->z_fuidp,
@@ -1789,7 +1787,7 @@ zfs_acl_ids_create(znode_t *dzp, int flag, vattr_t *vap, cred_t *cr,
                                            acl_ids->z_fgid, ZFS_GROUP);
                                }
                        } else {
-                               acl_ids->z_fgid = zfs_fuid_create_cred(zfsvfs,
+                               acl_ids->z_fgid = zfs_fuid_create_cred(zsb,
                                    ZFS_GROUP, cr, &acl_ids->z_fuidp);
                                gid = crgetgid(cr);
                        }
@@ -1805,7 +1803,7 @@ zfs_acl_ids_create(znode_t *dzp, int flag, vattr_t *vap, cred_t *cr,
         */
 
        if (!(flag & IS_ROOT_NODE) && (dzp->z_mode & S_ISGID) &&
-           (vap->va_type == VDIR)) {
+           (S_ISDIR(vap->va_mode))) {
                acl_ids->z_mode |= S_ISGID;
        } else {
                if ((acl_ids->z_mode & S_ISGID) &&
@@ -1816,13 +1814,13 @@ zfs_acl_ids_create(znode_t *dzp, int flag, vattr_t *vap, cred_t *cr,
        if (acl_ids->z_aclp == NULL) {
                mutex_enter(&dzp->z_acl_lock);
                mutex_enter(&dzp->z_lock);
-               if (!(flag & IS_ROOT_NODE) && (ZTOV(dzp)->v_type == VDIR &&
+               if (!(flag & IS_ROOT_NODE) && (S_ISDIR(ZTOI(dzp)->i_mode) &&
                    (dzp->z_pflags & ZFS_INHERIT_ACE)) &&
                    !(dzp->z_pflags & ZFS_XATTR)) {
                        VERIFY(0 == zfs_acl_node_read(dzp, B_TRUE,
                            &paclp, B_FALSE));
-                       acl_ids->z_aclp = zfs_acl_inherit(zfsvfs,
-                           vap->va_type, paclp, acl_ids->z_mode, &need_chmod);
+                       acl_ids->z_aclp = zfs_acl_inherit(zsb,
+                           vap->va_mode, paclp, acl_ids->z_mode, &need_chmod);
                        inherited = B_TRUE;
                } else {
                        acl_ids->z_aclp =
@@ -1832,9 +1830,9 @@ zfs_acl_ids_create(znode_t *dzp, int flag, vattr_t *vap, cred_t *cr,
                mutex_exit(&dzp->z_lock);
                mutex_exit(&dzp->z_acl_lock);
                if (need_chmod) {
-                       acl_ids->z_aclp->z_hints |= (vap->va_type == VDIR) ?
+                       acl_ids->z_aclp->z_hints |= S_ISDIR(vap->va_mode) ?
                            ZFS_ACL_AUTO_INHERIT : 0;
-                       zfs_acl_chmod(zfsvfs, acl_ids->z_mode, acl_ids->z_aclp);
+                       zfs_acl_chmod(zsb, acl_ids->z_mode, acl_ids->z_aclp);
                }
        }
 
@@ -1864,10 +1862,10 @@ zfs_acl_ids_free(zfs_acl_ids_t *acl_ids)
 }
 
 boolean_t
-zfs_acl_ids_overquota(zfsvfs_t *zfsvfs, zfs_acl_ids_t *acl_ids)
+zfs_acl_ids_overquota(zfs_sb_t *zsb, zfs_acl_ids_t *acl_ids)
 {
-       return (zfs_fuid_overquota(zfsvfs, B_FALSE, acl_ids->z_fuid) ||
-           zfs_fuid_overquota(zfsvfs, B_TRUE, acl_ids->z_fgid));
+       return (zfs_fuid_overquota(zsb, B_FALSE, acl_ids->z_fuid) ||
+           zfs_fuid_overquota(zsb, B_TRUE, acl_ids->z_fgid));
 }
 
 /*
@@ -1939,7 +1937,7 @@ zfs_getacl(znode_t *zp, vsecattr_t *vsecp, boolean_t skipaclchk, cred_t *cr)
                vsecp->vsa_aclentsz = aclsz;
 
                if (aclp->z_version == ZFS_ACL_VERSION_FUID)
-                       zfs_copy_fuid_2_ace(zp->z_zfsvfs, aclp, cr,
+                       zfs_copy_fuid_2_ace(ZTOZSB(zp), aclp, cr,
                            vsecp->vsa_aclentp, !(mask & VSA_ACE_ALLTYPES));
                else {
                        zfs_acl_node_t *aclnode;
@@ -1971,7 +1969,7 @@ zfs_getacl(znode_t *zp, vsecattr_t *vsecp, boolean_t skipaclchk, cred_t *cr)
 }
 
 int
-zfs_vsec_2_aclp(zfsvfs_t *zfsvfs, vtype_t obj_type,
+zfs_vsec_2_aclp(zfs_sb_t *zsb, umode_t obj_mode,
     vsecattr_t *vsecp, cred_t *cr, zfs_fuid_info_t **fuidp, zfs_acl_t **zaclp)
 {
        zfs_acl_t *aclp;
@@ -1982,12 +1980,12 @@ zfs_vsec_2_aclp(zfsvfs_t *zfsvfs, vtype_t obj_type,
        if (vsecp->vsa_aclcnt > MAX_ACL_ENTRIES || vsecp->vsa_aclcnt <= 0)
                return (EINVAL);
 
-       aclp = zfs_acl_alloc(zfs_acl_version(zfsvfs->z_version));
+       aclp = zfs_acl_alloc(zfs_acl_version(zsb->z_version));
 
        aclp->z_hints = 0;
        aclnode = zfs_acl_node_alloc(aclcnt * sizeof (zfs_object_ace_t));
        if (aclp->z_version == ZFS_ACL_VERSION_INITIAL) {
-               if ((error = zfs_copy_ace_2_oldace(obj_type, aclp,
+               if ((error = zfs_copy_ace_2_oldace(obj_mode, aclp,
                    (ace_t *)vsecp->vsa_aclentp, aclnode->z_acldata,
                    aclcnt, &aclnode->z_size)) != 0) {
                        zfs_acl_free(aclp);
@@ -1995,7 +1993,7 @@ zfs_vsec_2_aclp(zfsvfs_t *zfsvfs, vtype_t obj_type,
                        return (error);
                }
        } else {
-               if ((error = zfs_copy_ace_2_fuid(zfsvfs, obj_type, aclp,
+               if ((error = zfs_copy_ace_2_fuid(zsb, obj_mode, aclp,
                    vsecp->vsa_aclentp, aclnode->z_acldata, aclcnt,
                    &aclnode->z_size, fuidp, cr)) != 0) {
                        zfs_acl_free(aclp);
@@ -2031,8 +2029,8 @@ zfs_vsec_2_aclp(zfsvfs_t *zfsvfs, vtype_t obj_type,
 int
 zfs_setacl(znode_t *zp, vsecattr_t *vsecp, boolean_t skipaclchk, cred_t *cr)
 {
-       zfsvfs_t        *zfsvfs = zp->z_zfsvfs;
-       zilog_t         *zilog = zfsvfs->z_log;
+       zfs_sb_t        *zsb = ZTOZSB(zp);
+       zilog_t         *zilog = zsb->z_log;
        ulong_t         mask = vsecp->vsa_mask & (VSA_ACE | VSA_ACECNT);
        dmu_tx_t        *tx;
        int             error;
@@ -2050,7 +2048,7 @@ zfs_setacl(znode_t *zp, vsecattr_t *vsecp, boolean_t skipaclchk, cred_t *cr)
        if ((error = zfs_zaccess(zp, ACE_WRITE_ACL, 0, skipaclchk, cr)))
                return (error);
 
-       error = zfs_vsec_2_aclp(zfsvfs, ZTOV(zp)->v_type, vsecp, cr, &fuidp,
+       error = zfs_vsec_2_aclp(zsb, ZTOI(zp)->i_mode, vsecp, cr, &fuidp,
            &aclp);
        if (error)
                return (error);
@@ -2067,13 +2065,13 @@ top:
        mutex_enter(&zp->z_acl_lock);
        mutex_enter(&zp->z_lock);
 
-       tx = dmu_tx_create(zfsvfs->z_os);
+       tx = dmu_tx_create(zsb->z_os);
 
        dmu_tx_hold_sa(tx, zp->z_sa_hdl, B_TRUE);
 
-       fuid_dirtied = zfsvfs->z_fuid_dirty;
+       fuid_dirtied = zsb->z_fuid_dirty;
        if (fuid_dirtied)
-               zfs_fuid_txhold(zfsvfs, tx);
+               zfs_fuid_txhold(zsb, tx);
 
        /*
         * If old version and ACL won't fit in bonus and we aren't
@@ -2081,7 +2079,7 @@ top:
         */
 
        if ((acl_obj = zfs_external_acl(zp)) != 0) {
-               if (zfsvfs->z_version >= ZPL_VERSION_FUID &&
+               if (zsb->z_version >= ZPL_VERSION_FUID &&
                    zfs_znode_acl_version(zp) <= ZFS_ACL_VERSION_INITIAL) {
                        dmu_tx_hold_free(tx, acl_obj, 0,
                            DMU_OBJECT_END);
@@ -2116,7 +2114,7 @@ top:
        zp->z_acl_cached = aclp;
 
        if (fuid_dirtied)
-               zfs_fuid_sync(zfsvfs, tx);
+               zfs_fuid_sync(zsb, tx);
 
        zfs_log_acl(zilog, tx, zp, vsecp, fuidp);
 
@@ -2138,10 +2136,9 @@ top:
 static int
 zfs_zaccess_dataset_check(znode_t *zp, uint32_t v4_mode)
 {
-       if ((v4_mode & WRITE_MASK) &&
-           (zp->z_zfsvfs->z_vfs->vfs_flag & VFS_RDONLY) &&
-           (!IS_DEVVP(ZTOV(zp)) ||
-           (IS_DEVVP(ZTOV(zp)) && (v4_mode & WRITE_MASK_ATTRS)))) {
+       if ((v4_mode & WRITE_MASK) && (zfs_is_readonly(ZTOZSB(zp))) &&
+           (!S_ISDEV(ZTOI(zp)->i_mode) ||
+           (S_ISDEV(ZTOI(zp)->i_mode) && (v4_mode & WRITE_MASK_ATTRS)))) {
                return (EROFS);
        }
 
@@ -2149,9 +2146,9 @@ zfs_zaccess_dataset_check(znode_t *zp, uint32_t v4_mode)
         * Only check for READONLY on non-directories.
         */
        if ((v4_mode & WRITE_MASK_DATA) &&
-           (((ZTOV(zp)->v_type != VDIR) &&
+           ((!S_ISDIR(ZTOI(zp)->i_mode) &&
            (zp->z_pflags & (ZFS_READONLY | ZFS_IMMUTABLE))) ||
-           (ZTOV(zp)->v_type == VDIR &&
+           (S_ISDIR(ZTOI(zp)->i_mode) &&
            (zp->z_pflags & ZFS_IMMUTABLE)))) {
                return (EPERM);
        }
@@ -2198,11 +2195,11 @@ static int
 zfs_zaccess_aces_check(znode_t *zp, uint32_t *working_mode,
     boolean_t anyaccess, cred_t *cr)
 {
-       zfsvfs_t        *zfsvfs = zp->z_zfsvfs;
+       zfs_sb_t        *zsb = ZTOZSB(zp);
        zfs_acl_t       *aclp;
        int             error;
        uid_t           uid = crgetuid(cr);
-       uint64_t        who;
+       uint64_t        who;
        uint16_t        type, iflags;
        uint16_t        entry_type;
        uint32_t        access_mask;
@@ -2231,7 +2228,8 @@ zfs_zaccess_aces_check(znode_t *zp, uint32_t *working_mode,
                if (!zfs_acl_valid_ace_type(type, iflags))
                        continue;
 
-               if (ZTOV(zp)->v_type == VDIR && (iflags & ACE_INHERIT_ONLY_ACE))
+               if (S_ISDIR(ZTOI(zp)->i_mode) &&
+                   (iflags & ACE_INHERIT_ONLY_ACE))
                        continue;
 
                /* Skip ACE if it does not affect any AoI */
@@ -2252,7 +2250,7 @@ zfs_zaccess_aces_check(znode_t *zp, uint32_t *working_mode,
                        who = gowner;
                        /*FALLTHROUGH*/
                case ACE_IDENTIFIER_GROUP:
-                       checkit = zfs_groupmember(zfsvfs, who, cr);
+                       checkit = zfs_groupmember(zsb, who, cr);
                        break;
                case ACE_EVERYONE:
                        checkit = B_TRUE;
@@ -2263,7 +2261,7 @@ zfs_zaccess_aces_check(znode_t *zp, uint32_t *working_mode,
                        if (entry_type == 0) {
                                uid_t newid;
 
-                               newid = zfs_fuid_map_id(zfsvfs, who, cr,
+                               newid = zfs_fuid_map_id(zsb, who, cr,
                                    ZFS_ACE_USER);
                                if (newid != IDMAP_WK_CREATOR_OWNER_UID &&
                                    uid == newid)
@@ -2325,8 +2323,8 @@ zfs_has_access(znode_t *zp, cred_t *cr)
        if (zfs_zaccess_aces_check(zp, &have, B_TRUE, cr) != 0) {
                uid_t owner;
 
-               owner = zfs_fuid_map_id(zp->z_zfsvfs, zp->z_uid, cr, ZFS_OWNER);
-               return (secpolicy_vnode_any_access(cr, ZTOV(zp), owner) == 0);
+               owner = zfs_fuid_map_id(ZTOZSB(zp), zp->z_uid, cr, ZFS_OWNER);
+               return (secpolicy_vnode_any_access(cr, ZTOI(zp), owner) == 0);
        }
        return (B_TRUE);
 }
@@ -2335,7 +2333,7 @@ static int
 zfs_zaccess_common(znode_t *zp, uint32_t v4_mode, uint32_t *working_mode,
     boolean_t *check_privs, boolean_t skipaclchk, cred_t *cr)
 {
-       zfsvfs_t *zfsvfs = zp->z_zfsvfs;
+       zfs_sb_t *zsb = ZTOZSB(zp);
        int err;
 
        *working_mode = v4_mode;
@@ -2344,7 +2342,7 @@ zfs_zaccess_common(znode_t *zp, uint32_t v4_mode, uint32_t *working_mode,
        /*
         * Short circuit empty requests
         */
-       if (v4_mode == 0 || zfsvfs->z_replay) {
+       if (v4_mode == 0 || zsb->z_replay) {
                *working_mode = 0;
                return (0);
        }
@@ -2391,7 +2389,7 @@ zfs_fastaccesschk_execute(znode_t *zdp, cred_t *cr)
                return (EACCES);
 
        is_attr = ((zdp->z_pflags & ZFS_XATTR) &&
-           (ZTOV(zdp)->v_type == VDIR));
+           (S_ISDIR(ZTOI(zdp)->i_mode)));
        if (is_attr)
                goto slow;
 
@@ -2439,9 +2437,9 @@ zfs_fastaccesschk_execute(znode_t *zdp, cred_t *cr)
 
 slow:
        DTRACE_PROBE(zfs__fastpath__execute__access__miss);
-       ZFS_ENTER(zdp->z_zfsvfs);
+       ZFS_ENTER(ZTOZSB(zdp));
        error = zfs_zaccess(zdp, ACE_EXECUTE, 0, B_FALSE, cr);
-       ZFS_EXIT(zdp->z_zfsvfs);
+       ZFS_EXIT(ZTOZSB(zdp));
        return (error);
 }
 
@@ -2455,32 +2453,52 @@ zfs_zaccess(znode_t *zp, int mode, int flags, boolean_t skipaclchk, cred_t *cr)
 {
        uint32_t        working_mode;
        int             error;
-       int             is_attr;
-       boolean_t       check_privs;
-       znode_t         *xzp;
-       znode_t         *check_zp = zp;
+       boolean_t       check_privs;
+       znode_t         *check_zp = zp;
        mode_t          needed_bits;
        uid_t           owner;
 
-       is_attr = ((zp->z_pflags & ZFS_XATTR) && (ZTOV(zp)->v_type == VDIR));
-
        /*
         * If attribute then validate against base file
         */
-       if (is_attr) {
+       if ((zp->z_pflags & ZFS_XATTR) && S_ISDIR(ZTOI(zp)->i_mode)) {
                uint64_t        parent;
 
-               if ((error = sa_lookup(zp->z_sa_hdl,
-                   SA_ZPL_PARENT(zp->z_zfsvfs), &parent,
-                   sizeof (parent))) != 0)
-                       return (error);
+               rw_enter(&zp->z_xattr_lock, RW_READER);
+               if (zp->z_xattr_parent) {
+                       check_zp = zp->z_xattr_parent;
+                       rw_exit(&zp->z_xattr_lock);
 
-               if ((error = zfs_zget(zp->z_zfsvfs,
-                   parent, &xzp)) != 0)        {
-                       return (error);
-               }
+                       /*
+                        * Verify a lookup yields the same znode.
+                        */
+                       ASSERT3S(sa_lookup(zp->z_sa_hdl, SA_ZPL_PARENT(
+                           ZTOZSB(zp)), &parent, sizeof (parent)), ==, 0);
+                       ASSERT3U(check_zp->z_id, ==, parent);
+               } else {
+                       rw_exit(&zp->z_xattr_lock);
+
+                       error = sa_lookup(zp->z_sa_hdl, SA_ZPL_PARENT(
+                           ZTOZSB(zp)), &parent, sizeof (parent));
+                       if (error)
+                               return (error);
 
-               check_zp = xzp;
+                       /*
+                        * Cache the lookup on the parent file znode as
+                        * zp->z_xattr_parent and hold a reference.  This
+                        * effectively pins the parent in memory until all
+                        * child xattr znodes have been destroyed and
+                        * release their references in zfs_inode_destroy().
+                        */
+                       error = zfs_zget(ZTOZSB(zp), parent, &check_zp);
+                       if (error)
+                               return (error);
+
+                       rw_enter(&zp->z_xattr_lock, RW_WRITER);
+                       if (zp->z_xattr_parent == NULL)
+                               zp->z_xattr_parent = check_zp;
+                       rw_exit(&zp->z_xattr_lock);
+               }
 
                /*
                 * fixup mode to map to xattr perms
@@ -2497,11 +2515,11 @@ zfs_zaccess(znode_t *zp, int mode, int flags, boolean_t skipaclchk, cred_t *cr)
                }
        }
 
-       owner = zfs_fuid_map_id(zp->z_zfsvfs, zp->z_uid, cr, ZFS_OWNER);
+       owner = zfs_fuid_map_id(ZTOZSB(zp), zp->z_uid, cr, ZFS_OWNER);
        /*
-        * Map the bits required to the standard vnode flags VREAD|VWRITE|VEXEC
-        * in needed_bits.  Map the bits mapped by working_mode (currently
-        * missing) in missing_bits.
+        * Map the bits required to the standard inode flags
+        * S_IRUSR|S_IWUSR|S_IXUSR in the needed_bits.  Map the bits
+        * mapped by working_mode (currently missing) in missing_bits.
         * Call secpolicy_vnode_access2() with (needed_bits & ~checkmode),
         * needed_bits.
         */
@@ -2514,24 +2532,20 @@ zfs_zaccess(znode_t *zp, int mode, int flags, boolean_t skipaclchk, cred_t *cr)
 
        if (working_mode & (ACE_READ_DATA|ACE_READ_NAMED_ATTRS|
            ACE_READ_ACL|ACE_READ_ATTRIBUTES|ACE_SYNCHRONIZE))
-               needed_bits |= VREAD;
+               needed_bits |= S_IRUSR;
        if (working_mode & (ACE_WRITE_DATA|ACE_WRITE_NAMED_ATTRS|
            ACE_APPEND_DATA|ACE_WRITE_ATTRIBUTES|ACE_SYNCHRONIZE))
-               needed_bits |= VWRITE;
+               needed_bits |= S_IWUSR;
        if (working_mode & ACE_EXECUTE)
-               needed_bits |= VEXEC;
+               needed_bits |= S_IXUSR;
 
        if ((error = zfs_zaccess_common(check_zp, mode, &working_mode,
            &check_privs, skipaclchk, cr)) == 0) {
-               if (is_attr)
-                       VN_RELE(ZTOV(xzp));
-               return (secpolicy_vnode_access2(cr, ZTOV(zp), owner,
+               return (secpolicy_vnode_access2(cr, ZTOI(zp), owner,
                    needed_bits, needed_bits));
        }
 
        if (error && !check_privs) {
-               if (is_attr)
-                       VN_RELE(ZTOV(xzp));
                return (error);
        }
 
@@ -2556,14 +2570,14 @@ zfs_zaccess(znode_t *zp, int mode, int flags, boolean_t skipaclchk, cred_t *cr)
 
                if (working_mode & (ACE_READ_DATA|ACE_READ_NAMED_ATTRS|
                    ACE_READ_ACL|ACE_READ_ATTRIBUTES|ACE_SYNCHRONIZE))
-                       checkmode |= VREAD;
+                       checkmode |= S_IRUSR;
                if (working_mode & (ACE_WRITE_DATA|ACE_WRITE_NAMED_ATTRS|
                    ACE_APPEND_DATA|ACE_WRITE_ATTRIBUTES|ACE_SYNCHRONIZE))
-                       checkmode |= VWRITE;
+                       checkmode |= S_IWUSR;
                if (working_mode & ACE_EXECUTE)
-                       checkmode |= VEXEC;
+                       checkmode |= S_IXUSR;
 
-               error = secpolicy_vnode_access2(cr, ZTOV(check_zp), owner,
+               error = secpolicy_vnode_access2(cr, ZTOI(check_zp), owner,
                    needed_bits & ~checkmode, needed_bits);
 
                if (error == 0 && (working_mode & ACE_WRITE_OWNER))
@@ -2588,19 +2602,15 @@ zfs_zaccess(znode_t *zp, int mode, int flags, boolean_t skipaclchk, cred_t *cr)
                        }
                }
        } else if (error == 0) {
-               error = secpolicy_vnode_access2(cr, ZTOV(zp), owner,
+               error = secpolicy_vnode_access2(cr, ZTOI(zp), owner,
                    needed_bits, needed_bits);
        }
 
-
-       if (is_attr)
-               VN_RELE(ZTOV(xzp));
-
        return (error);
 }
 
 /*
- * Translate traditional unix VREAD/VWRITE/VEXEC mode into
+ * Translate traditional unix S_IRUSR/S_IWUSR/S_IXUSR mode into
  * native ACL format and call zfs_zaccess()
  */
 int
@@ -2627,10 +2637,10 @@ zfs_delete_final_check(znode_t *zp, znode_t *dzp,
        int error;
        uid_t downer;
 
-       downer = zfs_fuid_map_id(dzp->z_zfsvfs, dzp->z_uid, cr, ZFS_OWNER);
+       downer = zfs_fuid_map_id(ZTOZSB(dzp), dzp->z_uid, cr, ZFS_OWNER);
 
-       error = secpolicy_vnode_access2(cr, ZTOV(dzp),
-           downer, available_perms, VWRITE|VEXEC);
+       error = secpolicy_vnode_access2(cr, ZTOI(dzp),
+           downer, available_perms, S_IWUSR|S_IXUSR);
 
        if (error == 0)
                error = zfs_sticky_remove_access(dzp, zp, cr);
@@ -2750,8 +2760,8 @@ zfs_zaccess_delete(znode_t *dzp, znode_t *zp, cred_t *cr)
         * Fourth row
         */
 
-       available_perms = (dzp_working_mode & ACE_WRITE_DATA) ? 0 : VWRITE;
-       available_perms |= (dzp_working_mode & ACE_EXECUTE) ? 0 : VEXEC;
+       available_perms = (dzp_working_mode & ACE_WRITE_DATA) ? 0 : S_IWUSR;
+       available_perms |= (dzp_working_mode & ACE_EXECUTE) ? 0 : S_IXUSR;
 
        return (zfs_delete_final_check(zp, dzp, available_perms, cr));
 
@@ -2767,7 +2777,7 @@ zfs_zaccess_rename(znode_t *sdzp, znode_t *szp, znode_t *tdzp,
        if (szp->z_pflags & ZFS_AV_QUARANTINED)
                return (EACCES);
 
-       add_perm = (ZTOV(szp)->v_type == VDIR) ?
+       add_perm = S_ISDIR(ZTOI(szp)->i_mode) ?
            ACE_ADD_SUBDIRECTORY : ACE_ADD_FILE;
 
        /*