2013-10-15 02:17:52 +04:00
/*
* Copyright ( c ) 2000 - 2006 Silicon Graphics , Inc .
* Copyright ( c ) 2013 Red Hat , Inc .
* All Rights Reserved .
*
* This program is free software ; you can redistribute it and / or
* modify it under the terms of the GNU General Public License as
* published by the Free Software Foundation .
*
* 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 .
*
* 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
*/
# include "xfs.h"
# include "xfs_fs.h"
2013-10-29 15:11:58 +04:00
# include "xfs_shared.h"
2013-10-15 02:17:52 +04:00
# include "xfs_format.h"
2013-10-23 03:50:10 +04:00
# include "xfs_log_format.h"
# include "xfs_trans_resv.h"
2013-10-15 02:17:52 +04:00
# include "xfs_mount.h"
# include "xfs_inode.h"
# include "xfs_quota.h"
2013-10-23 03:50:10 +04:00
# include "xfs_trans.h"
2013-10-15 02:17:52 +04:00
# include "xfs_qm.h"
# include "xfs_error.h"
# include "xfs_cksum.h"
# include "xfs_trace.h"
int
xfs_calc_dquots_per_chunk (
unsigned int nbblks ) /* basic block units */
{
ASSERT ( nbblks > 0 ) ;
2017-04-07 02:01:47 +03:00
return BBTOB ( nbblks ) / sizeof ( xfs_dqblk_t ) ;
2013-10-15 02:17:52 +04:00
}
/*
* Do some primitive error checking on ondisk dquot data structures .
2018-05-07 19:20:18 +03:00
*
* The xfs_dqblk structure / contains / the xfs_disk_dquot structure ;
* we verify them separately because at some points we have only the
* smaller xfs_disk_dquot structure available .
2013-10-15 02:17:52 +04:00
*/
2018-05-07 19:20:18 +03:00
2018-01-08 21:51:25 +03:00
xfs_failaddr_t
xfs_dquot_verify (
2013-10-15 02:17:52 +04:00
struct xfs_mount * mp ,
xfs_disk_dquot_t * ddq ,
xfs_dqid_t id ,
2018-05-07 19:20:17 +03:00
uint type ) /* used only during quotacheck */
2013-10-15 02:17:52 +04:00
{
/*
* We can encounter an uninitialized dquot buffer for 2 reasons :
* 1. If we crash while deleting the quotainode ( s ) , and those blks got
* used for user data . This is because we take the path of regular
* file deletion ; however , the size field of quotainodes is never
* updated , so all the tricks that we play in itruncate_finish
* don ' t quite matter .
*
* 2. We don ' t play the quota buffers when there ' s a quotaoff logitem .
* But the allocation will be replayed so we ' ll end up with an
* uninitialized quota block .
*
* This is all fine ; things are still consistent , and we haven ' t lost
* any quota information . Just don ' t complain about bad dquot blks .
*/
2018-01-08 21:51:25 +03:00
if ( ddq - > d_magic ! = cpu_to_be16 ( XFS_DQUOT_MAGIC ) )
return __this_address ;
if ( ddq - > d_version ! = XFS_DQUOT_VERSION )
return __this_address ;
2013-10-15 02:17:52 +04:00
2018-05-07 19:20:17 +03:00
if ( type & & ddq - > d_flags ! = type )
return __this_address ;
2013-10-15 02:17:52 +04:00
if ( ddq - > d_flags ! = XFS_DQ_USER & &
ddq - > d_flags ! = XFS_DQ_PROJ & &
2018-01-08 21:51:25 +03:00
ddq - > d_flags ! = XFS_DQ_GROUP )
return __this_address ;
2013-10-15 02:17:52 +04:00
2018-01-08 21:51:25 +03:00
if ( id ! = - 1 & & id ! = be32_to_cpu ( ddq - > d_id ) )
return __this_address ;
2013-10-15 02:17:52 +04:00
2018-01-08 21:51:25 +03:00
if ( ! ddq - > d_id )
return NULL ;
2013-10-15 02:17:52 +04:00
2018-01-08 21:51:25 +03:00
if ( ddq - > d_blk_softlimit & &
be64_to_cpu ( ddq - > d_bcount ) > be64_to_cpu ( ddq - > d_blk_softlimit ) & &
! ddq - > d_btimer )
return __this_address ;
if ( ddq - > d_ino_softlimit & &
be64_to_cpu ( ddq - > d_icount ) > be64_to_cpu ( ddq - > d_ino_softlimit ) & &
! ddq - > d_itimer )
return __this_address ;
if ( ddq - > d_rtb_softlimit & &
be64_to_cpu ( ddq - > d_rtbcount ) > be64_to_cpu ( ddq - > d_rtb_softlimit ) & &
! ddq - > d_rtbtimer )
return __this_address ;
return NULL ;
2018-01-08 21:51:24 +03:00
}
2018-05-07 19:20:18 +03:00
xfs_failaddr_t
xfs_dqblk_verify (
struct xfs_mount * mp ,
struct xfs_dqblk * dqb ,
xfs_dqid_t id ,
uint type ) /* used only during quotacheck */
{
if ( xfs_sb_version_hascrc ( & mp - > m_sb ) & &
! uuid_equal ( & dqb - > dd_uuid , & mp - > m_sb . sb_meta_uuid ) )
return __this_address ;
return xfs_dquot_verify ( mp , & dqb - > dd_diskdq , id , type ) ;
}
2018-01-08 21:51:24 +03:00
/*
* Do some primitive error checking on ondisk dquot data structures .
*/
int
2018-05-07 19:20:17 +03:00
xfs_dqblk_repair (
2018-01-08 21:51:24 +03:00
struct xfs_mount * mp ,
2018-05-07 19:20:17 +03:00
struct xfs_dqblk * dqb ,
2018-01-08 21:51:24 +03:00
xfs_dqid_t id ,
uint type )
{
2013-10-15 02:17:52 +04:00
/*
* Typically , a repair is only requested by quotacheck .
*/
ASSERT ( id ! = - 1 ) ;
2018-05-07 19:20:17 +03:00
memset ( dqb , 0 , sizeof ( xfs_dqblk_t ) ) ;
2013-10-15 02:17:52 +04:00
2018-05-07 19:20:17 +03:00
dqb - > dd_diskdq . d_magic = cpu_to_be16 ( XFS_DQUOT_MAGIC ) ;
dqb - > dd_diskdq . d_version = XFS_DQUOT_VERSION ;
dqb - > dd_diskdq . d_flags = type ;
dqb - > dd_diskdq . d_id = cpu_to_be32 ( id ) ;
2013-10-15 02:17:52 +04:00
if ( xfs_sb_version_hascrc ( & mp - > m_sb ) ) {
2018-05-07 19:20:17 +03:00
uuid_copy ( & dqb - > dd_uuid , & mp - > m_sb . sb_meta_uuid ) ;
xfs_update_cksum ( ( char * ) dqb , sizeof ( struct xfs_dqblk ) ,
2013-10-15 02:17:52 +04:00
XFS_DQUOT_CRC_OFF ) ;
}
2018-01-08 21:51:24 +03:00
return 0 ;
2013-10-15 02:17:52 +04:00
}
STATIC bool
xfs_dquot_buf_verify_crc (
struct xfs_mount * mp ,
struct xfs_buf * bp )
{
struct xfs_dqblk * d = ( struct xfs_dqblk * ) bp - > b_addr ;
int ndquots ;
int i ;
if ( ! xfs_sb_version_hascrc ( & mp - > m_sb ) )
return true ;
/*
* if we are in log recovery , the quota subsystem has not been
* initialised so we have no quotainfo structure . In that case , we need
* to manually calculate the number of dquots in the buffer .
*/
if ( mp - > m_quotainfo )
ndquots = mp - > m_quotainfo - > qi_dqperchunk ;
else
2016-10-20 07:46:18 +03:00
ndquots = xfs_calc_dquots_per_chunk ( bp - > b_length ) ;
2013-10-15 02:17:52 +04:00
for ( i = 0 ; i < ndquots ; i + + , d + + ) {
if ( ! xfs_verify_cksum ( ( char * ) d , sizeof ( struct xfs_dqblk ) ,
XFS_DQUOT_CRC_OFF ) )
return false ;
}
return true ;
}
2018-01-08 21:51:25 +03:00
STATIC xfs_failaddr_t
2013-10-15 02:17:52 +04:00
xfs_dquot_buf_verify (
struct xfs_mount * mp ,
2018-01-08 21:51:25 +03:00
struct xfs_buf * bp )
2013-10-15 02:17:52 +04:00
{
2018-05-07 19:20:18 +03:00
struct xfs_dqblk * dqb = bp - > b_addr ;
2018-01-08 21:51:25 +03:00
xfs_failaddr_t fa ;
2013-10-15 02:17:52 +04:00
xfs_dqid_t id = 0 ;
int ndquots ;
int i ;
/*
* if we are in log recovery , the quota subsystem has not been
* initialised so we have no quotainfo structure . In that case , we need
* to manually calculate the number of dquots in the buffer .
*/
if ( mp - > m_quotainfo )
ndquots = mp - > m_quotainfo - > qi_dqperchunk ;
else
2014-04-14 13:03:34 +04:00
ndquots = xfs_calc_dquots_per_chunk ( bp - > b_length ) ;
2013-10-15 02:17:52 +04:00
/*
* On the first read of the buffer , verify that each dquot is valid .
* We don ' t know what the id of the dquot is supposed to be , just that
* they should be increasing monotonically within the buffer . If the
* first id is corrupt , then it will fail on the second dquot in the
* buffer so corruptions could point to the wrong dquot in this case .
*/
for ( i = 0 ; i < ndquots ; i + + ) {
struct xfs_disk_dquot * ddq ;
2018-05-07 19:20:18 +03:00
ddq = & dqb [ i ] . dd_diskdq ;
2013-10-15 02:17:52 +04:00
if ( i = = 0 )
id = be32_to_cpu ( ddq - > d_id ) ;
2018-05-07 19:20:18 +03:00
fa = xfs_dqblk_verify ( mp , & dqb [ i ] , id + i , 0 ) ;
2018-01-08 21:51:25 +03:00
if ( fa )
return fa ;
2013-10-15 02:17:52 +04:00
}
2018-01-08 21:51:25 +03:00
return NULL ;
2013-10-15 02:17:52 +04:00
}
2018-01-08 21:51:08 +03:00
static xfs_failaddr_t
xfs_dquot_buf_verify_struct (
2018-01-08 21:51:25 +03:00
struct xfs_buf * bp )
2018-01-08 21:51:08 +03:00
{
struct xfs_mount * mp = bp - > b_target - > bt_mount ;
2018-01-08 21:51:25 +03:00
return xfs_dquot_buf_verify ( mp , bp ) ;
2018-01-08 21:51:08 +03:00
}
2013-10-15 02:17:52 +04:00
static void
xfs_dquot_buf_read_verify (
2018-01-08 21:51:25 +03:00
struct xfs_buf * bp )
2013-10-15 02:17:52 +04:00
{
struct xfs_mount * mp = bp - > b_target - > bt_mount ;
2018-01-08 21:51:25 +03:00
xfs_failaddr_t fa ;
2013-10-15 02:17:52 +04:00
2014-02-27 08:23:10 +04:00
if ( ! xfs_dquot_buf_verify_crc ( mp , bp ) )
2018-01-08 21:51:03 +03:00
xfs_verifier_error ( bp , - EFSBADCRC , __this_address ) ;
2018-01-08 21:51:25 +03:00
else {
fa = xfs_dquot_buf_verify ( mp , bp ) ;
if ( fa )
xfs_verifier_error ( bp , - EFSCORRUPTED , __this_address ) ;
}
2013-10-15 02:17:52 +04:00
}
2016-01-11 23:04:01 +03:00
/*
* readahead errors are silent and simply leave the buffer as ! done so a real
* read will then be run with the xfs_dquot_buf_ops verifier . See
* xfs_inode_buf_verify ( ) for why we use EIO and ~ XBF_DONE here rather than
* reporting the failure .
*/
static void
xfs_dquot_buf_readahead_verify (
struct xfs_buf * bp )
{
struct xfs_mount * mp = bp - > b_target - > bt_mount ;
if ( ! xfs_dquot_buf_verify_crc ( mp , bp ) | |
2018-01-08 21:51:25 +03:00
xfs_dquot_buf_verify ( mp , bp ) ! = NULL ) {
2016-01-11 23:04:01 +03:00
xfs_buf_ioerror ( bp , - EIO ) ;
bp - > b_flags & = ~ XBF_DONE ;
}
}
2013-10-15 02:17:52 +04:00
/*
* we don ' t calculate the CRC here as that is done when the dquot is flushed to
* the buffer after the update is done . This ensures that the dquot in the
* buffer always has an up - to - date CRC value .
*/
2013-10-29 15:11:58 +04:00
static void
2013-10-15 02:17:52 +04:00
xfs_dquot_buf_write_verify (
2018-01-08 21:51:25 +03:00
struct xfs_buf * bp )
2013-10-15 02:17:52 +04:00
{
struct xfs_mount * mp = bp - > b_target - > bt_mount ;
2018-01-08 21:51:25 +03:00
xfs_failaddr_t fa ;
2013-10-15 02:17:52 +04:00
2018-01-08 21:51:25 +03:00
fa = xfs_dquot_buf_verify ( mp , bp ) ;
if ( fa )
2018-01-08 21:51:03 +03:00
xfs_verifier_error ( bp , - EFSCORRUPTED , __this_address ) ;
2013-10-15 02:17:52 +04:00
}
const struct xfs_buf_ops xfs_dquot_buf_ops = {
2016-01-04 08:10:19 +03:00
. name = " xfs_dquot " ,
2013-10-15 02:17:52 +04:00
. verify_read = xfs_dquot_buf_read_verify ,
. verify_write = xfs_dquot_buf_write_verify ,
2018-01-08 21:51:08 +03:00
. verify_struct = xfs_dquot_buf_verify_struct ,
2013-10-15 02:17:52 +04:00
} ;
2016-01-11 23:04:01 +03:00
const struct xfs_buf_ops xfs_dquot_buf_ra_ops = {
. name = " xfs_dquot_ra " ,
. verify_read = xfs_dquot_buf_readahead_verify ,
. verify_write = xfs_dquot_buf_write_verify ,
} ;