From: Sunil Mushran <sunil.mushran@oracle.com>
To: ocfs2-devel@oss.oracle.com
Subject: [Ocfs2-devel] [PATCH 1/1] Ocfs2: Add new OCFS2_IOC_INFO ioctl for ocfs2 v7.
Date: Tue, 27 Apr 2010 13:07:30 -0700 [thread overview]
Message-ID: <4BD74402.6090406@oracle.com> (raw)
In-Reply-To: <1272284243-22492-1-git-send-email-tristan.ye@oracle.com>
sob... for this and the userspace series including --mkfs and
ioctl for journal size.
Tristan Ye wrote:
> The reason why we need this ioctl is to offer the none-privileged
> end-user a possibility to get filesys info gathering.
>
> We use OCFS2_IOC_INFO to manipulate the new ioctl, userspace passes a
> structure to kernel containing an array of request pointers and request
> count, such as,
>
> * From userspace:
>
> struct ocfs2_info_blocksize oib = {
> .ib_req = {
> .ir_magic = OCFS2_INFO_MAGIC,
> .ir_code = OCFS2_INFO_BLOCKSIZE,
> ...
> }
> ...
> }
>
> struct ocfs2_info_clustersize oic = {
> ...
> }
>
> uint64_t reqs[2] = {(unsigned long)&oib,
> (unsigned long)&oic};
>
> struct ocfs2_info info = {
> .oi_requests = reqs,
> .oi_count = 2,
> }
>
> ret = ioctl(fd, OCFS2_IOC_INFO, &info);
>
> * In kernel:
>
> Get the request pointers from *info*, then handle each request one bye one.
>
> Idea here is to make the spearated request small enough to guarantee
> a better backward&forward compatibility since a small piece of request
> would be less likely to be broken if filesys on raw disk get changed.
>
> Currently, following 8 ioctls get implemented per the requirement from
> userspace tool o2info, and I believe it will grow over time:-)
>
> OCFS2_INFO_CLUSTERSIZE
> OCFS2_INFO_BLOCKSIZE
> OCFS2_INFO_MAXSLOTS
> OCFS2_INFO_LABEL
> OCFS2_INFO_UUID
> OCFS2_INFO_FS_FEATURES
>
> This ioctl is only specific to OCFS2.
>
> Signed-off-by: Tristan Ye <tristan.ye@oracle.com>
> ---
> fs/ocfs2/ioctl.c | 317 ++++++++++++++++++++++++++++++++++++++++++++++++
> fs/ocfs2/ocfs2_ioctl.h | 85 +++++++++++++
> 2 files changed, 402 insertions(+), 0 deletions(-)
>
> diff --git a/fs/ocfs2/ioctl.c b/fs/ocfs2/ioctl.c
> index 7d9d9c1..e55e52d 100644
> --- a/fs/ocfs2/ioctl.c
> +++ b/fs/ocfs2/ioctl.c
> @@ -23,8 +23,18 @@
> #include "ioctl.h"
> #include "resize.h"
> #include "refcounttree.h"
> +#include "sysfile.h"
> +#include "buffer_head_io.h"
> +#include "suballoc.h"
> +
>
> #include <linux/ext2_fs.h>
> +#include <linux/compat.h>
> +
> +#define o2info_from_user(a, b) \
> + copy_from_user(&(a), (b), sizeof(a))
> +#define o2info_to_user(a, b) \
> + copy_to_user((typeof(a) __user *)b, &(a), sizeof(a))
>
> static int ocfs2_get_inode_attr(struct inode *inode, unsigned *flags)
> {
> @@ -109,6 +119,299 @@ bail:
> return status;
> }
>
> +int ocfs2_info_handle_blocksize(struct inode *inode,
> + struct ocfs2_info_request __user *req)
> +{
> + int status = -EFAULT;
> + struct ocfs2_info_blocksize oib;
> +
> + if (o2info_from_user(oib, req))
> + goto bail;
> +
> + oib.ib_blocksize = inode->i_sb->s_blocksize;
> + oib.ib_req.ir_flags |= OCFS2_INFO_FL_FILLED;
> +
> + if (o2info_to_user(oib, req))
> + goto bail;
> +
> + status = 0;
> +bail:
> + if (status) {
> + oib.ib_req.ir_flags |= OCFS2_INFO_FL_ERROR;
> + oib.ib_req.ir_error = status;
> + }
> +
> + return status;
> +}
> +
> +int ocfs2_info_handle_clustersize(struct inode *inode,
> + struct ocfs2_info_request __user *req)
> +{
> + int status = -EFAULT;
> + struct ocfs2_info_clustersize oic;
> + struct ocfs2_super *osb = OCFS2_SB(inode->i_sb);
> +
> + if (o2info_from_user(oic, req))
> + goto bail;
> +
> + oic.ic_clustersize = osb->s_clustersize;
> + oic.ic_req.ir_flags |= OCFS2_INFO_FL_FILLED;
> +
> + if (o2info_to_user(oic, req))
> + goto bail;
> +
> + status = 0;
> +bail:
> + if (status) {
> + oic.ic_req.ir_flags |= OCFS2_INFO_FL_ERROR;
> + oic.ic_req.ir_error = status;
> + }
> +
> + return status;
> +}
> +
> +int ocfs2_info_handle_maxslots(struct inode *inode,
> + struct ocfs2_info_request __user *req)
> +{
> + int status = -EFAULT;
> + struct ocfs2_info_maxslots oim;
> + struct ocfs2_super *osb = OCFS2_SB(inode->i_sb);
> +
> + if (o2info_from_user(oim, req))
> + goto bail;
> +
> + oim.im_max_slots = osb->max_slots;
> + oim.im_req.ir_flags |= OCFS2_INFO_FL_FILLED;
> +
> + if (o2info_to_user(oim, req))
> + goto bail;
> +
> + status = 0;
> +bail:
> + if (status) {
> + oim.im_req.ir_flags |= OCFS2_INFO_FL_ERROR;
> + oim.im_req.ir_error = status;
> + }
> +
> + return status;
> +}
> +
> +int ocfs2_info_handle_label(struct inode *inode,
> + struct ocfs2_info_request __user *req)
> +{
> + int status = -EFAULT;
> + struct ocfs2_info_label oil;
> + struct ocfs2_super *osb = OCFS2_SB(inode->i_sb);
> +
> + if (o2info_from_user(oil, req))
> + goto bail;
> +
> + memcpy(oil.il_label, osb->vol_label, OCFS2_MAX_VOL_LABEL_LEN);
> + oil.il_req.ir_flags |= OCFS2_INFO_FL_FILLED;
> +
> + if (o2info_to_user(oil, req))
> + goto bail;
> +
> + status = 0;
> +bail:
> + if (status) {
> + oil.il_req.ir_flags |= OCFS2_INFO_FL_ERROR;
> + oil.il_req.ir_error = status;
> + }
> +
> + return status;
> +}
> +
> +int ocfs2_info_handle_uuid(struct inode *inode,
> + struct ocfs2_info_request __user *req)
> +{
> + int status = -EFAULT;
> + struct ocfs2_info_uuid oiu;
> + struct ocfs2_super *osb = OCFS2_SB(inode->i_sb);
> +
> + if (o2info_from_user(oiu, req))
> + goto bail;
> +
> + memcpy(oiu.iu_uuid_str, osb->uuid_str, OCFS2_TEXT_UUID_LEN + 1);
> + oiu.iu_req.ir_flags |= OCFS2_INFO_FL_FILLED;
> +
> + if (o2info_to_user(oiu, req))
> + goto bail;
> +
> + status = 0;
> +bail:
> + if (status) {
> + oiu.iu_req.ir_flags |= OCFS2_INFO_FL_ERROR;
> + oiu.iu_req.ir_error = status;
> + }
> +
> + return status;
> +}
> +
> +int ocfs2_info_handle_fs_features(struct inode *inode,
> + struct ocfs2_info_request __user *req)
> +{
> + int status = -EFAULT;
> + struct ocfs2_info_fs_features oif;
> + struct ocfs2_super *osb = OCFS2_SB(inode->i_sb);
> +
> + if (o2info_from_user(oif, req))
> + goto bail;
> +
> + oif.if_compat_features = osb->s_feature_compat;
> + oif.if_incompat_features = osb->s_feature_incompat;
> + oif.if_ro_compat_features = osb->s_feature_ro_compat;
> + oif.if_req.ir_flags |= OCFS2_INFO_FL_FILLED;
> +
> + if (o2info_to_user(oif, req))
> + goto bail;
> +
> + status = 0;
> +bail:
> + if (status) {
> + oif.if_req.ir_flags |= OCFS2_INFO_FL_ERROR;
> + oif.if_req.ir_error = status;
> + }
> +
> + return status;
> +}
> +
> +int ocfs2_info_handle_unknown(struct inode *inode,
> + struct ocfs2_info_request __user *req)
> +{
> + int status = -EFAULT;
> + struct ocfs2_info_request oir;
> +
> + if (o2info_from_user(oir, req))
> + goto bail;
> +
> + oir.ir_flags &= ~OCFS2_INFO_FL_FILLED;
> +
> + if (o2info_to_user(oir, req))
> + goto bail;
> +
> + status = 0;
> +bail:
> + if (status) {
> + oir.ir_flags |= OCFS2_INFO_FL_ERROR;
> + oir.ir_error = status;
> + }
> +
> + return status;
> +}
> +
> +/*
> + * Validate and distinguish OCFS2_IOC_INFO requests.
> + *
> + * - validate the magic number.
> + * - distinguish different requests.
> + * - validate size of different requests.
> + */
> +int ocfs2_info_handle_request(struct inode *inode,
> + struct ocfs2_info_request __user *req)
> +{
> + int status = -EFAULT;
> + struct ocfs2_info_request oir;
> +
> + if (o2info_from_user(oir, req))
> + goto bail;
> +
> + status = -EINVAL;
> + if (oir.ir_magic != OCFS2_INFO_MAGIC)
> + goto bail;
> +
> + switch (oir.ir_code) {
> + case OCFS2_INFO_BLOCKSIZE:
> + if (oir.ir_size == sizeof(struct ocfs2_info_blocksize))
> + status = ocfs2_info_handle_blocksize(inode, req);
> + break;
> + case OCFS2_INFO_CLUSTERSIZE:
> + if (oir.ir_size == sizeof(struct ocfs2_info_clustersize))
> + status = ocfs2_info_handle_clustersize(inode, req);
> + break;
> + case OCFS2_INFO_MAXSLOTS:
> + if (oir.ir_size == sizeof(struct ocfs2_info_maxslots))
> + status = ocfs2_info_handle_maxslots(inode, req);
> + break;
> + case OCFS2_INFO_LABEL:
> + if (oir.ir_size == sizeof(struct ocfs2_info_label))
> + status = ocfs2_info_handle_label(inode, req);
> + break;
> + case OCFS2_INFO_UUID:
> + if (oir.ir_size == sizeof(struct ocfs2_info_uuid))
> + status = ocfs2_info_handle_uuid(inode, req);
> + break;
> + case OCFS2_INFO_FS_FEATURES:
> + if (oir.ir_size == sizeof(struct ocfs2_info_fs_features))
> + status = ocfs2_info_handle_fs_features(inode, req);
> + break;
> + default:
> + status = ocfs2_info_handle_unknown(inode, req);
> + break;
> + }
> +
> +bail:
> + return status;
> +}
> +
> +/*
> + * OCFS2_IOC_INFO handles an array of requests passed from userspace.
> + *
> + * ocfs2_info_handle() recevies a large info aggregation, grab and
> + * validate the request count from header, then break it into small
> + * pieces, later specific handlers can handle them one by one.
> + *
> + * Idea here is to make each separate request small enough to ensure
> + * a better backward&forward compatibility, since a small piece of
> + * request will be less likely to be broken if disk layout get changed.
> + */
> +int ocfs2_info_handle(struct inode *inode, struct ocfs2_info *info,
> + int compat_flag)
> +{
> + int i, status = 0;
> + u64 req_addr;
> + struct ocfs2_info_request __user *reqp;
> +
> + if ((info->oi_count > OCFS2_INFO_MAX_REQUEST) ||
> + (!info->oi_requests)) {
> + status = -EINVAL;
> + goto bail;
> + }
> +
> + for (i = 0; i < info->oi_count; i++) {
> + status = -EFAULT;
> + if (compat_flag) {
> + if (get_user(req_addr,
> + (u64 __user *)compat_ptr(info->oi_requests) + i))
> + goto bail;
> + } else {
> + if (get_user(req_addr,
> + (u64 __user *)(info->oi_requests) + i))
> + goto bail;
> + }
> +
> + reqp = (struct ocfs2_info_request *)req_addr;
> + if (!reqp) {
> + status = -EINVAL;
> + goto bail;
> + }
> +
> + status = ocfs2_info_handle_request(inode, reqp);
> + }
> +
> + /*
> + * Error of each request handling was kept separately inside
> + * the request itself, ioctl(2) however will be responsible
> + * for the error outside meaty request-filling operations.
> + *
> + * That way, all requests expect to be handled no matter error
> + * occurs or not in request-handling body.
> + */
> + status = 0;
> +bail:
> + return status;
> +}
> +
> long ocfs2_ioctl(struct file *filp, unsigned int cmd, unsigned long arg)
> {
> struct inode *inode = filp->f_path.dentry->d_inode;
> @@ -120,6 +423,7 @@ long ocfs2_ioctl(struct file *filp, unsigned int cmd, unsigned long arg)
> struct reflink_arguments args;
> const char *old_path, *new_path;
> bool preserve;
> + struct ocfs2_info info;
>
> switch (cmd) {
> case OCFS2_IOC_GETFLAGS:
> @@ -174,6 +478,12 @@ long ocfs2_ioctl(struct file *filp, unsigned int cmd, unsigned long arg)
> preserve = (args.preserve != 0);
>
> return ocfs2_reflink_ioctl(inode, old_path, new_path, preserve);
> + case OCFS2_IOC_INFO:
> + if (copy_from_user(&info, (struct ocfs2_info __user *)arg,
> + sizeof(struct ocfs2_info)))
> + return -EFAULT;
> +
> + return ocfs2_info_handle(inode, &info, 0);
> default:
> return -ENOTTY;
> }
> @@ -185,6 +495,7 @@ long ocfs2_compat_ioctl(struct file *file, unsigned cmd, unsigned long arg)
> bool preserve;
> struct reflink_arguments args;
> struct inode *inode = file->f_path.dentry->d_inode;
> + struct ocfs2_info info;
>
> switch (cmd) {
> case OCFS2_IOC32_GETFLAGS:
> @@ -209,6 +520,12 @@ long ocfs2_compat_ioctl(struct file *file, unsigned cmd, unsigned long arg)
>
> return ocfs2_reflink_ioctl(inode, compat_ptr(args.old_path),
> compat_ptr(args.new_path), preserve);
> + case OCFS2_IOC_INFO:
> + if (copy_from_user(&info, (struct ocfs2_info __user *)arg,
> + sizeof(struct ocfs2_info)))
> + return -EFAULT;
> +
> + return ocfs2_info_handle(inode, &info, 1);
> default:
> return -ENOIOCTLCMD;
> }
> diff --git a/fs/ocfs2/ocfs2_ioctl.h b/fs/ocfs2/ocfs2_ioctl.h
> index 2d3420a..4b922f3 100644
> --- a/fs/ocfs2/ocfs2_ioctl.h
> +++ b/fs/ocfs2/ocfs2_ioctl.h
> @@ -76,4 +76,89 @@ struct reflink_arguments {
> };
> #define OCFS2_IOC_REFLINK _IOW('o', 4, struct reflink_arguments)
>
> +/* Following definitions dedicated for ocfs2_info_request ioctls. */
> +#define OCFS2_INFO_MAX_REQUEST (50)
> +#define OCFS2_TEXT_UUID_LEN (OCFS2_VOL_UUID_LEN * 2)
> +
> +/* Magic number of all requests */
> +#define OCFS2_INFO_MAGIC (0x4F32494E)
> +
> +/*
> + * Always try to separate info request into small pieces to
> + * guarantee the backward&forward compatibility.
> + */
> +struct ocfs2_info {
> + __u64 oi_requests; /* Array of __u64 pointers to requests */
> + __u32 oi_count; /* Number of requests in info_requests */
> +};
> +
> +struct ocfs2_info_request {
> +/*00*/ __u32 ir_magic; /* Magic number */
> + __u32 ir_code; /* Info request code */
> + __u32 ir_size; /* Size of request */
> + __u32 ir_flags; /* Request flags */
> + __u32 ir_error; /* Error code */
> +/*14*/ /* Request specific fields */
> +};
> +
> +struct ocfs2_info_clustersize {
> + struct ocfs2_info_request ic_req;
> + __u32 ic_clustersize;
> +};
> +
> +struct ocfs2_info_blocksize {
> + struct ocfs2_info_request ib_req;
> + __u32 ib_blocksize;
> +};
> +
> +struct ocfs2_info_maxslots {
> + struct ocfs2_info_request im_req;
> + __u16 im_max_slots;
> +};
> +
> +struct ocfs2_info_label {
> + struct ocfs2_info_request il_req;
> + __u8 il_label[OCFS2_MAX_VOL_LABEL_LEN];
> +};
> +
> +struct ocfs2_info_uuid {
> + struct ocfs2_info_request iu_req;
> + __u8 iu_uuid_str[OCFS2_TEXT_UUID_LEN + 1];
> +};
> +
> +struct ocfs2_info_fs_features {
> + struct ocfs2_info_request if_req;
> + __u32 if_compat_features;
> + __u32 if_incompat_features;
> + __u32 if_ro_compat_features;
> +};
> +
> +/* Codes for ocfs2_info_request */
> +enum ocfs2_info_type {
> + OCFS2_INFO_CLUSTERSIZE = 1,
> + OCFS2_INFO_BLOCKSIZE,
> + OCFS2_INFO_MAXSLOTS,
> + OCFS2_INFO_LABEL,
> + OCFS2_INFO_UUID,
> + OCFS2_INFO_FS_FEATURES,
> + OCFS2_INFO_NUM_TYPES
> +};
> +
> +/* Flags for struct ocfs2_info_request */
> +/* Filled by the caller */
> +#define OCFS2_INFO_FL_NON_COHERENT (0x00000001) /* Cluster coherency not
> + required. This is a hint.
> + It is up to ocfs2 whether
> + the request can be fulfilled
> + without locking. */
> +/* Filled by ocfs2 */
> +#define OCFS2_INFO_FL_FILLED (0x40000000) /* Filesystem understood
> + this request and
> + filled in the answer */
> +
> +#define OCFS2_INFO_FL_ERROR (0x80000000) /* Error happened during
> + request handling. */
> +
> +#define OCFS2_IOC_INFO _IOR('o', 5, struct ocfs2_info)
> +
> #endif /* OCFS2_IOCTL_H */
>
next prev parent reply other threads:[~2010-04-27 20:07 UTC|newest]
Thread overview: 24+ messages / expand[flat|nested] mbox.gz Atom feed top
2010-04-26 12:17 [Ocfs2-devel] [PATCH 1/1] Ocfs2: Add new OCFS2_IOC_INFO ioctl for ocfs2 v7 Tristan Ye
2010-04-27 20:07 ` Sunil Mushran [this message]
2010-05-06 1:05 ` Joel Becker
2010-05-06 2:09 ` tristan
-- strict thread matches above, loose matches on Subject: below --
2010-05-19 2:28 Tristan Ye
2010-05-20 23:26 ` Joel Becker
2010-05-20 23:49 ` Joel Becker
2010-05-21 9:07 ` tristan
2010-05-21 10:22 ` Joel Becker
2010-05-21 1:30 ` tristan
2010-05-21 2:41 ` Joel Becker
2010-05-21 3:05 ` tristan
2010-05-11 7:21 [Ocfs2-devel] [PATCH 0/1] Ocfs2: o2info for kernel v7 Tristan Ye
2010-05-11 7:21 ` [Ocfs2-devel] [PATCH 1/1] Ocfs2: Add new OCFS2_IOC_INFO ioctl for ocfs2 v7 Tristan Ye
2010-05-11 20:40 ` Joel Becker
2010-05-18 23:55 ` Joel Becker
2010-05-19 3:03 ` tristan
2010-05-06 8:43 Tristan Ye
2010-05-10 20:01 ` Joel Becker
2010-05-11 2:12 ` tristan
2010-05-11 6:51 ` Joel Becker
2010-04-19 11:00 Tristan Ye
2010-04-19 20:16 ` Sunil Mushran
2010-04-20 2:31 ` tristan
2010-04-20 4:28 ` Sunil Mushran
Reply instructions:
You may reply publicly to this message via plain-text email
using any one of the following methods:
* Save the following mbox file, import it into your mail client,
and reply-to-all from there: mbox
Avoid top-posting and favor interleaved quoting:
https://en.wikipedia.org/wiki/Posting_style#Interleaved_style
* Reply using the --to, --cc, and --in-reply-to
switches of git-send-email(1):
git send-email \
--in-reply-to=4BD74402.6090406@oracle.com \
--to=sunil.mushran@oracle.com \
--cc=ocfs2-devel@oss.oracle.com \
/path/to/YOUR_REPLY
https://kernel.org/pub/software/scm/git/docs/git-send-email.html
* If your mail client supports setting the In-Reply-To header
via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line
before the message body.
This is an external index of several public inboxes,
see mirroring instructions on how to clone and mirror
all data and code used by this external index.