1 // SPDX-License-Identifier: GPL-2.0
3 * Copyright (c) 2000-2002,2005 Silicon Graphics, Inc.
4 * Copyright (C) 2017 Oracle.
9 #include "xfs_format.h"
10 #include "xfs_log_format.h"
11 #include "xfs_shared.h"
12 #include "xfs_trans_resv.h"
15 #include "xfs_mount.h"
16 #include "xfs_defer.h"
17 #include "xfs_inode.h"
18 #include "xfs_btree.h"
20 #include "xfs_alloc_btree.h"
21 #include "xfs_alloc.h"
22 #include "xfs_ialloc.h"
24 /* Find the size of the AG, in blocks. */
30 ASSERT(agno < mp->m_sb.sb_agcount);
32 if (agno < mp->m_sb.sb_agcount - 1)
33 return mp->m_sb.sb_agblocks;
34 return mp->m_sb.sb_dblocks - (agno * mp->m_sb.sb_agblocks);
38 * Verify that an AG block number pointer neither points outside the AG
39 * nor points at static metadata.
49 eoag = xfs_ag_block_count(mp, agno);
52 if (agbno <= XFS_AGFL_BLOCK(mp))
58 * Verify that an FS block number pointer neither points outside the
59 * filesystem nor points at static AG metadata.
66 xfs_agnumber_t agno = XFS_FSB_TO_AGNO(mp, fsbno);
68 if (agno >= mp->m_sb.sb_agcount)
70 return xfs_verify_agbno(mp, agno, XFS_FSB_TO_AGBNO(mp, fsbno));
73 /* Calculate the first and last possible inode number in an AG. */
84 eoag = xfs_ag_block_count(mp, agno);
87 * Calculate the first inode, which will be in the first
88 * cluster-aligned block after the AGFL.
90 bno = round_up(XFS_AGFL_BLOCK(mp) + 1, M_IGEO(mp)->cluster_align);
91 *first = XFS_AGB_TO_AGINO(mp, bno);
94 * Calculate the last inode, which will be at the end of the
95 * last (aligned) cluster that can be allocated in the AG.
97 bno = round_down(eoag, M_IGEO(mp)->cluster_align);
98 *last = XFS_AGB_TO_AGINO(mp, bno) - 1;
102 * Verify that an AG inode number pointer neither points outside the AG
103 * nor points at static metadata.
107 struct xfs_mount *mp,
114 xfs_agino_range(mp, agno, &first, &last);
115 return agino >= first && agino <= last;
119 * Verify that an AG inode number pointer neither points outside the AG
120 * nor points at static metadata, or is NULLAGINO.
123 xfs_verify_agino_or_null(
124 struct xfs_mount *mp,
128 return agino == NULLAGINO || xfs_verify_agino(mp, agno, agino);
132 * Verify that an FS inode number pointer neither points outside the
133 * filesystem nor points at static AG metadata.
137 struct xfs_mount *mp,
140 xfs_agnumber_t agno = XFS_INO_TO_AGNO(mp, ino);
141 xfs_agino_t agino = XFS_INO_TO_AGINO(mp, ino);
143 if (agno >= mp->m_sb.sb_agcount)
145 if (XFS_AGINO_TO_INO(mp, agno, agino) != ino)
147 return xfs_verify_agino(mp, agno, agino);
150 /* Is this an internal inode number? */
153 struct xfs_mount *mp,
156 return ino == mp->m_sb.sb_rbmino || ino == mp->m_sb.sb_rsumino ||
157 (xfs_sb_version_hasquota(&mp->m_sb) &&
158 xfs_is_quota_inode(&mp->m_sb, ino));
162 * Verify that a directory entry's inode number doesn't point at an internal
163 * inode, empty space, or static AG metadata.
167 struct xfs_mount *mp,
170 if (xfs_internal_inum(mp, ino))
172 return xfs_verify_ino(mp, ino);
176 * Verify that an realtime block number pointer doesn't point off the
177 * end of the realtime device.
181 struct xfs_mount *mp,
184 return rtbno < mp->m_sb.sb_rblocks;
187 /* Calculate the range of valid icount values. */
190 struct xfs_mount *mp,
191 unsigned long long *min,
192 unsigned long long *max)
194 unsigned long long nr_inos = 0;
197 /* root, rtbitmap, rtsum all live in the first chunk */
198 *min = XFS_INODES_PER_CHUNK;
200 for (agno = 0; agno < mp->m_sb.sb_agcount; agno++) {
201 xfs_agino_t first, last;
203 xfs_agino_range(mp, agno, &first, &last);
204 nr_inos += last - first + 1;
209 /* Sanity-checking of inode counts. */
212 struct xfs_mount *mp,
213 unsigned long long icount)
215 unsigned long long min, max;
217 xfs_icount_range(mp, &min, &max);
218 return icount >= min && icount <= max;
221 /* Sanity-checking of dir/attr block offsets. */
224 struct xfs_mount *mp,
227 xfs_dablk_t max_dablk = -1U;
229 return dabno <= max_dablk;