Fix nfs snapdir automount

The current implementation for allowing nfs to access snapdir is very buggy.
It uses a special fh for snapdirs, such that the next time nfsd does
fh_to_dentry, it actually returns the root inode inside the snapshot. So nfsd
never knows it cross a mountpoint.

The problem is that nfsd will not hold a reference on the vfsmount of the
snapshot. This cause auto unmounter to unmount the snapshot even though nfs is
still holding dentries in it.

To fix this, we return the inode for the snapdirs themselves. However, we also
trigger automount upon fh_to_dentry, and return ESTALE so nfsd will revalidate
and see the mountpoint and do crossmnt.

Because nfsd will now be aware that these are different filesystems users
must add crossmnt to their export options to access snapshot directories.

Reviewed-by: Brian Behlendorf <behlendorf1@llnl.gov>
Signed-off-by: Chunwei Chen <david.chen@osnexus.com>
Closes #3794
Closes #4716
Closes #5810 
Closes #5833
This commit is contained in:
Chunwei Chen 2017-03-08 09:26:33 -08:00 committed by Brian Behlendorf
parent 463009865f
commit 9b77d1c958
6 changed files with 94 additions and 107 deletions

View File

@ -78,8 +78,8 @@ extern int zfsctl_snapshot_mount(struct path *path, int flags);
extern int zfsctl_snapshot_unmount(char *snapname, int flags); extern int zfsctl_snapshot_unmount(char *snapname, int flags);
extern int zfsctl_snapshot_unmount_delay(spa_t *spa, uint64_t objsetid, extern int zfsctl_snapshot_unmount_delay(spa_t *spa, uint64_t objsetid,
int delay); int delay);
extern int zfsctl_lookup_objset(struct super_block *sb, uint64_t objsetid, extern int zfsctl_snapdir_vget(struct super_block *sb, uint64_t objsetid,
zfs_sb_t **zsb); int gen, struct inode **ipp);
/* zfsctl '.zfs/shares' functions */ /* zfsctl '.zfs/shares' functions */
extern int zfsctl_shares_lookup(struct inode *dip, char *name, extern int zfsctl_shares_lookup(struct inode *dip, char *name,

View File

@ -600,7 +600,7 @@ nfs_update_shareopts(sa_share_impl_t impl_share, const char *resource,
old_shareopts = FSINFO(impl_share, nfs_fstype)->shareopts; old_shareopts = FSINFO(impl_share, nfs_fstype)->shareopts;
if (strcmp(shareopts, "on") == 0) if (strcmp(shareopts, "on") == 0)
shareopts = "rw"; shareopts = "rw,crossmnt";
if (FSINFO(impl_share, nfs_fstype)->active && old_shareopts != NULL && if (FSINFO(impl_share, nfs_fstype)->active && old_shareopts != NULL &&
strcmp(old_shareopts, shareopts) != 0) { strcmp(old_shareopts, shareopts) != 0) {

View File

@ -1340,7 +1340,7 @@ Controls whether the file system is shared via \fBNFS\fR, and what options are u
.sp .sp
.in +4 .in +4
.nf .nf
/usr/sbin/exportfs -i -o sec=sys,rw,no_subtree_check,no_root_squash,mountpoint *:<mountpoint of dataset> /usr/sbin/exportfs -i -o sec=sys,rw,crossmnt,no_subtree_check,no_root_squash,mountpoint *:<mountpoint of dataset>
.fi .fi
.in -4 .in -4
.sp .sp
@ -3746,6 +3746,10 @@ The following commands show how to set \fBsharenfs\fR property options to enable
.LP .LP
If you are using \fBDNS\fR for host name resolution, specify the fully qualified hostname. If you are using \fBDNS\fR for host name resolution, specify the fully qualified hostname.
.sp
.LP
If you want to access snapdir through NFS, be sure to add \fBcrossmnt\fR to the options.
.LP .LP
\fBExample 17 \fRDelegating ZFS Administration Permissions on a ZFS Dataset \fBExample 17 \fRDelegating ZFS Administration Permissions on a ZFS Dataset
.sp .sp

View File

@ -590,27 +590,40 @@ zfsctl_root(znode_t *zp)
igrab(ZTOZSB(zp)->z_ctldir); igrab(ZTOZSB(zp)->z_ctldir);
return (ZTOZSB(zp)->z_ctldir); return (ZTOZSB(zp)->z_ctldir);
} }
/* /*
* Generate a long fid which includes the root object and objset of a * Generate a long fid to indicate a snapdir. We encode whether snapdir is
* snapshot but not the generation number. For the root object the * already monunted in gen field. We do this because nfsd lookup will not
* generation number is ignored when zero to avoid needing to open * trigger automount. Next time the nfsd does fh_to_dentry, we will notice
* the dataset when generating fids for the snapshot names. * this and do automount and return ESTALE to force nfsd revalidate and follow
* mount.
*/ */
static int static int
zfsctl_snapdir_fid(struct inode *ip, fid_t *fidp) zfsctl_snapdir_fid(struct inode *ip, fid_t *fidp)
{ {
zfs_sb_t *zsb = ITOZSB(ip);
zfid_short_t *zfid = (zfid_short_t *)fidp; zfid_short_t *zfid = (zfid_short_t *)fidp;
zfid_long_t *zlfid = (zfid_long_t *)fidp; zfid_long_t *zlfid = (zfid_long_t *)fidp;
uint32_t gen = 0; uint32_t gen = 0;
uint64_t object; uint64_t object;
uint64_t objsetid; uint64_t objsetid;
int i; int i;
struct dentry *dentry;
object = zsb->z_root; if (fidp->fid_len < LONG_FID_LEN) {
fidp->fid_len = LONG_FID_LEN;
return (SET_ERROR(ENOSPC));
}
object = ip->i_ino;
objsetid = ZFSCTL_INO_SNAPDIRS - ip->i_ino; objsetid = ZFSCTL_INO_SNAPDIRS - ip->i_ino;
zfid->zf_len = LONG_FID_LEN; zfid->zf_len = LONG_FID_LEN;
dentry = d_obtain_alias(igrab(ip));
if (!IS_ERR(dentry)) {
gen = !!d_mountpoint(dentry);
dput(dentry);
}
for (i = 0; i < sizeof (zfid->zf_object); i++) for (i = 0; i < sizeof (zfid->zf_object); i++)
zfid->zf_object[i] = (uint8_t)(object >> (8 * i)); zfid->zf_object[i] = (uint8_t)(object >> (8 * i));
@ -640,17 +653,17 @@ zfsctl_fid(struct inode *ip, fid_t *fidp)
ZFS_ENTER(zsb); ZFS_ENTER(zsb);
if (zfsctl_is_snapdir(ip)) {
ZFS_EXIT(zsb);
return (zfsctl_snapdir_fid(ip, fidp));
}
if (fidp->fid_len < SHORT_FID_LEN) { if (fidp->fid_len < SHORT_FID_LEN) {
fidp->fid_len = SHORT_FID_LEN; fidp->fid_len = SHORT_FID_LEN;
ZFS_EXIT(zsb); ZFS_EXIT(zsb);
return (SET_ERROR(ENOSPC)); return (SET_ERROR(ENOSPC));
} }
if (zfsctl_is_snapdir(ip)) {
ZFS_EXIT(zsb);
return (zfsctl_snapdir_fid(ip, fidp));
}
zfid = (zfid_short_t *)fidp; zfid = (zfid_short_t *)fidp;
zfid->zf_len = SHORT_FID_LEN; zfid->zf_len = SHORT_FID_LEN;
@ -1145,70 +1158,52 @@ error:
} }
/* /*
* Given the objset id of the snapshot return its zfs_sb_t as zsbp. * Get the snapdir inode from fid
*/ */
int int
zfsctl_lookup_objset(struct super_block *sb, uint64_t objsetid, zfs_sb_t **zsbp) zfsctl_snapdir_vget(struct super_block *sb, uint64_t objsetid, int gen,
struct inode **ipp)
{ {
zfs_snapentry_t *se;
int error; int error;
spa_t *spa = ((zfs_sb_t *)(sb->s_fs_info))->z_os->os_spa; struct path path;
char *mnt;
struct dentry *dentry;
mnt = kmem_alloc(MAXPATHLEN, KM_SLEEP);
error = zfsctl_snapshot_path_objset(sb->s_fs_info, objsetid,
MAXPATHLEN, mnt);
if (error)
goto out;
/* Trigger automount */
error = kern_path(mnt, LOOKUP_FOLLOW|LOOKUP_DIRECTORY, &path);
if (error)
goto out;
path_put(&path);
/* /*
* Verify that the snapshot is mounted then lookup the mounted root * Get the snapdir inode. Note, we don't want to use the above
* rather than the covered mount point. This may fail if the * path because it contains the root of the snapshot rather
* snapshot has just been unmounted by an unrelated user space * than the snapdir.
* process. This race cannot occur to an expired mount point
* because we hold the zfs_snapshot_lock to prevent the race.
*/ */
rw_enter(&zfs_snapshot_lock, RW_READER); *ipp = ilookup(sb, ZFSCTL_INO_SNAPDIRS - objsetid);
if ((se = zfsctl_snapshot_find_by_objsetid(spa, objsetid)) != NULL) { if (*ipp == NULL) {
zfs_sb_t *zsb; error = SET_ERROR(ENOENT);
goto out;
}
zsb = ITOZSB(se->se_root_dentry->d_inode); /* check gen, see zfsctl_snapdir_fid */
ASSERT3U(dmu_objset_id(zsb->z_os), ==, objsetid); dentry = d_obtain_alias(igrab(*ipp));
if (gen != (!IS_ERR(dentry) && d_mountpoint(dentry))) {
if (time_after(jiffies, zsb->z_snap_defer_time + iput(*ipp);
MAX(zfs_expire_snapshot * HZ / 2, HZ))) { *ipp = NULL;
zsb->z_snap_defer_time = jiffies;
zfsctl_snapshot_unmount_cancel(se);
zfsctl_snapshot_unmount_delay_impl(se,
zfs_expire_snapshot);
}
*zsbp = zsb;
zfsctl_snapshot_rele(se);
error = SET_ERROR(0);
} else {
error = SET_ERROR(ENOENT); error = SET_ERROR(ENOENT);
} }
rw_exit(&zfs_snapshot_lock); if (!IS_ERR(dentry))
dput(dentry);
/* out:
* Automount the snapshot given the objset id by constructing the kmem_free(mnt, MAXPATHLEN);
* full mount point and performing a traversal.
*/
if (error == ENOENT) {
struct path path;
char *mnt;
mnt = kmem_alloc(MAXPATHLEN, KM_SLEEP);
error = zfsctl_snapshot_path_objset(sb->s_fs_info, objsetid,
MAXPATHLEN, mnt);
if (error) {
kmem_free(mnt, MAXPATHLEN);
return (SET_ERROR(error));
}
error = kern_path(mnt, LOOKUP_FOLLOW|LOOKUP_DIRECTORY, &path);
if (error == 0) {
*zsbp = ITOZSB(path.dentry->d_inode);
path_put(&path);
}
kmem_free(mnt, MAXPATHLEN);
}
return (error); return (error);
} }

View File

@ -1597,8 +1597,19 @@ zfs_vget(struct super_block *sb, struct inode **ipp, fid_t *fidp)
*ipp = NULL; *ipp = NULL;
ZFS_ENTER(zsb); if (fidp->fid_len == SHORT_FID_LEN || fidp->fid_len == LONG_FID_LEN) {
zfid_short_t *zfid = (zfid_short_t *)fidp;
for (i = 0; i < sizeof (zfid->zf_object); i++)
object |= ((uint64_t)zfid->zf_object[i]) << (8 * i);
for (i = 0; i < sizeof (zfid->zf_gen); i++)
fid_gen |= ((uint64_t)zfid->zf_gen[i]) << (8 * i);
} else {
return (SET_ERROR(EINVAL));
}
/* LONG_FID_LEN means snapdirs */
if (fidp->fid_len == LONG_FID_LEN) { if (fidp->fid_len == LONG_FID_LEN) {
zfid_long_t *zlfid = (zfid_long_t *)fidp; zfid_long_t *zlfid = (zfid_long_t *)fidp;
uint64_t objsetid = 0; uint64_t objsetid = 0;
@ -1610,28 +1621,24 @@ zfs_vget(struct super_block *sb, struct inode **ipp, fid_t *fidp)
for (i = 0; i < sizeof (zlfid->zf_setgen); i++) for (i = 0; i < sizeof (zlfid->zf_setgen); i++)
setgen |= ((uint64_t)zlfid->zf_setgen[i]) << (8 * i); setgen |= ((uint64_t)zlfid->zf_setgen[i]) << (8 * i);
ZFS_EXIT(zsb); if (objsetid != ZFSCTL_INO_SNAPDIRS - object) {
dprintf("snapdir fid: objsetid (%llu) != "
"ZFSCTL_INO_SNAPDIRS (%llu) - object (%llu)\n",
objsetid, ZFSCTL_INO_SNAPDIRS, object);
err = zfsctl_lookup_objset(sb, objsetid, &zsb);
if (err)
return (SET_ERROR(EINVAL)); return (SET_ERROR(EINVAL));
}
ZFS_ENTER(zsb); if (fid_gen > 1 || setgen != 0) {
} dprintf("snapdir fid: fid_gen (%llu) and setgen "
"(%llu)\n", fid_gen, setgen);
if (fidp->fid_len == SHORT_FID_LEN || fidp->fid_len == LONG_FID_LEN) { return (SET_ERROR(EINVAL));
zfid_short_t *zfid = (zfid_short_t *)fidp; }
for (i = 0; i < sizeof (zfid->zf_object); i++) return (zfsctl_snapdir_vget(sb, objsetid, fid_gen, ipp));
object |= ((uint64_t)zfid->zf_object[i]) << (8 * i);
for (i = 0; i < sizeof (zfid->zf_gen); i++)
fid_gen |= ((uint64_t)zfid->zf_gen[i]) << (8 * i);
} else {
ZFS_EXIT(zsb);
return (SET_ERROR(EINVAL));
} }
ZFS_ENTER(zsb);
/* A zero fid_gen means we are in the .zfs control directories */ /* A zero fid_gen means we are in the .zfs control directories */
if (fid_gen == 0 && if (fid_gen == 0 &&
(object == ZFSCTL_INO_ROOT || object == ZFSCTL_INO_SNAPDIR)) { (object == ZFSCTL_INO_ROOT || object == ZFSCTL_INO_SNAPDIR)) {

View File

@ -4714,12 +4714,7 @@ zfs_fid(struct inode *ip, fid_t *fidp)
gen = (uint32_t)gen64; gen = (uint32_t)gen64;
size = (zsb->z_parent != zsb) ? LONG_FID_LEN : SHORT_FID_LEN; size = SHORT_FID_LEN;
if (fidp->fid_len < size) {
fidp->fid_len = size;
ZFS_EXIT(zsb);
return (SET_ERROR(ENOSPC));
}
zfid = (zfid_short_t *)fidp; zfid = (zfid_short_t *)fidp;
@ -4734,20 +4729,6 @@ zfs_fid(struct inode *ip, fid_t *fidp)
for (i = 0; i < sizeof (zfid->zf_gen); i++) for (i = 0; i < sizeof (zfid->zf_gen); i++)
zfid->zf_gen[i] = (uint8_t)(gen >> (8 * i)); zfid->zf_gen[i] = (uint8_t)(gen >> (8 * i));
if (size == LONG_FID_LEN) {
uint64_t objsetid = dmu_objset_id(zsb->z_os);
zfid_long_t *zlfid;
zlfid = (zfid_long_t *)fidp;
for (i = 0; i < sizeof (zlfid->zf_setid); i++)
zlfid->zf_setid[i] = (uint8_t)(objsetid >> (8 * i));
/* XXX - this should be the generation number for the objset */
for (i = 0; i < sizeof (zlfid->zf_setgen); i++)
zlfid->zf_setgen[i] = 0;
}
ZFS_EXIT(zsb); ZFS_EXIT(zsb);
return (0); return (0);
} }