2005-04-17 02:20:36 +04:00
/*
2005-11-02 06:58:39 +03:00
* Copyright ( c ) 2000 - 2003 , 2005 Silicon Graphics , Inc .
* All Rights Reserved .
2005-04-17 02:20:36 +04:00
*
2005-11-02 06:58:39 +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:58:39 +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:58:39 +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
*/
# ifndef __XFS_AG_H__
# define __XFS_AG_H__
/*
* Allocation group header
* This is divided into three structures , placed in sequential 512 - byte
* buffers after a copy of the superblock ( also in a 512 - byte buffer ) .
*/
struct xfs_buf ;
struct xfs_mount ;
struct xfs_trans ;
# define XFS_AGF_MAGIC 0x58414746 /* 'XAGF' */
# define XFS_AGI_MAGIC 0x58414749 /* 'XAGI' */
# define XFS_AGF_VERSION 1
# define XFS_AGI_VERSION 1
2005-11-02 06:38:42 +03:00
# define XFS_AGF_GOOD_VERSION(v) ((v) == XFS_AGF_VERSION)
# define XFS_AGI_GOOD_VERSION(v) ((v) == XFS_AGI_VERSION)
2005-04-17 02:20:36 +04:00
/*
* Btree number 0 is bno , 1 is cnt . This value gives the size of the
* arrays below .
*/
# define XFS_BTNUM_AGF ((int)XFS_BTNUM_CNTi + 1)
/*
* The second word of agf_levels in the first a . g . overlaps the EFS
* superblock ' s magic number . Since the magic numbers valid for EFS
* are > 64 k , our value cannot be confused for an EFS superblock ' s .
*/
2005-11-02 07:11:25 +03:00
typedef struct xfs_agf {
2005-04-17 02:20:36 +04:00
/*
* Common allocation group header information
*/
2005-11-02 07:11:25 +03:00
__be32 agf_magicnum ; /* magic number == XFS_AGF_MAGIC */
__be32 agf_versionnum ; /* header version == XFS_AGF_VERSION */
__be32 agf_seqno ; /* sequence # starting from 0 */
__be32 agf_length ; /* size in blocks of a.g. */
2005-04-17 02:20:36 +04:00
/*
* Freespace information
*/
2005-11-02 07:11:25 +03:00
__be32 agf_roots [ XFS_BTNUM_AGF ] ; /* root blocks */
__be32 agf_spare0 ; /* spare field */
__be32 agf_levels [ XFS_BTNUM_AGF ] ; /* btree levels */
__be32 agf_spare1 ; /* spare field */
__be32 agf_flfirst ; /* first freelist block's index */
__be32 agf_fllast ; /* last freelist block's index */
__be32 agf_flcount ; /* count of blocks in freelist */
__be32 agf_freeblks ; /* total free blocks */
__be32 agf_longest ; /* longest free space */
2005-04-17 02:20:36 +04:00
} xfs_agf_t ;
# define XFS_AGF_MAGICNUM 0x00000001
# define XFS_AGF_VERSIONNUM 0x00000002
# define XFS_AGF_SEQNO 0x00000004
# define XFS_AGF_LENGTH 0x00000008
# define XFS_AGF_ROOTS 0x00000010
# define XFS_AGF_LEVELS 0x00000020
# define XFS_AGF_FLFIRST 0x00000040
# define XFS_AGF_FLLAST 0x00000080
# define XFS_AGF_FLCOUNT 0x00000100
# define XFS_AGF_FREEBLKS 0x00000200
# define XFS_AGF_LONGEST 0x00000400
# define XFS_AGF_NUM_BITS 11
# define XFS_AGF_ALL_BITS ((1 << XFS_AGF_NUM_BITS) - 1)
/* disk block (xfs_daddr_t) in the AG */
# define XFS_AGF_DADDR(mp) ((xfs_daddr_t)(1 << (mp)->m_sectbb_log))
2005-11-02 06:38:42 +03:00
# define XFS_AGF_BLOCK(mp) XFS_HDR_BLOCK(mp, XFS_AGF_DADDR(mp))
# define XFS_BUF_TO_AGF(bp) ((xfs_agf_t *)XFS_BUF_PTR(bp))
2005-04-17 02:20:36 +04:00
/*
* Size of the unlinked inode hash table in the agi .
*/
# define XFS_AGI_UNLINKED_BUCKETS 64
2005-11-02 07:11:25 +03:00
typedef struct xfs_agi {
2005-04-17 02:20:36 +04:00
/*
* Common allocation group header information
*/
2005-11-02 07:11:25 +03:00
__be32 agi_magicnum ; /* magic number == XFS_AGI_MAGIC */
__be32 agi_versionnum ; /* header version == XFS_AGI_VERSION */
__be32 agi_seqno ; /* sequence # starting from 0 */
__be32 agi_length ; /* size in blocks of a.g. */
2005-04-17 02:20:36 +04:00
/*
* Inode information
* Inodes are mapped by interpreting the inode number , so no
* mapping data is needed here .
*/
2005-11-02 07:11:25 +03:00
__be32 agi_count ; /* count of allocated inodes */
__be32 agi_root ; /* root of inode btree */
__be32 agi_level ; /* levels in inode btree */
__be32 agi_freecount ; /* number of free inodes */
__be32 agi_newino ; /* new inode just allocated */
__be32 agi_dirino ; /* last directory inode chunk */
2005-04-17 02:20:36 +04:00
/*
* Hash table of inodes which have been unlinked but are
* still being referenced .
*/
2005-11-02 07:11:25 +03:00
__be32 agi_unlinked [ XFS_AGI_UNLINKED_BUCKETS ] ;
2005-04-17 02:20:36 +04:00
} xfs_agi_t ;
# define XFS_AGI_MAGICNUM 0x00000001
# define XFS_AGI_VERSIONNUM 0x00000002
# define XFS_AGI_SEQNO 0x00000004
# define XFS_AGI_LENGTH 0x00000008
# define XFS_AGI_COUNT 0x00000010
# define XFS_AGI_ROOT 0x00000020
# define XFS_AGI_LEVEL 0x00000040
# define XFS_AGI_FREECOUNT 0x00000080
# define XFS_AGI_NEWINO 0x00000100
# define XFS_AGI_DIRINO 0x00000200
# define XFS_AGI_UNLINKED 0x00000400
# define XFS_AGI_NUM_BITS 11
# define XFS_AGI_ALL_BITS ((1 << XFS_AGI_NUM_BITS) - 1)
/* disk block (xfs_daddr_t) in the AG */
# define XFS_AGI_DADDR(mp) ((xfs_daddr_t)(2 << (mp)->m_sectbb_log))
2005-11-02 06:38:42 +03:00
# define XFS_AGI_BLOCK(mp) XFS_HDR_BLOCK(mp, XFS_AGI_DADDR(mp))
# define XFS_BUF_TO_AGI(bp) ((xfs_agi_t *)XFS_BUF_PTR(bp))
2005-04-17 02:20:36 +04:00
/*
* The third a . g . block contains the a . g . freelist , an array
* of block pointers to blocks owned by the allocation btree code .
*/
# define XFS_AGFL_DADDR(mp) ((xfs_daddr_t)(3 << (mp)->m_sectbb_log))
2005-11-02 06:38:42 +03:00
# define XFS_AGFL_BLOCK(mp) XFS_HDR_BLOCK(mp, XFS_AGFL_DADDR(mp))
2005-04-17 02:20:36 +04:00
# define XFS_AGFL_SIZE(mp) ((mp)->m_sb.sb_sectsize / sizeof(xfs_agblock_t))
2005-11-02 06:38:42 +03:00
# define XFS_BUF_TO_AGFL(bp) ((xfs_agfl_t *)XFS_BUF_PTR(bp))
2005-04-17 02:20:36 +04:00
typedef struct xfs_agfl {
xfs_agblock_t agfl_bno [ 1 ] ; /* actually XFS_AGFL_SIZE(mp) */
} xfs_agfl_t ;
/*
* Busy block / extent entry . Used in perag to mark blocks that have been freed
* but whose transactions aren ' t committed to disk yet .
*/
typedef struct xfs_perag_busy {
xfs_agblock_t busy_start ;
xfs_extlen_t busy_length ;
struct xfs_trans * busy_tp ; /* transaction that did the free */
} xfs_perag_busy_t ;
/*
* Per - ag incore structure , copies of information in agf and agi ,
* to improve the performance of allocation group selection .
*
* pick sizes which fit in allocation buckets well
*/
# if (BITS_PER_LONG == 32)
# define XFS_PAGB_NUM_SLOTS 84
# elif (BITS_PER_LONG == 64)
# define XFS_PAGB_NUM_SLOTS 128
# endif
typedef struct xfs_perag
{
char pagf_init ; /* this agf's entry is initialized */
char pagi_init ; /* this agi's entry is initialized */
char pagf_metadata ; /* the agf is prefered to be metadata */
char pagi_inodeok ; /* The agi is ok for inodes */
__uint8_t pagf_levels [ XFS_BTNUM_AGF ] ;
/* # of levels in bno & cnt btree */
__uint32_t pagf_flcount ; /* count of blocks in freelist */
xfs_extlen_t pagf_freeblks ; /* total free blocks */
xfs_extlen_t pagf_longest ; /* longest free space */
xfs_agino_t pagi_freecount ; /* number of free inodes */
# ifdef __KERNEL__
lock_t pagb_lock ; /* lock for pagb_list */
# endif
int pagb_count ; /* pagb slots in use */
xfs_perag_busy_t * pagb_list ; /* unstable blocks */
} xfs_perag_t ;
2005-11-02 06:38:42 +03:00
# define XFS_AG_MAXLEVELS(mp) ((mp)->m_ag_maxlevels)
# define XFS_MIN_FREELIST_RAW(bl,cl,mp) \
( MIN ( bl + 1 , XFS_AG_MAXLEVELS ( mp ) ) + MIN ( cl + 1 , XFS_AG_MAXLEVELS ( mp ) ) )
# define XFS_MIN_FREELIST(a,mp) \
( XFS_MIN_FREELIST_RAW ( \
2005-11-02 07:11:25 +03:00
be32_to_cpu ( ( a ) - > agf_levels [ XFS_BTNUM_BNOi ] ) , \
be32_to_cpu ( ( a ) - > agf_levels [ XFS_BTNUM_CNTi ] ) , mp ) )
2005-04-17 02:20:36 +04:00
# define XFS_MIN_FREELIST_PAG(pag,mp) \
2005-11-02 06:38:42 +03:00
( XFS_MIN_FREELIST_RAW ( \
( uint_t ) ( pag ) - > pagf_levels [ XFS_BTNUM_BNOi ] , \
( uint_t ) ( pag ) - > pagf_levels [ XFS_BTNUM_CNTi ] , mp ) )
2005-04-17 02:20:36 +04:00
2005-11-02 06:38:42 +03:00
# define XFS_AGB_TO_FSB(mp,agno,agbno) \
2005-04-17 02:20:36 +04:00
( ( ( xfs_fsblock_t ) ( agno ) < < ( mp ) - > m_sb . sb_agblklog ) | ( agbno ) )
2005-11-02 06:38:42 +03:00
# define XFS_FSB_TO_AGNO(mp,fsbno) \
2005-04-17 02:20:36 +04:00
( ( xfs_agnumber_t ) ( ( fsbno ) > > ( mp ) - > m_sb . sb_agblklog ) )
2005-11-02 06:38:42 +03:00
# define XFS_FSB_TO_AGBNO(mp,fsbno) \
2005-04-17 02:20:36 +04:00
( ( xfs_agblock_t ) ( ( fsbno ) & XFS_MASK32LO ( ( mp ) - > m_sb . sb_agblklog ) ) )
2005-11-02 06:38:42 +03:00
# define XFS_AGB_TO_DADDR(mp,agno,agbno) \
( ( xfs_daddr_t ) XFS_FSB_TO_BB ( mp , \
( xfs_fsblock_t ) ( agno ) * ( mp ) - > m_sb . sb_agblocks + ( agbno ) ) )
# define XFS_AG_DADDR(mp,agno,d) (XFS_AGB_TO_DADDR(mp, agno, 0) + (d))
2005-04-17 02:20:36 +04:00
/*
* For checking for bad ranges of xfs_daddr_t ' s , covering multiple
* allocation groups or a single xfs_daddr_t that ' s a superblock copy .
*/
# define XFS_AG_CHECK_DADDR(mp,d,len) \
( ( len ) = = 1 ? \
ASSERT ( ( d ) = = XFS_SB_DADDR | | \
XFS_DADDR_TO_AGBNO ( mp , d ) ! = XFS_SB_DADDR ) : \
ASSERT ( XFS_DADDR_TO_AGNO ( mp , d ) = = \
XFS_DADDR_TO_AGNO ( mp , ( d ) + ( len ) - 1 ) ) )
# endif /* __XFS_AG_H__ */