zfs: depessimize zfs_root with rmlocks

Currently vfs calls the root method on each absolute lookup and when
crossing mount points.

zfs_root ends up looking up the inode internally as if it was not
instantianted which results in significant lock contention on systems
like EPYC.

Store the vnode in the mount point and protect the access with rmlocks.
This is a temporary hack for 12.0.

Sample result:

before:
make -s -j 128 buildkernel 2778.09s user 3319.45s system 8370% cpu 1:12.85 total

after:
make -s -j 128 buildkernel 3199.57s user 1772.78s system 8232% cpu 1:00.40 total

Tested by:	pho (zfs mount/unmount tests)
Reviewed by:	kib, mav, sef (different parts)
Approved by:	re (gjb)
Differential Revision:	https://reviews.freebsd.org/D17233
This commit is contained in:
Mateusz Guzik 2018-09-25 17:58:06 +00:00
parent a78849183c
commit af534f8d99
Notes: svn2git 2020-12-20 02:59:44 +00:00
svn path=/head/; revision=338927
2 changed files with 68 additions and 2 deletions

View File

@ -46,6 +46,8 @@ struct zfsvfs {
zfsvfs_t *z_parent; /* parent fs */
objset_t *z_os; /* objset reference */
uint64_t z_root; /* id of root znode */
struct vnode *z_rootvnode; /* root vnode */
struct rmlock z_rootvnodelock;/* protection for root vnode */
uint64_t z_unlinkedobj; /* id of unlinked zapobj */
uint64_t z_max_blksz; /* maximum block size for files */
uint64_t z_fuid_obj; /* fuid table object number */

View File

@ -65,6 +65,7 @@
#include <sys/spa_boot.h>
#include <sys/jail.h>
#include <ufs/ufs/quota.h>
#include <sys/rmlock.h>
#include "zfs_comutil.h"
@ -92,6 +93,9 @@ static int zfs_version_zpl = ZPL_VERSION;
SYSCTL_INT(_vfs_zfs_version, OID_AUTO, zpl, CTLFLAG_RD, &zfs_version_zpl, 0,
"ZPL_VERSION");
static int zfs_root_setvnode(zfsvfs_t *zfsvfs);
static void zfs_root_dropvnode(zfsvfs_t *zfsvfs);
static int zfs_quotactl(vfs_t *vfsp, int cmds, uid_t id, void *arg);
static int zfs_mount(vfs_t *vfsp);
static int zfs_umount(vfs_t *vfsp, int fflag);
@ -1209,6 +1213,8 @@ zfsvfs_create_impl(zfsvfs_t **zfvp, zfsvfs_t *zfsvfs, objset_t *os)
for (int i = 0; i != ZFS_OBJ_MTX_SZ; i++)
mutex_init(&zfsvfs->z_hold_mtx[i], NULL, MUTEX_DEFAULT, NULL);
rm_init(&zfsvfs->z_rootvnodelock, "zfs root vnode lock");
error = zfsvfs_init(zfsvfs, os);
if (error != 0) {
*zfvp = NULL;
@ -1315,6 +1321,8 @@ zfsvfs_free(zfsvfs_t *zfsvfs)
rw_enter(&zfsvfs_lock, RW_READER);
rw_exit(&zfsvfs_lock);
rm_destroy(&zfsvfs->z_rootvnodelock);
zfs_fuid_destroy(zfsvfs);
mutex_destroy(&zfsvfs->z_znodes_lock);
@ -1921,6 +1929,8 @@ zfs_mount(vfs_t *vfsp)
error = zfs_domount(vfsp, osname);
PICKUP_GIANT();
zfs_root_setvnode((zfsvfs_t *)vfsp->vfs_data);
#ifdef illumos
/*
* Add an extra VFS_HOLD on our parent vfs so that it can't
@ -1993,14 +2003,65 @@ zfs_statfs(vfs_t *vfsp, struct statfs *statp)
}
static int
zfs_root(vfs_t *vfsp, int flags, vnode_t **vpp)
zfs_root_setvnode(zfsvfs_t *zfsvfs)
{
zfsvfs_t *zfsvfs = vfsp->vfs_data;
znode_t *rootzp;
int error;
ZFS_ENTER(zfsvfs);
error = zfs_zget(zfsvfs, zfsvfs->z_root, &rootzp);
if (error != 0)
panic("could not zfs_zget for root vnode");
ZFS_EXIT(zfsvfs);
rm_wlock(&zfsvfs->z_rootvnodelock);
if (zfsvfs->z_rootvnode != NULL)
panic("zfs mount point already has a root vnode: %p\n",
zfsvfs->z_rootvnode);
zfsvfs->z_rootvnode = ZTOV(rootzp);
rm_wunlock(&zfsvfs->z_rootvnodelock);
return (0);
}
static void
zfs_root_putvnode(zfsvfs_t *zfsvfs)
{
struct vnode *vp;
rm_wlock(&zfsvfs->z_rootvnodelock);
vp = zfsvfs->z_rootvnode;
zfsvfs->z_rootvnode = NULL;
rm_wunlock(&zfsvfs->z_rootvnodelock);
if (vp != NULL)
vrele(vp);
}
static int
zfs_root(vfs_t *vfsp, int flags, vnode_t **vpp)
{
struct rm_priotracker tracker;
zfsvfs_t *zfsvfs = vfsp->vfs_data;
znode_t *rootzp;
int error;
rm_rlock(&zfsvfs->z_rootvnodelock, &tracker);
*vpp = zfsvfs->z_rootvnode;
if (*vpp != NULL && (((*vpp)->v_iflag & VI_DOOMED) == 0)) {
vrefact(*vpp);
rm_runlock(&zfsvfs->z_rootvnodelock, &tracker);
goto lock;
}
rm_runlock(&zfsvfs->z_rootvnodelock, &tracker);
/*
* We found the vnode but did not like it.
*/
if (*vpp != NULL) {
*vpp = NULL;
zfs_root_putvnode(zfsvfs);
}
ZFS_ENTER(zfsvfs);
error = zfs_zget(zfsvfs, zfsvfs->z_root, &rootzp);
if (error == 0)
*vpp = ZTOV(rootzp);
@ -2008,6 +2069,7 @@ zfs_root(vfs_t *vfsp, int flags, vnode_t **vpp)
ZFS_EXIT(zfsvfs);
if (error == 0) {
lock:
error = vn_lock(*vpp, flags);
if (error != 0) {
VN_RELE(*vpp);
@ -2126,6 +2188,8 @@ zfs_umount(vfs_t *vfsp, int fflag)
cred_t *cr = td->td_ucred;
int ret;
zfs_root_putvnode(zfsvfs);
ret = secpolicy_fs_unmount(cr, vfsp);
if (ret) {
if (dsl_deleg_access((char *)refstr_value(vfsp->vfs_resource),