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
*/
# ifndef __XFS_BTREE_H__
# define __XFS_BTREE_H__
struct xfs_buf ;
struct xfs_bmap_free ;
struct xfs_inode ;
struct xfs_mount ;
struct xfs_trans ;
2007-11-23 16:28:09 +11:00
extern kmem_zone_t * xfs_btree_cur_zone ;
2005-04-16 15:20:36 -07:00
/*
* This nonsense is to make - wlint happy .
*/
# define XFS_LOOKUP_EQ ((xfs_lookup_t)XFS_LOOKUP_EQi)
# define XFS_LOOKUP_LE ((xfs_lookup_t)XFS_LOOKUP_LEi)
# define XFS_LOOKUP_GE ((xfs_lookup_t)XFS_LOOKUP_GEi)
# define XFS_BTNUM_BNO ((xfs_btnum_t)XFS_BTNUM_BNOi)
# define XFS_BTNUM_CNT ((xfs_btnum_t)XFS_BTNUM_CNTi)
# define XFS_BTNUM_BMAP ((xfs_btnum_t)XFS_BTNUM_BMAPi)
# define XFS_BTNUM_INO ((xfs_btnum_t)XFS_BTNUM_INOi)
/*
2008-10-30 17:14:34 +11:00
* Generic btree header .
*
2009-03-29 09:55:42 +02:00
* This is a combination of the actual format used on disk for short and long
2008-10-30 17:14:34 +11:00
* format btrees . The first three fields are shared by both format , but
* the pointers are different and should be used with care .
*
* To get the size of the actual short or long form headers please use
* the size macros below . Never use sizeof ( xfs_btree_block ) .
2005-04-16 15:20:36 -07:00
*/
2008-10-30 17:14:34 +11:00
struct xfs_btree_block {
2005-11-02 15:11:25 +11:00
__be32 bb_magic ; /* magic number for block type */
__be16 bb_level ; /* 0 is a leaf */
__be16 bb_numrecs ; /* current # of data records */
union {
struct {
__be32 bb_leftsib ;
__be32 bb_rightsib ;
} s ; /* short form pointers */
2005-04-16 15:20:36 -07:00
struct {
2005-11-02 15:11:25 +11:00
__be64 bb_leftsib ;
__be64 bb_rightsib ;
} l ; /* long form pointers */
} bb_u ; /* rest */
2008-10-30 17:14:34 +11:00
} ;
# define XFS_BTREE_SBLOCK_LEN 16 /* size of a short form block */
# define XFS_BTREE_LBLOCK_LEN 24 /* size of a long form block */
2005-04-16 15:20:36 -07:00
2008-10-30 16:54:12 +11:00
/*
* Generic key , ptr and record wrapper structures .
*
* These are disk format structures , and are converted where necessary
* by the btree specific code that needs to interpret them .
*/
union xfs_btree_ptr {
__be32 s ; /* short form ptr */
__be64 l ; /* long form ptr */
} ;
union xfs_btree_key {
xfs_bmbt_key_t bmbt ;
xfs_bmdr_key_t bmbr ; /* bmbt root block */
xfs_alloc_key_t alloc ;
xfs_inobt_key_t inobt ;
} ;
union xfs_btree_rec {
xfs_bmbt_rec_t bmbt ;
xfs_bmdr_rec_t bmbr ; /* bmbt root block */
xfs_alloc_rec_t alloc ;
xfs_inobt_rec_t inobt ;
} ;
2005-04-16 15:20:36 -07:00
/*
* For logging record fields .
*/
# define XFS_BB_MAGIC 0x01
# define XFS_BB_LEVEL 0x02
# define XFS_BB_NUMRECS 0x04
# define XFS_BB_LEFTSIB 0x08
# define XFS_BB_RIGHTSIB 0x10
# define XFS_BB_NUM_BITS 5
# define XFS_BB_ALL_BITS ((1 << XFS_BB_NUM_BITS) - 1)
/*
* Magic numbers for btree blocks .
*/
extern const __uint32_t xfs_magics [ ] ;
2008-10-30 16:55:03 +11:00
/*
* Generic stats interface
*/
# define __XFS_BTREE_STATS_INC(type, stat) \
XFS_STATS_INC ( xs_ # # type # # _2_ # # stat )
# define XFS_BTREE_STATS_INC(cur, stat) \
do { \
switch ( cur - > bc_btnum ) { \
case XFS_BTNUM_BNO : __XFS_BTREE_STATS_INC ( abtb , stat ) ; break ; \
case XFS_BTNUM_CNT : __XFS_BTREE_STATS_INC ( abtc , stat ) ; break ; \
case XFS_BTNUM_BMAP : __XFS_BTREE_STATS_INC ( bmbt , stat ) ; break ; \
case XFS_BTNUM_INO : __XFS_BTREE_STATS_INC ( ibt , stat ) ; break ; \
case XFS_BTNUM_MAX : ASSERT ( 0 ) ; /* fucking gcc */ ; break ; \
} \
} while ( 0 )
# define __XFS_BTREE_STATS_ADD(type, stat, val) \
XFS_STATS_ADD ( xs_ # # type # # _2_ # # stat , val )
# define XFS_BTREE_STATS_ADD(cur, stat, val) \
do { \
switch ( cur - > bc_btnum ) { \
case XFS_BTNUM_BNO : __XFS_BTREE_STATS_ADD ( abtb , stat , val ) ; break ; \
case XFS_BTNUM_CNT : __XFS_BTREE_STATS_ADD ( abtc , stat , val ) ; break ; \
case XFS_BTNUM_BMAP : __XFS_BTREE_STATS_ADD ( bmbt , stat , val ) ; break ; \
case XFS_BTNUM_INO : __XFS_BTREE_STATS_ADD ( ibt , stat , val ) ; break ; \
case XFS_BTNUM_MAX : ASSERT ( 0 ) ; /* fucking gcc */ ; break ; \
} \
} while ( 0 )
2005-04-16 15:20:36 -07:00
# define XFS_BTREE_MAXLEVELS 8 /* max of all btrees */
2008-10-30 16:53:59 +11:00
struct xfs_btree_ops {
2008-10-30 16:55:34 +11:00
/* size of the key and record structures */
size_t key_len ;
size_t rec_len ;
2008-10-30 16:53:59 +11:00
/* cursor operations */
struct xfs_btree_cur * ( * dup_cursor ) ( struct xfs_btree_cur * ) ;
2008-10-30 16:57:40 +11:00
void ( * update_cursor ) ( struct xfs_btree_cur * src ,
struct xfs_btree_cur * dst ) ;
2008-10-30 16:55:13 +11:00
2008-10-30 16:57:16 +11:00
/* update btree root pointer */
void ( * set_root ) ( struct xfs_btree_cur * cur ,
union xfs_btree_ptr * nptr , int level_change ) ;
2008-10-30 16:58:01 +11:00
int ( * kill_root ) ( struct xfs_btree_cur * cur , struct xfs_buf * bp ,
int level , union xfs_btree_ptr * newroot ) ;
2008-10-30 16:57:16 +11:00
2008-10-30 16:57:03 +11:00
/* block allocation / freeing */
int ( * alloc_block ) ( struct xfs_btree_cur * cur ,
union xfs_btree_ptr * start_bno ,
union xfs_btree_ptr * new_bno ,
int length , int * stat ) ;
2008-10-30 16:57:51 +11:00
int ( * free_block ) ( struct xfs_btree_cur * cur , struct xfs_buf * bp ) ;
2008-10-30 16:57:03 +11:00
2008-10-30 16:56:32 +11:00
/* update last record information */
void ( * update_lastrec ) ( struct xfs_btree_cur * cur ,
struct xfs_btree_block * block ,
union xfs_btree_rec * rec ,
int ptr , int reason ) ;
2008-10-30 16:55:23 +11:00
/* records in block/level */
2008-10-30 16:58:01 +11:00
int ( * get_minrecs ) ( struct xfs_btree_cur * cur , int level ) ;
2008-10-30 16:55:23 +11:00
int ( * get_maxrecs ) ( struct xfs_btree_cur * cur , int level ) ;
2008-10-30 16:57:40 +11:00
/* records on disk. Matter for the root in inode case. */
int ( * get_dmaxrecs ) ( struct xfs_btree_cur * cur , int level ) ;
2008-10-30 16:56:09 +11:00
/* init values of btree structures */
void ( * init_key_from_rec ) ( union xfs_btree_key * key ,
union xfs_btree_rec * rec ) ;
2008-10-30 16:57:40 +11:00
void ( * init_rec_from_key ) ( union xfs_btree_key * key ,
union xfs_btree_rec * rec ) ;
void ( * init_rec_from_cur ) ( struct xfs_btree_cur * cur ,
union xfs_btree_rec * rec ) ;
2008-10-30 16:56:09 +11:00
void ( * init_ptr_from_cur ) ( struct xfs_btree_cur * cur ,
union xfs_btree_ptr * ptr ) ;
/* difference between key value and cursor value */
__int64_t ( * key_diff ) ( struct xfs_btree_cur * cur ,
union xfs_btree_key * key ) ;
2008-10-30 16:58:32 +11:00
# ifdef DEBUG
/* check that k1 is lower than k2 */
int ( * keys_inorder ) ( struct xfs_btree_cur * cur ,
union xfs_btree_key * k1 ,
union xfs_btree_key * k2 ) ;
/* check that r1 is lower than r2 */
int ( * recs_inorder ) ( struct xfs_btree_cur * cur ,
union xfs_btree_rec * r1 ,
union xfs_btree_rec * r2 ) ;
# endif
2008-10-30 16:55:13 +11:00
/* btree tracing */
# ifdef XFS_BTREE_TRACE
void ( * trace_enter ) ( struct xfs_btree_cur * , const char * ,
char * , int , int , __psunsigned_t ,
__psunsigned_t , __psunsigned_t ,
__psunsigned_t , __psunsigned_t ,
__psunsigned_t , __psunsigned_t ,
__psunsigned_t , __psunsigned_t ,
__psunsigned_t , __psunsigned_t ) ;
void ( * trace_cursor ) ( struct xfs_btree_cur * , __uint32_t * ,
__uint64_t * , __uint64_t * ) ;
void ( * trace_key ) ( struct xfs_btree_cur * ,
union xfs_btree_key * , __uint64_t * ,
__uint64_t * ) ;
void ( * trace_record ) ( struct xfs_btree_cur * ,
union xfs_btree_rec * , __uint64_t * ,
__uint64_t * , __uint64_t * ) ;
# endif
2008-10-30 16:53:59 +11:00
} ;
2008-10-30 16:56:32 +11:00
/*
* Reasons for the update_lastrec method to be called .
*/
# define LASTREC_UPDATE 0
2008-10-30 16:57:40 +11:00
# define LASTREC_INSREC 1
2008-10-30 16:58:01 +11:00
# define LASTREC_DELREC 2
2008-10-30 16:56:32 +11:00
2005-04-16 15:20:36 -07:00
/*
* Btree cursor structure .
* This collects all information needed by the btree code in one place .
*/
typedef struct xfs_btree_cur
{
struct xfs_trans * bc_tp ; /* transaction we're in, if any */
struct xfs_mount * bc_mp ; /* file system mount struct */
2008-10-30 16:53:59 +11:00
const struct xfs_btree_ops * bc_ops ;
2008-10-30 16:54:22 +11:00
uint bc_flags ; /* btree features - below */
2005-04-16 15:20:36 -07:00
union {
2005-11-02 15:11:25 +11:00
xfs_alloc_rec_incore_t a ;
2005-04-16 15:20:36 -07:00
xfs_bmbt_irec_t b ;
2006-09-28 10:57:04 +10:00
xfs_inobt_rec_incore_t i ;
2005-04-16 15:20:36 -07:00
} bc_rec ; /* current insert/search record value */
struct xfs_buf * bc_bufs [ XFS_BTREE_MAXLEVELS ] ; /* buf ptr per level */
int bc_ptrs [ XFS_BTREE_MAXLEVELS ] ; /* key/record # */
__uint8_t bc_ra [ XFS_BTREE_MAXLEVELS ] ; /* readahead bits */
# define XFS_BTCUR_LEFTRA 1 /* left sibling has been read-ahead */
# define XFS_BTCUR_RIGHTRA 2 /* right sibling has been read-ahead */
__uint8_t bc_nlevels ; /* number of levels in the tree */
__uint8_t bc_blocklog ; /* log2(blocksize) of btree blocks */
xfs_btnum_t bc_btnum ; /* identifies which btree type */
union {
2008-08-13 16:25:27 +10:00
struct { /* needed for BNO, CNT, INO */
struct xfs_buf * agbp ; /* agf/agi buffer pointer */
2005-04-16 15:20:36 -07:00
xfs_agnumber_t agno ; /* ag number */
} a ;
struct { /* needed for BMAP */
struct xfs_inode * ip ; /* pointer to our inode */
struct xfs_bmap_free * flist ; /* list to free after */
xfs_fsblock_t firstblock ; /* 1st blk allocated */
int allocated ; /* count of alloced */
short forksize ; /* fork's inode space */
char whichfork ; /* data or attr fork */
char flags ; /* flags */
# define XFS_BTCUR_BPRV_WASDEL 1 /* was delayed */
} b ;
} bc_private ; /* per-btree type data */
} xfs_btree_cur_t ;
2008-10-30 16:54:22 +11:00
/* cursor flags */
2008-10-30 16:54:33 +11:00
# define XFS_BTREE_LONG_PTRS (1<<0) /* pointers are 64bits long */
2008-10-30 16:54:22 +11:00
# define XFS_BTREE_ROOT_IN_INODE (1<<1) /* root may be variable size */
2008-10-30 16:56:32 +11:00
# define XFS_BTREE_LASTREC_UPDATE (1<<2) /* track last rec externally */
2008-10-30 16:54:22 +11:00
2005-04-16 15:20:36 -07:00
# define XFS_BTREE_NOERROR 0
# define XFS_BTREE_ERROR 1
/*
* Convert from buffer to btree block header .
*/
2008-10-30 17:14:34 +11:00
# define XFS_BUF_TO_BLOCK(bp) ((struct xfs_btree_block *)XFS_BUF_PTR(bp))
2005-04-16 15:20:36 -07:00
/*
2008-10-30 16:54:53 +11:00
* Check that block header is ok .
2005-04-16 15:20:36 -07:00
*/
2008-10-30 16:54:53 +11:00
int
xfs_btree_check_block (
struct xfs_btree_cur * cur , /* btree cursor */
struct xfs_btree_block * block , /* generic btree block pointer */
2005-04-16 15:20:36 -07:00
int level , /* level of the btree block */
struct xfs_buf * bp ) ; /* buffer containing block, if any */
/*
2008-10-30 16:54:53 +11:00
* Check that ( long ) pointer is ok .
2005-04-16 15:20:36 -07:00
*/
int /* error (0 or EFSCORRUPTED) */
xfs_btree_check_lptr (
2008-10-30 16:54:53 +11:00
struct xfs_btree_cur * cur , /* btree cursor */
2005-04-16 15:20:36 -07:00
xfs_dfsbno_t ptr , /* btree block disk address */
int level ) ; /* btree block level */
/*
* Delete the btree cursor .
*/
void
xfs_btree_del_cursor (
xfs_btree_cur_t * cur , /* btree cursor */
int error ) ; /* del because of error */
/*
* Duplicate the btree cursor .
* Allocate a new one , copy the record , re - get the buffers .
*/
int /* error */
xfs_btree_dup_cursor (
xfs_btree_cur_t * cur , /* input cursor */
xfs_btree_cur_t * * ncur ) ; /* output cursor */
/*
* Get a buffer for the block , return it with no data read .
* Long - form addressing .
*/
struct xfs_buf * /* buffer for fsbno */
xfs_btree_get_bufl (
struct xfs_mount * mp , /* file system mount point */
struct xfs_trans * tp , /* transaction pointer */
xfs_fsblock_t fsbno , /* file system block number */
uint lock ) ; /* lock flags for get_buf */
/*
* Get a buffer for the block , return it with no data read .
* Short - form addressing .
*/
struct xfs_buf * /* buffer for agno/agbno */
xfs_btree_get_bufs (
struct xfs_mount * mp , /* file system mount point */
struct xfs_trans * tp , /* transaction pointer */
xfs_agnumber_t agno , /* allocation group number */
xfs_agblock_t agbno , /* allocation group block number */
uint lock ) ; /* lock flags for get_buf */
/*
* Check for the cursor referring to the last block at the given level .
*/
int /* 1=is last block, 0=not last block */
xfs_btree_islastblock (
xfs_btree_cur_t * cur , /* btree cursor */
int level ) ; /* level to check */
/*
* Compute first and last byte offsets for the fields given .
* Interprets the offsets table , which contains struct field offsets .
*/
void
xfs_btree_offsets (
__int64_t fields , /* bitmask of fields */
const short * offsets , /* table of field offsets */
int nbits , /* number of bits to inspect */
int * first , /* output: first byte offset */
int * last ) ; /* output: last byte offset */
/*
* Get a buffer for the block , return it read in .
* Long - form addressing .
*/
int /* error */
xfs_btree_read_bufl (
struct xfs_mount * mp , /* file system mount point */
struct xfs_trans * tp , /* transaction pointer */
xfs_fsblock_t fsbno , /* file system block number */
uint lock , /* lock flags for read_buf */
struct xfs_buf * * bpp , /* buffer for fsbno */
int refval ) ; /* ref count value for buffer */
/*
* Get a buffer for the block , return it read in .
* Short - form addressing .
*/
int /* error */
xfs_btree_read_bufs (
struct xfs_mount * mp , /* file system mount point */
struct xfs_trans * tp , /* transaction pointer */
xfs_agnumber_t agno , /* allocation group number */
xfs_agblock_t agbno , /* allocation group block number */
uint lock , /* lock flags for read_buf */
struct xfs_buf * * bpp , /* buffer for agno/agbno */
int refval ) ; /* ref count value for buffer */
/*
* Read - ahead the block , don ' t wait for it , don ' t return a buffer .
* Long - form addressing .
*/
void /* error */
xfs_btree_reada_bufl (
struct xfs_mount * mp , /* file system mount point */
xfs_fsblock_t fsbno , /* file system block number */
xfs_extlen_t count ) ; /* count of filesystem blocks */
/*
* Read - ahead the block , don ' t wait for it , don ' t return a buffer .
* Short - form addressing .
*/
void /* error */
xfs_btree_reada_bufs (
struct xfs_mount * mp , /* file system mount point */
xfs_agnumber_t agno , /* allocation group number */
xfs_agblock_t agbno , /* allocation group block number */
xfs_extlen_t count ) ; /* count of filesystem blocks */
/*
* Set the buffer for level " lev " in the cursor to bp , releasing
* any previous buffer .
*/
void
xfs_btree_setbuf (
xfs_btree_cur_t * cur , /* btree cursor */
int lev , /* level in btree */
struct xfs_buf * bp ) ; /* new buffer to set */
2008-10-30 16:55:34 +11:00
2008-10-30 16:55:45 +11:00
/*
* Common btree core entry points .
*/
int xfs_btree_increment ( struct xfs_btree_cur * , int , int * ) ;
2008-10-30 16:55:58 +11:00
int xfs_btree_decrement ( struct xfs_btree_cur * , int , int * ) ;
2008-10-30 16:56:09 +11:00
int xfs_btree_lookup ( struct xfs_btree_cur * , xfs_lookup_t , int * ) ;
2008-10-30 16:56:32 +11:00
int xfs_btree_update ( struct xfs_btree_cur * , union xfs_btree_rec * ) ;
2008-10-30 16:57:28 +11:00
int xfs_btree_new_iroot ( struct xfs_btree_cur * , int * , int * ) ;
2008-10-30 16:57:51 +11:00
int xfs_btree_kill_iroot ( struct xfs_btree_cur * ) ;
2008-10-30 16:57:40 +11:00
int xfs_btree_insert ( struct xfs_btree_cur * , int * ) ;
2008-10-30 16:58:01 +11:00
int xfs_btree_delete ( struct xfs_btree_cur * , int * ) ;
2008-10-30 16:58:11 +11:00
int xfs_btree_get_rec ( struct xfs_btree_cur * , union xfs_btree_rec * * , int * ) ;
2008-10-30 16:55:45 +11:00
2008-10-30 16:58:21 +11:00
/*
* Internal btree helpers also used by xfs_bmap . c .
*/
void xfs_btree_log_block ( struct xfs_btree_cur * , struct xfs_buf * , int ) ;
void xfs_btree_log_recs ( struct xfs_btree_cur * , struct xfs_buf * , int , int ) ;
2008-10-30 16:55:34 +11:00
/*
* Helpers .
*/
2008-10-30 16:55:45 +11:00
static inline int xfs_btree_get_numrecs ( struct xfs_btree_block * block )
{
return be16_to_cpu ( block - > bb_numrecs ) ;
}
2008-10-30 16:56:43 +11:00
static inline void xfs_btree_set_numrecs ( struct xfs_btree_block * block ,
__uint16_t numrecs )
{
block - > bb_numrecs = cpu_to_be16 ( numrecs ) ;
}
2008-10-30 16:55:34 +11:00
static inline int xfs_btree_get_level ( struct xfs_btree_block * block )
{
return be16_to_cpu ( block - > bb_level ) ;
}
2005-04-16 15:20:36 -07:00
/*
* Min and max functions for extlen , agblock , fileoff , and filblks types .
*/
2007-06-28 16:43:39 +10:00
# define XFS_EXTLEN_MIN(a,b) min_t(xfs_extlen_t, (a), (b))
# define XFS_EXTLEN_MAX(a,b) max_t(xfs_extlen_t, (a), (b))
# define XFS_AGBLOCK_MIN(a,b) min_t(xfs_agblock_t, (a), (b))
# define XFS_AGBLOCK_MAX(a,b) max_t(xfs_agblock_t, (a), (b))
# define XFS_FILEOFF_MIN(a,b) min_t(xfs_fileoff_t, (a), (b))
# define XFS_FILEOFF_MAX(a,b) max_t(xfs_fileoff_t, (a), (b))
# define XFS_FILBLKS_MIN(a,b) min_t(xfs_filblks_t, (a), (b))
# define XFS_FILBLKS_MAX(a,b) max_t(xfs_filblks_t, (a), (b))
2005-11-02 14:38:42 +11:00
2005-04-16 15:20:36 -07:00
# define XFS_FSB_SANITY_CHECK(mp,fsb) \
( XFS_FSB_TO_AGNO ( mp , fsb ) < mp - > m_sb . sb_agcount & & \
2005-11-02 14:38:42 +11:00
XFS_FSB_TO_AGBNO ( mp , fsb ) < mp - > m_sb . sb_agblocks )
2005-04-16 15:20:36 -07:00
# endif /* __XFS_BTREE_H__ */