2005-04-17 02:20:36 +04:00
/*
2005-11-02 06:59:41 +03:00
* Copyright ( c ) 2000 - 2005 Silicon Graphics , Inc .
* All Rights Reserved .
2005-04-17 02:20:36 +04:00
*
2005-11-02 06:59:41 +03:00
* This program is free software ; you can redistribute it and / or
* modify it under the terms of the GNU General Public License as
2005-04-17 02:20:36 +04:00
* published by the Free Software Foundation .
*
2005-11-02 06:59:41 +03:00
* This program is distributed in the hope that it would be useful ,
* but WITHOUT ANY WARRANTY ; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
* GNU General Public License for more details .
2005-04-17 02:20:36 +04:00
*
2005-11-02 06:59:41 +03:00
* You should have received a copy of the GNU General Public License
* along with this program ; if not , write the Free Software Foundation ,
* Inc . , 51 Franklin St , Fifth Floor , Boston , MA 02110 - 1301 USA
2005-04-17 02:20:36 +04:00
*/
# include "xfs.h"
# include "xfs_fs.h"
2005-11-02 06:38:42 +03:00
# include "xfs_bit.h"
2005-04-17 02:20:36 +04:00
# include "xfs_log.h"
2005-11-02 06:38:42 +03:00
# include "xfs_inum.h"
2005-04-17 02:20:36 +04:00
# include "xfs_trans.h"
# include "xfs_sb.h"
# include "xfs_ag.h"
# include "xfs_alloc.h"
# include "xfs_quota.h"
# include "xfs_mount.h"
# include "xfs_bmap_btree.h"
# include "xfs_ialloc_btree.h"
# include "xfs_dinode.h"
# include "xfs_inode.h"
2005-11-02 06:38:42 +03:00
# include "xfs_ialloc.h"
# include "xfs_itable.h"
2005-04-17 02:20:36 +04:00
# include "xfs_rtalloc.h"
# include "xfs_error.h"
2005-11-02 06:38:42 +03:00
# include "xfs_bmap.h"
2005-04-17 02:20:36 +04:00
# include "xfs_attr.h"
# include "xfs_buf_item.h"
# include "xfs_trans_space.h"
# include "xfs_utils.h"
# include "xfs_qm.h"
2009-12-15 02:14:59 +03:00
# include "xfs_trace.h"
2005-04-17 02:20:36 +04:00
/*
* The global quota manager . There is only one of these for the entire
* system , _not_ one per file system . XQM keeps track of the overall
* quota functionality , including maintaining the freelist and hash
* tables of dquots .
*/
2009-03-29 11:51:00 +04:00
struct mutex xfs_Gqm_lock ;
2005-04-17 02:20:36 +04:00
struct xfs_qm * xfs_Gqm ;
2005-11-02 07:08:25 +03:00
uint ndquot ;
2005-04-17 02:20:36 +04:00
kmem_zone_t * qm_dqzone ;
kmem_zone_t * qm_dqtrxzone ;
STATIC void xfs_qm_list_init ( xfs_dqlist_t * , char * , int ) ;
STATIC void xfs_qm_list_destroy ( xfs_dqlist_t * ) ;
STATIC int xfs_qm_init_quotainos ( xfs_mount_t * ) ;
2005-06-21 09:36:52 +04:00
STATIC int xfs_qm_init_quotainfo ( xfs_mount_t * ) ;
2011-05-25 04:12:27 +04:00
STATIC int xfs_qm_shake ( struct shrinker * , struct shrink_control * ) ;
2005-04-17 02:20:36 +04:00
2007-07-17 15:03:17 +04:00
static struct shrinker xfs_qm_shaker = {
. shrink = xfs_qm_shake ,
. seeks = DEFAULT_SEEKS ,
} ;
2005-04-17 02:20:36 +04:00
/*
* Initialize the XQM structure .
* Note that there is not one quota manager per file system .
*/
STATIC struct xfs_qm *
xfs_Gqm_init ( void )
{
2005-11-02 07:08:25 +03:00
xfs_dqhash_t * udqhash , * gdqhash ;
xfs_qm_t * xqm ;
2006-09-28 05:04:43 +04:00
size_t hsize ;
uint i ;
2005-04-17 02:20:36 +04:00
/*
* Initialize the dquot hash tables .
*/
2006-09-28 05:03:27 +04:00
udqhash = kmem_zalloc_greedy ( & hsize ,
2007-08-16 10:49:11 +04:00
XFS_QM_HASHSIZE_LOW * sizeof ( xfs_dqhash_t ) ,
2010-01-21 00:55:30 +03:00
XFS_QM_HASHSIZE_HIGH * sizeof ( xfs_dqhash_t ) ) ;
if ( ! udqhash )
goto out ;
gdqhash = kmem_zalloc_large ( hsize ) ;
2010-02-06 11:45:15 +03:00
if ( ! gdqhash )
2010-01-21 00:55:30 +03:00
goto out_free_udqhash ;
2006-09-28 05:03:27 +04:00
hsize / = sizeof ( xfs_dqhash_t ) ;
2005-11-02 07:08:25 +03:00
ndquot = hsize < < 8 ;
2005-04-17 02:20:36 +04:00
2005-11-02 07:08:25 +03:00
xqm = kmem_zalloc ( sizeof ( xfs_qm_t ) , KM_SLEEP ) ;
xqm - > qm_dqhashmask = hsize - 1 ;
xqm - > qm_usr_dqhtable = udqhash ;
xqm - > qm_grp_dqhtable = gdqhash ;
2005-04-17 02:20:36 +04:00
ASSERT ( xqm - > qm_usr_dqhtable ! = NULL ) ;
ASSERT ( xqm - > qm_grp_dqhtable ! = NULL ) ;
for ( i = 0 ; i < hsize ; i + + ) {
xfs_qm_list_init ( & ( xqm - > qm_usr_dqhtable [ i ] ) , " uxdqh " , i ) ;
xfs_qm_list_init ( & ( xqm - > qm_grp_dqhtable [ i ] ) , " gxdqh " , i ) ;
}
/*
* Freelist of all dquots of all file systems
*/
2010-04-13 09:06:52 +04:00
INIT_LIST_HEAD ( & xqm - > qm_dqfrlist ) ;
xqm - > qm_dqfrlist_cnt = 0 ;
mutex_init ( & xqm - > qm_dqfrlist_lock ) ;
2005-04-17 02:20:36 +04:00
/*
* dquot zone . we register our own low - memory callback .
*/
if ( ! qm_dqzone ) {
xqm - > qm_dqzone = kmem_zone_init ( sizeof ( xfs_dquot_t ) ,
" xfs_dquots " ) ;
qm_dqzone = xqm - > qm_dqzone ;
} else
xqm - > qm_dqzone = qm_dqzone ;
2007-07-17 15:03:17 +04:00
register_shrinker ( & xfs_qm_shaker ) ;
2005-04-17 02:20:36 +04:00
/*
* The t_dqinfo portion of transactions .
*/
if ( ! qm_dqtrxzone ) {
xqm - > qm_dqtrxzone = kmem_zone_init ( sizeof ( xfs_dquot_acct_t ) ,
" xfs_dqtrx " ) ;
qm_dqtrxzone = xqm - > qm_dqtrxzone ;
} else
xqm - > qm_dqtrxzone = qm_dqtrxzone ;
atomic_set ( & xqm - > qm_totaldquots , 0 ) ;
xqm - > qm_dqfree_ratio = XFS_QM_DQFREE_RATIO ;
xqm - > qm_nrefs = 0 ;
return xqm ;
2010-01-21 00:55:30 +03:00
out_free_udqhash :
kmem_free_large ( udqhash ) ;
out :
return NULL ;
2005-04-17 02:20:36 +04:00
}
/*
* Destroy the global quota manager when its reference count goes to zero .
*/
2005-06-21 09:36:52 +04:00
STATIC void
2005-04-17 02:20:36 +04:00
xfs_qm_destroy (
struct xfs_qm * xqm )
{
2010-04-13 09:06:52 +04:00
struct xfs_dquot * dqp , * n ;
2005-04-17 02:20:36 +04:00
int hsize , i ;
ASSERT ( xqm ! = NULL ) ;
ASSERT ( xqm - > qm_nrefs = = 0 ) ;
2007-07-17 15:03:17 +04:00
unregister_shrinker ( & xfs_qm_shaker ) ;
2005-04-17 02:20:36 +04:00
hsize = xqm - > qm_dqhashmask + 1 ;
for ( i = 0 ; i < hsize ; i + + ) {
xfs_qm_list_destroy ( & ( xqm - > qm_usr_dqhtable [ i ] ) ) ;
xfs_qm_list_destroy ( & ( xqm - > qm_grp_dqhtable [ i ] ) ) ;
}
2010-01-21 00:55:30 +03:00
kmem_free_large ( xqm - > qm_usr_dqhtable ) ;
kmem_free_large ( xqm - > qm_grp_dqhtable ) ;
2005-04-17 02:20:36 +04:00
xqm - > qm_usr_dqhtable = NULL ;
xqm - > qm_grp_dqhtable = NULL ;
xqm - > qm_dqhashmask = 0 ;
2010-04-13 09:06:52 +04:00
/* frlist cleanup */
mutex_lock ( & xqm - > qm_dqfrlist_lock ) ;
list_for_each_entry_safe ( dqp , n , & xqm - > qm_dqfrlist , q_freelist ) {
xfs_dqlock ( dqp ) ;
list_del_init ( & dqp - > q_freelist ) ;
xfs_Gqm - > qm_dqfrlist_cnt - - ;
xfs_dqunlock ( dqp ) ;
xfs_qm_dqdestroy ( dqp ) ;
}
mutex_unlock ( & xqm - > qm_dqfrlist_lock ) ;
mutex_destroy ( & xqm - > qm_dqfrlist_lock ) ;
2008-05-19 10:31:57 +04:00
kmem_free ( xqm ) ;
2005-04-17 02:20:36 +04:00
}
/*
* Called at mount time to let XQM know that another file system is
* starting quotas . This isn ' t crucial information as the individual mount
* structures are pretty independent , but it helps the XQM keep a
* global view of what ' s going on .
*/
/* ARGSUSED */
STATIC int
xfs_qm_hold_quotafs_ref (
struct xfs_mount * mp )
{
/*
* Need to lock the xfs_Gqm structure for things like this . For example ,
* the structure could disappear between the entry to this routine and
* a HOLD operation if not locked .
*/
2009-02-09 10:38:39 +03:00
mutex_lock ( & xfs_Gqm_lock ) ;
2005-04-17 02:20:36 +04:00
2010-01-21 00:55:30 +03:00
if ( ! xfs_Gqm ) {
2005-04-17 02:20:36 +04:00
xfs_Gqm = xfs_Gqm_init ( ) ;
2010-05-26 19:57:23 +04:00
if ( ! xfs_Gqm ) {
mutex_unlock ( & xfs_Gqm_lock ) ;
2010-01-21 00:55:30 +03:00
return ENOMEM ;
2010-05-26 19:57:23 +04:00
}
2010-01-21 00:55:30 +03:00
}
2005-04-17 02:20:36 +04:00
/*
* We can keep a list of all filesystems with quotas mounted for
* debugging and statistical purposes , but . . .
* Just take a reference and get out .
*/
2009-02-09 10:38:39 +03:00
xfs_Gqm - > qm_nrefs + + ;
mutex_unlock ( & xfs_Gqm_lock ) ;
2005-04-17 02:20:36 +04:00
return 0 ;
}
/*
* Release the reference that a filesystem took at mount time ,
* so that we know when we need to destroy the entire quota manager .
*/
/* ARGSUSED */
STATIC void
xfs_qm_rele_quotafs_ref (
struct xfs_mount * mp )
{
2010-04-13 09:06:52 +04:00
xfs_dquot_t * dqp , * n ;
2005-04-17 02:20:36 +04:00
ASSERT ( xfs_Gqm ) ;
ASSERT ( xfs_Gqm - > qm_nrefs > 0 ) ;
/*
* Go thru the freelist and destroy all inactive dquots .
*/
2010-04-13 09:06:52 +04:00
mutex_lock ( & xfs_Gqm - > qm_dqfrlist_lock ) ;
2005-04-17 02:20:36 +04:00
2010-04-13 09:06:52 +04:00
list_for_each_entry_safe ( dqp , n , & xfs_Gqm - > qm_dqfrlist , q_freelist ) {
2005-04-17 02:20:36 +04:00
xfs_dqlock ( dqp ) ;
if ( dqp - > dq_flags & XFS_DQ_INACTIVE ) {
ASSERT ( dqp - > q_mount = = NULL ) ;
ASSERT ( ! XFS_DQ_IS_DIRTY ( dqp ) ) ;
2010-04-13 09:06:51 +04:00
ASSERT ( list_empty ( & dqp - > q_hashlist ) ) ;
2010-04-13 09:06:48 +04:00
ASSERT ( list_empty ( & dqp - > q_mplist ) ) ;
2010-04-13 09:06:52 +04:00
list_del_init ( & dqp - > q_freelist ) ;
xfs_Gqm - > qm_dqfrlist_cnt - - ;
2005-04-17 02:20:36 +04:00
xfs_dqunlock ( dqp ) ;
xfs_qm_dqdestroy ( dqp ) ;
} else {
xfs_dqunlock ( dqp ) ;
}
}
2010-04-13 09:06:52 +04:00
mutex_unlock ( & xfs_Gqm - > qm_dqfrlist_lock ) ;
2005-04-17 02:20:36 +04:00
/*
* Destroy the entire XQM . If somebody mounts with quotaon , this ' ll
* be restarted .
*/
2009-02-09 10:38:39 +03:00
mutex_lock ( & xfs_Gqm_lock ) ;
if ( - - xfs_Gqm - > qm_nrefs = = 0 ) {
2005-04-17 02:20:36 +04:00
xfs_qm_destroy ( xfs_Gqm ) ;
xfs_Gqm = NULL ;
}
2009-02-09 10:38:39 +03:00
mutex_unlock ( & xfs_Gqm_lock ) ;
2005-04-17 02:20:36 +04:00
}
/*
* Just destroy the quotainfo structure .
*/
void
2009-06-08 17:33:32 +04:00
xfs_qm_unmount (
struct xfs_mount * mp )
2005-04-17 02:20:36 +04:00
{
2009-06-08 17:33:32 +04:00
if ( mp - > m_quotainfo ) {
2010-04-20 11:02:29 +04:00
xfs_qm_dqpurge_all ( mp , XFS_QMOPT_QUOTALL ) ;
2005-04-17 02:20:36 +04:00
xfs_qm_destroy_quotainfo ( mp ) ;
2009-06-08 17:33:32 +04:00
}
2005-04-17 02:20:36 +04:00
}
/*
* This is called from xfs_mountfs to start quotas and initialize all
* necessary data structures like quotainfo . This is also responsible for
* running a quotacheck as necessary . We are guaranteed that the superblock
* is consistently read in at this point .
2008-04-10 06:20:31 +04:00
*
* If we fail here , the mount will continue with quota turned off . We don ' t
* need to inidicate success or failure at all .
2005-04-17 02:20:36 +04:00
*/
2008-04-10 06:20:31 +04:00
void
2005-04-17 02:20:36 +04:00
xfs_qm_mount_quotas (
2008-08-13 10:49:32 +04:00
xfs_mount_t * mp )
2005-04-17 02:20:36 +04:00
{
int error = 0 ;
uint sbf ;
/*
* If quotas on realtime volumes is not supported , we disable
* quotas immediately .
*/
if ( mp - > m_sb . sb_rextents ) {
2011-03-07 02:08:35 +03:00
xfs_notice ( mp , " Cannot turn on quotas for realtime filesystem " ) ;
2005-04-17 02:20:36 +04:00
mp - > m_qflags = 0 ;
goto write_changes ;
}
ASSERT ( XFS_IS_QUOTA_RUNNING ( mp ) ) ;
2005-09-02 10:43:48 +04:00
2005-04-17 02:20:36 +04:00
/*
* Allocate the quotainfo structure inside the mount struct , and
* create quotainode ( s ) , and change / rev superblock if necessary .
*/
2008-04-10 06:20:31 +04:00
error = xfs_qm_init_quotainfo ( mp ) ;
if ( error ) {
2005-04-17 02:20:36 +04:00
/*
* We must turn off quotas .
*/
ASSERT ( mp - > m_quotainfo = = NULL ) ;
mp - > m_qflags = 0 ;
goto write_changes ;
}
/*
* If any of the quotas are not consistent , do a quotacheck .
*/
2008-08-13 10:49:32 +04:00
if ( XFS_QM_NEED_QUOTACHECK ( mp ) ) {
2008-04-10 06:20:31 +04:00
error = xfs_qm_quotacheck ( mp ) ;
if ( error ) {
/* Quotacheck failed and disabled quotas. */
return ;
2005-04-17 02:20:36 +04:00
}
}
2007-05-08 07:49:09 +04:00
/*
* If one type of quotas is off , then it will lose its
* quotachecked status , since we won ' t be doing accounting for
* that type anymore .
*/
2008-04-10 06:20:31 +04:00
if ( ! XFS_IS_UQUOTA_ON ( mp ) )
2007-05-08 07:49:09 +04:00
mp - > m_qflags & = ~ XFS_UQUOTA_CHKD ;
2008-04-10 06:20:31 +04:00
if ( ! ( XFS_IS_GQUOTA_ON ( mp ) | | XFS_IS_PQUOTA_ON ( mp ) ) )
2007-05-08 07:49:09 +04:00
mp - > m_qflags & = ~ XFS_OQUOTA_CHKD ;
2005-09-02 10:43:48 +04:00
2005-04-17 02:20:36 +04:00
write_changes :
/*
2007-10-11 11:42:32 +04:00
* We actually don ' t have to acquire the m_sb_lock at all .
2005-04-17 02:20:36 +04:00
* This can only be called from mount , and that ' s single threaded . XXX
*/
2007-10-11 11:42:32 +04:00
spin_lock ( & mp - > m_sb_lock ) ;
2005-04-17 02:20:36 +04:00
sbf = mp - > m_sb . sb_qflags ;
mp - > m_sb . sb_qflags = mp - > m_qflags & XFS_MOUNT_QUOTA_ALL ;
2007-10-11 11:42:32 +04:00
spin_unlock ( & mp - > m_sb_lock ) ;
2005-04-17 02:20:36 +04:00
if ( sbf ! = ( mp - > m_qflags & XFS_MOUNT_QUOTA_ALL ) ) {
if ( xfs_qm_write_sb_changes ( mp , XFS_SB_QFLAGS ) ) {
/*
* We could only have been turning quotas off .
* We aren ' t in very good shape actually because
* the incore structures are convinced that quotas are
* off , but the on disk superblock doesn ' t know that !
*/
ASSERT ( ! ( XFS_IS_QUOTA_RUNNING ( mp ) ) ) ;
2011-03-07 02:05:35 +03:00
xfs_alert ( mp , " %s: Superblock update failed! " ,
__func__ ) ;
2005-04-17 02:20:36 +04:00
}
}
if ( error ) {
2011-03-07 02:05:35 +03:00
xfs_warn ( mp , " Failed to initialize disk quotas. " ) ;
2009-06-08 17:33:32 +04:00
return ;
2005-04-17 02:20:36 +04:00
}
}
/*
* Called from the vfsops layer .
*/
2008-12-03 14:20:36 +03:00
void
2005-04-17 02:20:36 +04:00
xfs_qm_unmount_quotas (
xfs_mount_t * mp )
{
/*
* Release the dquots that root inode , et al might be holding ,
* before we flush quotas and blow away the quotainfo structure .
*/
ASSERT ( mp - > m_rootip ) ;
xfs_qm_dqdetach ( mp - > m_rootip ) ;
if ( mp - > m_rbmip )
xfs_qm_dqdetach ( mp - > m_rbmip ) ;
if ( mp - > m_rsumip )
xfs_qm_dqdetach ( mp - > m_rsumip ) ;
/*
2008-12-03 14:20:36 +03:00
* Release the quota inodes .
2005-04-17 02:20:36 +04:00
*/
if ( mp - > m_quotainfo ) {
2008-12-03 14:20:36 +03:00
if ( mp - > m_quotainfo - > qi_uquotaip ) {
IRELE ( mp - > m_quotainfo - > qi_uquotaip ) ;
mp - > m_quotainfo - > qi_uquotaip = NULL ;
2005-04-17 02:20:36 +04:00
}
2008-12-03 14:20:36 +03:00
if ( mp - > m_quotainfo - > qi_gquotaip ) {
IRELE ( mp - > m_quotainfo - > qi_gquotaip ) ;
mp - > m_quotainfo - > qi_gquotaip = NULL ;
2005-04-17 02:20:36 +04:00
}
}
}
/*
* Flush all dquots of the given file system to disk . The dquots are
* _not_ purged from memory here , just their data written to disk .
*/
2005-06-21 09:36:52 +04:00
STATIC int
2005-04-17 02:20:36 +04:00
xfs_qm_dqflush_all (
2010-04-20 11:01:30 +04:00
struct xfs_mount * mp ,
int sync_mode )
2005-04-17 02:20:36 +04:00
{
2010-04-20 11:01:30 +04:00
struct xfs_quotainfo * q = mp - > m_quotainfo ;
int recl ;
struct xfs_dquot * dqp ;
int error ;
2005-04-17 02:20:36 +04:00
2010-04-20 11:01:30 +04:00
if ( ! q )
2006-01-15 04:37:08 +03:00
return 0 ;
2005-04-17 02:20:36 +04:00
again :
2010-04-20 11:01:30 +04:00
mutex_lock ( & q - > qi_dqlist_lock ) ;
list_for_each_entry ( dqp , & q - > qi_dqlist , q_mplist ) {
2005-04-17 02:20:36 +04:00
xfs_dqlock ( dqp ) ;
if ( ! XFS_DQ_IS_DIRTY ( dqp ) ) {
xfs_dqunlock ( dqp ) ;
continue ;
}
2009-12-15 02:14:59 +03:00
2005-04-17 02:20:36 +04:00
/* XXX a sentinel would be better */
2010-04-20 11:01:30 +04:00
recl = q - > qi_dqreclaims ;
2008-08-13 10:41:43 +04:00
if ( ! xfs_dqflock_nowait ( dqp ) ) {
2005-04-17 02:20:36 +04:00
/*
* If we can ' t grab the flush lock then check
* to see if the dquot has been flushed delayed
* write . If so , grab its buffer and send it
* out immediately . We ' ll be able to acquire
* the flush lock when the I / O completes .
*/
xfs_qm_dqflock_pushbuf_wait ( dqp ) ;
}
/*
* Let go of the mplist lock . We don ' t want to hold it
* across a disk write .
*/
2010-04-20 11:01:30 +04:00
mutex_unlock ( & q - > qi_dqlist_lock ) ;
2010-02-04 01:48:58 +03:00
error = xfs_qm_dqflush ( dqp , sync_mode ) ;
2005-04-17 02:20:36 +04:00
xfs_dqunlock ( dqp ) ;
if ( error )
2006-01-15 04:37:08 +03:00
return error ;
2005-04-17 02:20:36 +04:00
2010-04-20 11:01:30 +04:00
mutex_lock ( & q - > qi_dqlist_lock ) ;
if ( recl ! = q - > qi_dqreclaims ) {
mutex_unlock ( & q - > qi_dqlist_lock ) ;
2005-04-17 02:20:36 +04:00
/* XXX restart limit */
goto again ;
}
}
2010-04-20 11:01:30 +04:00
mutex_unlock ( & q - > qi_dqlist_lock ) ;
2005-04-17 02:20:36 +04:00
/* return ! busy */
2006-01-15 04:37:08 +03:00
return 0 ;
2005-04-17 02:20:36 +04:00
}
/*
* Release the group dquot pointers the user dquots may be
* carrying around as a hint . mplist is locked on entry and exit .
*/
STATIC void
xfs_qm_detach_gdquots (
2010-04-20 11:01:30 +04:00
struct xfs_mount * mp )
2005-04-17 02:20:36 +04:00
{
2010-04-20 11:01:30 +04:00
struct xfs_quotainfo * q = mp - > m_quotainfo ;
struct xfs_dquot * dqp , * gdqp ;
int nrecl ;
2005-04-17 02:20:36 +04:00
again :
2010-04-20 11:01:30 +04:00
ASSERT ( mutex_is_locked ( & q - > qi_dqlist_lock ) ) ;
list_for_each_entry ( dqp , & q - > qi_dqlist , q_mplist ) {
2005-04-17 02:20:36 +04:00
xfs_dqlock ( dqp ) ;
if ( ( gdqp = dqp - > q_gdquot ) ) {
xfs_dqlock ( gdqp ) ;
dqp - > q_gdquot = NULL ;
}
xfs_dqunlock ( dqp ) ;
if ( gdqp ) {
/*
* Can ' t hold the mplist lock across a dqput .
* XXXmust convert to marker based iterations here .
*/
2010-04-20 11:01:30 +04:00
nrecl = q - > qi_dqreclaims ;
mutex_unlock ( & q - > qi_dqlist_lock ) ;
2005-04-17 02:20:36 +04:00
xfs_qm_dqput ( gdqp ) ;
2010-04-20 11:01:30 +04:00
mutex_lock ( & q - > qi_dqlist_lock ) ;
if ( nrecl ! = q - > qi_dqreclaims )
2005-04-17 02:20:36 +04:00
goto again ;
}
}
}
/*
* Go through all the incore dquots of this file system and take them
* off the mplist and hashlist , if the dquot type matches the dqtype
* parameter . This is used when turning off quota accounting for
* users and / or groups , as well as when the filesystem is unmounting .
*/
STATIC int
xfs_qm_dqpurge_int (
2010-04-20 11:01:30 +04:00
struct xfs_mount * mp ,
uint flags )
2005-04-17 02:20:36 +04:00
{
2010-04-20 11:01:30 +04:00
struct xfs_quotainfo * q = mp - > m_quotainfo ;
struct xfs_dquot * dqp , * n ;
uint dqtype ;
int nrecl ;
int nmisses ;
2005-04-17 02:20:36 +04:00
2010-04-20 11:01:30 +04:00
if ( ! q )
2006-01-15 04:37:08 +03:00
return 0 ;
2005-04-17 02:20:36 +04:00
dqtype = ( flags & XFS_QMOPT_UQUOTA ) ? XFS_DQ_USER : 0 ;
2005-06-21 09:38:48 +04:00
dqtype | = ( flags & XFS_QMOPT_PQUOTA ) ? XFS_DQ_PROJ : 0 ;
2005-04-17 02:20:36 +04:00
dqtype | = ( flags & XFS_QMOPT_GQUOTA ) ? XFS_DQ_GROUP : 0 ;
2010-04-20 11:01:30 +04:00
mutex_lock ( & q - > qi_dqlist_lock ) ;
2005-04-17 02:20:36 +04:00
/*
* In the first pass through all incore dquots of this filesystem ,
* we release the group dquot pointers the user dquots may be
* carrying around as a hint . We need to do this irrespective of
* what ' s being turned off .
*/
xfs_qm_detach_gdquots ( mp ) ;
again :
nmisses = 0 ;
2010-04-20 11:01:30 +04:00
ASSERT ( mutex_is_locked ( & q - > qi_dqlist_lock ) ) ;
2005-04-17 02:20:36 +04:00
/*
* Try to get rid of all of the unwanted dquots . The idea is to
* get them off mplist and hashlist , but leave them on freelist .
*/
2010-04-20 11:01:30 +04:00
list_for_each_entry_safe ( dqp , n , & q - > qi_dqlist , q_mplist ) {
2005-04-17 02:20:36 +04:00
/*
* It ' s OK to look at the type without taking dqlock here .
* We ' re holding the mplist lock here , and that ' s needed for
* a dqreclaim .
*/
2010-04-13 09:06:48 +04:00
if ( ( dqp - > dq_flags & dqtype ) = = 0 )
2005-04-17 02:20:36 +04:00
continue ;
2009-02-09 10:47:22 +03:00
if ( ! mutex_trylock ( & dqp - > q_hash - > qh_lock ) ) {
2010-04-20 11:01:30 +04:00
nrecl = q - > qi_dqreclaims ;
mutex_unlock ( & q - > qi_dqlist_lock ) ;
2009-02-09 10:47:22 +03:00
mutex_lock ( & dqp - > q_hash - > qh_lock ) ;
2010-04-20 11:01:30 +04:00
mutex_lock ( & q - > qi_dqlist_lock ) ;
2005-04-17 02:20:36 +04:00
/*
* XXXTheoretically , we can get into a very long
* ping pong game here .
* No one can be adding dquots to the mplist at
* this point , but somebody might be taking things off .
*/
2010-04-20 11:01:30 +04:00
if ( nrecl ! = q - > qi_dqreclaims ) {
2009-02-09 10:47:22 +03:00
mutex_unlock ( & dqp - > q_hash - > qh_lock ) ;
2005-04-17 02:20:36 +04:00
goto again ;
}
}
/*
* Take the dquot off the mplist and hashlist . It may remain on
* freelist in INACTIVE state .
*/
2008-05-19 10:34:04 +04:00
nmisses + = xfs_qm_dqpurge ( dqp ) ;
2005-04-17 02:20:36 +04:00
}
2010-04-20 11:01:30 +04:00
mutex_unlock ( & q - > qi_dqlist_lock ) ;
2005-04-17 02:20:36 +04:00
return nmisses ;
}
int
xfs_qm_dqpurge_all (
xfs_mount_t * mp ,
uint flags )
{
int ndquots ;
/*
* Purge the dquot cache .
* None of the dquots should really be busy at this point .
*/
if ( mp - > m_quotainfo ) {
while ( ( ndquots = xfs_qm_dqpurge_int ( mp , flags ) ) ) {
delay ( ndquots * 10 ) ;
}
}
return 0 ;
}
STATIC int
xfs_qm_dqattach_one (
xfs_inode_t * ip ,
xfs_dqid_t id ,
uint type ,
uint doalloc ,
xfs_dquot_t * udqhint , /* hint */
xfs_dquot_t * * IO_idqpp )
{
xfs_dquot_t * dqp ;
int error ;
2008-04-22 11:34:00 +04:00
ASSERT ( xfs_isilocked ( ip , XFS_ILOCK_EXCL ) ) ;
2005-04-17 02:20:36 +04:00
error = 0 ;
2009-02-08 23:51:42 +03:00
2005-04-17 02:20:36 +04:00
/*
* See if we already have it in the inode itself . IO_idqpp is
* & i_udquot or & i_gdquot . This made the code look weird , but
* made the logic a lot simpler .
*/
2009-02-08 23:51:42 +03:00
dqp = * IO_idqpp ;
if ( dqp ) {
2009-12-15 02:14:59 +03:00
trace_xfs_dqattach_found ( dqp ) ;
2009-02-08 23:51:42 +03:00
return 0 ;
2005-04-17 02:20:36 +04:00
}
/*
* udqhint is the i_udquot field in inode , and is non - NULL only
2005-06-21 09:38:48 +04:00
* when the type arg is group / project . Its purpose is to save a
2005-04-17 02:20:36 +04:00
* lookup by dqid ( xfs_qm_dqget ) by caching a group dquot inside
* the user dquot .
*/
2009-02-08 23:51:42 +03:00
if ( udqhint ) {
ASSERT ( type = = XFS_DQ_GROUP | | type = = XFS_DQ_PROJ ) ;
2005-04-17 02:20:36 +04:00
xfs_dqlock ( udqhint ) ;
2009-02-08 23:51:42 +03:00
/*
* No need to take dqlock to look at the id .
*
* The ID can ' t change until it gets reclaimed , and it won ' t
* be reclaimed as long as we have a ref from inode and we
* hold the ilock .
*/
dqp = udqhint - > q_gdquot ;
if ( dqp & & be32_to_cpu ( dqp - > q_core . d_id ) = = id ) {
xfs_dqlock ( dqp ) ;
XFS_DQHOLD ( dqp ) ;
ASSERT ( * IO_idqpp = = NULL ) ;
* IO_idqpp = dqp ;
2005-04-17 02:20:36 +04:00
xfs_dqunlock ( dqp ) ;
xfs_dqunlock ( udqhint ) ;
2009-02-08 23:51:42 +03:00
return 0 ;
2005-04-17 02:20:36 +04:00
}
2009-02-08 23:51:42 +03:00
/*
* We can ' t hold a dquot lock when we call the dqget code .
* We ' ll deadlock in no time , because of ( not conforming to )
* lock ordering - the inodelock comes before any dquot lock ,
* and we may drop and reacquire the ilock in xfs_qm_dqget ( ) .
*/
2005-04-17 02:20:36 +04:00
xfs_dqunlock ( udqhint ) ;
2009-02-08 23:51:42 +03:00
}
2005-04-17 02:20:36 +04:00
/*
* Find the dquot from somewhere . This bumps the
* reference count of dquot and returns it locked .
* This can return ENOENT if dquot didn ' t exist on
* disk and we didn ' t ask it to allocate ;
* ESRCH if quotas got turned off suddenly .
*/
2009-02-08 23:51:42 +03:00
error = xfs_qm_dqget ( ip - > i_mount , ip , id , type , XFS_QMOPT_DOWARN , & dqp ) ;
if ( error )
return error ;
2005-04-17 02:20:36 +04:00
2009-12-15 02:14:59 +03:00
trace_xfs_dqattach_get ( dqp ) ;
2009-02-08 23:51:42 +03:00
2005-04-17 02:20:36 +04:00
/*
* dqget may have dropped and re - acquired the ilock , but it guarantees
* that the dquot returned is the one that should go in the inode .
*/
* IO_idqpp = dqp ;
2009-02-08 23:51:42 +03:00
xfs_dqunlock ( dqp ) ;
return 0 ;
2005-04-17 02:20:36 +04:00
}
/*
* Given a udquot and gdquot , attach a ptr to the group dquot in the
* udquot as a hint for future lookups . The idea sounds simple , but the
* execution isn ' t , because the udquot might have a group dquot attached
2006-03-29 02:55:14 +04:00
* already and getting rid of that gets us into lock ordering constraints .
2005-04-17 02:20:36 +04:00
* The process is complicated more by the fact that the dquots may or may not
* be locked on entry .
*/
STATIC void
xfs_qm_dqattach_grouphint (
xfs_dquot_t * udq ,
2009-02-08 23:51:42 +03:00
xfs_dquot_t * gdq )
2005-04-17 02:20:36 +04:00
{
xfs_dquot_t * tmp ;
2009-02-08 23:51:42 +03:00
xfs_dqlock ( udq ) ;
2005-04-17 02:20:36 +04:00
if ( ( tmp = udq - > q_gdquot ) ) {
if ( tmp = = gdq ) {
2009-02-08 23:51:42 +03:00
xfs_dqunlock ( udq ) ;
2005-04-17 02:20:36 +04:00
return ;
}
udq - > q_gdquot = NULL ;
/*
* We can ' t keep any dqlocks when calling dqrele ,
* because the freelist lock comes before dqlocks .
*/
xfs_dqunlock ( udq ) ;
/*
* we took a hard reference once upon a time in dqget ,
* so give it back when the udquot no longer points at it
* dqput ( ) does the unlocking of the dquot .
*/
xfs_qm_dqrele ( tmp ) ;
xfs_dqlock ( udq ) ;
xfs_dqlock ( gdq ) ;
} else {
ASSERT ( XFS_DQ_IS_LOCKED ( udq ) ) ;
2009-02-08 23:51:42 +03:00
xfs_dqlock ( gdq ) ;
2005-04-17 02:20:36 +04:00
}
ASSERT ( XFS_DQ_IS_LOCKED ( udq ) ) ;
ASSERT ( XFS_DQ_IS_LOCKED ( gdq ) ) ;
/*
* Somebody could have attached a gdquot here ,
* when we dropped the uqlock . If so , just do nothing .
*/
if ( udq - > q_gdquot = = NULL ) {
XFS_DQHOLD ( gdq ) ;
udq - > q_gdquot = gdq ;
}
2009-02-08 23:51:42 +03:00
xfs_dqunlock ( gdq ) ;
xfs_dqunlock ( udq ) ;
2005-04-17 02:20:36 +04:00
}
/*
2005-06-21 09:38:48 +04:00
* Given a locked inode , attach dquot ( s ) to it , taking U / G / P - QUOTAON
* into account .
2005-04-17 02:20:36 +04:00
* If XFS_QMOPT_DQALLOC , the dquot ( s ) will be allocated if needed .
* Inode may get unlocked and relocked in here , and the caller must deal with
* the consequences .
*/
int
2009-06-08 17:33:32 +04:00
xfs_qm_dqattach_locked (
2005-04-17 02:20:36 +04:00
xfs_inode_t * ip ,
uint flags )
{
xfs_mount_t * mp = ip - > i_mount ;
uint nquotas = 0 ;
int error = 0 ;
2009-06-08 17:33:32 +04:00
if ( ! XFS_IS_QUOTA_RUNNING ( mp ) | |
! XFS_IS_QUOTA_ON ( mp ) | |
! XFS_NOT_DQATTACHED ( mp , ip ) | |
ip - > i_ino = = mp - > m_sb . sb_uquotino | |
ip - > i_ino = = mp - > m_sb . sb_gquotino )
2006-01-15 04:37:08 +03:00
return 0 ;
2005-04-17 02:20:36 +04:00
2009-06-08 17:33:32 +04:00
ASSERT ( xfs_isilocked ( ip , XFS_ILOCK_EXCL ) ) ;
2005-04-17 02:20:36 +04:00
if ( XFS_IS_UQUOTA_ON ( mp ) ) {
error = xfs_qm_dqattach_one ( ip , ip - > i_d . di_uid , XFS_DQ_USER ,
flags & XFS_QMOPT_DQALLOC ,
NULL , & ip - > i_udquot ) ;
if ( error )
goto done ;
nquotas + + ;
}
2008-04-22 11:34:00 +04:00
ASSERT ( xfs_isilocked ( ip , XFS_ILOCK_EXCL ) ) ;
2005-06-21 09:38:48 +04:00
if ( XFS_IS_OQUOTA_ON ( mp ) ) {
error = XFS_IS_GQUOTA_ON ( mp ) ?
xfs_qm_dqattach_one ( ip , ip - > i_d . di_gid , XFS_DQ_GROUP ,
flags & XFS_QMOPT_DQALLOC ,
ip - > i_udquot , & ip - > i_gdquot ) :
2010-09-26 10:10:18 +04:00
xfs_qm_dqattach_one ( ip , xfs_get_projid ( ip ) , XFS_DQ_PROJ ,
2005-04-17 02:20:36 +04:00
flags & XFS_QMOPT_DQALLOC ,
ip - > i_udquot , & ip - > i_gdquot ) ;
/*
* Don ' t worry about the udquot that we may have
* attached above . It ' ll get detached , if not already .
*/
if ( error )
goto done ;
nquotas + + ;
}
/*
* Attach this group quota to the user quota as a hint .
* This WON ' T , in general , result in a thrash .
*/
if ( nquotas = = 2 ) {
2008-04-22 11:34:00 +04:00
ASSERT ( xfs_isilocked ( ip , XFS_ILOCK_EXCL ) ) ;
2005-04-17 02:20:36 +04:00
ASSERT ( ip - > i_udquot ) ;
ASSERT ( ip - > i_gdquot ) ;
/*
* We may or may not have the i_udquot locked at this point ,
* but this check is OK since we don ' t depend on the i_gdquot to
* be accurate 100 % all the time . It is just a hint , and this
* will succeed in general .
*/
if ( ip - > i_udquot - > q_gdquot = = ip - > i_gdquot )
goto done ;
/*
* Attach i_gdquot to the gdquot hint inside the i_udquot .
*/
2009-02-08 23:51:42 +03:00
xfs_qm_dqattach_grouphint ( ip - > i_udquot , ip - > i_gdquot ) ;
2005-04-17 02:20:36 +04:00
}
2009-06-08 17:33:32 +04:00
done :
2011-07-13 15:43:50 +04:00
# ifdef DEBUG
if ( ! error ) {
2005-04-17 02:20:36 +04:00
if ( XFS_IS_UQUOTA_ON ( mp ) )
ASSERT ( ip - > i_udquot ) ;
2005-06-21 09:38:48 +04:00
if ( XFS_IS_OQUOTA_ON ( mp ) )
2005-04-17 02:20:36 +04:00
ASSERT ( ip - > i_gdquot ) ;
}
2009-06-08 17:33:32 +04:00
ASSERT ( xfs_isilocked ( ip , XFS_ILOCK_EXCL ) ) ;
2005-04-17 02:20:36 +04:00
# endif
2009-06-08 17:33:32 +04:00
return error ;
}
2005-04-17 02:20:36 +04:00
2009-06-08 17:33:32 +04:00
int
xfs_qm_dqattach (
struct xfs_inode * ip ,
uint flags )
{
int error ;
xfs_ilock ( ip , XFS_ILOCK_EXCL ) ;
error = xfs_qm_dqattach_locked ( ip , flags ) ;
xfs_iunlock ( ip , XFS_ILOCK_EXCL ) ;
2005-04-17 02:20:36 +04:00
2006-01-15 04:37:08 +03:00
return error ;
2005-04-17 02:20:36 +04:00
}
/*
* Release dquots ( and their references ) if any .
* The inode should be locked EXCL except when this ' s called by
* xfs_ireclaim .
*/
void
xfs_qm_dqdetach (
xfs_inode_t * ip )
{
if ( ! ( ip - > i_udquot | | ip - > i_gdquot ) )
return ;
2009-12-15 02:14:59 +03:00
trace_xfs_dquot_dqdetach ( ip ) ;
2005-04-17 02:20:36 +04:00
ASSERT ( ip - > i_ino ! = ip - > i_mount - > m_sb . sb_uquotino ) ;
ASSERT ( ip - > i_ino ! = ip - > i_mount - > m_sb . sb_gquotino ) ;
if ( ip - > i_udquot ) {
xfs_qm_dqrele ( ip - > i_udquot ) ;
ip - > i_udquot = NULL ;
}
if ( ip - > i_gdquot ) {
xfs_qm_dqrele ( ip - > i_gdquot ) ;
ip - > i_gdquot = NULL ;
}
}
int
xfs_qm_sync (
2010-04-20 11:01:30 +04:00
struct xfs_mount * mp ,
int flags )
2005-04-17 02:20:36 +04:00
{
2010-04-20 11:01:30 +04:00
struct xfs_quotainfo * q = mp - > m_quotainfo ;
int recl , restarts ;
struct xfs_dquot * dqp ;
int error ;
2005-04-17 02:20:36 +04:00
2009-06-08 17:33:32 +04:00
if ( ! XFS_IS_QUOTA_RUNNING ( mp ) | | ! XFS_IS_QUOTA_ON ( mp ) )
2007-11-23 08:27:42 +03:00
return 0 ;
2005-04-17 02:20:36 +04:00
restarts = 0 ;
again :
2010-04-20 11:01:30 +04:00
mutex_lock ( & q - > qi_dqlist_lock ) ;
2005-04-17 02:20:36 +04:00
/*
* dqpurge_all ( ) also takes the mplist lock and iterate thru all dquots
* in quotaoff . However , if the QUOTA_ACTIVE bits are not cleared
* when we have the mplist lock , we know that dquots will be consistent
* as long as we have it locked .
*/
2010-04-13 09:06:48 +04:00
if ( ! XFS_IS_QUOTA_ON ( mp ) ) {
2010-04-20 11:01:30 +04:00
mutex_unlock ( & q - > qi_dqlist_lock ) ;
2006-01-15 04:37:08 +03:00
return 0 ;
2005-04-17 02:20:36 +04:00
}
2010-04-20 11:01:30 +04:00
ASSERT ( mutex_is_locked ( & q - > qi_dqlist_lock ) ) ;
list_for_each_entry ( dqp , & q - > qi_dqlist , q_mplist ) {
2005-04-17 02:20:36 +04:00
/*
* If this is vfs_sync calling , then skip the dquots that
* don ' t ' seem ' to be dirty . ie . don ' t acquire dqlock .
* This is very similar to what xfs_sync does with inodes .
*/
2009-06-08 17:37:16 +04:00
if ( flags & SYNC_TRYLOCK ) {
if ( ! XFS_DQ_IS_DIRTY ( dqp ) )
2005-04-17 02:20:36 +04:00
continue ;
2009-06-08 17:37:16 +04:00
if ( ! xfs_qm_dqlock_nowait ( dqp ) )
2005-04-17 02:20:36 +04:00
continue ;
} else {
xfs_dqlock ( dqp ) ;
}
/*
* Now , find out for sure if this dquot is dirty or not .
*/
if ( ! XFS_DQ_IS_DIRTY ( dqp ) ) {
xfs_dqunlock ( dqp ) ;
continue ;
}
/* XXX a sentinel would be better */
2010-04-20 11:01:30 +04:00
recl = q - > qi_dqreclaims ;
2008-08-13 10:41:43 +04:00
if ( ! xfs_dqflock_nowait ( dqp ) ) {
2009-06-08 17:37:16 +04:00
if ( flags & SYNC_TRYLOCK ) {
2005-04-17 02:20:36 +04:00
xfs_dqunlock ( dqp ) ;
continue ;
}
/*
* If we can ' t grab the flush lock then if the caller
2006-03-29 02:55:14 +04:00
* really wanted us to give this our best shot , so
2005-04-17 02:20:36 +04:00
* see if we can give a push to the buffer before we wait
* on the flush lock . At this point , we know that
2006-03-29 02:55:14 +04:00
* even though the dquot is being flushed ,
2005-04-17 02:20:36 +04:00
* it has ( new ) dirty data .
*/
xfs_qm_dqflock_pushbuf_wait ( dqp ) ;
}
/*
* Let go of the mplist lock . We don ' t want to hold it
* across a disk write
*/
2010-04-20 11:01:30 +04:00
mutex_unlock ( & q - > qi_dqlist_lock ) ;
2010-02-04 01:48:58 +03:00
error = xfs_qm_dqflush ( dqp , flags ) ;
2005-04-17 02:20:36 +04:00
xfs_dqunlock ( dqp ) ;
if ( error & & XFS_FORCED_SHUTDOWN ( mp ) )
2006-01-15 04:37:08 +03:00
return 0 ; /* Need to prevent umount failure */
2005-04-17 02:20:36 +04:00
else if ( error )
2006-01-15 04:37:08 +03:00
return error ;
2005-04-17 02:20:36 +04:00
2010-04-20 11:01:30 +04:00
mutex_lock ( & q - > qi_dqlist_lock ) ;
if ( recl ! = q - > qi_dqreclaims ) {
2005-04-17 02:20:36 +04:00
if ( + + restarts > = XFS_QM_SYNC_MAX_RESTARTS )
break ;
2010-04-20 11:01:30 +04:00
mutex_unlock ( & q - > qi_dqlist_lock ) ;
2005-04-17 02:20:36 +04:00
goto again ;
}
}
2010-04-20 11:01:30 +04:00
mutex_unlock ( & q - > qi_dqlist_lock ) ;
2006-01-15 04:37:08 +03:00
return 0 ;
2005-04-17 02:20:36 +04:00
}
2009-01-19 04:03:11 +03:00
/*
* The hash chains and the mplist use the same xfs_dqhash structure as
* their list head , but we can take the mplist qh_lock and one of the
* hash qh_locks at the same time without any problem as they aren ' t
* related .
*/
static struct lock_class_key xfs_quota_mplist_class ;
2005-04-17 02:20:36 +04:00
/*
* This initializes all the quota information that ' s kept in the
* mount structure
*/
2005-06-21 09:36:52 +04:00
STATIC int
2005-04-17 02:20:36 +04:00
xfs_qm_init_quotainfo (
xfs_mount_t * mp )
{
xfs_quotainfo_t * qinf ;
int error ;
xfs_dquot_t * dqp ;
ASSERT ( XFS_IS_QUOTA_RUNNING ( mp ) ) ;
/*
* Tell XQM that we exist as soon as possible .
*/
if ( ( error = xfs_qm_hold_quotafs_ref ( mp ) ) ) {
2006-01-15 04:37:08 +03:00
return error ;
2005-04-17 02:20:36 +04:00
}
qinf = mp - > m_quotainfo = kmem_zalloc ( sizeof ( xfs_quotainfo_t ) , KM_SLEEP ) ;
/*
* See if quotainodes are setup , and if not , allocate them ,
* and change the superblock accordingly .
*/
if ( ( error = xfs_qm_init_quotainos ( mp ) ) ) {
2008-05-19 10:31:57 +04:00
kmem_free ( qinf ) ;
2005-04-17 02:20:36 +04:00
mp - > m_quotainfo = NULL ;
2006-01-15 04:37:08 +03:00
return error ;
2005-04-17 02:20:36 +04:00
}
2010-04-13 09:06:48 +04:00
INIT_LIST_HEAD ( & qinf - > qi_dqlist ) ;
mutex_init ( & qinf - > qi_dqlist_lock ) ;
lockdep_set_class ( & qinf - > qi_dqlist_lock , & xfs_quota_mplist_class ) ;
2009-01-19 04:03:11 +03:00
2005-04-17 02:20:36 +04:00
qinf - > qi_dqreclaims = 0 ;
/* mutex used to serialize quotaoffs */
2006-01-10 02:59:21 +03:00
mutex_init ( & qinf - > qi_quotaofflock ) ;
2005-04-17 02:20:36 +04:00
/* Precalc some constants */
qinf - > qi_dqchunklen = XFS_FSB_TO_BB ( mp , XFS_DQUOT_CLUSTER_SIZE_FSB ) ;
ASSERT ( qinf - > qi_dqchunklen ) ;
qinf - > qi_dqperchunk = BBTOB ( qinf - > qi_dqchunklen ) ;
do_div ( qinf - > qi_dqperchunk , sizeof ( xfs_dqblk_t ) ) ;
mp - > m_qflags | = ( mp - > m_sb . sb_qflags & XFS_ALL_QUOTA_CHKD ) ;
/*
* We try to get the limits from the superuser ' s limits fields .
* This is quite hacky , but it is standard quota practice .
* We look at the USR dquot with id = = 0 first , but if user quotas
* are not enabled we goto the GRP dquot with id = = 0.
* We don ' t really care to keep separate default limits for user
* and group quotas , at least not at this point .
*/
error = xfs_qm_dqget ( mp , NULL , ( xfs_dqid_t ) 0 ,
2005-06-21 09:38:48 +04:00
XFS_IS_UQUOTA_RUNNING ( mp ) ? XFS_DQ_USER :
( XFS_IS_GQUOTA_RUNNING ( mp ) ? XFS_DQ_GROUP :
XFS_DQ_PROJ ) ,
2005-04-17 02:20:36 +04:00
XFS_QMOPT_DQSUSER | XFS_QMOPT_DOWARN ,
& dqp ) ;
if ( ! error ) {
xfs_disk_dquot_t * ddqp = & dqp - > q_core ;
/*
* The warnings and timers set the grace period given to
* a user or group before he or she can not perform any
* more writing . If it is zero , a default is used .
*/
2005-11-02 07:01:12 +03:00
qinf - > qi_btimelimit = ddqp - > d_btimer ?
be32_to_cpu ( ddqp - > d_btimer ) : XFS_QM_BTIMELIMIT ;
qinf - > qi_itimelimit = ddqp - > d_itimer ?
be32_to_cpu ( ddqp - > d_itimer ) : XFS_QM_ITIMELIMIT ;
qinf - > qi_rtbtimelimit = ddqp - > d_rtbtimer ?
be32_to_cpu ( ddqp - > d_rtbtimer ) : XFS_QM_RTBTIMELIMIT ;
qinf - > qi_bwarnlimit = ddqp - > d_bwarns ?
be16_to_cpu ( ddqp - > d_bwarns ) : XFS_QM_BWARNLIMIT ;
qinf - > qi_iwarnlimit = ddqp - > d_iwarns ?
be16_to_cpu ( ddqp - > d_iwarns ) : XFS_QM_IWARNLIMIT ;
qinf - > qi_rtbwarnlimit = ddqp - > d_rtbwarns ?
be16_to_cpu ( ddqp - > d_rtbwarns ) : XFS_QM_RTBWARNLIMIT ;
qinf - > qi_bhardlimit = be64_to_cpu ( ddqp - > d_blk_hardlimit ) ;
qinf - > qi_bsoftlimit = be64_to_cpu ( ddqp - > d_blk_softlimit ) ;
qinf - > qi_ihardlimit = be64_to_cpu ( ddqp - > d_ino_hardlimit ) ;
qinf - > qi_isoftlimit = be64_to_cpu ( ddqp - > d_ino_softlimit ) ;
qinf - > qi_rtbhardlimit = be64_to_cpu ( ddqp - > d_rtb_hardlimit ) ;
qinf - > qi_rtbsoftlimit = be64_to_cpu ( ddqp - > d_rtb_softlimit ) ;
2005-04-17 02:20:36 +04:00
/*
* We sent the XFS_QMOPT_DQSUSER flag to dqget because
* we don ' t want this dquot cached . We haven ' t done a
* quotacheck yet , and quotacheck doesn ' t like incore dquots .
*/
xfs_qm_dqdestroy ( dqp ) ;
} else {
qinf - > qi_btimelimit = XFS_QM_BTIMELIMIT ;
qinf - > qi_itimelimit = XFS_QM_ITIMELIMIT ;
qinf - > qi_rtbtimelimit = XFS_QM_RTBTIMELIMIT ;
qinf - > qi_bwarnlimit = XFS_QM_BWARNLIMIT ;
qinf - > qi_iwarnlimit = XFS_QM_IWARNLIMIT ;
2005-06-21 09:48:47 +04:00
qinf - > qi_rtbwarnlimit = XFS_QM_RTBWARNLIMIT ;
2005-04-17 02:20:36 +04:00
}
2006-01-15 04:37:08 +03:00
return 0 ;
2005-04-17 02:20:36 +04:00
}
/*
* Gets called when unmounting a filesystem or when all quotas get
* turned off .
* This purges the quota inodes , destroys locks and frees itself .
*/
void
xfs_qm_destroy_quotainfo (
xfs_mount_t * mp )
{
xfs_quotainfo_t * qi ;
qi = mp - > m_quotainfo ;
ASSERT ( qi ! = NULL ) ;
ASSERT ( xfs_Gqm ! = NULL ) ;
/*
* Release the reference that XQM kept , so that we know
* when the XQM structure should be freed . We cannot assume
* that xfs_Gqm is non - null after this point .
*/
xfs_qm_rele_quotafs_ref ( mp ) ;
2010-04-13 09:06:48 +04:00
ASSERT ( list_empty ( & qi - > qi_dqlist ) ) ;
mutex_destroy ( & qi - > qi_dqlist_lock ) ;
2005-04-17 02:20:36 +04:00
if ( qi - > qi_uquotaip ) {
2008-07-18 11:12:43 +04:00
IRELE ( qi - > qi_uquotaip ) ;
2005-04-17 02:20:36 +04:00
qi - > qi_uquotaip = NULL ; /* paranoia */
}
if ( qi - > qi_gquotaip ) {
2008-07-18 11:12:43 +04:00
IRELE ( qi - > qi_gquotaip ) ;
2005-04-17 02:20:36 +04:00
qi - > qi_gquotaip = NULL ;
}
mutex_destroy ( & qi - > qi_quotaofflock ) ;
2008-05-19 10:31:57 +04:00
kmem_free ( qi ) ;
2005-04-17 02:20:36 +04:00
mp - > m_quotainfo = NULL ;
}
/* ------------------- PRIVATE STATIC FUNCTIONS ----------------------- */
/* ARGSUSED */
STATIC void
xfs_qm_list_init (
xfs_dqlist_t * list ,
char * str ,
int n )
{
2006-01-10 02:59:21 +03:00
mutex_init ( & list - > qh_lock ) ;
2010-04-13 09:06:51 +04:00
INIT_LIST_HEAD ( & list - > qh_list ) ;
2005-04-17 02:20:36 +04:00
list - > qh_version = 0 ;
list - > qh_nelems = 0 ;
}
STATIC void
xfs_qm_list_destroy (
xfs_dqlist_t * list )
{
mutex_destroy ( & ( list - > qh_lock ) ) ;
}
/*
* Create an inode and return with a reference already taken , but unlocked
* This is how we create quota inodes
*/
STATIC int
xfs_qm_qino_alloc (
xfs_mount_t * mp ,
xfs_inode_t * * ip ,
__int64_t sbfields ,
uint flags )
{
xfs_trans_t * tp ;
int error ;
int committed ;
2006-01-11 07:27:50 +03:00
tp = xfs_trans_alloc ( mp , XFS_TRANS_QM_QINOCREATE ) ;
2005-04-17 02:20:36 +04:00
if ( ( error = xfs_trans_reserve ( tp ,
XFS_QM_QINOCREATE_SPACE_RES ( mp ) ,
XFS_CREATE_LOG_RES ( mp ) , 0 ,
XFS_TRANS_PERM_LOG_RES ,
XFS_CREATE_LOG_COUNT ) ) ) {
xfs_trans_cancel ( tp , 0 ) ;
2006-01-15 04:37:08 +03:00
return error ;
2005-04-17 02:20:36 +04:00
}
2010-10-06 22:41:17 +04:00
error = xfs_dir_ialloc ( & tp , NULL , S_IFREG , 1 , 0 , 0 , 1 , ip , & committed ) ;
if ( error ) {
2005-04-17 02:20:36 +04:00
xfs_trans_cancel ( tp , XFS_TRANS_RELEASE_LOG_RES |
XFS_TRANS_ABORT ) ;
2006-01-15 04:37:08 +03:00
return error ;
2005-04-17 02:20:36 +04:00
}
/*
* Make the changes in the superblock , and log those too .
* sbfields arg may contain fields other than * QUOTINO ;
* VERSIONNUM for example .
*/
2007-10-11 11:42:32 +04:00
spin_lock ( & mp - > m_sb_lock ) ;
2005-04-17 02:20:36 +04:00
if ( flags & XFS_QMOPT_SBVERSION ) {
2008-03-06 05:44:28 +03:00
ASSERT ( ! xfs_sb_version_hasquota ( & mp - > m_sb ) ) ;
2005-04-17 02:20:36 +04:00
ASSERT ( ( sbfields & ( XFS_SB_VERSIONNUM | XFS_SB_UQUOTINO |
XFS_SB_GQUOTINO | XFS_SB_QFLAGS ) ) = =
( XFS_SB_VERSIONNUM | XFS_SB_UQUOTINO |
XFS_SB_GQUOTINO | XFS_SB_QFLAGS ) ) ;
2008-03-06 05:44:28 +03:00
xfs_sb_version_addquota ( & mp - > m_sb ) ;
2005-04-17 02:20:36 +04:00
mp - > m_sb . sb_uquotino = NULLFSINO ;
mp - > m_sb . sb_gquotino = NULLFSINO ;
/* qflags will get updated _after_ quotacheck */
mp - > m_sb . sb_qflags = 0 ;
}
if ( flags & XFS_QMOPT_UQUOTA )
mp - > m_sb . sb_uquotino = ( * ip ) - > i_ino ;
else
mp - > m_sb . sb_gquotino = ( * ip ) - > i_ino ;
2007-10-11 11:42:32 +04:00
spin_unlock ( & mp - > m_sb_lock ) ;
2005-04-17 02:20:36 +04:00
xfs_mod_sb ( tp , sbfields ) ;
2007-05-08 07:48:42 +04:00
if ( ( error = xfs_trans_commit ( tp , XFS_TRANS_RELEASE_LOG_RES ) ) ) {
2011-03-07 02:05:35 +03:00
xfs_alert ( mp , " %s failed (error %d)! " , __func__ , error ) ;
2006-01-15 04:37:08 +03:00
return error ;
2005-04-17 02:20:36 +04:00
}
2006-01-15 04:37:08 +03:00
return 0 ;
2005-04-17 02:20:36 +04:00
}
2008-04-10 06:20:10 +04:00
STATIC void
2005-04-17 02:20:36 +04:00
xfs_qm_reset_dqcounts (
xfs_mount_t * mp ,
xfs_buf_t * bp ,
xfs_dqid_t id ,
uint type )
{
xfs_disk_dquot_t * ddq ;
int j ;
2009-12-15 02:14:59 +03:00
trace_xfs_reset_dqcounts ( bp , _RET_IP_ ) ;
2005-04-17 02:20:36 +04:00
/*
* Reset all counters and timers . They ' ll be
* started afresh by xfs_qm_quotacheck .
*/
# ifdef DEBUG
j = XFS_FSB_TO_B ( mp , XFS_DQUOT_CLUSTER_SIZE_FSB ) ;
do_div ( j , sizeof ( xfs_dqblk_t ) ) ;
2010-04-20 11:01:30 +04:00
ASSERT ( mp - > m_quotainfo - > qi_dqperchunk = = j ) ;
2005-04-17 02:20:36 +04:00
# endif
2011-07-23 03:40:15 +04:00
ddq = bp - > b_addr ;
2010-04-20 11:01:30 +04:00
for ( j = 0 ; j < mp - > m_quotainfo - > qi_dqperchunk ; j + + ) {
2005-04-17 02:20:36 +04:00
/*
* Do a sanity check , and if needed , repair the dqblk . Don ' t
* output any warnings because it ' s perfectly possible to
2006-03-29 02:55:14 +04:00
* find uninitialised dquot blks . See comment in xfs_qm_dqcheck .
2005-04-17 02:20:36 +04:00
*/
2011-03-07 02:01:35 +03:00
( void ) xfs_qm_dqcheck ( mp , ddq , id + j , type , XFS_QMOPT_DQREPAIR ,
2005-04-17 02:20:36 +04:00
" xfs_quotacheck " ) ;
2005-11-02 07:01:12 +03:00
ddq - > d_bcount = 0 ;
ddq - > d_icount = 0 ;
ddq - > d_rtbcount = 0 ;
ddq - > d_btimer = 0 ;
ddq - > d_itimer = 0 ;
ddq - > d_rtbtimer = 0 ;
ddq - > d_bwarns = 0 ;
ddq - > d_iwarns = 0 ;
ddq - > d_rtbwarns = 0 ;
2005-04-17 02:20:36 +04:00
ddq = ( xfs_disk_dquot_t * ) ( ( xfs_dqblk_t * ) ddq + 1 ) ;
}
}
STATIC int
xfs_qm_dqiter_bufs (
xfs_mount_t * mp ,
xfs_dqid_t firstid ,
xfs_fsblock_t bno ,
xfs_filblks_t blkcnt ,
uint flags )
{
xfs_buf_t * bp ;
int error ;
2005-06-21 09:38:48 +04:00
int type ;
2005-04-17 02:20:36 +04:00
ASSERT ( blkcnt > 0 ) ;
2005-06-21 09:38:48 +04:00
type = flags & XFS_QMOPT_UQUOTA ? XFS_DQ_USER :
( flags & XFS_QMOPT_PQUOTA ? XFS_DQ_PROJ : XFS_DQ_GROUP ) ;
2005-04-17 02:20:36 +04:00
error = 0 ;
/*
* Blkcnt arg can be a very big number , and might even be
* larger than the log itself . So , we have to break it up into
* manageable - sized transactions .
* Note that we don ' t start a permanent transaction here ; we might
* not be able to get a log reservation for the whole thing up front ,
* and we don ' t really care to either , because we just discard
* everything if we were to crash in the middle of this loop .
*/
while ( blkcnt - - ) {
error = xfs_trans_read_buf ( mp , NULL , mp - > m_ddev_targp ,
XFS_FSB_TO_DADDR ( mp , bno ) ,
2010-04-20 11:01:30 +04:00
mp - > m_quotainfo - > qi_dqchunklen , 0 , & bp ) ;
2005-04-17 02:20:36 +04:00
if ( error )
break ;
2008-04-10 06:20:10 +04:00
xfs_qm_reset_dqcounts ( mp , bp , firstid , type ) ;
2005-04-17 02:20:36 +04:00
xfs_bdwrite ( mp , bp ) ;
/*
* goto the next block .
*/
bno + + ;
2010-04-20 11:01:30 +04:00
firstid + = mp - > m_quotainfo - > qi_dqperchunk ;
2005-04-17 02:20:36 +04:00
}
2006-01-15 04:37:08 +03:00
return error ;
2005-04-17 02:20:36 +04:00
}
/*
2005-06-21 09:38:48 +04:00
* Iterate over all allocated USR / GRP / PRJ dquots in the system , calling a
2005-04-17 02:20:36 +04:00
* caller supplied function for every chunk of dquots that we find .
*/
STATIC int
xfs_qm_dqiterate (
xfs_mount_t * mp ,
xfs_inode_t * qip ,
uint flags )
{
xfs_bmbt_irec_t * map ;
int i , nmaps ; /* number of map entries */
int error ; /* return value */
xfs_fileoff_t lblkno ;
xfs_filblks_t maxlblkcnt ;
xfs_dqid_t firstid ;
xfs_fsblock_t rablkno ;
xfs_filblks_t rablkcnt ;
error = 0 ;
/*
2006-03-29 02:55:14 +04:00
* This looks racy , but we can ' t keep an inode lock across a
2005-04-17 02:20:36 +04:00
* trans_reserve . But , this gets called during quotacheck , and that
* happens only at mount time which is single threaded .
*/
if ( qip - > i_d . di_nblocks = = 0 )
2006-01-15 04:37:08 +03:00
return 0 ;
2005-04-17 02:20:36 +04:00
map = kmem_alloc ( XFS_DQITER_MAP_SIZE * sizeof ( * map ) , KM_SLEEP ) ;
lblkno = 0 ;
maxlblkcnt = XFS_B_TO_FSB ( mp , ( xfs_ufsize_t ) XFS_MAXIOFFSET ( mp ) ) ;
do {
nmaps = XFS_DQITER_MAP_SIZE ;
/*
* We aren ' t changing the inode itself . Just changing
* some of its data . No new blocks are added here , and
* the inode is never added to the transaction .
*/
xfs_ilock ( qip , XFS_ILOCK_SHARED ) ;
error = xfs_bmapi ( NULL , qip , lblkno ,
maxlblkcnt - lblkno ,
XFS_BMAPI_METADATA ,
NULL ,
2010-06-23 12:11:15 +04:00
0 , map , & nmaps , NULL ) ;
2005-04-17 02:20:36 +04:00
xfs_iunlock ( qip , XFS_ILOCK_SHARED ) ;
if ( error )
break ;
ASSERT ( nmaps < = XFS_DQITER_MAP_SIZE ) ;
for ( i = 0 ; i < nmaps ; i + + ) {
ASSERT ( map [ i ] . br_startblock ! = DELAYSTARTBLOCK ) ;
ASSERT ( map [ i ] . br_blockcount ) ;
lblkno + = map [ i ] . br_blockcount ;
if ( map [ i ] . br_startblock = = HOLESTARTBLOCK )
continue ;
firstid = ( xfs_dqid_t ) map [ i ] . br_startoff *
2010-04-20 11:01:30 +04:00
mp - > m_quotainfo - > qi_dqperchunk ;
2005-04-17 02:20:36 +04:00
/*
* Do a read - ahead on the next extent .
*/
if ( ( i + 1 < nmaps ) & &
( map [ i + 1 ] . br_startblock ! = HOLESTARTBLOCK ) ) {
rablkcnt = map [ i + 1 ] . br_blockcount ;
rablkno = map [ i + 1 ] . br_startblock ;
while ( rablkcnt - - ) {
2010-10-06 22:41:18 +04:00
xfs_buf_readahead ( mp - > m_ddev_targp ,
2005-04-17 02:20:36 +04:00
XFS_FSB_TO_DADDR ( mp , rablkno ) ,
2010-04-20 11:01:30 +04:00
mp - > m_quotainfo - > qi_dqchunklen ) ;
2005-04-17 02:20:36 +04:00
rablkno + + ;
}
}
/*
* Iterate thru all the blks in the extent and
* reset the counters of all the dquots inside them .
*/
if ( ( error = xfs_qm_dqiter_bufs ( mp ,
firstid ,
map [ i ] . br_startblock ,
map [ i ] . br_blockcount ,
flags ) ) ) {
break ;
}
}
if ( error )
break ;
} while ( nmaps > 0 ) ;
2008-05-19 10:31:57 +04:00
kmem_free ( map ) ;
2005-04-17 02:20:36 +04:00
2006-01-15 04:37:08 +03:00
return error ;
2005-04-17 02:20:36 +04:00
}
/*
* Called by dqusage_adjust in doing a quotacheck .
2010-09-06 05:44:22 +04:00
*
* Given the inode , and a dquot id this updates both the incore dqout as well
* as the buffer copy . This is so that once the quotacheck is done , we can
* just log all the buffers , as opposed to logging numerous updates to
* individual dquots .
2005-04-17 02:20:36 +04:00
*/
2010-09-06 05:44:22 +04:00
STATIC int
2005-04-17 02:20:36 +04:00
xfs_qm_quotacheck_dqadjust (
2010-09-06 05:44:22 +04:00
struct xfs_inode * ip ,
xfs_dqid_t id ,
uint type ,
2005-04-17 02:20:36 +04:00
xfs_qcnt_t nblks ,
xfs_qcnt_t rtblks )
{
2010-09-06 05:44:22 +04:00
struct xfs_mount * mp = ip - > i_mount ;
struct xfs_dquot * dqp ;
int error ;
error = xfs_qm_dqget ( mp , ip , id , type ,
XFS_QMOPT_DQALLOC | XFS_QMOPT_DOWARN , & dqp ) ;
if ( error ) {
/*
* Shouldn ' t be able to turn off quotas here .
*/
ASSERT ( error ! = ESRCH ) ;
ASSERT ( error ! = ENOENT ) ;
return error ;
}
2009-12-15 02:14:59 +03:00
trace_xfs_dqadjust ( dqp ) ;
2005-04-17 02:20:36 +04:00
/*
* Adjust the inode count and the block count to reflect this inode ' s
* resource usage .
*/
2008-02-14 02:03:29 +03:00
be64_add_cpu ( & dqp - > q_core . d_icount , 1 ) ;
2005-04-17 02:20:36 +04:00
dqp - > q_res_icount + + ;
if ( nblks ) {
2008-02-14 02:03:29 +03:00
be64_add_cpu ( & dqp - > q_core . d_bcount , nblks ) ;
2005-04-17 02:20:36 +04:00
dqp - > q_res_bcount + = nblks ;
}
if ( rtblks ) {
2008-02-14 02:03:29 +03:00
be64_add_cpu ( & dqp - > q_core . d_rtbcount , rtblks ) ;
2005-04-17 02:20:36 +04:00
dqp - > q_res_rtbcount + = rtblks ;
}
/*
* Set default limits , adjust timers ( since we changed usages )
2010-04-20 11:01:53 +04:00
*
* There are no timers for the default values set in the root dquot .
2005-04-17 02:20:36 +04:00
*/
2010-04-20 11:01:53 +04:00
if ( dqp - > q_core . d_id ) {
2010-09-06 05:44:22 +04:00
xfs_qm_adjust_dqlimits ( mp , & dqp - > q_core ) ;
xfs_qm_adjust_dqtimers ( mp , & dqp - > q_core ) ;
2005-04-17 02:20:36 +04:00
}
dqp - > dq_flags | = XFS_DQ_DIRTY ;
2010-09-06 05:44:22 +04:00
xfs_qm_dqput ( dqp ) ;
return 0 ;
2005-04-17 02:20:36 +04:00
}
STATIC int
xfs_qm_get_rtblks (
xfs_inode_t * ip ,
xfs_qcnt_t * O_rtblks )
{
xfs_filblks_t rtblks ; /* total rt blks */
2006-03-14 05:29:52 +03:00
xfs_extnum_t idx ; /* extent record index */
2005-04-17 02:20:36 +04:00
xfs_ifork_t * ifp ; /* inode fork pointer */
xfs_extnum_t nextents ; /* number of extent entries */
int error ;
ASSERT ( XFS_IS_REALTIME_INODE ( ip ) ) ;
ifp = XFS_IFORK_PTR ( ip , XFS_DATA_FORK ) ;
if ( ! ( ifp - > if_flags & XFS_IFEXTENTS ) ) {
if ( ( error = xfs_iread_extents ( NULL , ip , XFS_DATA_FORK ) ) )
2006-01-15 04:37:08 +03:00
return error ;
2005-04-17 02:20:36 +04:00
}
rtblks = 0 ;
2006-03-14 05:29:52 +03:00
nextents = ifp - > if_bytes / ( uint ) sizeof ( xfs_bmbt_rec_t ) ;
2007-08-16 10:23:40 +04:00
for ( idx = 0 ; idx < nextents ; idx + + )
rtblks + = xfs_bmbt_get_blockcount ( xfs_iext_get_ext ( ifp , idx ) ) ;
2005-04-17 02:20:36 +04:00
* O_rtblks = ( xfs_qcnt_t ) rtblks ;
2006-01-15 04:37:08 +03:00
return 0 ;
2005-04-17 02:20:36 +04:00
}
/*
* callback routine supplied to bulkstat ( ) . Given an inumber , find its
* dquots and update them to account for resources taken by that inode .
*/
/* ARGSUSED */
STATIC int
xfs_qm_dqusage_adjust (
xfs_mount_t * mp , /* mount point for filesystem */
xfs_ino_t ino , /* inode number to get data for */
void __user * buffer , /* not used */
int ubsize , /* not used */
int * ubused , /* not used */
int * res ) /* result code value */
{
xfs_inode_t * ip ;
2010-09-06 05:44:22 +04:00
xfs_qcnt_t nblks , rtblks = 0 ;
2005-04-17 02:20:36 +04:00
int error ;
ASSERT ( XFS_IS_QUOTA_RUNNING ( mp ) ) ;
/*
* rootino must have its resources accounted for , not so with the quota
* inodes .
*/
if ( ino = = mp - > m_sb . sb_uquotino | | ino = = mp - > m_sb . sb_gquotino ) {
* res = BULKSTAT_RV_NOTHING ;
return XFS_ERROR ( EINVAL ) ;
}
/*
* We don ' t _need_ to take the ilock EXCL . However , the xfs_qm_dqget
* interface expects the inode to be exclusively locked because that ' s
* the case in all other instances . It ' s OK that we do this because
* quotacheck is done only at mount time .
*/
2010-09-06 05:44:22 +04:00
error = xfs_iget ( mp , NULL , ino , 0 , XFS_ILOCK_EXCL , & ip ) ;
if ( error ) {
2005-04-17 02:20:36 +04:00
* res = BULKSTAT_RV_NOTHING ;
2006-01-15 04:37:08 +03:00
return error ;
2005-04-17 02:20:36 +04:00
}
2010-09-06 05:44:22 +04:00
ASSERT ( ip - > i_delayed_blks = = 0 ) ;
2005-04-17 02:20:36 +04:00
2010-09-06 05:44:22 +04:00
if ( XFS_IS_REALTIME_INODE ( ip ) ) {
2005-04-17 02:20:36 +04:00
/*
* Walk thru the extent list and count the realtime blocks .
*/
2010-09-06 05:44:22 +04:00
error = xfs_qm_get_rtblks ( ip , & rtblks ) ;
if ( error )
goto error0 ;
2005-04-17 02:20:36 +04:00
}
2010-09-06 05:44:22 +04:00
nblks = ( xfs_qcnt_t ) ip - > i_d . di_nblocks - rtblks ;
2005-04-17 02:20:36 +04:00
/*
* Add the ( disk blocks and inode ) resources occupied by this
* inode to its dquots . We do this adjustment in the incore dquot ,
* and also copy the changes to its buffer .
* We don ' t care about putting these changes in a transaction
* envelope because if we crash in the middle of a ' quotacheck '
* we have to start from the beginning anyway .
* Once we ' re done , we ' ll log all the dquot bufs .
*
2006-03-29 02:55:14 +04:00
* The * QUOTA_ON checks below may look pretty racy , but quotachecks
2005-04-17 02:20:36 +04:00
* and quotaoffs don ' t race . ( Quotachecks happen at mount time only ) .
*/
if ( XFS_IS_UQUOTA_ON ( mp ) ) {
2010-09-06 05:44:22 +04:00
error = xfs_qm_quotacheck_dqadjust ( ip , ip - > i_d . di_uid ,
XFS_DQ_USER , nblks , rtblks ) ;
if ( error )
goto error0 ;
2005-04-17 02:20:36 +04:00
}
2010-09-06 05:44:22 +04:00
if ( XFS_IS_GQUOTA_ON ( mp ) ) {
error = xfs_qm_quotacheck_dqadjust ( ip , ip - > i_d . di_gid ,
XFS_DQ_GROUP , nblks , rtblks ) ;
if ( error )
goto error0 ;
2005-04-17 02:20:36 +04:00
}
2010-09-06 05:44:22 +04:00
if ( XFS_IS_PQUOTA_ON ( mp ) ) {
2010-09-26 10:10:18 +04:00
error = xfs_qm_quotacheck_dqadjust ( ip , xfs_get_projid ( ip ) ,
2010-09-06 05:44:22 +04:00
XFS_DQ_PROJ , nblks , rtblks ) ;
if ( error )
goto error0 ;
}
xfs_iunlock ( ip , XFS_ILOCK_EXCL ) ;
IRELE ( ip ) ;
2005-04-17 02:20:36 +04:00
* res = BULKSTAT_RV_DIDONE ;
2006-01-15 04:37:08 +03:00
return 0 ;
2010-09-06 05:44:22 +04:00
error0 :
xfs_iunlock ( ip , XFS_ILOCK_EXCL ) ;
IRELE ( ip ) ;
* res = BULKSTAT_RV_GIVEUP ;
return error ;
2005-04-17 02:20:36 +04:00
}
/*
* Walk thru all the filesystem inodes and construct a consistent view
* of the disk quota world . If the quotacheck fails , disable quotas .
*/
int
xfs_qm_quotacheck (
xfs_mount_t * mp )
{
int done , count , error ;
xfs_ino_t lastino ;
size_t structsz ;
xfs_inode_t * uip , * gip ;
uint flags ;
count = INT_MAX ;
structsz = 1 ;
lastino = 0 ;
flags = 0 ;
2010-04-20 11:01:30 +04:00
ASSERT ( mp - > m_quotainfo - > qi_uquotaip | | mp - > m_quotainfo - > qi_gquotaip ) ;
2005-04-17 02:20:36 +04:00
ASSERT ( XFS_IS_QUOTA_RUNNING ( mp ) ) ;
/*
* There should be no cached dquots . The ( simplistic ) quotacheck
* algorithm doesn ' t like that .
*/
2010-04-13 09:06:48 +04:00
ASSERT ( list_empty ( & mp - > m_quotainfo - > qi_dqlist ) ) ;
2005-04-17 02:20:36 +04:00
2011-03-07 02:08:35 +03:00
xfs_notice ( mp , " Quotacheck needed: Please wait. " ) ;
2005-04-17 02:20:36 +04:00
/*
2005-06-21 09:38:48 +04:00
* First we go thru all the dquots on disk , USR and GRP / PRJ , and reset
2005-04-17 02:20:36 +04:00
* their counters to zero . We need a clean slate .
* We don ' t log our changes till later .
*/
2010-04-20 11:01:30 +04:00
uip = mp - > m_quotainfo - > qi_uquotaip ;
if ( uip ) {
error = xfs_qm_dqiterate ( mp , uip , XFS_QMOPT_UQUOTA ) ;
if ( error )
2005-04-17 02:20:36 +04:00
goto error_return ;
flags | = XFS_UQUOTA_CHKD ;
}
2010-04-20 11:01:30 +04:00
gip = mp - > m_quotainfo - > qi_gquotaip ;
if ( gip ) {
error = xfs_qm_dqiterate ( mp , gip , XFS_IS_GQUOTA_ON ( mp ) ?
XFS_QMOPT_GQUOTA : XFS_QMOPT_PQUOTA ) ;
if ( error )
2005-04-17 02:20:36 +04:00
goto error_return ;
2005-06-21 09:38:48 +04:00
flags | = XFS_OQUOTA_CHKD ;
2005-04-17 02:20:36 +04:00
}
do {
/*
* Iterate thru all the inodes in the file system ,
* adjusting the corresponding dquot counters in core .
*/
2010-06-23 12:11:11 +04:00
error = xfs_bulkstat ( mp , & lastino , & count ,
xfs_qm_dqusage_adjust ,
structsz , NULL , & done ) ;
if ( error )
2005-04-17 02:20:36 +04:00
break ;
2010-06-23 12:11:11 +04:00
} while ( ! done ) ;
2005-04-17 02:20:36 +04:00
2008-04-10 06:20:17 +04:00
/*
* We ' ve made all the changes that we need to make incore .
* Flush them down to disk buffers if everything was updated
* successfully .
*/
if ( ! error )
2010-02-04 01:48:58 +03:00
error = xfs_qm_dqflush_all ( mp , 0 ) ;
2008-04-10 06:20:17 +04:00
2005-04-17 02:20:36 +04:00
/*
* We can get this error if we couldn ' t do a dquot allocation inside
* xfs_qm_dqusage_adjust ( via bulkstat ) . We don ' t care about the
* dirty dquots that might be cached , we just want to get rid of them
* and turn quotaoff . The dquots won ' t be attached to any of the inodes
* at this point ( because we intentionally didn ' t in dqget_noattach ) .
*/
if ( error ) {
2010-04-20 11:02:29 +04:00
xfs_qm_dqpurge_all ( mp , XFS_QMOPT_QUOTALL ) ;
2005-04-17 02:20:36 +04:00
goto error_return ;
}
/*
* We didn ' t log anything , because if we crashed , we ' ll have to
* start the quotacheck from scratch anyway . However , we must make
* sure that our dquot changes are secure before we put the
* quotacheck ' d stamp on the superblock . So , here we do a synchronous
* flush .
*/
XFS_bflush ( mp - > m_ddev_targp ) ;
/*
* If one type of quotas is off , then it will lose its
* quotachecked status , since we won ' t be doing accounting for
* that type anymore .
*/
2005-06-21 09:38:48 +04:00
mp - > m_qflags & = ~ ( XFS_OQUOTA_CHKD | XFS_UQUOTA_CHKD ) ;
2005-04-17 02:20:36 +04:00
mp - > m_qflags | = flags ;
error_return :
if ( error ) {
2011-03-07 02:08:35 +03:00
xfs_warn ( mp ,
" Quotacheck: Unsuccessful (Error %d): Disabling quotas. " ,
error ) ;
2005-04-17 02:20:36 +04:00
/*
* We must turn off quotas .
*/
ASSERT ( mp - > m_quotainfo ! = NULL ) ;
ASSERT ( xfs_Gqm ! = NULL ) ;
xfs_qm_destroy_quotainfo ( mp ) ;
2008-04-10 06:20:38 +04:00
if ( xfs_mount_reset_sbqflags ( mp ) ) {
2011-03-07 02:08:35 +03:00
xfs_warn ( mp ,
" Quotacheck: Failed to reset quota flags. " ) ;
2008-04-10 06:20:38 +04:00
}
2011-03-07 02:08:35 +03:00
} else
xfs_notice ( mp , " Quotacheck: Done. " ) ;
2005-04-17 02:20:36 +04:00
return ( error ) ;
}
/*
* This is called after the superblock has been read in and we ' re ready to
* iget the quota inodes .
*/
STATIC int
xfs_qm_init_quotainos (
xfs_mount_t * mp )
{
xfs_inode_t * uip , * gip ;
int error ;
__int64_t sbflags ;
uint flags ;
ASSERT ( mp - > m_quotainfo ) ;
uip = gip = NULL ;
sbflags = 0 ;
flags = 0 ;
/*
* Get the uquota and gquota inodes
*/
2008-03-06 05:44:28 +03:00
if ( xfs_sb_version_hasquota ( & mp - > m_sb ) ) {
2005-04-17 02:20:36 +04:00
if ( XFS_IS_UQUOTA_ON ( mp ) & &
mp - > m_sb . sb_uquotino ! = NULLFSINO ) {
ASSERT ( mp - > m_sb . sb_uquotino > 0 ) ;
if ( ( error = xfs_iget ( mp , NULL , mp - > m_sb . sb_uquotino ,
2010-06-24 05:35:17 +04:00
0 , 0 , & uip ) ) )
2005-04-17 02:20:36 +04:00
return XFS_ERROR ( error ) ;
}
2005-06-21 09:38:48 +04:00
if ( XFS_IS_OQUOTA_ON ( mp ) & &
2005-04-17 02:20:36 +04:00
mp - > m_sb . sb_gquotino ! = NULLFSINO ) {
ASSERT ( mp - > m_sb . sb_gquotino > 0 ) ;
if ( ( error = xfs_iget ( mp , NULL , mp - > m_sb . sb_gquotino ,
2010-06-24 05:35:17 +04:00
0 , 0 , & gip ) ) ) {
2005-04-17 02:20:36 +04:00
if ( uip )
2008-03-27 10:01:08 +03:00
IRELE ( uip ) ;
2005-04-17 02:20:36 +04:00
return XFS_ERROR ( error ) ;
}
}
} else {
flags | = XFS_QMOPT_SBVERSION ;
sbflags | = ( XFS_SB_VERSIONNUM | XFS_SB_UQUOTINO |
XFS_SB_GQUOTINO | XFS_SB_QFLAGS ) ;
}
/*
* Create the two inodes , if they don ' t exist already . The changes
* made above will get added to a transaction and logged in one of
* the qino_alloc calls below . If the device is readonly ,
* temporarily switch to read - write to do this .
*/
if ( XFS_IS_UQUOTA_ON ( mp ) & & uip = = NULL ) {
if ( ( error = xfs_qm_qino_alloc ( mp , & uip ,
sbflags | XFS_SB_UQUOTINO ,
flags | XFS_QMOPT_UQUOTA ) ) )
return XFS_ERROR ( error ) ;
flags & = ~ XFS_QMOPT_SBVERSION ;
}
2005-06-21 09:38:48 +04:00
if ( XFS_IS_OQUOTA_ON ( mp ) & & gip = = NULL ) {
flags | = ( XFS_IS_GQUOTA_ON ( mp ) ?
XFS_QMOPT_GQUOTA : XFS_QMOPT_PQUOTA ) ;
error = xfs_qm_qino_alloc ( mp , & gip ,
sbflags | XFS_SB_GQUOTINO , flags ) ;
if ( error ) {
2005-04-17 02:20:36 +04:00
if ( uip )
2008-03-27 10:01:08 +03:00
IRELE ( uip ) ;
2005-04-17 02:20:36 +04:00
return XFS_ERROR ( error ) ;
}
}
2010-04-20 11:01:30 +04:00
mp - > m_quotainfo - > qi_uquotaip = uip ;
mp - > m_quotainfo - > qi_gquotaip = gip ;
2005-04-17 02:20:36 +04:00
2006-01-15 04:37:08 +03:00
return 0 ;
2005-04-17 02:20:36 +04:00
}
2010-04-13 09:06:50 +04:00
2005-04-17 02:20:36 +04:00
/*
2010-04-13 09:06:50 +04:00
* Just pop the least recently used dquot off the freelist and
* recycle it . The returned dquot is locked .
2005-04-17 02:20:36 +04:00
*/
2010-04-13 09:06:50 +04:00
STATIC xfs_dquot_t *
xfs_qm_dqreclaim_one ( void )
2005-04-17 02:20:36 +04:00
{
2010-04-13 09:06:50 +04:00
xfs_dquot_t * dqpout ;
xfs_dquot_t * dqp ;
2005-04-17 02:20:36 +04:00
int restarts ;
2011-01-28 03:20:46 +03:00
int startagain ;
2005-04-17 02:20:36 +04:00
restarts = 0 ;
2010-04-13 09:06:50 +04:00
dqpout = NULL ;
2005-04-17 02:20:36 +04:00
2010-04-13 09:06:50 +04:00
/* lockorder: hashchainlock, freelistlock, mplistlock, dqlock, dqflock */
2011-01-28 03:20:46 +03:00
again :
startagain = 0 ;
2010-04-13 09:06:52 +04:00
mutex_lock ( & xfs_Gqm - > qm_dqfrlist_lock ) ;
2005-04-17 02:20:36 +04:00
2010-04-13 09:06:52 +04:00
list_for_each_entry ( dqp , & xfs_Gqm - > qm_dqfrlist , q_freelist ) {
2010-04-13 09:06:48 +04:00
struct xfs_mount * mp = dqp - > q_mount ;
2005-04-17 02:20:36 +04:00
xfs_dqlock ( dqp ) ;
/*
* We are racing with dqlookup here . Naturally we don ' t
2010-04-13 09:06:50 +04:00
* want to reclaim a dquot that lookup wants . We release the
* freelist lock and start over , so that lookup will grab
* both the dquot and the freelistlock .
2005-04-17 02:20:36 +04:00
*/
if ( dqp - > dq_flags & XFS_DQ_WANT ) {
2010-04-13 09:06:50 +04:00
ASSERT ( ! ( dqp - > dq_flags & XFS_DQ_INACTIVE ) ) ;
trace_xfs_dqreclaim_want ( dqp ) ;
2005-04-17 02:20:36 +04:00
XQM_STATS_INC ( xqmstats . xs_qm_dqwants ) ;
2011-01-28 03:20:46 +03:00
restarts + + ;
startagain = 1 ;
goto dqunlock ;
2005-04-17 02:20:36 +04:00
}
/*
* If the dquot is inactive , we are assured that it is
* not on the mplist or the hashlist , and that makes our
* life easier .
*/
if ( dqp - > dq_flags & XFS_DQ_INACTIVE ) {
2010-04-13 09:06:48 +04:00
ASSERT ( mp = = NULL ) ;
2005-04-17 02:20:36 +04:00
ASSERT ( ! XFS_DQ_IS_DIRTY ( dqp ) ) ;
2010-04-13 09:06:51 +04:00
ASSERT ( list_empty ( & dqp - > q_hashlist ) ) ;
2010-04-13 09:06:48 +04:00
ASSERT ( list_empty ( & dqp - > q_mplist ) ) ;
2010-04-13 09:06:52 +04:00
list_del_init ( & dqp - > q_freelist ) ;
xfs_Gqm - > qm_dqfrlist_cnt - - ;
2010-04-13 09:06:50 +04:00
dqpout = dqp ;
2005-04-17 02:20:36 +04:00
XQM_STATS_INC ( xqmstats . xs_qm_dqinact_reclaims ) ;
2011-01-28 03:20:46 +03:00
goto dqunlock ;
2005-04-17 02:20:36 +04:00
}
2010-04-13 09:06:50 +04:00
ASSERT ( dqp - > q_hash ) ;
2010-04-13 09:06:48 +04:00
ASSERT ( ! list_empty ( & dqp - > q_mplist ) ) ;
2010-04-13 09:06:50 +04:00
2005-04-17 02:20:36 +04:00
/*
2011-01-28 03:20:46 +03:00
* Try to grab the flush lock . If this dquot is in the process
* of getting flushed to disk , we don ' t want to reclaim it .
2005-04-17 02:20:36 +04:00
*/
2011-01-28 03:20:46 +03:00
if ( ! xfs_dqflock_nowait ( dqp ) )
goto dqunlock ;
2005-04-17 02:20:36 +04:00
/*
* We have the flush lock so we know that this is not in the
* process of being flushed . So , if this is dirty , flush it
* DELWRI so that we don ' t get a freelist infested with
* dirty dquots .
*/
if ( XFS_DQ_IS_DIRTY ( dqp ) ) {
2008-04-10 06:20:24 +04:00
int error ;
2009-12-15 02:14:59 +03:00
2010-04-13 09:06:50 +04:00
trace_xfs_dqreclaim_dirty ( dqp ) ;
2009-12-15 02:14:59 +03:00
2005-04-17 02:20:36 +04:00
/*
* We flush it delayed write , so don ' t bother
2010-04-13 09:06:50 +04:00
* releasing the freelist lock .
2005-04-17 02:20:36 +04:00
*/
2010-02-04 01:48:58 +03:00
error = xfs_qm_dqflush ( dqp , 0 ) ;
2008-04-10 06:20:24 +04:00
if ( error ) {
2011-03-07 02:05:35 +03:00
xfs_warn ( mp , " %s: dquot %p flush failed " ,
__func__ , dqp ) ;
2008-04-10 06:20:24 +04:00
}
2011-01-28 03:20:46 +03:00
goto dqunlock ;
2005-04-17 02:20:36 +04:00
}
2010-04-13 09:06:50 +04:00
2005-04-17 02:20:36 +04:00
/*
* We ' re trying to get the hashlock out of order . This races
* with dqlookup ; so , we giveup and goto the next dquot if
* we couldn ' t get the hashlock . This way , we won ' t starve
* a dqlookup process that holds the hashlock that is
* waiting for the freelist lock .
*/
2009-02-09 10:47:22 +03:00
if ( ! mutex_trylock ( & dqp - > q_hash - > qh_lock ) ) {
2010-04-13 09:06:50 +04:00
restarts + + ;
goto dqfunlock ;
2005-04-17 02:20:36 +04:00
}
2010-04-13 09:06:50 +04:00
2005-04-17 02:20:36 +04:00
/*
* This races with dquot allocation code as well as dqflush_all
* and reclaim code . So , if we failed to grab the mplist lock ,
* giveup everything and start over .
*/
2010-04-13 09:06:48 +04:00
if ( ! mutex_trylock ( & mp - > m_quotainfo - > qi_dqlist_lock ) ) {
2010-04-13 09:06:50 +04:00
restarts + + ;
2011-01-28 03:20:46 +03:00
startagain = 1 ;
goto qhunlock ;
2005-04-17 02:20:36 +04:00
}
2009-12-15 02:14:59 +03:00
2005-04-17 02:20:36 +04:00
ASSERT ( dqp - > q_nrefs = = 0 ) ;
2010-04-13 09:06:48 +04:00
list_del_init ( & dqp - > q_mplist ) ;
mp - > m_quotainfo - > qi_dquots - - ;
mp - > m_quotainfo - > qi_dqreclaims + + ;
2010-04-13 09:06:51 +04:00
list_del_init ( & dqp - > q_hashlist ) ;
dqp - > q_hash - > qh_version + + ;
2010-04-13 09:06:52 +04:00
list_del_init ( & dqp - > q_freelist ) ;
xfs_Gqm - > qm_dqfrlist_cnt - - ;
2010-04-13 09:06:50 +04:00
dqpout = dqp ;
mutex_unlock ( & mp - > m_quotainfo - > qi_dqlist_lock ) ;
2011-01-28 03:20:46 +03:00
qhunlock :
2010-04-13 09:06:50 +04:00
mutex_unlock ( & dqp - > q_hash - > qh_lock ) ;
dqfunlock :
xfs_dqfunlock ( dqp ) ;
2011-01-28 03:20:46 +03:00
dqunlock :
2005-04-17 02:20:36 +04:00
xfs_dqunlock ( dqp ) ;
2010-04-13 09:06:50 +04:00
if ( dqpout )
break ;
if ( restarts > = XFS_QM_RECLAIM_MAX_RESTARTS )
2011-01-28 03:20:46 +03:00
break ;
if ( startagain ) {
mutex_unlock ( & xfs_Gqm - > qm_dqfrlist_lock ) ;
goto again ;
}
2005-04-17 02:20:36 +04:00
}
2010-04-13 09:06:52 +04:00
mutex_unlock ( & xfs_Gqm - > qm_dqfrlist_lock ) ;
2010-04-13 09:06:50 +04:00
return dqpout ;
2005-04-17 02:20:36 +04:00
}
2010-04-13 09:06:50 +04:00
/*
* Traverse the freelist of dquots and attempt to reclaim a maximum of
* ' howmany ' dquots . This operation races with dqlookup ( ) , and attempts to
* favor the lookup function . . .
*/
STATIC int
xfs_qm_shake_freelist (
int howmany )
{
int nreclaimed = 0 ;
xfs_dquot_t * dqp ;
if ( howmany < = 0 )
return 0 ;
while ( nreclaimed < howmany ) {
dqp = xfs_qm_dqreclaim_one ( ) ;
if ( ! dqp )
return nreclaimed ;
xfs_qm_dqdestroy ( dqp ) ;
nreclaimed + + ;
}
return nreclaimed ;
}
2005-04-17 02:20:36 +04:00
/*
* The kmem_shake interface is invoked when memory is running low .
*/
/* ARGSUSED */
STATIC int
2010-07-19 08:56:17 +04:00
xfs_qm_shake (
struct shrinker * shrink ,
2011-05-25 04:12:27 +04:00
struct shrink_control * sc )
2005-04-17 02:20:36 +04:00
{
int ndqused , nfree , n ;
2011-05-25 04:12:27 +04:00
gfp_t gfp_mask = sc - > gfp_mask ;
2005-04-17 02:20:36 +04:00
if ( ! kmem_shake_allow ( gfp_mask ) )
2006-01-15 04:37:08 +03:00
return 0 ;
2005-04-17 02:20:36 +04:00
if ( ! xfs_Gqm )
2006-01-15 04:37:08 +03:00
return 0 ;
2005-04-17 02:20:36 +04:00
2010-04-13 09:06:52 +04:00
nfree = xfs_Gqm - > qm_dqfrlist_cnt ; /* free dquots */
2005-04-17 02:20:36 +04:00
/* incore dquots in all f/s's */
ndqused = atomic_read ( & xfs_Gqm - > qm_totaldquots ) - nfree ;
ASSERT ( ndqused > = 0 ) ;
if ( nfree < = ndqused & & nfree < ndquot )
2006-01-15 04:37:08 +03:00
return 0 ;
2005-04-17 02:20:36 +04:00
ndqused * = xfs_Gqm - > qm_dqfree_ratio ; /* target # of free dquots */
n = nfree - ndqused - ndquot ; /* # over target */
return xfs_qm_shake_freelist ( MAX ( nfree , n ) ) ;
}
/*------------------------------------------------------------------*/
/*
* Return a new incore dquot . Depending on the number of
* dquots in the system , we either allocate a new one on the kernel heap ,
* or reclaim a free one .
* Return value is B_TRUE if we allocated a new dquot , B_FALSE if we managed
* to reclaim an existing one from the freelist .
*/
boolean_t
xfs_qm_dqalloc_incore (
xfs_dquot_t * * O_dqpp )
{
xfs_dquot_t * dqp ;
/*
* Check against high water mark to see if we want to pop
* a nincompoop dquot off the freelist .
*/
if ( atomic_read ( & xfs_Gqm - > qm_totaldquots ) > = ndquot ) {
/*
* Try to recycle a dquot from the freelist .
*/
if ( ( dqp = xfs_qm_dqreclaim_one ( ) ) ) {
XQM_STATS_INC ( xqmstats . xs_qm_dqreclaims ) ;
/*
* Just zero the core here . The rest will get
* reinitialized by caller . XXX we shouldn ' t even
* do this zero . . .
*/
memset ( & dqp - > q_core , 0 , sizeof ( dqp - > q_core ) ) ;
* O_dqpp = dqp ;
2006-01-15 04:37:08 +03:00
return B_FALSE ;
2005-04-17 02:20:36 +04:00
}
XQM_STATS_INC ( xqmstats . xs_qm_dqreclaim_misses ) ;
}
/*
* Allocate a brand new dquot on the kernel heap and return it
* to the caller to initialize .
*/
ASSERT ( xfs_Gqm - > qm_dqzone ! = NULL ) ;
* O_dqpp = kmem_zone_zalloc ( xfs_Gqm - > qm_dqzone , KM_SLEEP ) ;
atomic_inc ( & xfs_Gqm - > qm_totaldquots ) ;
2006-01-15 04:37:08 +03:00
return B_TRUE ;
2005-04-17 02:20:36 +04:00
}
/*
* Start a transaction and write the incore superblock changes to
* disk . flags parameter indicates which fields have changed .
*/
int
xfs_qm_write_sb_changes (
xfs_mount_t * mp ,
__int64_t flags )
{
xfs_trans_t * tp ;
int error ;
tp = xfs_trans_alloc ( mp , XFS_TRANS_QM_SBCHANGE ) ;
if ( ( error = xfs_trans_reserve ( tp , 0 ,
mp - > m_sb . sb_sectsize + 128 , 0 ,
0 ,
XFS_DEFAULT_LOG_COUNT ) ) ) {
xfs_trans_cancel ( tp , 0 ) ;
2006-01-15 04:37:08 +03:00
return error ;
2005-04-17 02:20:36 +04:00
}
xfs_mod_sb ( tp , flags ) ;
2008-04-10 06:21:18 +04:00
error = xfs_trans_commit ( tp , 0 ) ;
2005-04-17 02:20:36 +04:00
2008-04-10 06:21:18 +04:00
return error ;
2005-04-17 02:20:36 +04:00
}
/* --------------- utility functions for vnodeops ---------------- */
/*
2010-10-06 22:41:17 +04:00
* Given an inode , a uid , gid and prid make sure that we have
2005-04-17 02:20:36 +04:00
* allocated relevant dquot ( s ) on disk , and that we won ' t exceed inode
* quotas by creating this file .
* This also attaches dquot ( s ) to the given inode after locking it ,
* and returns the dquots corresponding to the uid and / or gid .
*
* in : inode ( unlocked )
* out : udquot , gdquot with references taken and unlocked
*/
int
xfs_qm_vop_dqalloc (
2009-06-08 17:33:32 +04:00
struct xfs_inode * ip ,
uid_t uid ,
gid_t gid ,
prid_t prid ,
uint flags ,
struct xfs_dquot * * O_udqpp ,
struct xfs_dquot * * O_gdqpp )
2005-04-17 02:20:36 +04:00
{
2009-06-08 17:33:32 +04:00
struct xfs_mount * mp = ip - > i_mount ;
struct xfs_dquot * uq , * gq ;
int error ;
uint lockflags ;
2005-04-17 02:20:36 +04:00
2009-06-08 17:33:32 +04:00
if ( ! XFS_IS_QUOTA_RUNNING ( mp ) | | ! XFS_IS_QUOTA_ON ( mp ) )
2005-04-17 02:20:36 +04:00
return 0 ;
lockflags = XFS_ILOCK_EXCL ;
xfs_ilock ( ip , lockflags ) ;
2007-08-30 11:21:12 +04:00
if ( ( flags & XFS_QMOPT_INHERIT ) & & XFS_INHERIT_GID ( ip ) )
2005-04-17 02:20:36 +04:00
gid = ip - > i_d . di_gid ;
/*
* Attach the dquot ( s ) to this inode , doing a dquot allocation
* if necessary . The dquot ( s ) will not be locked .
*/
if ( XFS_NOT_DQATTACHED ( mp , ip ) ) {
2009-06-08 17:33:32 +04:00
error = xfs_qm_dqattach_locked ( ip , XFS_QMOPT_DQALLOC ) ;
if ( error ) {
2005-04-17 02:20:36 +04:00
xfs_iunlock ( ip , lockflags ) ;
2006-01-15 04:37:08 +03:00
return error ;
2005-04-17 02:20:36 +04:00
}
}
uq = gq = NULL ;
2005-06-21 09:38:48 +04:00
if ( ( flags & XFS_QMOPT_UQUOTA ) & & XFS_IS_UQUOTA_ON ( mp ) ) {
2005-04-17 02:20:36 +04:00
if ( ip - > i_d . di_uid ! = uid ) {
/*
* What we need is the dquot that has this uid , and
* if we send the inode to dqget , the uid of the inode
* takes priority over what ' s sent in the uid argument .
* We must unlock inode here before calling dqget if
* we ' re not sending the inode , because otherwise
* we ' ll deadlock by doing trans_reserve while
* holding ilock .
*/
xfs_iunlock ( ip , lockflags ) ;
if ( ( error = xfs_qm_dqget ( mp , NULL , ( xfs_dqid_t ) uid ,
XFS_DQ_USER ,
XFS_QMOPT_DQALLOC |
XFS_QMOPT_DOWARN ,
& uq ) ) ) {
ASSERT ( error ! = ENOENT ) ;
2006-01-15 04:37:08 +03:00
return error ;
2005-04-17 02:20:36 +04:00
}
/*
* Get the ilock in the right order .
*/
xfs_dqunlock ( uq ) ;
lockflags = XFS_ILOCK_SHARED ;
xfs_ilock ( ip , lockflags ) ;
} else {
/*
* Take an extra reference , because we ' ll return
* this to caller
*/
ASSERT ( ip - > i_udquot ) ;
uq = ip - > i_udquot ;
xfs_dqlock ( uq ) ;
XFS_DQHOLD ( uq ) ;
xfs_dqunlock ( uq ) ;
}
}
2005-06-21 09:38:48 +04:00
if ( ( flags & XFS_QMOPT_GQUOTA ) & & XFS_IS_GQUOTA_ON ( mp ) ) {
2005-04-17 02:20:36 +04:00
if ( ip - > i_d . di_gid ! = gid ) {
xfs_iunlock ( ip , lockflags ) ;
if ( ( error = xfs_qm_dqget ( mp , NULL , ( xfs_dqid_t ) gid ,
XFS_DQ_GROUP ,
XFS_QMOPT_DQALLOC |
XFS_QMOPT_DOWARN ,
& gq ) ) ) {
if ( uq )
xfs_qm_dqrele ( uq ) ;
ASSERT ( error ! = ENOENT ) ;
2006-01-15 04:37:08 +03:00
return error ;
2005-04-17 02:20:36 +04:00
}
xfs_dqunlock ( gq ) ;
lockflags = XFS_ILOCK_SHARED ;
xfs_ilock ( ip , lockflags ) ;
} else {
ASSERT ( ip - > i_gdquot ) ;
gq = ip - > i_gdquot ;
xfs_dqlock ( gq ) ;
XFS_DQHOLD ( gq ) ;
xfs_dqunlock ( gq ) ;
}
2005-06-21 09:38:48 +04:00
} else if ( ( flags & XFS_QMOPT_PQUOTA ) & & XFS_IS_PQUOTA_ON ( mp ) ) {
2010-09-26 10:10:18 +04:00
if ( xfs_get_projid ( ip ) ! = prid ) {
2005-06-21 09:38:48 +04:00
xfs_iunlock ( ip , lockflags ) ;
if ( ( error = xfs_qm_dqget ( mp , NULL , ( xfs_dqid_t ) prid ,
XFS_DQ_PROJ ,
XFS_QMOPT_DQALLOC |
XFS_QMOPT_DOWARN ,
& gq ) ) ) {
if ( uq )
xfs_qm_dqrele ( uq ) ;
ASSERT ( error ! = ENOENT ) ;
return ( error ) ;
}
xfs_dqunlock ( gq ) ;
lockflags = XFS_ILOCK_SHARED ;
xfs_ilock ( ip , lockflags ) ;
} else {
ASSERT ( ip - > i_gdquot ) ;
gq = ip - > i_gdquot ;
xfs_dqlock ( gq ) ;
XFS_DQHOLD ( gq ) ;
xfs_dqunlock ( gq ) ;
}
2005-04-17 02:20:36 +04:00
}
if ( uq )
2009-12-15 02:14:59 +03:00
trace_xfs_dquot_dqalloc ( ip ) ;
2005-04-17 02:20:36 +04:00
xfs_iunlock ( ip , lockflags ) ;
if ( O_udqpp )
* O_udqpp = uq ;
else if ( uq )
xfs_qm_dqrele ( uq ) ;
if ( O_gdqpp )
* O_gdqpp = gq ;
else if ( gq )
xfs_qm_dqrele ( gq ) ;
2006-01-15 04:37:08 +03:00
return 0 ;
2005-04-17 02:20:36 +04:00
}
/*
* Actually transfer ownership , and do dquot modifications .
* These were already reserved .
*/
xfs_dquot_t *
xfs_qm_vop_chown (
xfs_trans_t * tp ,
xfs_inode_t * ip ,
xfs_dquot_t * * IO_olddq ,
xfs_dquot_t * newdq )
{
xfs_dquot_t * prevdq ;
2005-06-21 09:48:47 +04:00
uint bfield = XFS_IS_REALTIME_INODE ( ip ) ?
XFS_TRANS_DQ_RTBCOUNT : XFS_TRANS_DQ_BCOUNT ;
2009-06-08 17:33:32 +04:00
2008-04-22 11:34:00 +04:00
ASSERT ( xfs_isilocked ( ip , XFS_ILOCK_EXCL ) ) ;
2005-04-17 02:20:36 +04:00
ASSERT ( XFS_IS_QUOTA_RUNNING ( ip - > i_mount ) ) ;
/* old dquot */
prevdq = * IO_olddq ;
ASSERT ( prevdq ) ;
ASSERT ( prevdq ! = newdq ) ;
2005-06-21 09:48:47 +04:00
xfs_trans_mod_dquot ( tp , prevdq , bfield , - ( ip - > i_d . di_nblocks ) ) ;
xfs_trans_mod_dquot ( tp , prevdq , XFS_TRANS_DQ_ICOUNT , - 1 ) ;
2005-04-17 02:20:36 +04:00
/* the sparkling new dquot */
2005-06-21 09:48:47 +04:00
xfs_trans_mod_dquot ( tp , newdq , bfield , ip - > i_d . di_nblocks ) ;
xfs_trans_mod_dquot ( tp , newdq , XFS_TRANS_DQ_ICOUNT , 1 ) ;
2005-04-17 02:20:36 +04:00
/*
* Take an extra reference , because the inode
* is going to keep this dquot pointer even
* after the trans_commit .
*/
xfs_dqlock ( newdq ) ;
XFS_DQHOLD ( newdq ) ;
xfs_dqunlock ( newdq ) ;
* IO_olddq = newdq ;
2006-01-15 04:37:08 +03:00
return prevdq ;
2005-04-17 02:20:36 +04:00
}
/*
2005-06-21 09:38:48 +04:00
* Quota reservations for setattr ( AT_UID | AT_GID | AT_PROJID ) .
2005-04-17 02:20:36 +04:00
*/
int
xfs_qm_vop_chown_reserve (
xfs_trans_t * tp ,
xfs_inode_t * ip ,
xfs_dquot_t * udqp ,
xfs_dquot_t * gdqp ,
uint flags )
{
2009-06-08 17:33:32 +04:00
xfs_mount_t * mp = ip - > i_mount ;
2006-03-31 07:04:49 +04:00
uint delblks , blkflags , prjflags = 0 ;
2005-04-17 02:20:36 +04:00
xfs_dquot_t * unresudq , * unresgdq , * delblksudq , * delblksgdq ;
2009-06-08 17:33:32 +04:00
int error ;
2005-04-17 02:20:36 +04:00
2008-04-22 11:34:00 +04:00
ASSERT ( xfs_isilocked ( ip , XFS_ILOCK_EXCL | XFS_ILOCK_SHARED ) ) ;
2005-04-17 02:20:36 +04:00
ASSERT ( XFS_IS_QUOTA_RUNNING ( mp ) ) ;
delblks = ip - > i_delayed_blks ;
delblksudq = delblksgdq = unresudq = unresgdq = NULL ;
2005-06-21 09:48:47 +04:00
blkflags = XFS_IS_REALTIME_INODE ( ip ) ?
XFS_QMOPT_RES_RTBLKS : XFS_QMOPT_RES_REGBLKS ;
2005-04-17 02:20:36 +04:00
if ( XFS_IS_UQUOTA_ON ( mp ) & & udqp & &
2005-11-02 07:01:12 +03:00
ip - > i_d . di_uid ! = ( uid_t ) be32_to_cpu ( udqp - > q_core . d_id ) ) {
2005-04-17 02:20:36 +04:00
delblksudq = udqp ;
/*
* If there are delayed allocation blocks , then we have to
* unreserve those from the old dquot , and add them to the
* new dquot .
*/
if ( delblks ) {
ASSERT ( ip - > i_udquot ) ;
unresudq = ip - > i_udquot ;
}
}
2005-06-21 09:38:48 +04:00
if ( XFS_IS_OQUOTA_ON ( ip - > i_mount ) & & gdqp ) {
2006-03-31 07:04:49 +04:00
if ( XFS_IS_PQUOTA_ON ( ip - > i_mount ) & &
2010-09-26 10:10:18 +04:00
xfs_get_projid ( ip ) ! = be32_to_cpu ( gdqp - > q_core . d_id ) )
2006-03-31 07:04:49 +04:00
prjflags = XFS_QMOPT_ENOSPC ;
if ( prjflags | |
( XFS_IS_GQUOTA_ON ( ip - > i_mount ) & &
ip - > i_d . di_gid ! = be32_to_cpu ( gdqp - > q_core . d_id ) ) ) {
2005-06-21 09:38:48 +04:00
delblksgdq = gdqp ;
if ( delblks ) {
ASSERT ( ip - > i_gdquot ) ;
unresgdq = ip - > i_gdquot ;
}
2005-04-17 02:20:36 +04:00
}
}
if ( ( error = xfs_trans_reserve_quota_bydquots ( tp , ip - > i_mount ,
delblksudq , delblksgdq , ip - > i_d . di_nblocks , 1 ,
2006-03-31 07:04:49 +04:00
flags | blkflags | prjflags ) ) )
2005-04-17 02:20:36 +04:00
return ( error ) ;
/*
* Do the delayed blks reservations / unreservations now . Since , these
* are done without the help of a transaction , if a reservation fails
* its previous reservations won ' t be automatically undone by trans
* code . So , we have to do it manually here .
*/
if ( delblks ) {
/*
* Do the reservations first . Unreservation can ' t fail .
*/
ASSERT ( delblksudq | | delblksgdq ) ;
ASSERT ( unresudq | | unresgdq ) ;
if ( ( error = xfs_trans_reserve_quota_bydquots ( NULL , ip - > i_mount ,
delblksudq , delblksgdq , ( xfs_qcnt_t ) delblks , 0 ,
2006-03-31 07:04:49 +04:00
flags | blkflags | prjflags ) ) )
2005-04-17 02:20:36 +04:00
return ( error ) ;
xfs_trans_reserve_quota_bydquots ( NULL , ip - > i_mount ,
unresudq , unresgdq , - ( ( xfs_qcnt_t ) delblks ) , 0 ,
2005-06-21 09:48:47 +04:00
blkflags ) ;
2005-04-17 02:20:36 +04:00
}
return ( 0 ) ;
}
int
xfs_qm_vop_rename_dqattach (
2009-06-08 17:33:32 +04:00
struct xfs_inode * * i_tab )
2005-04-17 02:20:36 +04:00
{
2009-06-08 17:33:32 +04:00
struct xfs_mount * mp = i_tab [ 0 ] - > i_mount ;
int i ;
2005-04-17 02:20:36 +04:00
2009-06-08 17:33:32 +04:00
if ( ! XFS_IS_QUOTA_RUNNING ( mp ) | | ! XFS_IS_QUOTA_ON ( mp ) )
2006-01-15 04:37:08 +03:00
return 0 ;
2005-04-17 02:20:36 +04:00
2009-06-08 17:33:32 +04:00
for ( i = 0 ; ( i < 4 & & i_tab [ i ] ) ; i + + ) {
struct xfs_inode * ip = i_tab [ i ] ;
int error ;
2005-04-17 02:20:36 +04:00
/*
* Watch out for duplicate entries in the table .
*/
2009-06-08 17:33:32 +04:00
if ( i = = 0 | | ip ! = i_tab [ i - 1 ] ) {
if ( XFS_NOT_DQATTACHED ( mp , ip ) ) {
2005-04-17 02:20:36 +04:00
error = xfs_qm_dqattach ( ip , 0 ) ;
if ( error )
2006-01-15 04:37:08 +03:00
return error ;
2005-04-17 02:20:36 +04:00
}
}
}
2006-01-15 04:37:08 +03:00
return 0 ;
2005-04-17 02:20:36 +04:00
}
void
2009-06-08 17:33:32 +04:00
xfs_qm_vop_create_dqattach (
struct xfs_trans * tp ,
struct xfs_inode * ip ,
struct xfs_dquot * udqp ,
struct xfs_dquot * gdqp )
2005-04-17 02:20:36 +04:00
{
2009-06-08 17:33:32 +04:00
struct xfs_mount * mp = tp - > t_mountp ;
if ( ! XFS_IS_QUOTA_RUNNING ( mp ) | | ! XFS_IS_QUOTA_ON ( mp ) )
2005-04-17 02:20:36 +04:00
return ;
2008-04-22 11:34:00 +04:00
ASSERT ( xfs_isilocked ( ip , XFS_ILOCK_EXCL ) ) ;
2009-06-08 17:33:32 +04:00
ASSERT ( XFS_IS_QUOTA_RUNNING ( mp ) ) ;
2005-04-17 02:20:36 +04:00
if ( udqp ) {
xfs_dqlock ( udqp ) ;
XFS_DQHOLD ( udqp ) ;
xfs_dqunlock ( udqp ) ;
ASSERT ( ip - > i_udquot = = NULL ) ;
ip - > i_udquot = udqp ;
2009-06-08 17:33:32 +04:00
ASSERT ( XFS_IS_UQUOTA_ON ( mp ) ) ;
2005-11-02 07:01:12 +03:00
ASSERT ( ip - > i_d . di_uid = = be32_to_cpu ( udqp - > q_core . d_id ) ) ;
2005-04-17 02:20:36 +04:00
xfs_trans_mod_dquot ( tp , udqp , XFS_TRANS_DQ_ICOUNT , 1 ) ;
}
if ( gdqp ) {
xfs_dqlock ( gdqp ) ;
XFS_DQHOLD ( gdqp ) ;
xfs_dqunlock ( gdqp ) ;
ASSERT ( ip - > i_gdquot = = NULL ) ;
ip - > i_gdquot = gdqp ;
2009-06-08 17:33:32 +04:00
ASSERT ( XFS_IS_OQUOTA_ON ( mp ) ) ;
ASSERT ( ( XFS_IS_GQUOTA_ON ( mp ) ?
2010-09-26 10:10:18 +04:00
ip - > i_d . di_gid : xfs_get_projid ( ip ) ) = =
2006-01-11 07:33:36 +03:00
be32_to_cpu ( gdqp - > q_core . d_id ) ) ;
2005-04-17 02:20:36 +04:00
xfs_trans_mod_dquot ( tp , gdqp , XFS_TRANS_DQ_ICOUNT , 1 ) ;
}
}