2005-04-16 15:20:36 -07:00
/*
2005-11-02 14:58:39 +11:00
* Copyright ( c ) 2000 - 2001 , 2005 Silicon Graphics , Inc .
* All Rights Reserved .
2005-04-16 15:20:36 -07:00
*
2005-11-02 14:58:39 +11: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-16 15:20:36 -07:00
* published by the Free Software Foundation .
*
2005-11-02 14:58:39 +11: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-16 15:20:36 -07:00
*
2005-11-02 14:58:39 +11: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-16 15:20:36 -07:00
*/
# include "xfs.h"
2005-11-02 14:38:42 +11:00
# include "xfs_fs.h"
2013-10-29 22:11:58 +11:00
# include "xfs_shared.h"
2013-10-23 10:51:50 +11:00
# include "xfs_format.h"
2013-10-23 10:50:10 +11:00
# include "xfs_log_format.h"
# include "xfs_trans_resv.h"
2005-11-02 14:38:42 +11:00
# include "xfs_bit.h"
2005-04-16 15:20:36 -07:00
# include "xfs_mount.h"
2005-11-02 14:38:42 +11:00
# include "xfs_inode.h"
2005-04-16 15:20:36 -07:00
# include "xfs_btree.h"
# include "xfs_ialloc.h"
2013-10-23 10:51:50 +11:00
# include "xfs_ialloc_btree.h"
2005-04-16 15:20:36 -07:00
# include "xfs_alloc.h"
# include "xfs_error.h"
2012-11-12 22:54:08 +11:00
# include "xfs_trace.h"
2013-04-21 14:53:46 -05:00
# include "xfs_cksum.h"
2013-10-23 10:50:10 +11:00
# include "xfs_trans.h"
2005-04-16 15:20:36 -07:00
2008-10-30 16:58:01 +11:00
STATIC int
xfs_inobt_get_minrecs (
struct xfs_btree_cur * cur ,
int level )
{
return cur - > bc_mp - > m_inobt_mnr [ level ! = 0 ] ;
}
2005-04-16 15:20:36 -07:00
2008-10-30 16:53:59 +11:00
STATIC struct xfs_btree_cur *
xfs_inobt_dup_cursor (
struct xfs_btree_cur * cur )
{
return xfs_inobt_init_cursor ( cur - > bc_mp , cur - > bc_tp ,
2014-04-24 16:00:50 +10:00
cur - > bc_private . a . agbp , cur - > bc_private . a . agno ,
cur - > bc_btnum ) ;
2008-10-30 16:53:59 +11:00
}
2008-10-30 16:57:16 +11:00
STATIC void
xfs_inobt_set_root (
struct xfs_btree_cur * cur ,
union xfs_btree_ptr * nptr ,
int inc ) /* level change */
{
struct xfs_buf * agbp = cur - > bc_private . a . agbp ;
struct xfs_agi * agi = XFS_BUF_TO_AGI ( agbp ) ;
agi - > agi_root = nptr - > s ;
be32_add_cpu ( & agi - > agi_level , inc ) ;
xfs_ialloc_log_agi ( cur - > bc_tp , agbp , XFS_AGI_ROOT | XFS_AGI_LEVEL ) ;
}
2014-04-24 16:00:52 +10:00
STATIC void
xfs_finobt_set_root (
struct xfs_btree_cur * cur ,
union xfs_btree_ptr * nptr ,
int inc ) /* level change */
{
struct xfs_buf * agbp = cur - > bc_private . a . agbp ;
struct xfs_agi * agi = XFS_BUF_TO_AGI ( agbp ) ;
agi - > agi_free_root = nptr - > s ;
be32_add_cpu ( & agi - > agi_free_level , inc ) ;
xfs_ialloc_log_agi ( cur - > bc_tp , agbp ,
XFS_AGI_FREE_ROOT | XFS_AGI_FREE_LEVEL ) ;
}
2008-10-30 16:57:03 +11:00
STATIC int
xfs_inobt_alloc_block (
struct xfs_btree_cur * cur ,
union xfs_btree_ptr * start ,
union xfs_btree_ptr * new ,
int * stat )
{
xfs_alloc_arg_t args ; /* block allocation args */
int error ; /* error return value */
xfs_agblock_t sbno = be32_to_cpu ( start - > s ) ;
XFS_BTREE_TRACE_CURSOR ( cur , XBT_ENTRY ) ;
memset ( & args , 0 , sizeof ( args ) ) ;
args . tp = cur - > bc_tp ;
args . mp = cur - > bc_mp ;
args . fsbno = XFS_AGB_TO_FSB ( args . mp , cur - > bc_private . a . agno , sbno ) ;
args . minlen = 1 ;
args . maxlen = 1 ;
args . prod = 1 ;
args . type = XFS_ALLOCTYPE_NEAR_BNO ;
error = xfs_alloc_vextent ( & args ) ;
if ( error ) {
XFS_BTREE_TRACE_CURSOR ( cur , XBT_ERROR ) ;
return error ;
}
if ( args . fsbno = = NULLFSBLOCK ) {
XFS_BTREE_TRACE_CURSOR ( cur , XBT_EXIT ) ;
* stat = 0 ;
return 0 ;
}
ASSERT ( args . len = = 1 ) ;
XFS_BTREE_TRACE_CURSOR ( cur , XBT_EXIT ) ;
new - > s = cpu_to_be32 ( XFS_FSB_TO_AGBNO ( args . mp , args . fsbno ) ) ;
* stat = 1 ;
return 0 ;
}
2008-10-30 16:57:51 +11:00
STATIC int
xfs_inobt_free_block (
struct xfs_btree_cur * cur ,
struct xfs_buf * bp )
{
xfs_fsblock_t fsbno ;
int error ;
fsbno = XFS_DADDR_TO_FSB ( cur - > bc_mp , XFS_BUF_ADDR ( bp ) ) ;
error = xfs_free_extent ( cur - > bc_tp , fsbno , 1 ) ;
if ( error )
return error ;
xfs_trans_binval ( cur - > bc_tp , bp ) ;
return error ;
}
2008-10-30 16:57:03 +11:00
2008-10-30 16:55:23 +11:00
STATIC int
xfs_inobt_get_maxrecs (
struct xfs_btree_cur * cur ,
int level )
{
return cur - > bc_mp - > m_inobt_mxr [ level ! = 0 ] ;
}
2008-10-30 16:56:09 +11:00
STATIC void
xfs_inobt_init_key_from_rec (
union xfs_btree_key * key ,
union xfs_btree_rec * rec )
{
key - > inobt . ir_startino = rec - > inobt . ir_startino ;
}
2008-10-30 16:57:40 +11:00
STATIC void
xfs_inobt_init_rec_from_key (
union xfs_btree_key * key ,
union xfs_btree_rec * rec )
{
rec - > inobt . ir_startino = key - > inobt . ir_startino ;
}
STATIC void
xfs_inobt_init_rec_from_cur (
struct xfs_btree_cur * cur ,
union xfs_btree_rec * rec )
{
rec - > inobt . ir_startino = cpu_to_be32 ( cur - > bc_rec . i . ir_startino ) ;
2015-05-29 09:03:04 +10:00
if ( xfs_sb_version_hassparseinodes ( & cur - > bc_mp - > m_sb ) ) {
rec - > inobt . ir_u . sp . ir_holemask =
cpu_to_be16 ( cur - > bc_rec . i . ir_holemask ) ;
rec - > inobt . ir_u . sp . ir_count = cur - > bc_rec . i . ir_count ;
rec - > inobt . ir_u . sp . ir_freecount = cur - > bc_rec . i . ir_freecount ;
} else {
/* ir_holemask/ir_count not supported on-disk */
rec - > inobt . ir_u . f . ir_freecount =
cpu_to_be32 ( cur - > bc_rec . i . ir_freecount ) ;
}
2008-10-30 16:57:40 +11:00
rec - > inobt . ir_free = cpu_to_be64 ( cur - > bc_rec . i . ir_free ) ;
}
2008-10-30 16:56:09 +11:00
/*
2009-03-29 09:55:42 +02:00
* initial value of ptr for lookup
2008-10-30 16:56:09 +11:00
*/
STATIC void
xfs_inobt_init_ptr_from_cur (
struct xfs_btree_cur * cur ,
union xfs_btree_ptr * ptr )
{
struct xfs_agi * agi = XFS_BUF_TO_AGI ( cur - > bc_private . a . agbp ) ;
ASSERT ( cur - > bc_private . a . agno = = be32_to_cpu ( agi - > agi_seqno ) ) ;
ptr - > s = agi - > agi_root ;
}
2014-04-24 16:00:52 +10:00
STATIC void
xfs_finobt_init_ptr_from_cur (
struct xfs_btree_cur * cur ,
union xfs_btree_ptr * ptr )
{
struct xfs_agi * agi = XFS_BUF_TO_AGI ( cur - > bc_private . a . agbp ) ;
ASSERT ( cur - > bc_private . a . agno = = be32_to_cpu ( agi - > agi_seqno ) ) ;
ptr - > s = agi - > agi_free_root ;
}
2008-10-30 16:56:09 +11:00
STATIC __int64_t
xfs_inobt_key_diff (
struct xfs_btree_cur * cur ,
union xfs_btree_key * key )
{
return ( __int64_t ) be32_to_cpu ( key - > inobt . ir_startino ) -
cur - > bc_rec . i . ir_startino ;
}
2013-04-21 14:53:46 -05:00
static int
2012-11-14 17:52:32 +11:00
xfs_inobt_verify (
2012-11-12 22:54:08 +11:00
struct xfs_buf * bp )
{
struct xfs_mount * mp = bp - > b_target - > bt_mount ;
struct xfs_btree_block * block = XFS_BUF_TO_BLOCK ( bp ) ;
2013-04-21 14:53:46 -05:00
struct xfs_perag * pag = bp - > b_pag ;
2012-11-12 22:54:08 +11:00
unsigned int level ;
2013-04-21 14:53:46 -05:00
/*
* During growfs operations , we can ' t verify the exact owner as the
* perag is not fully initialised and hence not attached to the buffer .
*
* Similarly , during log recovery we will have a perag structure
* attached , but the agi information will not yet have been initialised
* from the on disk AGI . We don ' t currently use any of this information ,
* but beware of the landmine ( i . e . need to check pag - > pagi_init ) if we
* ever do .
*/
switch ( block - > bb_magic ) {
case cpu_to_be32 ( XFS_IBT_CRC_MAGIC ) :
2014-04-24 16:00:52 +10:00
case cpu_to_be32 ( XFS_FIBT_CRC_MAGIC ) :
2013-04-21 14:53:46 -05:00
if ( ! xfs_sb_version_hascrc ( & mp - > m_sb ) )
return false ;
2015-07-29 11:53:31 +10:00
if ( ! uuid_equal ( & block - > bb_u . s . bb_uuid , & mp - > m_sb . sb_meta_uuid ) )
2013-04-21 14:53:46 -05:00
return false ;
if ( block - > bb_u . s . bb_blkno ! = cpu_to_be64 ( bp - > b_bn ) )
return false ;
if ( pag & &
be32_to_cpu ( block - > bb_u . s . bb_owner ) ! = pag - > pag_agno )
return false ;
/* fall through */
case cpu_to_be32 ( XFS_IBT_MAGIC ) :
2014-04-24 16:00:52 +10:00
case cpu_to_be32 ( XFS_FIBT_MAGIC ) :
2013-04-21 14:53:46 -05:00
break ;
default :
return 0 ;
}
2012-11-12 22:54:08 +11:00
2013-04-21 14:53:46 -05:00
/* numrecs and level verification */
level = be16_to_cpu ( block - > bb_level ) ;
if ( level > = mp - > m_in_maxlevels )
return false ;
if ( be16_to_cpu ( block - > bb_numrecs ) > mp - > m_inobt_mxr [ level ! = 0 ] )
return false ;
2012-11-12 22:54:08 +11:00
/* sibling pointer verification */
2013-04-21 14:53:46 -05:00
if ( ! block - > bb_u . s . bb_leftsib | |
( be32_to_cpu ( block - > bb_u . s . bb_leftsib ) > = mp - > m_sb . sb_agblocks & &
block - > bb_u . s . bb_leftsib ! = cpu_to_be32 ( NULLAGBLOCK ) ) )
return false ;
if ( ! block - > bb_u . s . bb_rightsib | |
( be32_to_cpu ( block - > bb_u . s . bb_rightsib ) > = mp - > m_sb . sb_agblocks & &
block - > bb_u . s . bb_rightsib ! = cpu_to_be32 ( NULLAGBLOCK ) ) )
return false ;
return true ;
2012-11-14 17:52:32 +11:00
}
static void
2012-11-14 17:54:40 +11:00
xfs_inobt_read_verify (
2012-11-14 17:52:32 +11:00
struct xfs_buf * bp )
{
2014-02-27 15:23:10 +11:00
if ( ! xfs_btree_sblock_verify_crc ( bp ) )
2014-06-25 14:58:08 +10:00
xfs_buf_ioerror ( bp , - EFSBADCRC ) ;
2014-02-27 15:23:10 +11:00
else if ( ! xfs_inobt_verify ( bp ) )
2014-06-25 14:58:08 +10:00
xfs_buf_ioerror ( bp , - EFSCORRUPTED ) ;
2014-02-27 15:23:10 +11:00
if ( bp - > b_error ) {
trace_xfs_btree_corrupt ( bp , _RET_IP_ ) ;
xfs_verifier_error ( bp ) ;
2013-04-21 14:53:46 -05:00
}
2012-11-14 17:52:32 +11:00
}
2012-11-12 22:54:08 +11:00
2012-11-14 17:54:40 +11:00
static void
xfs_inobt_write_verify (
2012-11-14 17:52:32 +11:00
struct xfs_buf * bp )
{
2013-04-21 14:53:46 -05:00
if ( ! xfs_inobt_verify ( bp ) ) {
trace_xfs_btree_corrupt ( bp , _RET_IP_ ) ;
2014-06-25 14:58:08 +10:00
xfs_buf_ioerror ( bp , - EFSCORRUPTED ) ;
2014-02-27 15:23:10 +11:00
xfs_verifier_error ( bp ) ;
2014-02-27 15:14:31 +11:00
return ;
2013-04-21 14:53:46 -05:00
}
xfs_btree_sblock_calc_crc ( bp ) ;
2012-11-12 22:54:08 +11:00
}
2012-11-14 17:54:40 +11:00
const struct xfs_buf_ops xfs_inobt_buf_ops = {
2016-01-04 16:10:19 +11:00
. name = " xfs_inobt " ,
2012-11-14 17:54:40 +11:00
. verify_read = xfs_inobt_read_verify ,
. verify_write = xfs_inobt_write_verify ,
} ;
2013-04-30 21:39:34 +10:00
# if defined(DEBUG) || defined(XFS_WARN)
2008-10-30 16:58:32 +11:00
STATIC int
xfs_inobt_keys_inorder (
struct xfs_btree_cur * cur ,
union xfs_btree_key * k1 ,
union xfs_btree_key * k2 )
{
return be32_to_cpu ( k1 - > inobt . ir_startino ) <
be32_to_cpu ( k2 - > inobt . ir_startino ) ;
}
STATIC int
xfs_inobt_recs_inorder (
struct xfs_btree_cur * cur ,
union xfs_btree_rec * r1 ,
union xfs_btree_rec * r2 )
{
return be32_to_cpu ( r1 - > inobt . ir_startino ) + XFS_INODES_PER_CHUNK < =
be32_to_cpu ( r2 - > inobt . ir_startino ) ;
}
# endif /* DEBUG */
2008-10-30 16:53:59 +11:00
static const struct xfs_btree_ops xfs_inobt_ops = {
2008-10-30 16:55:34 +11:00
. rec_len = sizeof ( xfs_inobt_rec_t ) ,
. key_len = sizeof ( xfs_inobt_key_t ) ,
2008-10-30 16:53:59 +11:00
. dup_cursor = xfs_inobt_dup_cursor ,
2008-10-30 16:57:16 +11:00
. set_root = xfs_inobt_set_root ,
2008-10-30 16:57:03 +11:00
. alloc_block = xfs_inobt_alloc_block ,
2008-10-30 16:57:51 +11:00
. free_block = xfs_inobt_free_block ,
2008-10-30 16:58:01 +11:00
. get_minrecs = xfs_inobt_get_minrecs ,
2008-10-30 16:55:23 +11:00
. get_maxrecs = xfs_inobt_get_maxrecs ,
2008-10-30 16:56:09 +11:00
. init_key_from_rec = xfs_inobt_init_key_from_rec ,
2008-10-30 16:57:40 +11:00
. init_rec_from_key = xfs_inobt_init_rec_from_key ,
. init_rec_from_cur = xfs_inobt_init_rec_from_cur ,
2008-10-30 16:56:09 +11:00
. init_ptr_from_cur = xfs_inobt_init_ptr_from_cur ,
. key_diff = xfs_inobt_key_diff ,
2012-11-14 17:54:40 +11:00
. buf_ops = & xfs_inobt_buf_ops ,
2013-04-30 21:39:34 +10:00
# if defined(DEBUG) || defined(XFS_WARN)
2008-10-30 16:58:32 +11:00
. keys_inorder = xfs_inobt_keys_inorder ,
. recs_inorder = xfs_inobt_recs_inorder ,
# endif
2008-10-30 16:53:59 +11:00
} ;
2014-04-24 16:00:52 +10:00
static const struct xfs_btree_ops xfs_finobt_ops = {
. rec_len = sizeof ( xfs_inobt_rec_t ) ,
. key_len = sizeof ( xfs_inobt_key_t ) ,
. dup_cursor = xfs_inobt_dup_cursor ,
. set_root = xfs_finobt_set_root ,
. alloc_block = xfs_inobt_alloc_block ,
. free_block = xfs_inobt_free_block ,
. get_minrecs = xfs_inobt_get_minrecs ,
. get_maxrecs = xfs_inobt_get_maxrecs ,
. init_key_from_rec = xfs_inobt_init_key_from_rec ,
. init_rec_from_key = xfs_inobt_init_rec_from_key ,
. init_rec_from_cur = xfs_inobt_init_rec_from_cur ,
. init_ptr_from_cur = xfs_finobt_init_ptr_from_cur ,
. key_diff = xfs_inobt_key_diff ,
. buf_ops = & xfs_inobt_buf_ops ,
# if defined(DEBUG) || defined(XFS_WARN)
. keys_inorder = xfs_inobt_keys_inorder ,
. recs_inorder = xfs_inobt_recs_inorder ,
# endif
} ;
2008-10-30 16:53:59 +11:00
/*
* Allocate a new inode btree cursor .
*/
struct xfs_btree_cur * /* new inode btree cursor */
xfs_inobt_init_cursor (
struct xfs_mount * mp , /* file system mount point */
struct xfs_trans * tp , /* transaction pointer */
struct xfs_buf * agbp , /* buffer for agi structure */
2014-04-24 16:00:50 +10:00
xfs_agnumber_t agno , /* allocation group number */
xfs_btnum_t btnum ) /* ialloc or free ino btree */
2008-10-30 16:53:59 +11:00
{
struct xfs_agi * agi = XFS_BUF_TO_AGI ( agbp ) ;
struct xfs_btree_cur * cur ;
cur = kmem_zone_zalloc ( xfs_btree_cur_zone , KM_SLEEP ) ;
cur - > bc_tp = tp ;
cur - > bc_mp = mp ;
2014-04-24 16:00:50 +10:00
cur - > bc_btnum = btnum ;
2014-04-24 16:00:52 +10:00
if ( btnum = = XFS_BTNUM_INO ) {
cur - > bc_nlevels = be32_to_cpu ( agi - > agi_level ) ;
cur - > bc_ops = & xfs_inobt_ops ;
} else {
cur - > bc_nlevels = be32_to_cpu ( agi - > agi_free_level ) ;
cur - > bc_ops = & xfs_finobt_ops ;
}
2008-10-30 16:53:59 +11:00
cur - > bc_blocklog = mp - > m_sb . sb_blocklog ;
2013-04-21 14:53:46 -05:00
if ( xfs_sb_version_hascrc ( & mp - > m_sb ) )
cur - > bc_flags | = XFS_BTREE_CRC_BLOCKS ;
2008-10-30 16:53:59 +11:00
cur - > bc_private . a . agbp = agbp ;
cur - > bc_private . a . agno = agno ;
return cur ;
}
2008-10-30 17:11:19 +11:00
/*
* Calculate number of records in an inobt btree block .
*/
int
xfs_inobt_maxrecs (
struct xfs_mount * mp ,
int blocklen ,
int leaf )
{
2008-10-30 17:14:34 +11:00
blocklen - = XFS_INOBT_BLOCK_LEN ( mp ) ;
2008-10-30 17:11:19 +11:00
if ( leaf )
return blocklen / sizeof ( xfs_inobt_rec_t ) ;
return blocklen / ( sizeof ( xfs_inobt_key_t ) + sizeof ( xfs_inobt_ptr_t ) ) ;
}
2015-05-29 09:09:05 +10:00
/*
* Convert the inode record holemask to an inode allocation bitmap . The inode
* allocation bitmap is inode granularity and specifies whether an inode is
* physically allocated on disk ( not whether the inode is considered allocated
* or free by the fs ) .
*
* A bit value of 1 means the inode is allocated , a value of 0 means it is free .
*/
uint64_t
xfs_inobt_irec_to_allocmask (
struct xfs_inobt_rec_incore * rec )
{
uint64_t bitmap = 0 ;
uint64_t inodespbit ;
int nextbit ;
uint allocbitmap ;
/*
* The holemask has 16 - bits for a 64 inode record . Therefore each
* holemask bit represents multiple inodes . Create a mask of bits to set
* in the allocmask for each holemask bit .
*/
inodespbit = ( 1 < < XFS_INODES_PER_HOLEMASK_BIT ) - 1 ;
/*
* Allocated inodes are represented by 0 bits in holemask . Invert the 0
* bits to 1 and convert to a uint so we can use xfs_next_bit ( ) . Mask
* anything beyond the 16 holemask bits since this casts to a larger
* type .
*/
allocbitmap = ~ rec - > ir_holemask & ( ( 1 < < XFS_INOBT_HOLEMASK_BITS ) - 1 ) ;
/*
* allocbitmap is the inverted holemask so every set bit represents
* allocated inodes . To expand from 16 - bit holemask granularity to
* 64 - bit ( e . g . , bit - per - inode ) , set inodespbit bits in the target
* bitmap for every holemask bit .
*/
nextbit = xfs_next_bit ( & allocbitmap , 1 , 0 ) ;
while ( nextbit ! = - 1 ) {
ASSERT ( nextbit < ( sizeof ( rec - > ir_holemask ) * NBBY ) ) ;
bitmap | = ( inodespbit < <
( nextbit * XFS_INODES_PER_HOLEMASK_BIT ) ) ;
nextbit = xfs_next_bit ( & allocbitmap , 1 , nextbit + 1 ) ;
}
return bitmap ;
}
2015-05-29 09:18:32 +10:00
# if defined(DEBUG) || defined(XFS_WARN)
/*
* Verify that an in - core inode record has a valid inode count .
*/
int
xfs_inobt_rec_check_count (
struct xfs_mount * mp ,
struct xfs_inobt_rec_incore * rec )
{
int inocount = 0 ;
int nextbit = 0 ;
uint64_t allocbmap ;
int wordsz ;
wordsz = sizeof ( allocbmap ) / sizeof ( unsigned int ) ;
allocbmap = xfs_inobt_irec_to_allocmask ( rec ) ;
nextbit = xfs_next_bit ( ( uint * ) & allocbmap , wordsz , nextbit ) ;
while ( nextbit ! = - 1 ) {
inocount + + ;
nextbit = xfs_next_bit ( ( uint * ) & allocbmap , wordsz ,
nextbit + 1 ) ;
}
if ( inocount ! = rec - > ir_count )
return - EFSCORRUPTED ;
return 0 ;
}
# endif /* DEBUG */