mirror of
git://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
synced 2025-09-04 20:19:47 +08:00

Create an in-memory btree of rmap records instead of an array. This enables us to do live record collection instead of freezing the fs. Signed-off-by: "Darrick J. Wong" <djwong@kernel.org> Reviewed-by: Christoph Hellwig <hch@lst.de>
211 lines
5.8 KiB
C
211 lines
5.8 KiB
C
/* SPDX-License-Identifier: GPL-2.0-or-later */
|
|
/*
|
|
* Copyright (c) 2018-2024 Oracle. All Rights Reserved.
|
|
* Author: Darrick J. Wong <djwong@kernel.org>
|
|
*/
|
|
#ifndef __XFS_RTRMAP_BTREE_H__
|
|
#define __XFS_RTRMAP_BTREE_H__
|
|
|
|
struct xfs_buf;
|
|
struct xfs_btree_cur;
|
|
struct xfs_mount;
|
|
struct xbtree_ifakeroot;
|
|
struct xfs_rtgroup;
|
|
struct xfbtree;
|
|
|
|
/* rmaps only exist on crc enabled filesystems */
|
|
#define XFS_RTRMAP_BLOCK_LEN XFS_BTREE_LBLOCK_CRC_LEN
|
|
|
|
struct xfs_btree_cur *xfs_rtrmapbt_init_cursor(struct xfs_trans *tp,
|
|
struct xfs_rtgroup *rtg);
|
|
struct xfs_btree_cur *xfs_rtrmapbt_stage_cursor(struct xfs_mount *mp,
|
|
struct xfs_rtgroup *rtg, struct xfs_inode *ip,
|
|
struct xbtree_ifakeroot *ifake);
|
|
void xfs_rtrmapbt_commit_staged_btree(struct xfs_btree_cur *cur,
|
|
struct xfs_trans *tp);
|
|
unsigned int xfs_rtrmapbt_maxrecs(struct xfs_mount *mp, unsigned int blocklen,
|
|
bool leaf);
|
|
void xfs_rtrmapbt_compute_maxlevels(struct xfs_mount *mp);
|
|
unsigned int xfs_rtrmapbt_droot_maxrecs(unsigned int blocklen, bool leaf);
|
|
|
|
/*
|
|
* Addresses of records, keys, and pointers within an incore rtrmapbt block.
|
|
*
|
|
* (note that some of these may appear unused, but they are used in userspace)
|
|
*/
|
|
static inline struct xfs_rmap_rec *
|
|
xfs_rtrmap_rec_addr(
|
|
struct xfs_btree_block *block,
|
|
unsigned int index)
|
|
{
|
|
return (struct xfs_rmap_rec *)
|
|
((char *)block + XFS_RTRMAP_BLOCK_LEN +
|
|
(index - 1) * sizeof(struct xfs_rmap_rec));
|
|
}
|
|
|
|
static inline struct xfs_rmap_key *
|
|
xfs_rtrmap_key_addr(
|
|
struct xfs_btree_block *block,
|
|
unsigned int index)
|
|
{
|
|
return (struct xfs_rmap_key *)
|
|
((char *)block + XFS_RTRMAP_BLOCK_LEN +
|
|
(index - 1) * 2 * sizeof(struct xfs_rmap_key));
|
|
}
|
|
|
|
static inline struct xfs_rmap_key *
|
|
xfs_rtrmap_high_key_addr(
|
|
struct xfs_btree_block *block,
|
|
unsigned int index)
|
|
{
|
|
return (struct xfs_rmap_key *)
|
|
((char *)block + XFS_RTRMAP_BLOCK_LEN +
|
|
sizeof(struct xfs_rmap_key) +
|
|
(index - 1) * 2 * sizeof(struct xfs_rmap_key));
|
|
}
|
|
|
|
static inline xfs_rtrmap_ptr_t *
|
|
xfs_rtrmap_ptr_addr(
|
|
struct xfs_btree_block *block,
|
|
unsigned int index,
|
|
unsigned int maxrecs)
|
|
{
|
|
return (xfs_rtrmap_ptr_t *)
|
|
((char *)block + XFS_RTRMAP_BLOCK_LEN +
|
|
maxrecs * 2 * sizeof(struct xfs_rmap_key) +
|
|
(index - 1) * sizeof(xfs_rtrmap_ptr_t));
|
|
}
|
|
|
|
unsigned int xfs_rtrmapbt_maxlevels_ondisk(void);
|
|
|
|
int __init xfs_rtrmapbt_init_cur_cache(void);
|
|
void xfs_rtrmapbt_destroy_cur_cache(void);
|
|
|
|
xfs_filblks_t xfs_rtrmapbt_calc_reserves(struct xfs_mount *mp);
|
|
|
|
/* Addresses of key, pointers, and records within an ondisk rtrmapbt block. */
|
|
|
|
static inline struct xfs_rmap_rec *
|
|
xfs_rtrmap_droot_rec_addr(
|
|
struct xfs_rtrmap_root *block,
|
|
unsigned int index)
|
|
{
|
|
return (struct xfs_rmap_rec *)
|
|
((char *)(block + 1) +
|
|
(index - 1) * sizeof(struct xfs_rmap_rec));
|
|
}
|
|
|
|
static inline struct xfs_rmap_key *
|
|
xfs_rtrmap_droot_key_addr(
|
|
struct xfs_rtrmap_root *block,
|
|
unsigned int index)
|
|
{
|
|
return (struct xfs_rmap_key *)
|
|
((char *)(block + 1) +
|
|
(index - 1) * 2 * sizeof(struct xfs_rmap_key));
|
|
}
|
|
|
|
static inline xfs_rtrmap_ptr_t *
|
|
xfs_rtrmap_droot_ptr_addr(
|
|
struct xfs_rtrmap_root *block,
|
|
unsigned int index,
|
|
unsigned int maxrecs)
|
|
{
|
|
return (xfs_rtrmap_ptr_t *)
|
|
((char *)(block + 1) +
|
|
maxrecs * 2 * sizeof(struct xfs_rmap_key) +
|
|
(index - 1) * sizeof(xfs_rtrmap_ptr_t));
|
|
}
|
|
|
|
/*
|
|
* Address of pointers within the incore btree root.
|
|
*
|
|
* These are to be used when we know the size of the block and
|
|
* we don't have a cursor.
|
|
*/
|
|
static inline xfs_rtrmap_ptr_t *
|
|
xfs_rtrmap_broot_ptr_addr(
|
|
struct xfs_mount *mp,
|
|
struct xfs_btree_block *bb,
|
|
unsigned int index,
|
|
unsigned int block_size)
|
|
{
|
|
return xfs_rtrmap_ptr_addr(bb, index,
|
|
xfs_rtrmapbt_maxrecs(mp, block_size, false));
|
|
}
|
|
|
|
/*
|
|
* Compute the space required for the incore btree root containing the given
|
|
* number of records.
|
|
*/
|
|
static inline size_t
|
|
xfs_rtrmap_broot_space_calc(
|
|
struct xfs_mount *mp,
|
|
unsigned int level,
|
|
unsigned int nrecs)
|
|
{
|
|
size_t sz = XFS_RTRMAP_BLOCK_LEN;
|
|
|
|
if (level > 0)
|
|
return sz + nrecs * (2 * sizeof(struct xfs_rmap_key) +
|
|
sizeof(xfs_rtrmap_ptr_t));
|
|
return sz + nrecs * sizeof(struct xfs_rmap_rec);
|
|
}
|
|
|
|
/*
|
|
* Compute the space required for the incore btree root given the ondisk
|
|
* btree root block.
|
|
*/
|
|
static inline size_t
|
|
xfs_rtrmap_broot_space(struct xfs_mount *mp, struct xfs_rtrmap_root *bb)
|
|
{
|
|
return xfs_rtrmap_broot_space_calc(mp, be16_to_cpu(bb->bb_level),
|
|
be16_to_cpu(bb->bb_numrecs));
|
|
}
|
|
|
|
/* Compute the space required for the ondisk root block. */
|
|
static inline size_t
|
|
xfs_rtrmap_droot_space_calc(
|
|
unsigned int level,
|
|
unsigned int nrecs)
|
|
{
|
|
size_t sz = sizeof(struct xfs_rtrmap_root);
|
|
|
|
if (level > 0)
|
|
return sz + nrecs * (2 * sizeof(struct xfs_rmap_key) +
|
|
sizeof(xfs_rtrmap_ptr_t));
|
|
return sz + nrecs * sizeof(struct xfs_rmap_rec);
|
|
}
|
|
|
|
/*
|
|
* Compute the space required for the ondisk root block given an incore root
|
|
* block.
|
|
*/
|
|
static inline size_t
|
|
xfs_rtrmap_droot_space(struct xfs_btree_block *bb)
|
|
{
|
|
return xfs_rtrmap_droot_space_calc(be16_to_cpu(bb->bb_level),
|
|
be16_to_cpu(bb->bb_numrecs));
|
|
}
|
|
|
|
int xfs_iformat_rtrmap(struct xfs_inode *ip, struct xfs_dinode *dip);
|
|
void xfs_rtrmapbt_to_disk(struct xfs_mount *mp, struct xfs_btree_block *rblock,
|
|
unsigned int rblocklen, struct xfs_rtrmap_root *dblock,
|
|
unsigned int dblocklen);
|
|
void xfs_iflush_rtrmap(struct xfs_inode *ip, struct xfs_dinode *dip);
|
|
|
|
int xfs_rtrmapbt_create(struct xfs_rtgroup *rtg, struct xfs_inode *ip,
|
|
struct xfs_trans *tp, bool init);
|
|
int xfs_rtrmapbt_init_rtsb(struct xfs_mount *mp, struct xfs_rtgroup *rtg,
|
|
struct xfs_trans *tp);
|
|
|
|
unsigned long long xfs_rtrmapbt_calc_size(struct xfs_mount *mp,
|
|
unsigned long long len);
|
|
|
|
struct xfs_btree_cur *xfs_rtrmapbt_mem_cursor(struct xfs_rtgroup *rtg,
|
|
struct xfs_trans *tp, struct xfbtree *xfbtree);
|
|
int xfs_rtrmapbt_mem_init(struct xfs_mount *mp, struct xfbtree *xfbtree,
|
|
struct xfs_buftarg *btp, xfs_rgnumber_t rgno);
|
|
|
|
#endif /* __XFS_RTRMAP_BTREE_H__ */
|