On Fri 22-01-16 12:25:32, Eric Sandeen wrote: > Q_GETNEXTQUOTA is exactly like Q_GETQUOTA, except that it > will return quota information for the id equal to or greater > than the id requested. In other words, if the requested id has > no quota, the command will return quota information for the > next higher id which does have a quota set. If no higher id > has an active quota, -ESRCH is returned. > > This allows filesystems to do efficient iteration in kernelspace, > much like extN filesystems do in userspace when asked to report > all active quotas. > > This does require a new data structure for userspace, as the > current structure does not include an ID for the returned quota > information. > > Today, Ext4 with a hidden quota inode requires getpwent-style > iterations, and for systems which have i.e. LDAP backends, > this can be very slow, or even impossible if iteration is not > allowed in the configuration. > > Signed-off-by: Eric Sandeen <sandeen@xxxxxxxxxx> The patch seems to be missing addition of Q_GETNEXTQUOTA into quotactl_cmd_write(). Otherwise it looks good to me. Honza > --- > fs/quota/quota.c | 30 ++++++++++++++++++++++++++++++ > include/uapi/linux/quota.h | 14 ++++++++++++++ > 2 files changed, 44 insertions(+), 0 deletions(-) > > diff --git a/fs/quota/quota.c b/fs/quota/quota.c > index 0a6dd71..ffa4e0b 100644 > --- a/fs/quota/quota.c > +++ b/fs/quota/quota.c > @@ -222,6 +222,34 @@ static int quota_getquota(struct super_block *sb, int type, qid_t id, > return 0; > } > > +/* > + * Return quota for next active quota >= this id, if any exists, > + * otherwise return -ESRCH via ->get_nextdqblk > + */ > +static int quota_getnextquota(struct super_block *sb, int type, qid_t id, > + void __user *addr) > +{ > + struct kqid qid; > + struct qc_dqblk fdq; > + struct if_nextdqblk idq; > + int ret; > + > + if (!sb->s_qcop->get_nextdqblk) > + return -ENOSYS; > + qid = make_kqid(current_user_ns(), type, id); > + if (!qid_valid(qid)) > + return -EINVAL; > + ret = sb->s_qcop->get_nextdqblk(sb, &qid, &fdq); > + if (ret) > + return ret; > + /* struct if_nextdqblk is a superset of struct if_dqblk */ > + copy_to_if_dqblk((struct if_dqblk *)&idq, &fdq); > + idq.dqb_id = from_kqid(current_user_ns(), qid); > + if (copy_to_user(addr, &idq, sizeof(idq))) > + return -EFAULT; > + return 0; > +} > + > static void copy_from_if_dqblk(struct qc_dqblk *dst, struct if_dqblk *src) > { > dst->d_spc_hardlimit = qbtos(src->dqb_bhardlimit); > @@ -698,6 +726,8 @@ static int do_quotactl(struct super_block *sb, int type, int cmd, qid_t id, > return quota_setinfo(sb, type, addr); > case Q_GETQUOTA: > return quota_getquota(sb, type, id, addr); > + case Q_GETNEXTQUOTA: > + return quota_getnextquota(sb, type, id, addr); > case Q_SETQUOTA: > return quota_setquota(sb, type, id, addr); > case Q_SYNC: > diff --git a/include/uapi/linux/quota.h b/include/uapi/linux/quota.h > index 9c95b2c..38baddb 100644 > --- a/include/uapi/linux/quota.h > +++ b/include/uapi/linux/quota.h > @@ -71,6 +71,7 @@ > #define Q_SETINFO 0x800006 /* set information about quota files */ > #define Q_GETQUOTA 0x800007 /* get user quota structure */ > #define Q_SETQUOTA 0x800008 /* set user quota structure */ > +#define Q_GETNEXTQUOTA 0x800009 /* get disk limits and usage >= ID */ > > /* Quota format type IDs */ > #define QFMT_VFS_OLD 1 > @@ -119,6 +120,19 @@ struct if_dqblk { > __u32 dqb_valid; > }; > > +struct if_nextdqblk { > + __u64 dqb_bhardlimit; > + __u64 dqb_bsoftlimit; > + __u64 dqb_curspace; > + __u64 dqb_ihardlimit; > + __u64 dqb_isoftlimit; > + __u64 dqb_curinodes; > + __u64 dqb_btime; > + __u64 dqb_itime; > + __u32 dqb_valid; > + __u32 dqb_id; > +}; > + > /* > * Structure used for setting quota information about file via quotactl > * Following flags are used to specify which fields are valid > -- > 1.7.1 > -- Jan Kara <jack@xxxxxxxx> SUSE Labs, CR -- To unsubscribe from this list: send the line "unsubscribe linux-fsdevel" in the body of a message to majordomo@xxxxxxxxxxxxxxx More majordomo info at http://vger.kernel.org/majordomo-info.html