diff options
| author | Sage Weil <sage@inktank.com> | 2013-08-15 11:11:45 -0700 | 
|---|---|---|
| committer | Sage Weil <sage@inktank.com> | 2013-08-15 11:11:45 -0700 | 
| commit | ee3e542fec6e69bc9fb668698889a37d93950ddf (patch) | |
| tree | e74ee766a4764769ef1d3d45d266b4dea64101d3 /fs/xfs/xfs_quota.h | |
| parent | fe2a801b50c0bb8039d627e5ae1fec249d10ff39 (diff) | |
| parent | f1d6e17f540af37bb1891480143669ba7636c4cf (diff) | |
Merge remote-tracking branch 'linus/master' into testing
Diffstat (limited to 'fs/xfs/xfs_quota.h')
| -rw-r--r-- | fs/xfs/xfs_quota.h | 104 | 
1 files changed, 64 insertions, 40 deletions
| diff --git a/fs/xfs/xfs_quota.h b/fs/xfs/xfs_quota.h index c38068f26c55..b14f42c714b6 100644 --- a/fs/xfs/xfs_quota.h +++ b/fs/xfs/xfs_quota.h @@ -108,11 +108,28 @@ typedef struct xfs_dqblk {  	{ XFS_DQ_FREEING,	"FREEING" }  /* - * In the worst case, when both user and group quotas are on, - * we can have a max of three dquots changing in a single transaction. + * We have the possibility of all three quota types being active at once, and + * hence free space modification requires modification of all three current + * dquots in a single transaction. For this case we need to have a reservation + * of at least 3 dquots. + * + * However, a chmod operation can change both UID and GID in a single + * transaction, resulting in requiring {old, new} x {uid, gid} dquots to be + * modified. Hence for this case we need to reserve space for at least 4 dquots. + * + * And in the worst case, there's a rename operation that can be modifying up to + * 4 inodes with dquots attached to them. In reality, the only inodes that can + * have their dquots modified are the source and destination directory inodes + * due to directory name creation and removal. That can require space allocation + * and/or freeing on both directory inodes, and hence all three dquots on each + * inode can be modified. And if the directories are world writeable, all the + * dquots can be unique and so 6 dquots can be modified.... + * + * And, of course, we also need to take into account the dquot log format item + * used to describe each dquot.   */ -#define XFS_DQUOT_LOGRES(mp)	(sizeof(xfs_disk_dquot_t) * 3) - +#define XFS_DQUOT_LOGRES(mp)	\ +	((sizeof(struct xfs_dq_logformat) + sizeof(struct xfs_disk_dquot)) * 6)  /*   * These are the structures used to lay out dquots and quotaoff @@ -161,30 +178,42 @@ typedef struct xfs_qoff_logformat {  #define XFS_GQUOTA_ACCT	0x0040  /* group quota accounting ON */  /* + * Conversion to and from the combined OQUOTA flag (if necessary) + * is done only in xfs_sb_qflags_to_disk() and xfs_sb_qflags_from_disk() + */ +#define XFS_GQUOTA_ENFD	0x0080  /* group quota limits enforced */ +#define XFS_GQUOTA_CHKD	0x0100  /* quotacheck run on group quotas */ +#define XFS_PQUOTA_ENFD	0x0200  /* project quota limits enforced */ +#define XFS_PQUOTA_CHKD	0x0400  /* quotacheck run on project quotas */ + +/*   * Quota Accounting/Enforcement flags   */  #define XFS_ALL_QUOTA_ACCT	\  		(XFS_UQUOTA_ACCT | XFS_GQUOTA_ACCT | XFS_PQUOTA_ACCT) -#define XFS_ALL_QUOTA_ENFD	(XFS_UQUOTA_ENFD | XFS_OQUOTA_ENFD) -#define XFS_ALL_QUOTA_CHKD	(XFS_UQUOTA_CHKD | XFS_OQUOTA_CHKD) +#define XFS_ALL_QUOTA_ENFD	\ +		(XFS_UQUOTA_ENFD | XFS_GQUOTA_ENFD | XFS_PQUOTA_ENFD) +#define XFS_ALL_QUOTA_CHKD	\ +		(XFS_UQUOTA_CHKD | XFS_GQUOTA_CHKD | XFS_PQUOTA_CHKD)  #define XFS_IS_QUOTA_RUNNING(mp)	((mp)->m_qflags & XFS_ALL_QUOTA_ACCT)  #define XFS_IS_UQUOTA_RUNNING(mp)	((mp)->m_qflags & XFS_UQUOTA_ACCT)  #define XFS_IS_PQUOTA_RUNNING(mp)	((mp)->m_qflags & XFS_PQUOTA_ACCT)  #define XFS_IS_GQUOTA_RUNNING(mp)	((mp)->m_qflags & XFS_GQUOTA_ACCT)  #define XFS_IS_UQUOTA_ENFORCED(mp)	((mp)->m_qflags & XFS_UQUOTA_ENFD) -#define XFS_IS_OQUOTA_ENFORCED(mp)	((mp)->m_qflags & XFS_OQUOTA_ENFD) +#define XFS_IS_GQUOTA_ENFORCED(mp)	((mp)->m_qflags & XFS_GQUOTA_ENFD) +#define XFS_IS_PQUOTA_ENFORCED(mp)	((mp)->m_qflags & XFS_PQUOTA_ENFD)  /*   * Incore only flags for quotaoff - these bits get cleared when quota(s)   * are in the process of getting turned off. These flags are in m_qflags but   * never in sb_qflags.   */ -#define XFS_UQUOTA_ACTIVE	0x0100  /* uquotas are being turned off */ -#define XFS_PQUOTA_ACTIVE	0x0200  /* pquotas are being turned off */ -#define XFS_GQUOTA_ACTIVE	0x0400  /* gquotas are being turned off */ +#define XFS_UQUOTA_ACTIVE	0x1000  /* uquotas are being turned off */ +#define XFS_GQUOTA_ACTIVE	0x2000  /* gquotas are being turned off */ +#define XFS_PQUOTA_ACTIVE	0x4000  /* pquotas are being turned off */  #define XFS_ALL_QUOTA_ACTIVE	\ -	(XFS_UQUOTA_ACTIVE | XFS_PQUOTA_ACTIVE | XFS_GQUOTA_ACTIVE) +	(XFS_UQUOTA_ACTIVE | XFS_GQUOTA_ACTIVE | XFS_PQUOTA_ACTIVE)  /*   * Checking XFS_IS_*QUOTA_ON() while holding any inode lock guarantees @@ -259,33 +288,24 @@ typedef struct xfs_qoff_logformat {   * we didn't have the inode locked, the appropriate dquot(s) will be   * attached atomically.   */ -#define XFS_NOT_DQATTACHED(mp, ip) ((XFS_IS_UQUOTA_ON(mp) &&\ -				     (ip)->i_udquot == NULL) || \ -				    (XFS_IS_OQUOTA_ON(mp) && \ -				     (ip)->i_gdquot == NULL)) +#define XFS_NOT_DQATTACHED(mp, ip) \ +	((XFS_IS_UQUOTA_ON(mp) && (ip)->i_udquot == NULL) || \ +	 (XFS_IS_GQUOTA_ON(mp) && (ip)->i_gdquot == NULL) || \ +	 (XFS_IS_PQUOTA_ON(mp) && (ip)->i_pdquot == NULL))  #define XFS_QM_NEED_QUOTACHECK(mp) \  	((XFS_IS_UQUOTA_ON(mp) && \  		(mp->m_sb.sb_qflags & XFS_UQUOTA_CHKD) == 0) || \  	 (XFS_IS_GQUOTA_ON(mp) && \ -		((mp->m_sb.sb_qflags & XFS_OQUOTA_CHKD) == 0 || \ -		 (mp->m_sb.sb_qflags & XFS_PQUOTA_ACCT))) || \ +		(mp->m_sb.sb_qflags & XFS_GQUOTA_CHKD) == 0) || \  	 (XFS_IS_PQUOTA_ON(mp) && \ -		((mp->m_sb.sb_qflags & XFS_OQUOTA_CHKD) == 0 || \ -		 (mp->m_sb.sb_qflags & XFS_GQUOTA_ACCT)))) - -#define XFS_MOUNT_QUOTA_SET1	(XFS_UQUOTA_ACCT|XFS_UQUOTA_ENFD|\ -				 XFS_UQUOTA_CHKD|XFS_PQUOTA_ACCT|\ -				 XFS_OQUOTA_ENFD|XFS_OQUOTA_CHKD) - -#define XFS_MOUNT_QUOTA_SET2	(XFS_UQUOTA_ACCT|XFS_UQUOTA_ENFD|\ -				 XFS_UQUOTA_CHKD|XFS_GQUOTA_ACCT|\ -				 XFS_OQUOTA_ENFD|XFS_OQUOTA_CHKD) +		(mp->m_sb.sb_qflags & XFS_PQUOTA_CHKD) == 0))  #define XFS_MOUNT_QUOTA_ALL	(XFS_UQUOTA_ACCT|XFS_UQUOTA_ENFD|\ -				 XFS_UQUOTA_CHKD|XFS_PQUOTA_ACCT|\ -				 XFS_OQUOTA_ENFD|XFS_OQUOTA_CHKD|\ -				 XFS_GQUOTA_ACCT) +				 XFS_UQUOTA_CHKD|XFS_GQUOTA_ACCT|\ +				 XFS_GQUOTA_ENFD|XFS_GQUOTA_CHKD|\ +				 XFS_PQUOTA_ACCT|XFS_PQUOTA_ENFD|\ +				 XFS_PQUOTA_CHKD)  /* @@ -318,17 +338,18 @@ extern int xfs_trans_reserve_quota_nblks(struct xfs_trans *,  		struct xfs_inode *, long, long, uint);  extern int xfs_trans_reserve_quota_bydquots(struct xfs_trans *,  		struct xfs_mount *, struct xfs_dquot *, -		struct xfs_dquot *, long, long, uint); +		struct xfs_dquot *, struct xfs_dquot *, long, long, uint);  extern int xfs_qm_vop_dqalloc(struct xfs_inode *, uid_t, gid_t, prid_t, uint, -		struct xfs_dquot **, struct xfs_dquot **); +		struct xfs_dquot **, struct xfs_dquot **, struct xfs_dquot **);  extern void xfs_qm_vop_create_dqattach(struct xfs_trans *, struct xfs_inode *, -		struct xfs_dquot *, struct xfs_dquot *); +		struct xfs_dquot *, struct xfs_dquot *, struct xfs_dquot *);  extern int xfs_qm_vop_rename_dqattach(struct xfs_inode **);  extern struct xfs_dquot *xfs_qm_vop_chown(struct xfs_trans *,  		struct xfs_inode *, struct xfs_dquot **, struct xfs_dquot *);  extern int xfs_qm_vop_chown_reserve(struct xfs_trans *, struct xfs_inode *, -		struct xfs_dquot *, struct xfs_dquot *, uint); +		struct xfs_dquot *, struct xfs_dquot *, +		struct xfs_dquot *, uint);  extern int xfs_qm_dqattach(struct xfs_inode *, uint);  extern int xfs_qm_dqattach_locked(struct xfs_inode *, uint);  extern void xfs_qm_dqdetach(struct xfs_inode *); @@ -342,10 +363,12 @@ extern void xfs_qm_unmount_quotas(struct xfs_mount *);  #else  static inline int  xfs_qm_vop_dqalloc(struct xfs_inode *ip, uid_t uid, gid_t gid, prid_t prid, -		uint flags, struct xfs_dquot **udqp, struct xfs_dquot **gdqp) +		uint flags, struct xfs_dquot **udqp, struct xfs_dquot **gdqp, +		struct xfs_dquot **pdqp)  {  	*udqp = NULL;  	*gdqp = NULL; +	*pdqp = NULL;  	return 0;  }  #define xfs_trans_dup_dqinfo(tp, tp2) @@ -360,14 +383,15 @@ static inline int xfs_trans_reserve_quota_nblks(struct xfs_trans *tp,  }  static inline int xfs_trans_reserve_quota_bydquots(struct xfs_trans *tp,  		struct xfs_mount *mp, struct xfs_dquot *udqp, -		struct xfs_dquot *gdqp, long nblks, long nions, uint flags) +		struct xfs_dquot *gdqp, struct xfs_dquot *pdqp, +		long nblks, long nions, uint flags)  {  	return 0;  } -#define xfs_qm_vop_create_dqattach(tp, ip, u, g) +#define xfs_qm_vop_create_dqattach(tp, ip, u, g, p)  #define xfs_qm_vop_rename_dqattach(it)					(0)  #define xfs_qm_vop_chown(tp, ip, old, new)				(NULL) -#define xfs_qm_vop_chown_reserve(tp, ip, u, g, fl)			(0) +#define xfs_qm_vop_chown_reserve(tp, ip, u, g, p, fl)			(0)  #define xfs_qm_dqattach(ip, fl)						(0)  #define xfs_qm_dqattach_locked(ip, fl)					(0)  #define xfs_qm_dqdetach(ip) @@ -381,8 +405,8 @@ static inline int xfs_trans_reserve_quota_bydquots(struct xfs_trans *tp,  #define xfs_trans_unreserve_quota_nblks(tp, ip, nblks, ninos, flags) \  	xfs_trans_reserve_quota_nblks(tp, ip, -(nblks), -(ninos), flags) -#define xfs_trans_reserve_quota(tp, mp, ud, gd, nb, ni, f) \ -	xfs_trans_reserve_quota_bydquots(tp, mp, ud, gd, nb, ni, \ +#define xfs_trans_reserve_quota(tp, mp, ud, gd, pd, nb, ni, f) \ +	xfs_trans_reserve_quota_bydquots(tp, mp, ud, gd, pd, nb, ni, \  				f | XFS_QMOPT_RES_REGBLKS)  extern int xfs_qm_dqcheck(struct xfs_mount *, xfs_disk_dquot_t *, | 
