Commit 4346cdd4 authored by Christoph Hellwig's avatar Christoph Hellwig Committed by Christoph Hellwig

xfs: cleanup xfs_find_handle

Remove the superflous igrab by keeping a reference on the path/file all the
time and clean up various bits of surrounding code.
Signed-off-by: default avatarChristoph Hellwig <hch@lst.de>
Reviewed-by: default avatarFelix Blyakher <felixb@sgi.com>
parent ef8f7fc5
...@@ -78,92 +78,74 @@ xfs_find_handle( ...@@ -78,92 +78,74 @@ xfs_find_handle(
int hsize; int hsize;
xfs_handle_t handle; xfs_handle_t handle;
struct inode *inode; struct inode *inode;
struct file *file = NULL;
struct path path;
int error;
struct xfs_inode *ip;
memset((char *)&handle, 0, sizeof(handle)); if (cmd == XFS_IOC_FD_TO_HANDLE) {
file = fget(hreq->fd);
switch (cmd) { if (!file)
case XFS_IOC_PATH_TO_FSHANDLE: return -EBADF;
case XFS_IOC_PATH_TO_HANDLE: { inode = file->f_path.dentry->d_inode;
struct path path; } else {
int error = user_lpath((const char __user *)hreq->path, &path); error = user_lpath((const char __user *)hreq->path, &path);
if (error) if (error)
return error; return error;
inode = path.dentry->d_inode;
ASSERT(path.dentry);
ASSERT(path.dentry->d_inode);
inode = igrab(path.dentry->d_inode);
path_put(&path);
break;
} }
ip = XFS_I(inode);
case XFS_IOC_FD_TO_HANDLE: { /*
struct file *file; * We can only generate handles for inodes residing on a XFS filesystem,
* and only for regular files, directories or symbolic links.
file = fget(hreq->fd); */
if (!file) error = -EINVAL;
return -EBADF; if (inode->i_sb->s_magic != XFS_SB_MAGIC)
goto out_put;
ASSERT(file->f_path.dentry); error = -EBADF;
ASSERT(file->f_path.dentry->d_inode); if (!S_ISREG(inode->i_mode) &&
inode = igrab(file->f_path.dentry->d_inode); !S_ISDIR(inode->i_mode) &&
fput(file); !S_ISLNK(inode->i_mode))
break; goto out_put;
}
default:
ASSERT(0);
return -XFS_ERROR(EINVAL);
}
if (inode->i_sb->s_magic != XFS_SB_MAGIC) { memcpy(&handle.ha_fsid, ip->i_mount->m_fixedfsid, sizeof(xfs_fsid_t));
/* we're not in XFS anymore, Toto */
iput(inode);
return -XFS_ERROR(EINVAL);
}
switch (inode->i_mode & S_IFMT) { if (cmd == XFS_IOC_PATH_TO_FSHANDLE) {
case S_IFREG: /*
case S_IFDIR: * This handle only contains an fsid, zero the rest.
case S_IFLNK: */
break; memset(&handle.ha_fid, 0, sizeof(handle.ha_fid));
default: hsize = sizeof(xfs_fsid_t);
iput(inode); } else {
return -XFS_ERROR(EBADF);
}
/* now we can grab the fsid */
memcpy(&handle.ha_fsid, XFS_I(inode)->i_mount->m_fixedfsid,
sizeof(xfs_fsid_t));
hsize = sizeof(xfs_fsid_t);
if (cmd != XFS_IOC_PATH_TO_FSHANDLE) {
xfs_inode_t *ip = XFS_I(inode);
int lock_mode; int lock_mode;
/* need to get access to the xfs_inode to read the generation */
lock_mode = xfs_ilock_map_shared(ip); lock_mode = xfs_ilock_map_shared(ip);
/* fill in fid section of handle from inode */
handle.ha_fid.fid_len = sizeof(xfs_fid_t) - handle.ha_fid.fid_len = sizeof(xfs_fid_t) -
sizeof(handle.ha_fid.fid_len); sizeof(handle.ha_fid.fid_len);
handle.ha_fid.fid_pad = 0; handle.ha_fid.fid_pad = 0;
handle.ha_fid.fid_gen = ip->i_d.di_gen; handle.ha_fid.fid_gen = ip->i_d.di_gen;
handle.ha_fid.fid_ino = ip->i_ino; handle.ha_fid.fid_ino = ip->i_ino;
xfs_iunlock_map_shared(ip, lock_mode); xfs_iunlock_map_shared(ip, lock_mode);
hsize = XFS_HSIZE(handle); hsize = XFS_HSIZE(handle);
} }
/* now copy our handle into the user buffer & write out the size */ error = -EFAULT;
if (copy_to_user(hreq->ohandle, &handle, hsize) || if (copy_to_user(hreq->ohandle, &handle, hsize) ||
copy_to_user(hreq->ohandlen, &hsize, sizeof(__s32))) { copy_to_user(hreq->ohandlen, &hsize, sizeof(__s32)))
iput(inode); goto out_put;
return -XFS_ERROR(EFAULT);
}
iput(inode); error = 0;
return 0;
out_put:
if (cmd == XFS_IOC_FD_TO_HANDLE)
fput(file);
else
path_put(&path);
return error;
} }
/* /*
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment