2005-04-16 15:20:36 -07:00
/*
2005-11-02 14:58:39 +11:00
* Copyright ( c ) 2000 - 2003 , 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"
2005-04-16 15:20:36 -07:00
# include "xfs_types.h"
2005-11-02 14:38:42 +11:00
# include "xfs_bit.h"
2005-04-16 15:20:36 -07:00
# include "xfs_log.h"
# include "xfs_trans.h"
# include "xfs_sb.h"
# include "xfs_ag.h"
# include "xfs_mount.h"
# include "xfs_bmap_btree.h"
2005-11-02 14:38:42 +11:00
# include "xfs_alloc_btree.h"
2005-04-16 15:20:36 -07:00
# include "xfs_ialloc_btree.h"
# include "xfs_dinode.h"
# include "xfs_inode.h"
2005-11-02 14:38:42 +11:00
# include "xfs_inode_item.h"
2005-04-16 15:20:36 -07:00
# include "xfs_alloc.h"
2005-11-02 14:38:42 +11:00
# include "xfs_btree.h"
# include "xfs_itable.h"
2005-04-16 15:20:36 -07:00
# include "xfs_bmap.h"
# include "xfs_error.h"
# include "xfs_quota.h"
2012-11-12 22:54:08 +11:00
# include "xfs_trace.h"
2005-04-16 15:20:36 -07:00
/*
* Determine the extent state .
*/
/* ARGSUSED */
STATIC xfs_exntst_t
xfs_extent_state (
xfs_filblks_t blks ,
int extent_flag )
{
if ( extent_flag ) {
ASSERT ( blks ! = 0 ) ; /* saved for DMIG */
return XFS_EXT_UNWRITTEN ;
}
return XFS_EXT_NORM ;
}
/*
* Convert on - disk form of btree root to in - memory form .
*/
void
xfs_bmdr_to_bmbt (
2008-10-30 17:11:19 +11:00
struct xfs_mount * mp ,
2005-04-16 15:20:36 -07:00
xfs_bmdr_block_t * dblock ,
int dblocklen ,
2008-10-30 17:14:34 +11:00
struct xfs_btree_block * rblock ,
2005-04-16 15:20:36 -07:00
int rblocklen )
{
int dmxr ;
xfs_bmbt_key_t * fkp ;
2006-09-28 10:58:06 +10:00
__be64 * fpp ;
2005-04-16 15:20:36 -07:00
xfs_bmbt_key_t * tkp ;
2006-09-28 10:58:06 +10:00
__be64 * tpp ;
2005-04-16 15:20:36 -07:00
2005-11-02 15:11:25 +11:00
rblock - > bb_magic = cpu_to_be32 ( XFS_BMAP_MAGIC ) ;
rblock - > bb_level = dblock - > bb_level ;
ASSERT ( be16_to_cpu ( rblock - > bb_level ) > 0 ) ;
rblock - > bb_numrecs = dblock - > bb_numrecs ;
2008-10-30 17:14:34 +11:00
rblock - > bb_u . l . bb_leftsib = cpu_to_be64 ( NULLDFSBNO ) ;
rblock - > bb_u . l . bb_rightsib = cpu_to_be64 ( NULLDFSBNO ) ;
2008-10-30 17:11:19 +11:00
dmxr = xfs_bmdr_maxrecs ( mp , dblocklen , 0 ) ;
2008-10-30 17:11:40 +11:00
fkp = XFS_BMDR_KEY_ADDR ( dblock , 1 ) ;
tkp = XFS_BMBT_KEY_ADDR ( mp , rblock , 1 ) ;
fpp = XFS_BMDR_PTR_ADDR ( dblock , 1 , dmxr ) ;
2008-10-30 17:11:19 +11:00
tpp = XFS_BMAP_BROOT_PTR_ADDR ( mp , rblock , 1 , rblocklen ) ;
2005-11-02 15:11:25 +11:00
dmxr = be16_to_cpu ( dblock - > bb_numrecs ) ;
2005-04-16 15:20:36 -07:00
memcpy ( tkp , fkp , sizeof ( * fkp ) * dmxr ) ;
2006-09-28 10:58:06 +10:00
memcpy ( tpp , fpp , sizeof ( * fpp ) * dmxr ) ;
2005-04-16 15:20:36 -07:00
}
/*
* Convert a compressed bmap extent record to an uncompressed form .
* This code must be in sync with the routines xfs_bmbt_get_startoff ,
* xfs_bmbt_get_startblock , xfs_bmbt_get_blockcount and xfs_bmbt_get_state .
*/
2009-11-14 16:17:22 +00:00
STATIC void
2005-04-16 15:20:36 -07:00
__xfs_bmbt_get_all (
__uint64_t l0 ,
__uint64_t l1 ,
xfs_bmbt_irec_t * s )
{
int ext_flag ;
xfs_exntst_t st ;
ext_flag = ( int ) ( l0 > > ( 64 - BMBT_EXNTFLAG_BITLEN ) ) ;
s - > br_startoff = ( ( xfs_fileoff_t ) l0 &
2009-01-09 15:53:54 +11:00
xfs_mask64lo ( 64 - BMBT_EXNTFLAG_BITLEN ) ) > > 9 ;
2005-04-16 15:20:36 -07:00
# if XFS_BIG_BLKNOS
2009-01-09 15:53:54 +11:00
s - > br_startblock = ( ( ( xfs_fsblock_t ) l0 & xfs_mask64lo ( 9 ) ) < < 43 ) |
2005-04-16 15:20:36 -07:00
( ( ( xfs_fsblock_t ) l1 ) > > 21 ) ;
# else
# ifdef DEBUG
{
xfs_dfsbno_t b ;
2009-01-09 15:53:54 +11:00
b = ( ( ( xfs_dfsbno_t ) l0 & xfs_mask64lo ( 9 ) ) < < 43 ) |
2005-04-16 15:20:36 -07:00
( ( ( xfs_dfsbno_t ) l1 ) > > 21 ) ;
2009-01-14 23:22:07 -06:00
ASSERT ( ( b > > 32 ) = = 0 | | isnulldstartblock ( b ) ) ;
2005-04-16 15:20:36 -07:00
s - > br_startblock = ( xfs_fsblock_t ) b ;
}
# else /* !DEBUG */
s - > br_startblock = ( xfs_fsblock_t ) ( ( ( xfs_dfsbno_t ) l1 ) > > 21 ) ;
# endif /* DEBUG */
# endif /* XFS_BIG_BLKNOS */
2009-01-09 15:53:54 +11:00
s - > br_blockcount = ( xfs_filblks_t ) ( l1 & xfs_mask64lo ( 21 ) ) ;
2005-04-16 15:20:36 -07:00
/* This is xfs_extent_state() in-line */
if ( ext_flag ) {
ASSERT ( s - > br_blockcount ! = 0 ) ; /* saved for DMIG */
st = XFS_EXT_UNWRITTEN ;
} else
st = XFS_EXT_NORM ;
s - > br_state = st ;
}
void
xfs_bmbt_get_all (
2007-08-16 16:23:40 +10:00
xfs_bmbt_rec_host_t * r ,
2005-04-16 15:20:36 -07:00
xfs_bmbt_irec_t * s )
{
__xfs_bmbt_get_all ( r - > l0 , r - > l1 , s ) ;
}
/*
* Extract the blockcount field from an in memory bmap extent record .
*/
xfs_filblks_t
xfs_bmbt_get_blockcount (
2007-08-16 16:23:40 +10:00
xfs_bmbt_rec_host_t * r )
2005-04-16 15:20:36 -07:00
{
2009-01-09 15:53:54 +11:00
return ( xfs_filblks_t ) ( r - > l1 & xfs_mask64lo ( 21 ) ) ;
2005-04-16 15:20:36 -07:00
}
/*
* Extract the startblock field from an in memory bmap extent record .
*/
xfs_fsblock_t
xfs_bmbt_get_startblock (
2007-08-16 16:23:40 +10:00
xfs_bmbt_rec_host_t * r )
2005-04-16 15:20:36 -07:00
{
# if XFS_BIG_BLKNOS
2009-01-09 15:53:54 +11:00
return ( ( ( xfs_fsblock_t ) r - > l0 & xfs_mask64lo ( 9 ) ) < < 43 ) |
2005-04-16 15:20:36 -07:00
( ( ( xfs_fsblock_t ) r - > l1 ) > > 21 ) ;
# else
# ifdef DEBUG
xfs_dfsbno_t b ;
2009-01-09 15:53:54 +11:00
b = ( ( ( xfs_dfsbno_t ) r - > l0 & xfs_mask64lo ( 9 ) ) < < 43 ) |
2005-04-16 15:20:36 -07:00
( ( ( xfs_dfsbno_t ) r - > l1 ) > > 21 ) ;
2009-01-14 23:22:07 -06:00
ASSERT ( ( b > > 32 ) = = 0 | | isnulldstartblock ( b ) ) ;
2005-04-16 15:20:36 -07:00
return ( xfs_fsblock_t ) b ;
# else /* !DEBUG */
return ( xfs_fsblock_t ) ( ( ( xfs_dfsbno_t ) r - > l1 ) > > 21 ) ;
# endif /* DEBUG */
# endif /* XFS_BIG_BLKNOS */
}
/*
* Extract the startoff field from an in memory bmap extent record .
*/
xfs_fileoff_t
xfs_bmbt_get_startoff (
2007-08-16 16:23:40 +10:00
xfs_bmbt_rec_host_t * r )
2005-04-16 15:20:36 -07:00
{
return ( ( xfs_fileoff_t ) r - > l0 &
2009-01-09 15:53:54 +11:00
xfs_mask64lo ( 64 - BMBT_EXNTFLAG_BITLEN ) ) > > 9 ;
2005-04-16 15:20:36 -07:00
}
xfs_exntst_t
xfs_bmbt_get_state (
2007-08-16 16:23:40 +10:00
xfs_bmbt_rec_host_t * r )
2005-04-16 15:20:36 -07:00
{
int ext_flag ;
ext_flag = ( int ) ( ( r - > l0 ) > > ( 64 - BMBT_EXNTFLAG_BITLEN ) ) ;
return xfs_extent_state ( xfs_bmbt_get_blockcount ( r ) ,
ext_flag ) ;
}
/*
* Extract the blockcount field from an on disk bmap extent record .
*/
xfs_filblks_t
xfs_bmbt_disk_get_blockcount (
xfs_bmbt_rec_t * r )
{
2009-01-09 15:53:54 +11:00
return ( xfs_filblks_t ) ( be64_to_cpu ( r - > l1 ) & xfs_mask64lo ( 21 ) ) ;
2005-04-16 15:20:36 -07:00
}
/*
* Extract the startoff field from a disk format bmap extent record .
*/
xfs_fileoff_t
xfs_bmbt_disk_get_startoff (
xfs_bmbt_rec_t * r )
{
2007-08-16 16:24:15 +10:00
return ( ( xfs_fileoff_t ) be64_to_cpu ( r - > l0 ) &
2009-01-09 15:53:54 +11:00
xfs_mask64lo ( 64 - BMBT_EXNTFLAG_BITLEN ) ) > > 9 ;
2005-04-16 15:20:36 -07:00
}
/*
* Set all the fields in a bmap extent record from the arguments .
*/
void
xfs_bmbt_set_allf (
2007-08-16 16:23:53 +10:00
xfs_bmbt_rec_host_t * r ,
xfs_fileoff_t startoff ,
xfs_fsblock_t startblock ,
xfs_filblks_t blockcount ,
xfs_exntst_t state )
2005-04-16 15:20:36 -07:00
{
2007-08-16 16:23:53 +10:00
int extent_flag = ( state = = XFS_EXT_NORM ) ? 0 : 1 ;
ASSERT ( state = = XFS_EXT_NORM | | state = = XFS_EXT_UNWRITTEN ) ;
2009-01-09 15:53:54 +11:00
ASSERT ( ( startoff & xfs_mask64hi ( 64 - BMBT_STARTOFF_BITLEN ) ) = = 0 ) ;
ASSERT ( ( blockcount & xfs_mask64hi ( 64 - BMBT_BLOCKCOUNT_BITLEN ) ) = = 0 ) ;
2005-04-16 15:20:36 -07:00
# if XFS_BIG_BLKNOS
2009-01-09 15:53:54 +11:00
ASSERT ( ( startblock & xfs_mask64hi ( 64 - BMBT_STARTBLOCK_BITLEN ) ) = = 0 ) ;
2007-08-16 16:23:53 +10:00
2005-04-16 15:20:36 -07:00
r - > l0 = ( ( xfs_bmbt_rec_base_t ) extent_flag < < 63 ) |
2007-08-16 16:23:53 +10:00
( ( xfs_bmbt_rec_base_t ) startoff < < 9 ) |
( ( xfs_bmbt_rec_base_t ) startblock > > 43 ) ;
r - > l1 = ( ( xfs_bmbt_rec_base_t ) startblock < < 21 ) |
( ( xfs_bmbt_rec_base_t ) blockcount &
2009-01-09 15:53:54 +11:00
( xfs_bmbt_rec_base_t ) xfs_mask64lo ( 21 ) ) ;
2005-04-16 15:20:36 -07:00
# else /* !XFS_BIG_BLKNOS */
2009-01-14 23:22:07 -06:00
if ( isnullstartblock ( startblock ) ) {
2005-04-16 15:20:36 -07:00
r - > l0 = ( ( xfs_bmbt_rec_base_t ) extent_flag < < 63 ) |
2007-08-16 16:23:53 +10:00
( ( xfs_bmbt_rec_base_t ) startoff < < 9 ) |
2009-01-09 15:53:54 +11:00
( xfs_bmbt_rec_base_t ) xfs_mask64lo ( 9 ) ;
r - > l1 = xfs_mask64hi ( 11 ) |
2007-08-16 16:23:53 +10:00
( ( xfs_bmbt_rec_base_t ) startblock < < 21 ) |
( ( xfs_bmbt_rec_base_t ) blockcount &
2009-01-09 15:53:54 +11:00
( xfs_bmbt_rec_base_t ) xfs_mask64lo ( 21 ) ) ;
2005-04-16 15:20:36 -07:00
} else {
r - > l0 = ( ( xfs_bmbt_rec_base_t ) extent_flag < < 63 ) |
2007-08-16 16:23:53 +10:00
( ( xfs_bmbt_rec_base_t ) startoff < < 9 ) ;
r - > l1 = ( ( xfs_bmbt_rec_base_t ) startblock < < 21 ) |
( ( xfs_bmbt_rec_base_t ) blockcount &
2009-01-09 15:53:54 +11:00
( xfs_bmbt_rec_base_t ) xfs_mask64lo ( 21 ) ) ;
2005-04-16 15:20:36 -07:00
}
# endif /* XFS_BIG_BLKNOS */
}
/*
* Set all the fields in a bmap extent record from the uncompressed form .
*/
void
2007-08-16 16:23:53 +10:00
xfs_bmbt_set_all (
xfs_bmbt_rec_host_t * r ,
xfs_bmbt_irec_t * s )
2005-04-16 15:20:36 -07:00
{
2007-08-16 16:23:53 +10:00
xfs_bmbt_set_allf ( r , s - > br_startoff , s - > br_startblock ,
s - > br_blockcount , s - > br_state ) ;
2005-04-16 15:20:36 -07:00
}
2007-08-16 16:23:53 +10:00
2005-04-16 15:20:36 -07:00
/*
* Set all the fields in a disk format bmap extent record from the arguments .
*/
void
xfs_bmbt_disk_set_allf (
2007-08-16 16:23:53 +10:00
xfs_bmbt_rec_t * r ,
xfs_fileoff_t startoff ,
xfs_fsblock_t startblock ,
xfs_filblks_t blockcount ,
xfs_exntst_t state )
2005-04-16 15:20:36 -07:00
{
2007-08-16 16:23:53 +10:00
int extent_flag = ( state = = XFS_EXT_NORM ) ? 0 : 1 ;
ASSERT ( state = = XFS_EXT_NORM | | state = = XFS_EXT_UNWRITTEN ) ;
2009-01-09 15:53:54 +11:00
ASSERT ( ( startoff & xfs_mask64hi ( 64 - BMBT_STARTOFF_BITLEN ) ) = = 0 ) ;
ASSERT ( ( blockcount & xfs_mask64hi ( 64 - BMBT_BLOCKCOUNT_BITLEN ) ) = = 0 ) ;
2005-04-16 15:20:36 -07:00
# if XFS_BIG_BLKNOS
2009-01-09 15:53:54 +11:00
ASSERT ( ( startblock & xfs_mask64hi ( 64 - BMBT_STARTBLOCK_BITLEN ) ) = = 0 ) ;
2007-08-16 16:23:53 +10:00
2007-08-16 16:24:15 +10:00
r - > l0 = cpu_to_be64 (
2007-08-16 16:23:53 +10:00
( ( xfs_bmbt_rec_base_t ) extent_flag < < 63 ) |
( ( xfs_bmbt_rec_base_t ) startoff < < 9 ) |
( ( xfs_bmbt_rec_base_t ) startblock > > 43 ) ) ;
2007-08-16 16:24:15 +10:00
r - > l1 = cpu_to_be64 (
2007-08-16 16:23:53 +10:00
( ( xfs_bmbt_rec_base_t ) startblock < < 21 ) |
( ( xfs_bmbt_rec_base_t ) blockcount &
2009-01-09 15:53:54 +11:00
( xfs_bmbt_rec_base_t ) xfs_mask64lo ( 21 ) ) ) ;
2005-04-16 15:20:36 -07:00
# else /* !XFS_BIG_BLKNOS */
2009-01-14 23:22:07 -06:00
if ( isnullstartblock ( startblock ) ) {
2007-08-16 16:24:15 +10:00
r - > l0 = cpu_to_be64 (
2007-08-16 16:23:53 +10:00
( ( xfs_bmbt_rec_base_t ) extent_flag < < 63 ) |
( ( xfs_bmbt_rec_base_t ) startoff < < 9 ) |
2009-01-09 15:53:54 +11:00
( xfs_bmbt_rec_base_t ) xfs_mask64lo ( 9 ) ) ;
r - > l1 = cpu_to_be64 ( xfs_mask64hi ( 11 ) |
2007-08-16 16:23:53 +10:00
( ( xfs_bmbt_rec_base_t ) startblock < < 21 ) |
( ( xfs_bmbt_rec_base_t ) blockcount &
2009-01-09 15:53:54 +11:00
( xfs_bmbt_rec_base_t ) xfs_mask64lo ( 21 ) ) ) ;
2005-04-16 15:20:36 -07:00
} else {
2007-08-16 16:24:15 +10:00
r - > l0 = cpu_to_be64 (
2007-08-16 16:23:53 +10:00
( ( xfs_bmbt_rec_base_t ) extent_flag < < 63 ) |
( ( xfs_bmbt_rec_base_t ) startoff < < 9 ) ) ;
2007-08-16 16:24:15 +10:00
r - > l1 = cpu_to_be64 (
2007-08-16 16:23:53 +10:00
( ( xfs_bmbt_rec_base_t ) startblock < < 21 ) |
( ( xfs_bmbt_rec_base_t ) blockcount &
2009-01-09 15:53:54 +11:00
( xfs_bmbt_rec_base_t ) xfs_mask64lo ( 21 ) ) ) ;
2005-04-16 15:20:36 -07:00
}
# endif /* XFS_BIG_BLKNOS */
}
2007-08-16 16:23:53 +10:00
/*
* Set all the fields in a bmap extent record from the uncompressed form .
*/
2009-11-19 15:52:00 +00:00
STATIC void
2007-08-16 16:23:53 +10:00
xfs_bmbt_disk_set_all (
xfs_bmbt_rec_t * r ,
xfs_bmbt_irec_t * s )
{
xfs_bmbt_disk_set_allf ( r , s - > br_startoff , s - > br_startblock ,
s - > br_blockcount , s - > br_state ) ;
}
2005-04-16 15:20:36 -07:00
/*
* Set the blockcount field in a bmap extent record .
*/
void
xfs_bmbt_set_blockcount (
2007-08-16 16:23:40 +10:00
xfs_bmbt_rec_host_t * r ,
2005-04-16 15:20:36 -07:00
xfs_filblks_t v )
{
2009-01-09 15:53:54 +11:00
ASSERT ( ( v & xfs_mask64hi ( 43 ) ) = = 0 ) ;
r - > l1 = ( r - > l1 & ( xfs_bmbt_rec_base_t ) xfs_mask64hi ( 43 ) ) |
( xfs_bmbt_rec_base_t ) ( v & xfs_mask64lo ( 21 ) ) ;
2005-04-16 15:20:36 -07:00
}
/*
* Set the startblock field in a bmap extent record .
*/
void
xfs_bmbt_set_startblock (
2007-08-16 16:23:40 +10:00
xfs_bmbt_rec_host_t * r ,
2005-04-16 15:20:36 -07:00
xfs_fsblock_t v )
{
# if XFS_BIG_BLKNOS
2009-01-09 15:53:54 +11:00
ASSERT ( ( v & xfs_mask64hi ( 12 ) ) = = 0 ) ;
r - > l0 = ( r - > l0 & ( xfs_bmbt_rec_base_t ) xfs_mask64hi ( 55 ) ) |
2005-04-16 15:20:36 -07:00
( xfs_bmbt_rec_base_t ) ( v > > 43 ) ;
2009-01-09 15:53:54 +11:00
r - > l1 = ( r - > l1 & ( xfs_bmbt_rec_base_t ) xfs_mask64lo ( 21 ) ) |
2005-04-16 15:20:36 -07:00
( xfs_bmbt_rec_base_t ) ( v < < 21 ) ;
# else /* !XFS_BIG_BLKNOS */
2009-01-14 23:22:07 -06:00
if ( isnullstartblock ( v ) ) {
2009-01-09 15:53:54 +11:00
r - > l0 | = ( xfs_bmbt_rec_base_t ) xfs_mask64lo ( 9 ) ;
r - > l1 = ( xfs_bmbt_rec_base_t ) xfs_mask64hi ( 11 ) |
2005-04-16 15:20:36 -07:00
( ( xfs_bmbt_rec_base_t ) v < < 21 ) |
2009-01-09 15:53:54 +11:00
( r - > l1 & ( xfs_bmbt_rec_base_t ) xfs_mask64lo ( 21 ) ) ;
2005-04-16 15:20:36 -07:00
} else {
2009-01-09 15:53:54 +11:00
r - > l0 & = ~ ( xfs_bmbt_rec_base_t ) xfs_mask64lo ( 9 ) ;
2005-04-16 15:20:36 -07:00
r - > l1 = ( ( xfs_bmbt_rec_base_t ) v < < 21 ) |
2009-01-09 15:53:54 +11:00
( r - > l1 & ( xfs_bmbt_rec_base_t ) xfs_mask64lo ( 21 ) ) ;
2005-04-16 15:20:36 -07:00
}
# endif /* XFS_BIG_BLKNOS */
}
/*
* Set the startoff field in a bmap extent record .
*/
void
xfs_bmbt_set_startoff (
2007-08-16 16:23:40 +10:00
xfs_bmbt_rec_host_t * r ,
2005-04-16 15:20:36 -07:00
xfs_fileoff_t v )
{
2009-01-09 15:53:54 +11:00
ASSERT ( ( v & xfs_mask64hi ( 9 ) ) = = 0 ) ;
r - > l0 = ( r - > l0 & ( xfs_bmbt_rec_base_t ) xfs_mask64hi ( 1 ) ) |
2005-04-16 15:20:36 -07:00
( ( xfs_bmbt_rec_base_t ) v < < 9 ) |
2009-01-09 15:53:54 +11:00
( r - > l0 & ( xfs_bmbt_rec_base_t ) xfs_mask64lo ( 9 ) ) ;
2005-04-16 15:20:36 -07:00
}
/*
* Set the extent state field in a bmap extent record .
*/
void
xfs_bmbt_set_state (
2007-08-16 16:23:40 +10:00
xfs_bmbt_rec_host_t * r ,
2005-04-16 15:20:36 -07:00
xfs_exntst_t v )
{
ASSERT ( v = = XFS_EXT_NORM | | v = = XFS_EXT_UNWRITTEN ) ;
if ( v = = XFS_EXT_NORM )
2009-01-09 15:53:54 +11:00
r - > l0 & = xfs_mask64lo ( 64 - BMBT_EXNTFLAG_BITLEN ) ;
2005-04-16 15:20:36 -07:00
else
2009-01-09 15:53:54 +11:00
r - > l0 | = xfs_mask64hi ( BMBT_EXNTFLAG_BITLEN ) ;
2005-04-16 15:20:36 -07:00
}
/*
* Convert in - memory form of btree root to on - disk form .
*/
void
xfs_bmbt_to_bmdr (
2008-10-30 17:11:19 +11:00
struct xfs_mount * mp ,
2008-10-30 17:14:34 +11:00
struct xfs_btree_block * rblock ,
2005-04-16 15:20:36 -07:00
int rblocklen ,
xfs_bmdr_block_t * dblock ,
int dblocklen )
{
int dmxr ;
xfs_bmbt_key_t * fkp ;
2006-09-28 10:58:06 +10:00
__be64 * fpp ;
2005-04-16 15:20:36 -07:00
xfs_bmbt_key_t * tkp ;
2006-09-28 10:58:06 +10:00
__be64 * tpp ;
2005-04-16 15:20:36 -07:00
2011-07-08 14:36:05 +02:00
ASSERT ( rblock - > bb_magic = = cpu_to_be32 ( XFS_BMAP_MAGIC ) ) ;
ASSERT ( rblock - > bb_u . l . bb_leftsib = = cpu_to_be64 ( NULLDFSBNO ) ) ;
ASSERT ( rblock - > bb_u . l . bb_rightsib = = cpu_to_be64 ( NULLDFSBNO ) ) ;
ASSERT ( rblock - > bb_level ! = 0 ) ;
2005-11-02 15:11:25 +11:00
dblock - > bb_level = rblock - > bb_level ;
dblock - > bb_numrecs = rblock - > bb_numrecs ;
2008-10-30 17:11:19 +11:00
dmxr = xfs_bmdr_maxrecs ( mp , dblocklen , 0 ) ;
2008-10-30 17:11:40 +11:00
fkp = XFS_BMBT_KEY_ADDR ( mp , rblock , 1 ) ;
tkp = XFS_BMDR_KEY_ADDR ( dblock , 1 ) ;
2008-10-30 17:11:19 +11:00
fpp = XFS_BMAP_BROOT_PTR_ADDR ( mp , rblock , 1 , rblocklen ) ;
2008-10-30 17:11:40 +11:00
tpp = XFS_BMDR_PTR_ADDR ( dblock , 1 , dmxr ) ;
2005-11-02 15:11:25 +11:00
dmxr = be16_to_cpu ( dblock - > bb_numrecs ) ;
2005-04-16 15:20:36 -07:00
memcpy ( tkp , fkp , sizeof ( * fkp ) * dmxr ) ;
2006-09-28 10:58:06 +10:00
memcpy ( tpp , fpp , sizeof ( * fpp ) * dmxr ) ;
2005-04-16 15:20:36 -07:00
}
/*
2006-03-14 13:29:52 +11:00
* Check extent records , which have just been read , for
2005-04-16 15:20:36 -07:00
* any bit in the extent flag field . ASSERT on debug
* kernels , as this condition should not occur .
* Return an error condition ( 1 ) if any flags found ,
* otherwise return 0.
*/
int
xfs_check_nostate_extents (
2006-03-14 13:29:52 +11:00
xfs_ifork_t * ifp ,
xfs_extnum_t idx ,
2005-04-16 15:20:36 -07:00
xfs_extnum_t num )
{
2006-03-14 13:29:52 +11:00
for ( ; num > 0 ; num - - , idx + + ) {
2007-08-16 16:23:40 +10:00
xfs_bmbt_rec_host_t * ep = xfs_iext_get_ext ( ifp , idx ) ;
2005-04-16 15:20:36 -07:00
if ( ( ep - > l0 > >
( 64 - BMBT_EXNTFLAG_BITLEN ) ) ! = 0 ) {
ASSERT ( 0 ) ;
return 1 ;
}
}
return 0 ;
}
2008-10-30 16:53:59 +11:00
STATIC struct xfs_btree_cur *
xfs_bmbt_dup_cursor (
struct xfs_btree_cur * cur )
{
struct xfs_btree_cur * new ;
new = xfs_bmbt_init_cursor ( cur - > bc_mp , cur - > bc_tp ,
cur - > bc_private . b . ip , cur - > bc_private . b . whichfork ) ;
/*
* Copy the firstblock , flist , and flags values ,
* since init cursor doesn ' t get them .
*/
new - > bc_private . b . firstblock = cur - > bc_private . b . firstblock ;
new - > bc_private . b . flist = cur - > bc_private . b . flist ;
new - > bc_private . b . flags = cur - > bc_private . b . flags ;
return new ;
}
2008-10-30 16:57:40 +11:00
STATIC void
xfs_bmbt_update_cursor (
struct xfs_btree_cur * src ,
struct xfs_btree_cur * dst )
{
ASSERT ( ( dst - > bc_private . b . firstblock ! = NULLFSBLOCK ) | |
( dst - > bc_private . b . ip - > i_d . di_flags & XFS_DIFLAG_REALTIME ) ) ;
ASSERT ( dst - > bc_private . b . flist = = src - > bc_private . b . flist ) ;
dst - > bc_private . b . allocated + = src - > bc_private . b . allocated ;
dst - > bc_private . b . firstblock = src - > bc_private . b . firstblock ;
src - > bc_private . b . allocated = 0 ;
}
2008-10-30 16:57:03 +11:00
STATIC int
xfs_bmbt_alloc_block (
struct xfs_btree_cur * cur ,
union xfs_btree_ptr * start ,
union xfs_btree_ptr * new ,
int length ,
int * stat )
{
xfs_alloc_arg_t args ; /* block allocation args */
int error ; /* error return value */
memset ( & args , 0 , sizeof ( args ) ) ;
args . tp = cur - > bc_tp ;
args . mp = cur - > bc_mp ;
args . fsbno = cur - > bc_private . b . firstblock ;
args . firstblock = args . fsbno ;
if ( args . fsbno = = NULLFSBLOCK ) {
args . fsbno = be64_to_cpu ( start - > l ) ;
args . type = XFS_ALLOCTYPE_START_BNO ;
/*
* Make sure there is sufficient room left in the AG to
* complete a full tree split for an extent insert . If
* we are converting the middle part of an extent then
* we may need space for two tree splits .
*
* We are relying on the caller to make the correct block
* reservation for this operation to succeed . If the
* reservation amount is insufficient then we may fail a
* block allocation here and corrupt the filesystem .
*/
args . minleft = xfs_trans_get_block_res ( args . tp ) ;
} else if ( cur - > bc_private . b . flist - > xbf_low ) {
args . type = XFS_ALLOCTYPE_START_BNO ;
} else {
args . type = XFS_ALLOCTYPE_NEAR_BNO ;
}
args . minlen = args . maxlen = args . prod = 1 ;
args . wasdel = cur - > bc_private . b . flags & XFS_BTCUR_BPRV_WASDEL ;
if ( ! args . wasdel & & xfs_trans_get_block_res ( args . tp ) = = 0 ) {
error = XFS_ERROR ( ENOSPC ) ;
goto error0 ;
}
error = xfs_alloc_vextent ( & args ) ;
if ( error )
goto error0 ;
if ( args . fsbno = = NULLFSBLOCK & & args . minleft ) {
/*
* Could not find an AG with enough free space to satisfy
* a full btree split . Try again without minleft and if
* successful activate the lowspace algorithm .
*/
args . fsbno = 0 ;
args . type = XFS_ALLOCTYPE_FIRST_AG ;
args . minleft = 0 ;
error = xfs_alloc_vextent ( & args ) ;
if ( error )
goto error0 ;
cur - > bc_private . b . flist - > xbf_low = 1 ;
}
if ( args . fsbno = = NULLFSBLOCK ) {
XFS_BTREE_TRACE_CURSOR ( cur , XBT_EXIT ) ;
* stat = 0 ;
return 0 ;
}
ASSERT ( args . len = = 1 ) ;
cur - > bc_private . b . firstblock = args . fsbno ;
cur - > bc_private . b . allocated + + ;
cur - > bc_private . b . ip - > i_d . di_nblocks + + ;
xfs_trans_log_inode ( args . tp , cur - > bc_private . b . ip , XFS_ILOG_CORE ) ;
2009-06-08 15:33:32 +02:00
xfs_trans_mod_dquot_byino ( args . tp , cur - > bc_private . b . ip ,
2008-10-30 16:57:03 +11:00
XFS_TRANS_DQ_BCOUNT , 1L ) ;
new - > l = cpu_to_be64 ( args . fsbno ) ;
XFS_BTREE_TRACE_CURSOR ( cur , XBT_EXIT ) ;
* stat = 1 ;
return 0 ;
error0 :
XFS_BTREE_TRACE_CURSOR ( cur , XBT_ERROR ) ;
return error ;
}
2008-10-30 16:57:51 +11:00
STATIC int
xfs_bmbt_free_block (
struct xfs_btree_cur * cur ,
struct xfs_buf * bp )
{
struct xfs_mount * mp = cur - > bc_mp ;
struct xfs_inode * ip = cur - > bc_private . b . ip ;
struct xfs_trans * tp = cur - > bc_tp ;
xfs_fsblock_t fsbno = XFS_DADDR_TO_FSB ( mp , XFS_BUF_ADDR ( bp ) ) ;
xfs_bmap_add_free ( fsbno , 1 , cur - > bc_private . b . flist , mp ) ;
ip - > i_d . di_nblocks - - ;
xfs_trans_log_inode ( tp , ip , XFS_ILOG_CORE ) ;
2009-06-08 15:33:32 +02:00
xfs_trans_mod_dquot_byino ( tp , ip , XFS_TRANS_DQ_BCOUNT , - 1L ) ;
2008-10-30 16:57:51 +11:00
xfs_trans_binval ( tp , bp ) ;
return 0 ;
}
2008-10-30 16:58:01 +11:00
STATIC int
xfs_bmbt_get_minrecs (
struct xfs_btree_cur * cur ,
int level )
{
2008-10-30 17:11:19 +11:00
if ( level = = cur - > bc_nlevels - 1 ) {
struct xfs_ifork * ifp ;
ifp = XFS_IFORK_PTR ( cur - > bc_private . b . ip ,
cur - > bc_private . b . whichfork ) ;
return xfs_bmbt_maxrecs ( cur - > bc_mp ,
ifp - > if_broot_bytes , level = = 0 ) / 2 ;
}
return cur - > bc_mp - > m_bmap_dmnr [ level ! = 0 ] ;
2008-10-30 16:58:01 +11:00
}
2008-10-30 17:11:19 +11:00
int
2008-10-30 16:55:23 +11:00
xfs_bmbt_get_maxrecs (
struct xfs_btree_cur * cur ,
int level )
{
2008-10-30 17:11:19 +11:00
if ( level = = cur - > bc_nlevels - 1 ) {
struct xfs_ifork * ifp ;
ifp = XFS_IFORK_PTR ( cur - > bc_private . b . ip ,
cur - > bc_private . b . whichfork ) ;
return xfs_bmbt_maxrecs ( cur - > bc_mp ,
ifp - > if_broot_bytes , level = = 0 ) ;
}
return cur - > bc_mp - > m_bmap_dmxr [ level ! = 0 ] ;
2008-10-30 16:55:23 +11:00
}
2008-10-30 16:57:40 +11:00
/*
* Get the maximum records we could store in the on - disk format .
*
* For non - root nodes this is equivalent to xfs_bmbt_get_maxrecs , but
* for the root node this checks the available space in the dinode fork
* so that we can resize the in - memory buffer to match it . After a
* resize to the maximum size this function returns the same value
* as xfs_bmbt_get_maxrecs for the root node , too .
*/
STATIC int
xfs_bmbt_get_dmaxrecs (
struct xfs_btree_cur * cur ,
int level )
{
2008-10-30 17:11:19 +11:00
if ( level ! = cur - > bc_nlevels - 1 )
return cur - > bc_mp - > m_bmap_dmxr [ level ! = 0 ] ;
return xfs_bmdr_maxrecs ( cur - > bc_mp , cur - > bc_private . b . forksize ,
level = = 0 ) ;
2008-10-30 16:57:40 +11:00
}
2008-10-30 16:56:09 +11:00
STATIC void
xfs_bmbt_init_key_from_rec (
union xfs_btree_key * key ,
union xfs_btree_rec * rec )
{
key - > bmbt . br_startoff =
cpu_to_be64 ( xfs_bmbt_disk_get_startoff ( & rec - > bmbt ) ) ;
}
2008-10-30 16:57:40 +11:00
STATIC void
xfs_bmbt_init_rec_from_key (
union xfs_btree_key * key ,
union xfs_btree_rec * rec )
{
ASSERT ( key - > bmbt . br_startoff ! = 0 ) ;
xfs_bmbt_disk_set_allf ( & rec - > bmbt , be64_to_cpu ( key - > bmbt . br_startoff ) ,
0 , 0 , XFS_EXT_NORM ) ;
}
STATIC void
xfs_bmbt_init_rec_from_cur (
struct xfs_btree_cur * cur ,
union xfs_btree_rec * rec )
{
xfs_bmbt_disk_set_all ( & rec - > bmbt , & cur - > bc_rec . b ) ;
}
2008-10-30 16:56:09 +11:00
STATIC void
xfs_bmbt_init_ptr_from_cur (
struct xfs_btree_cur * cur ,
union xfs_btree_ptr * ptr )
{
ptr - > l = 0 ;
}
STATIC __int64_t
xfs_bmbt_key_diff (
struct xfs_btree_cur * cur ,
union xfs_btree_key * key )
{
return ( __int64_t ) be64_to_cpu ( key - > bmbt . br_startoff ) -
cur - > bc_rec . b . br_startoff ;
}
2012-11-14 17:52:32 +11:00
static void
xfs_bmbt_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 ) ;
unsigned int level ;
int lblock_ok ; /* block passes checks */
/* magic number and level verification.
*
* We don ' t know waht fork we belong to , so just verify that the level
* is less than the maximum of the two . Later checks will be more
* precise .
*/
level = be16_to_cpu ( block - > bb_level ) ;
lblock_ok = block - > bb_magic = = cpu_to_be32 ( XFS_BMAP_MAGIC ) & &
level < max ( mp - > m_bm_maxlevels [ 0 ] , mp - > m_bm_maxlevels [ 1 ] ) ;
/* numrecs verification */
lblock_ok = lblock_ok & &
be16_to_cpu ( block - > bb_numrecs ) < = mp - > m_bmap_dmxr [ level ! = 0 ] ;
/* sibling pointer verification */
lblock_ok = lblock_ok & &
block - > bb_u . l . bb_leftsib & &
( block - > bb_u . l . bb_leftsib = = cpu_to_be64 ( NULLDFSBNO ) | |
XFS_FSB_SANITY_CHECK ( mp ,
be64_to_cpu ( block - > bb_u . l . bb_leftsib ) ) ) & &
block - > bb_u . l . bb_rightsib & &
( block - > bb_u . l . bb_rightsib = = cpu_to_be64 ( NULLDFSBNO ) | |
XFS_FSB_SANITY_CHECK ( mp ,
be64_to_cpu ( block - > bb_u . l . bb_rightsib ) ) ) ;
if ( ! lblock_ok ) {
trace_xfs_btree_corrupt ( bp , _RET_IP_ ) ;
2012-11-14 17:52:32 +11:00
XFS_CORRUPTION_ERROR ( __func__ , XFS_ERRLEVEL_LOW , mp , block ) ;
2012-11-12 22:54:08 +11:00
xfs_buf_ioerror ( bp , EFSCORRUPTED ) ;
}
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_bmbt_read_verify (
2012-11-14 17:52:32 +11:00
struct xfs_buf * bp )
{
xfs_bmbt_verify ( bp ) ;
}
2012-11-14 17:54:40 +11:00
static void
xfs_bmbt_write_verify (
2012-11-14 17:52:32 +11:00
struct xfs_buf * bp )
{
xfs_bmbt_verify ( bp ) ;
2012-11-12 22:54:08 +11:00
}
2012-11-14 17:54:40 +11:00
const struct xfs_buf_ops xfs_bmbt_buf_ops = {
. verify_read = xfs_bmbt_read_verify ,
. verify_write = xfs_bmbt_write_verify ,
} ;
2008-10-30 16:58:32 +11:00
# ifdef DEBUG
STATIC int
xfs_bmbt_keys_inorder (
struct xfs_btree_cur * cur ,
union xfs_btree_key * k1 ,
union xfs_btree_key * k2 )
{
return be64_to_cpu ( k1 - > bmbt . br_startoff ) <
be64_to_cpu ( k2 - > bmbt . br_startoff ) ;
}
STATIC int
xfs_bmbt_recs_inorder (
struct xfs_btree_cur * cur ,
union xfs_btree_rec * r1 ,
union xfs_btree_rec * r2 )
{
return xfs_bmbt_disk_get_startoff ( & r1 - > bmbt ) +
xfs_bmbt_disk_get_blockcount ( & r1 - > bmbt ) < =
xfs_bmbt_disk_get_startoff ( & r2 - > bmbt ) ;
}
# endif /* DEBUG */
2008-10-30 16:53:59 +11:00
static const struct xfs_btree_ops xfs_bmbt_ops = {
2008-10-30 16:55:34 +11:00
. rec_len = sizeof ( xfs_bmbt_rec_t ) ,
. key_len = sizeof ( xfs_bmbt_key_t ) ,
2008-10-30 16:53:59 +11:00
. dup_cursor = xfs_bmbt_dup_cursor ,
2008-10-30 16:57:40 +11:00
. update_cursor = xfs_bmbt_update_cursor ,
2008-10-30 16:57:03 +11:00
. alloc_block = xfs_bmbt_alloc_block ,
2008-10-30 16:57:51 +11:00
. free_block = xfs_bmbt_free_block ,
2008-10-30 16:55:23 +11:00
. get_maxrecs = xfs_bmbt_get_maxrecs ,
2008-10-30 16:58:01 +11:00
. get_minrecs = xfs_bmbt_get_minrecs ,
2008-10-30 16:57:40 +11:00
. get_dmaxrecs = xfs_bmbt_get_dmaxrecs ,
2008-10-30 16:56:09 +11:00
. init_key_from_rec = xfs_bmbt_init_key_from_rec ,
2008-10-30 16:57:40 +11:00
. init_rec_from_key = xfs_bmbt_init_rec_from_key ,
. init_rec_from_cur = xfs_bmbt_init_rec_from_cur ,
2008-10-30 16:56:09 +11:00
. init_ptr_from_cur = xfs_bmbt_init_ptr_from_cur ,
. key_diff = xfs_bmbt_key_diff ,
2012-11-14 17:54:40 +11:00
. buf_ops = & xfs_bmbt_buf_ops ,
2008-10-30 16:58:32 +11:00
# ifdef DEBUG
. keys_inorder = xfs_bmbt_keys_inorder ,
. recs_inorder = xfs_bmbt_recs_inorder ,
# endif
2008-10-30 16:53:59 +11:00
} ;
/*
* Allocate a new bmap btree cursor .
*/
struct xfs_btree_cur * /* new bmap btree cursor */
xfs_bmbt_init_cursor (
struct xfs_mount * mp , /* file system mount point */
struct xfs_trans * tp , /* transaction pointer */
struct xfs_inode * ip , /* inode owning the btree */
int whichfork ) /* data or attr fork */
{
struct xfs_ifork * ifp = XFS_IFORK_PTR ( ip , whichfork ) ;
struct xfs_btree_cur * cur ;
cur = kmem_zone_zalloc ( xfs_btree_cur_zone , KM_SLEEP ) ;
cur - > bc_tp = tp ;
cur - > bc_mp = mp ;
cur - > bc_nlevels = be16_to_cpu ( ifp - > if_broot - > bb_level ) + 1 ;
cur - > bc_btnum = XFS_BTNUM_BMAP ;
cur - > bc_blocklog = mp - > m_sb . sb_blocklog ;
cur - > bc_ops = & xfs_bmbt_ops ;
2008-10-30 16:54:33 +11:00
cur - > bc_flags = XFS_BTREE_LONG_PTRS | XFS_BTREE_ROOT_IN_INODE ;
2008-10-30 16:53:59 +11:00
cur - > bc_private . b . forksize = XFS_IFORK_SIZE ( ip , whichfork ) ;
cur - > bc_private . b . ip = ip ;
cur - > bc_private . b . firstblock = NULLFSBLOCK ;
cur - > bc_private . b . flist = NULL ;
cur - > bc_private . b . allocated = 0 ;
cur - > bc_private . b . flags = 0 ;
cur - > bc_private . b . whichfork = whichfork ;
return cur ;
}
2008-10-30 17:11:19 +11:00
/*
* Calculate number of records in a bmap btree block .
*/
int
xfs_bmbt_maxrecs (
struct xfs_mount * mp ,
int blocklen ,
int leaf )
{
2008-10-30 17:14:34 +11:00
blocklen - = XFS_BMBT_BLOCK_LEN ( mp ) ;
2008-10-30 17:11:19 +11:00
if ( leaf )
return blocklen / sizeof ( xfs_bmbt_rec_t ) ;
return blocklen / ( sizeof ( xfs_bmbt_key_t ) + sizeof ( xfs_bmbt_ptr_t ) ) ;
}
/*
* Calculate number of records in a bmap btree inode root .
*/
int
xfs_bmdr_maxrecs (
struct xfs_mount * mp ,
int blocklen ,
int leaf )
{
blocklen - = sizeof ( xfs_bmdr_block_t ) ;
if ( leaf )
return blocklen / sizeof ( xfs_bmdr_rec_t ) ;
return blocklen / ( sizeof ( xfs_bmdr_key_t ) + sizeof ( xfs_bmdr_ptr_t ) ) ;
}