Commit 232b5194 authored by Christoph Hellwig's avatar Christoph Hellwig Committed by Darrick J. Wong

xfs: simplify the xfs_getbmap interface

Instead of passing in a formatter callback allocate the bmap buffer
in the caller and process the entries there.  Additionally replace
the in-kernel buffer with a new much smaller structure, and unify
the implementation of the different ioctls in a single function.
Signed-off-by: default avatarChristoph Hellwig <hch@lst.de>
Reviewed-by: default avatarBrian Foster <bfoster@redhat.com>
Reviewed-by: default avatarDarrick J. Wong <darrick.wong@oracle.com>
Signed-off-by: default avatarDarrick J. Wong <darrick.wong@oracle.com>
parent abbf9e8a
...@@ -409,11 +409,11 @@ static int ...@@ -409,11 +409,11 @@ static int
xfs_getbmap_report_one( xfs_getbmap_report_one(
struct xfs_inode *ip, struct xfs_inode *ip,
struct getbmapx *bmv, struct getbmapx *bmv,
struct getbmapx *out, struct kgetbmap *out,
int64_t bmv_end, int64_t bmv_end,
struct xfs_bmbt_irec *got) struct xfs_bmbt_irec *got)
{ {
struct getbmapx *p = out + bmv->bmv_entries; struct kgetbmap *p = out + bmv->bmv_entries;
bool shared = false, trimmed = false; bool shared = false, trimmed = false;
int error; int error;
...@@ -460,12 +460,12 @@ static void ...@@ -460,12 +460,12 @@ static void
xfs_getbmap_report_hole( xfs_getbmap_report_hole(
struct xfs_inode *ip, struct xfs_inode *ip,
struct getbmapx *bmv, struct getbmapx *bmv,
struct getbmapx *out, struct kgetbmap *out,
int64_t bmv_end, int64_t bmv_end,
xfs_fileoff_t bno, xfs_fileoff_t bno,
xfs_fileoff_t end) xfs_fileoff_t end)
{ {
struct getbmapx *p = out + bmv->bmv_entries; struct kgetbmap *p = out + bmv->bmv_entries;
if (bmv->bmv_iflags & BMV_IF_NO_HOLES) if (bmv->bmv_iflags & BMV_IF_NO_HOLES)
return; return;
...@@ -513,47 +513,36 @@ xfs_getbmap_next_rec( ...@@ -513,47 +513,36 @@ xfs_getbmap_next_rec(
*/ */
int /* error code */ int /* error code */
xfs_getbmap( xfs_getbmap(
xfs_inode_t *ip, struct xfs_inode *ip,
struct getbmapx *bmv, /* user bmap structure */ struct getbmapx *bmv, /* user bmap structure */
xfs_bmap_format_t formatter, /* format to user */ struct kgetbmap *out)
void *arg) /* formatter arg */
{ {
struct xfs_mount *mp = ip->i_mount; struct xfs_mount *mp = ip->i_mount;
int iflags = bmv->bmv_iflags; int iflags = bmv->bmv_iflags;
int whichfork, lock, i, error = 0; int whichfork, lock, error = 0;
int64_t bmv_end, max_len; int64_t bmv_end, max_len;
xfs_fileoff_t bno, first_bno; xfs_fileoff_t bno, first_bno;
struct xfs_ifork *ifp; struct xfs_ifork *ifp;
struct getbmapx *out;
struct xfs_bmbt_irec got, rec; struct xfs_bmbt_irec got, rec;
xfs_filblks_t len; xfs_filblks_t len;
xfs_extnum_t idx; xfs_extnum_t idx;
if (bmv->bmv_iflags & ~BMV_IF_VALID)
return -EINVAL;
#ifndef DEBUG #ifndef DEBUG
/* Only allow CoW fork queries if we're debugging. */ /* Only allow CoW fork queries if we're debugging. */
if (iflags & BMV_IF_COWFORK) if (iflags & BMV_IF_COWFORK)
return -EINVAL; return -EINVAL;
#endif #endif
if ((iflags & BMV_IF_ATTRFORK) && (iflags & BMV_IF_COWFORK)) if ((iflags & BMV_IF_ATTRFORK) && (iflags & BMV_IF_COWFORK))
return -EINVAL; return -EINVAL;
if (bmv->bmv_count <= 1)
return -EINVAL;
if (bmv->bmv_count > ULONG_MAX / sizeof(struct getbmapx))
return -ENOMEM;
if (bmv->bmv_length < -1) if (bmv->bmv_length < -1)
return -EINVAL; return -EINVAL;
bmv->bmv_entries = 0; bmv->bmv_entries = 0;
if (bmv->bmv_length == 0) if (bmv->bmv_length == 0)
return 0; return 0;
out = kmem_zalloc_large(bmv->bmv_count * sizeof(struct getbmapx), 0);
if (!out)
return -ENOMEM;
if (iflags & BMV_IF_ATTRFORK) if (iflags & BMV_IF_ATTRFORK)
whichfork = XFS_ATTR_FORK; whichfork = XFS_ATTR_FORK;
else if (iflags & BMV_IF_COWFORK) else if (iflags & BMV_IF_COWFORK)
...@@ -700,15 +689,6 @@ xfs_getbmap( ...@@ -700,15 +689,6 @@ xfs_getbmap(
xfs_iunlock(ip, lock); xfs_iunlock(ip, lock);
out_unlock_iolock: out_unlock_iolock:
xfs_iunlock(ip, XFS_IOLOCK_SHARED); xfs_iunlock(ip, XFS_IOLOCK_SHARED);
for (i = 0; i < bmv->bmv_entries; i++) {
/* format results & advance arg */
error = formatter(&arg, &out[i]);
if (error)
break;
}
kmem_free(out);
return error; return error;
} }
......
...@@ -47,10 +47,14 @@ int xfs_bmap_eof(struct xfs_inode *ip, xfs_fileoff_t endoff, ...@@ -47,10 +47,14 @@ int xfs_bmap_eof(struct xfs_inode *ip, xfs_fileoff_t endoff,
int xfs_bmap_punch_delalloc_range(struct xfs_inode *ip, int xfs_bmap_punch_delalloc_range(struct xfs_inode *ip,
xfs_fileoff_t start_fsb, xfs_fileoff_t length); xfs_fileoff_t start_fsb, xfs_fileoff_t length);
/* bmap to userspace formatter - copy to user & advance pointer */ struct kgetbmap {
typedef int (*xfs_bmap_format_t)(void **, struct getbmapx *); __s64 bmv_offset; /* file offset of segment in blocks */
__s64 bmv_block; /* starting block (64-bit daddr_t) */
__s64 bmv_length; /* length of segment, blocks */
__s32 bmv_oflags; /* output flags */
};
int xfs_getbmap(struct xfs_inode *ip, struct getbmapx *bmv, int xfs_getbmap(struct xfs_inode *ip, struct getbmapx *bmv,
xfs_bmap_format_t formatter, void *arg); struct kgetbmap *out);
/* functions in xfs_bmap.c that are only needed by xfs_bmap_util.c */ /* functions in xfs_bmap.c that are only needed by xfs_bmap_util.c */
int xfs_bmap_extsize_align(struct xfs_mount *mp, struct xfs_bmbt_irec *gotp, int xfs_bmap_extsize_align(struct xfs_mount *mp, struct xfs_bmbt_irec *gotp,
......
...@@ -1540,17 +1540,26 @@ xfs_ioc_setxflags( ...@@ -1540,17 +1540,26 @@ xfs_ioc_setxflags(
return error; return error;
} }
STATIC int static bool
xfs_getbmap_format(void **ap, struct getbmapx *bmv) xfs_getbmap_format(
struct kgetbmap *p,
struct getbmapx __user *u,
size_t recsize)
{ {
struct getbmap __user *base = (struct getbmap __user *)*ap; if (put_user(p->bmv_offset, &u->bmv_offset) ||
put_user(p->bmv_block, &u->bmv_block) ||
/* copy only getbmap portion (not getbmapx) */ put_user(p->bmv_length, &u->bmv_length) ||
if (copy_to_user(base, bmv, sizeof(struct getbmap))) put_user(0, &u->bmv_count) ||
return -EFAULT; put_user(0, &u->bmv_entries))
return false;
*ap += sizeof(struct getbmap); if (recsize < sizeof(struct getbmapx))
return 0; return true;
if (put_user(0, &u->bmv_iflags) ||
put_user(p->bmv_oflags, &u->bmv_oflags) ||
put_user(0, &u->bmv_unused1) ||
put_user(0, &u->bmv_unused2))
return false;
return true;
} }
STATIC int STATIC int
...@@ -1560,68 +1569,57 @@ xfs_ioc_getbmap( ...@@ -1560,68 +1569,57 @@ xfs_ioc_getbmap(
void __user *arg) void __user *arg)
{ {
struct getbmapx bmx = { 0 }; struct getbmapx bmx = { 0 };
int error; struct kgetbmap *buf;
size_t recsize;
/* struct getbmap is a strict subset of struct getbmapx. */ int error, i;
if (copy_from_user(&bmx, arg, offsetof(struct getbmapx, bmv_iflags)))
return -EFAULT;
if (bmx.bmv_count < 2) switch (cmd) {
return -EINVAL; case XFS_IOC_GETBMAPA:
bmx.bmv_iflags = BMV_IF_ATTRFORK;
bmx.bmv_iflags = (cmd == XFS_IOC_GETBMAPA ? BMV_IF_ATTRFORK : 0); /*FALLTHRU*/
case XFS_IOC_GETBMAP:
if (file->f_mode & FMODE_NOCMTIME) if (file->f_mode & FMODE_NOCMTIME)
bmx.bmv_iflags |= BMV_IF_NO_DMAPI_READ; bmx.bmv_iflags |= BMV_IF_NO_DMAPI_READ;
/* struct getbmap is a strict subset of struct getbmapx. */
recsize = sizeof(struct getbmap);
break;
case XFS_IOC_GETBMAPX:
recsize = sizeof(struct getbmapx);
break;
default:
return -EINVAL;
}
error = xfs_getbmap(XFS_I(file_inode(file)), &bmx, xfs_getbmap_format, if (copy_from_user(&bmx, arg, recsize))
(__force struct getbmap *)arg+1);
if (error)
return error;
/* copy back header - only size of getbmap */
if (copy_to_user(arg, &bmx, sizeof(struct getbmap)))
return -EFAULT;
return 0;
}
STATIC int
xfs_getbmapx_format(void **ap, struct getbmapx *bmv)
{
struct getbmapx __user *base = (struct getbmapx __user *)*ap;
if (copy_to_user(base, bmv, sizeof(struct getbmapx)))
return -EFAULT;
*ap += sizeof(struct getbmapx);
return 0;
}
STATIC int
xfs_ioc_getbmapx(
struct xfs_inode *ip,
void __user *arg)
{
struct getbmapx bmx;
int error;
if (copy_from_user(&bmx, arg, sizeof(bmx)))
return -EFAULT; return -EFAULT;
if (bmx.bmv_count < 2) if (bmx.bmv_count < 2)
return -EINVAL; return -EINVAL;
if (bmx.bmv_count > ULONG_MAX / recsize)
return -ENOMEM;
if (bmx.bmv_iflags & (~BMV_IF_VALID)) buf = kmem_zalloc_large(bmx.bmv_count * sizeof(*buf), 0);
return -EINVAL; if (!buf)
return -ENOMEM;
error = xfs_getbmap(ip, &bmx, xfs_getbmapx_format, error = xfs_getbmap(XFS_I(file_inode(file)), &bmx, buf);
(__force struct getbmapx *)arg+1);
if (error) if (error)
return error; goto out_free_buf;
/* copy back header */ error = -EFAULT;
if (copy_to_user(arg, &bmx, sizeof(struct getbmapx))) if (copy_to_user(arg, &bmx, recsize))
return -EFAULT; goto out_free_buf;
arg += recsize;
for (i = 0; i < bmx.bmv_entries; i++) {
if (!xfs_getbmap_format(buf + i, arg, recsize))
goto out_free_buf;
arg += recsize;
}
error = 0;
out_free_buf:
kmem_free(buf);
return 0; return 0;
} }
...@@ -1878,10 +1876,8 @@ xfs_file_ioctl( ...@@ -1878,10 +1876,8 @@ xfs_file_ioctl(
case XFS_IOC_GETBMAP: case XFS_IOC_GETBMAP:
case XFS_IOC_GETBMAPA: case XFS_IOC_GETBMAPA:
return xfs_ioc_getbmap(filp, cmd, arg);
case XFS_IOC_GETBMAPX: case XFS_IOC_GETBMAPX:
return xfs_ioc_getbmapx(ip, arg); return xfs_ioc_getbmap(filp, cmd, arg);
case FS_IOC_GETFSMAP: case FS_IOC_GETFSMAP:
return xfs_ioc_getfsmap(ip, arg); return xfs_ioc_getfsmap(ip, arg);
......
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