diff options
| author | Dmitry Torokhov <dmitry.torokhov@gmail.com> | 2010-06-22 14:32:04 -0400 |
|---|---|---|
| committer | Dmitry Torokhov <dmitry.torokhov@gmail.com> | 2010-06-22 14:32:04 -0400 |
| commit | 0e789314f8c0b50bd19bf08dc5624b9604d60183 (patch) | |
| tree | 64b6b9655e944284937f5eeff643f0754f35d0af /fs/xfs/xfs_trans.c | |
| parent | f5dec51172b81db226a23f309bc737ad021af35b (diff) | |
| parent | 7e27d6e778cd87b6f2415515d7127eba53fe5d02 (diff) | |
Merge commit 'v2.6.35-rc3' into for-linus
Diffstat (limited to 'fs/xfs/xfs_trans.c')
| -rw-r--r-- | fs/xfs/xfs_trans.c | 446 |
1 files changed, 400 insertions, 46 deletions
diff --git a/fs/xfs/xfs_trans.c b/fs/xfs/xfs_trans.c index ce558efa2ea0..28547dfce037 100644 --- a/fs/xfs/xfs_trans.c +++ b/fs/xfs/xfs_trans.c | |||
| @@ -48,134 +48,489 @@ | |||
| 48 | 48 | ||
| 49 | kmem_zone_t *xfs_trans_zone; | 49 | kmem_zone_t *xfs_trans_zone; |
| 50 | 50 | ||
| 51 | |||
| 51 | /* | 52 | /* |
| 52 | * Reservation functions here avoid a huge stack in xfs_trans_init | 53 | * Various log reservation values. |
| 53 | * due to register overflow from temporaries in the calculations. | 54 | * |
| 55 | * These are based on the size of the file system block because that is what | ||
| 56 | * most transactions manipulate. Each adds in an additional 128 bytes per | ||
| 57 | * item logged to try to account for the overhead of the transaction mechanism. | ||
| 58 | * | ||
| 59 | * Note: Most of the reservations underestimate the number of allocation | ||
| 60 | * groups into which they could free extents in the xfs_bmap_finish() call. | ||
| 61 | * This is because the number in the worst case is quite high and quite | ||
| 62 | * unusual. In order to fix this we need to change xfs_bmap_finish() to free | ||
| 63 | * extents in only a single AG at a time. This will require changes to the | ||
| 64 | * EFI code as well, however, so that the EFI for the extents not freed is | ||
| 65 | * logged again in each transaction. See SGI PV #261917. | ||
| 66 | * | ||
| 67 | * Reservation functions here avoid a huge stack in xfs_trans_init due to | ||
| 68 | * register overflow from temporaries in the calculations. | ||
| 69 | */ | ||
| 70 | |||
| 71 | |||
| 72 | /* | ||
| 73 | * In a write transaction we can allocate a maximum of 2 | ||
| 74 | * extents. This gives: | ||
| 75 | * the inode getting the new extents: inode size | ||
| 76 | * the inode's bmap btree: max depth * block size | ||
| 77 | * the agfs of the ags from which the extents are allocated: 2 * sector | ||
| 78 | * the superblock free block counter: sector size | ||
| 79 | * the allocation btrees: 2 exts * 2 trees * (2 * max depth - 1) * block size | ||
| 80 | * And the bmap_finish transaction can free bmap blocks in a join: | ||
| 81 | * the agfs of the ags containing the blocks: 2 * sector size | ||
| 82 | * the agfls of the ags containing the blocks: 2 * sector size | ||
| 83 | * the super block free block counter: sector size | ||
| 84 | * the allocation btrees: 2 exts * 2 trees * (2 * max depth - 1) * block size | ||
| 54 | */ | 85 | */ |
| 55 | STATIC uint | 86 | STATIC uint |
| 56 | xfs_calc_write_reservation(xfs_mount_t *mp) | 87 | xfs_calc_write_reservation( |
| 88 | struct xfs_mount *mp) | ||
| 57 | { | 89 | { |
| 58 | return XFS_CALC_WRITE_LOG_RES(mp) + XFS_DQUOT_LOGRES(mp); | 90 | return XFS_DQUOT_LOGRES(mp) + |
| 91 | MAX((mp->m_sb.sb_inodesize + | ||
| 92 | XFS_FSB_TO_B(mp, XFS_BM_MAXLEVELS(mp, XFS_DATA_FORK)) + | ||
| 93 | 2 * mp->m_sb.sb_sectsize + | ||
| 94 | mp->m_sb.sb_sectsize + | ||
| 95 | XFS_ALLOCFREE_LOG_RES(mp, 2) + | ||
| 96 | 128 * (4 + XFS_BM_MAXLEVELS(mp, XFS_DATA_FORK) + | ||
| 97 | XFS_ALLOCFREE_LOG_COUNT(mp, 2))), | ||
| 98 | (2 * mp->m_sb.sb_sectsize + | ||
| 99 | 2 * mp->m_sb.sb_sectsize + | ||
| 100 | mp->m_sb.sb_sectsize + | ||
| 101 | XFS_ALLOCFREE_LOG_RES(mp, 2) + | ||
| 102 | 128 * (5 + XFS_ALLOCFREE_LOG_COUNT(mp, 2)))); | ||
| 59 | } | 103 | } |
| 60 | 104 | ||
| 105 | /* | ||
| 106 | * In truncating a file we free up to two extents at once. We can modify: | ||
| 107 | * the inode being truncated: inode size | ||
| 108 | * the inode's bmap btree: (max depth + 1) * block size | ||
| 109 | * And the bmap_finish transaction can free the blocks and bmap blocks: | ||
| 110 | * the agf for each of the ags: 4 * sector size | ||
| 111 | * the agfl for each of the ags: 4 * sector size | ||
| 112 | * the super block to reflect the freed blocks: sector size | ||
| 113 | * worst case split in allocation btrees per extent assuming 4 extents: | ||
| 114 | * 4 exts * 2 trees * (2 * max depth - 1) * block size | ||
| 115 | * the inode btree: max depth * blocksize | ||
| 116 | * the allocation btrees: 2 trees * (max depth - 1) * block size | ||
| 117 | */ | ||
| 61 | STATIC uint | 118 | STATIC uint |
| 62 | xfs_calc_itruncate_reservation(xfs_mount_t *mp) | 119 | xfs_calc_itruncate_reservation( |
| 120 | struct xfs_mount *mp) | ||
| 63 | { | 121 | { |
| 64 | return XFS_CALC_ITRUNCATE_LOG_RES(mp) + XFS_DQUOT_LOGRES(mp); | 122 | return XFS_DQUOT_LOGRES(mp) + |
| 123 | MAX((mp->m_sb.sb_inodesize + | ||
| 124 | XFS_FSB_TO_B(mp, XFS_BM_MAXLEVELS(mp, XFS_DATA_FORK) + 1) + | ||
| 125 | 128 * (2 + XFS_BM_MAXLEVELS(mp, XFS_DATA_FORK))), | ||
| 126 | (4 * mp->m_sb.sb_sectsize + | ||
| 127 | 4 * mp->m_sb.sb_sectsize + | ||
| 128 | mp->m_sb.sb_sectsize + | ||
| 129 | XFS_ALLOCFREE_LOG_RES(mp, 4) + | ||
| 130 | 128 * (9 + XFS_ALLOCFREE_LOG_COUNT(mp, 4)) + | ||
| 131 | 128 * 5 + | ||
| 132 | XFS_ALLOCFREE_LOG_RES(mp, 1) + | ||
| 133 | 128 * (2 + XFS_IALLOC_BLOCKS(mp) + mp->m_in_maxlevels + | ||
| 134 | XFS_ALLOCFREE_LOG_COUNT(mp, 1)))); | ||
| 65 | } | 135 | } |
| 66 | 136 | ||
| 137 | /* | ||
| 138 | * In renaming a files we can modify: | ||
| 139 | * the four inodes involved: 4 * inode size | ||
| 140 | * the two directory btrees: 2 * (max depth + v2) * dir block size | ||
| 141 | * the two directory bmap btrees: 2 * max depth * block size | ||
| 142 | * And the bmap_finish transaction can free dir and bmap blocks (two sets | ||
| 143 | * of bmap blocks) giving: | ||
| 144 | * the agf for the ags in which the blocks live: 3 * sector size | ||
| 145 | * the agfl for the ags in which the blocks live: 3 * sector size | ||
| 146 | * the superblock for the free block count: sector size | ||
| 147 | * the allocation btrees: 3 exts * 2 trees * (2 * max depth - 1) * block size | ||
| 148 | */ | ||
| 67 | STATIC uint | 149 | STATIC uint |
| 68 | xfs_calc_rename_reservation(xfs_mount_t *mp) | 150 | xfs_calc_rename_reservation( |
| 151 | struct xfs_mount *mp) | ||
| 69 | { | 152 | { |
| 70 | return XFS_CALC_RENAME_LOG_RES(mp) + XFS_DQUOT_LOGRES(mp); | 153 | return XFS_DQUOT_LOGRES(mp) + |
| 154 | MAX((4 * mp->m_sb.sb_inodesize + | ||
| 155 | 2 * XFS_DIROP_LOG_RES(mp) + | ||
| 156 | 128 * (4 + 2 * XFS_DIROP_LOG_COUNT(mp))), | ||
| 157 | (3 * mp->m_sb.sb_sectsize + | ||
| 158 | 3 * mp->m_sb.sb_sectsize + | ||
| 159 | mp->m_sb.sb_sectsize + | ||
| 160 | XFS_ALLOCFREE_LOG_RES(mp, 3) + | ||
| 161 | 128 * (7 + XFS_ALLOCFREE_LOG_COUNT(mp, 3)))); | ||
| 71 | } | 162 | } |
| 72 | 163 | ||
| 164 | /* | ||
| 165 | * For creating a link to an inode: | ||
| 166 | * the parent directory inode: inode size | ||
| 167 | * the linked inode: inode size | ||
| 168 | * the directory btree could split: (max depth + v2) * dir block size | ||
| 169 | * the directory bmap btree could join or split: (max depth + v2) * blocksize | ||
| 170 | * And the bmap_finish transaction can free some bmap blocks giving: | ||
| 171 | * the agf for the ag in which the blocks live: sector size | ||
| 172 | * the agfl for the ag in which the blocks live: sector size | ||
| 173 | * the superblock for the free block count: sector size | ||
| 174 | * the allocation btrees: 2 trees * (2 * max depth - 1) * block size | ||
| 175 | */ | ||
| 73 | STATIC uint | 176 | STATIC uint |
| 74 | xfs_calc_link_reservation(xfs_mount_t *mp) | 177 | xfs_calc_link_reservation( |
| 178 | struct xfs_mount *mp) | ||
| 75 | { | 179 | { |
| 76 | return XFS_CALC_LINK_LOG_RES(mp) + XFS_DQUOT_LOGRES(mp); | 180 | return XFS_DQUOT_LOGRES(mp) + |
| 181 | MAX((mp->m_sb.sb_inodesize + | ||
| 182 | mp->m_sb.sb_inodesize + | ||
| 183 | XFS_DIROP_LOG_RES(mp) + | ||
| 184 | 128 * (2 + XFS_DIROP_LOG_COUNT(mp))), | ||
| 185 | (mp->m_sb.sb_sectsize + | ||
| 186 | mp->m_sb.sb_sectsize + | ||
| 187 | mp->m_sb.sb_sectsize + | ||
| 188 | XFS_ALLOCFREE_LOG_RES(mp, 1) + | ||
| 189 | 128 * (3 + XFS_ALLOCFREE_LOG_COUNT(mp, 1)))); | ||
| 77 | } | 190 | } |
| 78 | 191 | ||
| 192 | /* | ||
| 193 | * For removing a directory entry we can modify: | ||
| 194 | * the parent directory inode: inode size | ||
| 195 | * the removed inode: inode size | ||
| 196 | * the directory btree could join: (max depth + v2) * dir block size | ||
| 197 | * the directory bmap btree could join or split: (max depth + v2) * blocksize | ||
| 198 | * And the bmap_finish transaction can free the dir and bmap blocks giving: | ||
| 199 | * the agf for the ag in which the blocks live: 2 * sector size | ||
| 200 | * the agfl for the ag in which the blocks live: 2 * sector size | ||
| 201 | * the superblock for the free block count: sector size | ||
| 202 | * the allocation btrees: 2 exts * 2 trees * (2 * max depth - 1) * block size | ||
| 203 | */ | ||
| 79 | STATIC uint | 204 | STATIC uint |
| 80 | xfs_calc_remove_reservation(xfs_mount_t *mp) | 205 | xfs_calc_remove_reservation( |
| 206 | struct xfs_mount *mp) | ||
| 81 | { | 207 | { |
| 82 | return XFS_CALC_REMOVE_LOG_RES(mp) + XFS_DQUOT_LOGRES(mp); | 208 | return XFS_DQUOT_LOGRES(mp) + |
| 209 | MAX((mp->m_sb.sb_inodesize + | ||
| 210 | mp->m_sb.sb_inodesize + | ||
| 211 | XFS_DIROP_LOG_RES(mp) + | ||
| 212 | 128 * (2 + XFS_DIROP_LOG_COUNT(mp))), | ||
| 213 | (2 * mp->m_sb.sb_sectsize + | ||
| 214 | 2 * mp->m_sb.sb_sectsize + | ||
| 215 | mp->m_sb.sb_sectsize + | ||
| 216 | XFS_ALLOCFREE_LOG_RES(mp, 2) + | ||
| 217 | 128 * (5 + XFS_ALLOCFREE_LOG_COUNT(mp, 2)))); | ||
| 83 | } | 218 | } |
| 84 | 219 | ||
| 220 | /* | ||
| 221 | * For symlink we can modify: | ||
| 222 | * the parent directory inode: inode size | ||
| 223 | * the new inode: inode size | ||
| 224 | * the inode btree entry: 1 block | ||
| 225 | * the directory btree: (max depth + v2) * dir block size | ||
| 226 | * the directory inode's bmap btree: (max depth + v2) * block size | ||
| 227 | * the blocks for the symlink: 1 kB | ||
| 228 | * Or in the first xact we allocate some inodes giving: | ||
| 229 | * the agi and agf of the ag getting the new inodes: 2 * sectorsize | ||
| 230 | * the inode blocks allocated: XFS_IALLOC_BLOCKS * blocksize | ||
| 231 | * the inode btree: max depth * blocksize | ||
| 232 | * the allocation btrees: 2 trees * (2 * max depth - 1) * block size | ||
| 233 | */ | ||
| 85 | STATIC uint | 234 | STATIC uint |
| 86 | xfs_calc_symlink_reservation(xfs_mount_t *mp) | 235 | xfs_calc_symlink_reservation( |
| 236 | struct xfs_mount *mp) | ||
| 87 | { | 237 | { |
| 88 | return XFS_CALC_SYMLINK_LOG_RES(mp) + XFS_DQUOT_LOGRES(mp); | 238 | return XFS_DQUOT_LOGRES(mp) + |
| 239 | MAX((mp->m_sb.sb_inodesize + | ||
| 240 | mp->m_sb.sb_inodesize + | ||
| 241 | XFS_FSB_TO_B(mp, 1) + | ||
| 242 | XFS_DIROP_LOG_RES(mp) + | ||
| 243 | 1024 + | ||
| 244 | 128 * (4 + XFS_DIROP_LOG_COUNT(mp))), | ||
| 245 | (2 * mp->m_sb.sb_sectsize + | ||
| 246 | XFS_FSB_TO_B(mp, XFS_IALLOC_BLOCKS(mp)) + | ||
| 247 | XFS_FSB_TO_B(mp, mp->m_in_maxlevels) + | ||
| 248 | XFS_ALLOCFREE_LOG_RES(mp, 1) + | ||
| 249 | 128 * (2 + XFS_IALLOC_BLOCKS(mp) + mp->m_in_maxlevels + | ||
| 250 | XFS_ALLOCFREE_LOG_COUNT(mp, 1)))); | ||
| 89 | } | 251 | } |
| 90 | 252 | ||
| 253 | /* | ||
| 254 | * For create we can modify: | ||
| 255 | * the parent directory inode: inode size | ||
| 256 | * the new inode: inode size | ||
| 257 | * the inode btree entry: block size | ||
| 258 | * the superblock for the nlink flag: sector size | ||
| 259 | * the directory btree: (max depth + v2) * dir block size | ||
| 260 | * the directory inode's bmap btree: (max depth + v2) * block size | ||
| 261 | * Or in the first xact we allocate some inodes giving: | ||
| 262 | * the agi and agf of the ag getting the new inodes: 2 * sectorsize | ||
| 263 | * the superblock for the nlink flag: sector size | ||
| 264 | * the inode blocks allocated: XFS_IALLOC_BLOCKS * blocksize | ||
| 265 | * the inode btree: max depth * blocksize | ||
| 266 | * the allocation btrees: 2 trees * (max depth - 1) * block size | ||
| 267 | */ | ||
| 91 | STATIC uint | 268 | STATIC uint |
| 92 | xfs_calc_create_reservation(xfs_mount_t *mp) | 269 | xfs_calc_create_reservation( |
| 270 | struct xfs_mount *mp) | ||
| 93 | { | 271 | { |
| 94 | return XFS_CALC_CREATE_LOG_RES(mp) + XFS_DQUOT_LOGRES(mp); | 272 | return XFS_DQUOT_LOGRES(mp) + |
| 273 | MAX((mp->m_sb.sb_inodesize + | ||
| 274 | mp->m_sb.sb_inodesize + | ||
| 275 | mp->m_sb.sb_sectsize + | ||
| 276 | XFS_FSB_TO_B(mp, 1) + | ||
| 277 | XFS_DIROP_LOG_RES(mp) + | ||
| 278 | 128 * (3 + XFS_DIROP_LOG_COUNT(mp))), | ||
| 279 | (3 * mp->m_sb.sb_sectsize + | ||
| 280 | XFS_FSB_TO_B(mp, XFS_IALLOC_BLOCKS(mp)) + | ||
| 281 | XFS_FSB_TO_B(mp, mp->m_in_maxlevels) + | ||
| 282 | XFS_ALLOCFREE_LOG_RES(mp, 1) + | ||
| 283 | 128 * (2 + XFS_IALLOC_BLOCKS(mp) + mp->m_in_maxlevels + | ||
| 284 | XFS_ALLOCFREE_LOG_COUNT(mp, 1)))); | ||
| 95 | } | 285 | } |
| 96 | 286 | ||
| 287 | /* | ||
| 288 | * Making a new directory is the same as creating a new file. | ||
| 289 | */ | ||
| 97 | STATIC uint | 290 | STATIC uint |
| 98 | xfs_calc_mkdir_reservation(xfs_mount_t *mp) | 291 | xfs_calc_mkdir_reservation( |
| 292 | struct xfs_mount *mp) | ||
| 99 | { | 293 | { |
| 100 | return XFS_CALC_MKDIR_LOG_RES(mp) + XFS_DQUOT_LOGRES(mp); | 294 | return xfs_calc_create_reservation(mp); |
| 101 | } | 295 | } |
| 102 | 296 | ||
| 297 | /* | ||
| 298 | * In freeing an inode we can modify: | ||
| 299 | * the inode being freed: inode size | ||
| 300 | * the super block free inode counter: sector size | ||
| 301 | * the agi hash list and counters: sector size | ||
| 302 | * the inode btree entry: block size | ||
| 303 | * the on disk inode before ours in the agi hash list: inode cluster size | ||
| 304 | * the inode btree: max depth * blocksize | ||
| 305 | * the allocation btrees: 2 trees * (max depth - 1) * block size | ||
| 306 | */ | ||
| 103 | STATIC uint | 307 | STATIC uint |
| 104 | xfs_calc_ifree_reservation(xfs_mount_t *mp) | 308 | xfs_calc_ifree_reservation( |
| 309 | struct xfs_mount *mp) | ||
| 105 | { | 310 | { |
| 106 | return XFS_CALC_IFREE_LOG_RES(mp) + XFS_DQUOT_LOGRES(mp); | 311 | return XFS_DQUOT_LOGRES(mp) + |
| 312 | mp->m_sb.sb_inodesize + | ||
| 313 | mp->m_sb.sb_sectsize + | ||
| 314 | mp->m_sb.sb_sectsize + | ||
| 315 | XFS_FSB_TO_B(mp, 1) + | ||
| 316 | MAX((__uint16_t)XFS_FSB_TO_B(mp, 1), | ||
| 317 | XFS_INODE_CLUSTER_SIZE(mp)) + | ||
| 318 | 128 * 5 + | ||
| 319 | XFS_ALLOCFREE_LOG_RES(mp, 1) + | ||
| 320 | 128 * (2 + XFS_IALLOC_BLOCKS(mp) + mp->m_in_maxlevels + | ||
| 321 | XFS_ALLOCFREE_LOG_COUNT(mp, 1)); | ||
| 107 | } | 322 | } |
| 108 | 323 | ||
| 324 | /* | ||
| 325 | * When only changing the inode we log the inode and possibly the superblock | ||
| 326 | * We also add a bit of slop for the transaction stuff. | ||
| 327 | */ | ||
| 109 | STATIC uint | 328 | STATIC uint |
| 110 | xfs_calc_ichange_reservation(xfs_mount_t *mp) | 329 | xfs_calc_ichange_reservation( |
| 330 | struct xfs_mount *mp) | ||
| 111 | { | 331 | { |
| 112 | return XFS_CALC_ICHANGE_LOG_RES(mp) + XFS_DQUOT_LOGRES(mp); | 332 | return XFS_DQUOT_LOGRES(mp) + |
| 333 | mp->m_sb.sb_inodesize + | ||
| 334 | mp->m_sb.sb_sectsize + | ||
| 335 | 512; | ||
| 336 | |||
| 113 | } | 337 | } |
| 114 | 338 | ||
| 339 | /* | ||
| 340 | * Growing the data section of the filesystem. | ||
| 341 | * superblock | ||
| 342 | * agi and agf | ||
| 343 | * allocation btrees | ||
| 344 | */ | ||
| 115 | STATIC uint | 345 | STATIC uint |
| 116 | xfs_calc_growdata_reservation(xfs_mount_t *mp) | 346 | xfs_calc_growdata_reservation( |
| 347 | struct xfs_mount *mp) | ||
| 117 | { | 348 | { |
| 118 | return XFS_CALC_GROWDATA_LOG_RES(mp); | 349 | return mp->m_sb.sb_sectsize * 3 + |
| 350 | XFS_ALLOCFREE_LOG_RES(mp, 1) + | ||
| 351 | 128 * (3 + XFS_ALLOCFREE_LOG_COUNT(mp, 1)); | ||
| 119 | } | 352 | } |
| 120 | 353 | ||
| 354 | /* | ||
| 355 | * Growing the rt section of the filesystem. | ||
| 356 | * In the first set of transactions (ALLOC) we allocate space to the | ||
| 357 | * bitmap or summary files. | ||
| 358 | * superblock: sector size | ||
| 359 | * agf of the ag from which the extent is allocated: sector size | ||
| 360 | * bmap btree for bitmap/summary inode: max depth * blocksize | ||
| 361 | * bitmap/summary inode: inode size | ||
| 362 | * allocation btrees for 1 block alloc: 2 * (2 * maxdepth - 1) * blocksize | ||
| 363 | */ | ||
| 121 | STATIC uint | 364 | STATIC uint |
| 122 | xfs_calc_growrtalloc_reservation(xfs_mount_t *mp) | 365 | xfs_calc_growrtalloc_reservation( |
| 366 | struct xfs_mount *mp) | ||
| 123 | { | 367 | { |
| 124 | return XFS_CALC_GROWRTALLOC_LOG_RES(mp); | 368 | return 2 * mp->m_sb.sb_sectsize + |
| 369 | XFS_FSB_TO_B(mp, XFS_BM_MAXLEVELS(mp, XFS_DATA_FORK)) + | ||
| 370 | mp->m_sb.sb_inodesize + | ||
| 371 | XFS_ALLOCFREE_LOG_RES(mp, 1) + | ||
| 372 | 128 * (3 + XFS_BM_MAXLEVELS(mp, XFS_DATA_FORK) + | ||
| 373 | XFS_ALLOCFREE_LOG_COUNT(mp, 1)); | ||
| 125 | } | 374 | } |
| 126 | 375 | ||
| 376 | /* | ||
| 377 | * Growing the rt section of the filesystem. | ||
| 378 | * In the second set of transactions (ZERO) we zero the new metadata blocks. | ||
| 379 | * one bitmap/summary block: blocksize | ||
| 380 | */ | ||
| 127 | STATIC uint | 381 | STATIC uint |
| 128 | xfs_calc_growrtzero_reservation(xfs_mount_t *mp) | 382 | xfs_calc_growrtzero_reservation( |
| 383 | struct xfs_mount *mp) | ||
| 129 | { | 384 | { |
| 130 | return XFS_CALC_GROWRTZERO_LOG_RES(mp); | 385 | return mp->m_sb.sb_blocksize + 128; |
| 131 | } | 386 | } |
| 132 | 387 | ||
| 388 | /* | ||
| 389 | * Growing the rt section of the filesystem. | ||
| 390 | * In the third set of transactions (FREE) we update metadata without | ||
| 391 | * allocating any new blocks. | ||
| 392 | * superblock: sector size | ||
| 393 | * bitmap inode: inode size | ||
| 394 | * summary inode: inode size | ||
| 395 | * one bitmap block: blocksize | ||
| 396 | * summary blocks: new summary size | ||
| 397 | */ | ||
| 133 | STATIC uint | 398 | STATIC uint |
| 134 | xfs_calc_growrtfree_reservation(xfs_mount_t *mp) | 399 | xfs_calc_growrtfree_reservation( |
| 400 | struct xfs_mount *mp) | ||
| 135 | { | 401 | { |
| 136 | return XFS_CALC_GROWRTFREE_LOG_RES(mp); | 402 | return mp->m_sb.sb_sectsize + |
| 403 | 2 * mp->m_sb.sb_inodesize + | ||
| 404 | mp->m_sb.sb_blocksize + | ||
| 405 | mp->m_rsumsize + | ||
| 406 | 128 * 5; | ||
| 137 | } | 407 | } |
| 138 | 408 | ||
| 409 | /* | ||
| 410 | * Logging the inode modification timestamp on a synchronous write. | ||
| 411 | * inode | ||
| 412 | */ | ||
| 139 | STATIC uint | 413 | STATIC uint |
| 140 | xfs_calc_swrite_reservation(xfs_mount_t *mp) | 414 | xfs_calc_swrite_reservation( |
| 415 | struct xfs_mount *mp) | ||
| 141 | { | 416 | { |
| 142 | return XFS_CALC_SWRITE_LOG_RES(mp); | 417 | return mp->m_sb.sb_inodesize + 128; |
| 143 | } | 418 | } |
| 144 | 419 | ||
| 420 | /* | ||
| 421 | * Logging the inode mode bits when writing a setuid/setgid file | ||
| 422 | * inode | ||
| 423 | */ | ||
| 145 | STATIC uint | 424 | STATIC uint |
| 146 | xfs_calc_writeid_reservation(xfs_mount_t *mp) | 425 | xfs_calc_writeid_reservation(xfs_mount_t *mp) |
| 147 | { | 426 | { |
| 148 | return XFS_CALC_WRITEID_LOG_RES(mp); | 427 | return mp->m_sb.sb_inodesize + 128; |
| 149 | } | 428 | } |
| 150 | 429 | ||
| 430 | /* | ||
| 431 | * Converting the inode from non-attributed to attributed. | ||
| 432 | * the inode being converted: inode size | ||
| 433 | * agf block and superblock (for block allocation) | ||
| 434 | * the new block (directory sized) | ||
| 435 | * bmap blocks for the new directory block | ||
| 436 | * allocation btrees | ||
| 437 | */ | ||
| 151 | STATIC uint | 438 | STATIC uint |
| 152 | xfs_calc_addafork_reservation(xfs_mount_t *mp) | 439 | xfs_calc_addafork_reservation( |
| 440 | struct xfs_mount *mp) | ||
| 153 | { | 441 | { |
| 154 | return XFS_CALC_ADDAFORK_LOG_RES(mp) + XFS_DQUOT_LOGRES(mp); | 442 | return XFS_DQUOT_LOGRES(mp) + |
| 443 | mp->m_sb.sb_inodesize + | ||
| 444 | mp->m_sb.sb_sectsize * 2 + | ||
| 445 | mp->m_dirblksize + | ||
| 446 | XFS_FSB_TO_B(mp, XFS_DAENTER_BMAP1B(mp, XFS_DATA_FORK) + 1) + | ||
| 447 | XFS_ALLOCFREE_LOG_RES(mp, 1) + | ||
| 448 | 128 * (4 + XFS_DAENTER_BMAP1B(mp, XFS_DATA_FORK) + 1 + | ||
| 449 | XFS_ALLOCFREE_LOG_COUNT(mp, 1)); | ||
| 155 | } | 450 | } |
| 156 | 451 | ||
| 452 | /* | ||
| 453 | * Removing the attribute fork of a file | ||
| 454 | * the inode being truncated: inode size | ||
| 455 | * the inode's bmap btree: max depth * block size | ||
| 456 | * And the bmap_finish transaction can free the blocks and bmap blocks: | ||
| 457 | * the agf for each of the ags: 4 * sector size | ||
| 458 | * the agfl for each of the ags: 4 * sector size | ||
| 459 | * the super block to reflect the freed blocks: sector size | ||
| 460 | * worst case split in allocation btrees per extent assuming 4 extents: | ||
| 461 | * 4 exts * 2 trees * (2 * max depth - 1) * block size | ||
| 462 | */ | ||
| 157 | STATIC uint | 463 | STATIC uint |
| 158 | xfs_calc_attrinval_reservation(xfs_mount_t *mp) | 464 | xfs_calc_attrinval_reservation( |
| 465 | struct xfs_mount *mp) | ||
| 159 | { | 466 | { |
| 160 | return XFS_CALC_ATTRINVAL_LOG_RES(mp); | 467 | return MAX((mp->m_sb.sb_inodesize + |
| 468 | XFS_FSB_TO_B(mp, XFS_BM_MAXLEVELS(mp, XFS_ATTR_FORK)) + | ||
| 469 | 128 * (1 + XFS_BM_MAXLEVELS(mp, XFS_ATTR_FORK))), | ||
| 470 | (4 * mp->m_sb.sb_sectsize + | ||
| 471 | 4 * mp->m_sb.sb_sectsize + | ||
| 472 | mp->m_sb.sb_sectsize + | ||
| 473 | XFS_ALLOCFREE_LOG_RES(mp, 4) + | ||
| 474 | 128 * (9 + XFS_ALLOCFREE_LOG_COUNT(mp, 4)))); | ||
| 161 | } | 475 | } |
| 162 | 476 | ||
| 477 | /* | ||
| 478 | * Setting an attribute. | ||
| 479 | * the inode getting the attribute | ||
| 480 | * the superblock for allocations | ||
| 481 | * the agfs extents are allocated from | ||
| 482 | * the attribute btree * max depth | ||
| 483 | * the inode allocation btree | ||
| 484 | * Since attribute transaction space is dependent on the size of the attribute, | ||
| 485 | * the calculation is done partially at mount time and partially at runtime. | ||
| 486 | */ | ||
| 163 | STATIC uint | 487 | STATIC uint |
| 164 | xfs_calc_attrset_reservation(xfs_mount_t *mp) | 488 | xfs_calc_attrset_reservation( |
| 489 | struct xfs_mount *mp) | ||
| 165 | { | 490 | { |
| 166 | return XFS_CALC_ATTRSET_LOG_RES(mp) + XFS_DQUOT_LOGRES(mp); | 491 | return XFS_DQUOT_LOGRES(mp) + |
| 492 | mp->m_sb.sb_inodesize + | ||
| 493 | mp->m_sb.sb_sectsize + | ||
| 494 | XFS_FSB_TO_B(mp, XFS_DA_NODE_MAXDEPTH) + | ||
| 495 | 128 * (2 + XFS_DA_NODE_MAXDEPTH); | ||
| 167 | } | 496 | } |
| 168 | 497 | ||
| 498 | /* | ||
| 499 | * Removing an attribute. | ||
| 500 | * the inode: inode size | ||
| 501 | * the attribute btree could join: max depth * block size | ||
| 502 | * the inode bmap btree could join or split: max depth * block size | ||
| 503 | * And the bmap_finish transaction can free the attr blocks freed giving: | ||
| 504 | * the agf for the ag in which the blocks live: 2 * sector size | ||
| 505 | * the agfl for the ag in which the blocks live: 2 * sector size | ||
| 506 | * the superblock for the free block count: sector size | ||
| 507 | * the allocation btrees: 2 exts * 2 trees * (2 * max depth - 1) * block size | ||
| 508 | */ | ||
| 169 | STATIC uint | 509 | STATIC uint |
| 170 | xfs_calc_attrrm_reservation(xfs_mount_t *mp) | 510 | xfs_calc_attrrm_reservation( |
| 511 | struct xfs_mount *mp) | ||
| 171 | { | 512 | { |
| 172 | return XFS_CALC_ATTRRM_LOG_RES(mp) + XFS_DQUOT_LOGRES(mp); | 513 | return XFS_DQUOT_LOGRES(mp) + |
| 514 | MAX((mp->m_sb.sb_inodesize + | ||
| 515 | XFS_FSB_TO_B(mp, XFS_DA_NODE_MAXDEPTH) + | ||
| 516 | XFS_FSB_TO_B(mp, XFS_BM_MAXLEVELS(mp, XFS_ATTR_FORK)) + | ||
| 517 | 128 * (1 + XFS_DA_NODE_MAXDEPTH + | ||
| 518 | XFS_BM_MAXLEVELS(mp, XFS_DATA_FORK))), | ||
| 519 | (2 * mp->m_sb.sb_sectsize + | ||
| 520 | 2 * mp->m_sb.sb_sectsize + | ||
| 521 | mp->m_sb.sb_sectsize + | ||
| 522 | XFS_ALLOCFREE_LOG_RES(mp, 2) + | ||
| 523 | 128 * (5 + XFS_ALLOCFREE_LOG_COUNT(mp, 2)))); | ||
| 173 | } | 524 | } |
| 174 | 525 | ||
| 526 | /* | ||
| 527 | * Clearing a bad agino number in an agi hash bucket. | ||
| 528 | */ | ||
| 175 | STATIC uint | 529 | STATIC uint |
| 176 | xfs_calc_clear_agi_bucket_reservation(xfs_mount_t *mp) | 530 | xfs_calc_clear_agi_bucket_reservation( |
| 531 | struct xfs_mount *mp) | ||
| 177 | { | 532 | { |
| 178 | return XFS_CALC_CLEAR_AGI_BUCKET_LOG_RES(mp); | 533 | return mp->m_sb.sb_sectsize + 128; |
| 179 | } | 534 | } |
| 180 | 535 | ||
| 181 | /* | 536 | /* |
| @@ -184,11 +539,10 @@ xfs_calc_clear_agi_bucket_reservation(xfs_mount_t *mp) | |||
| 184 | */ | 539 | */ |
| 185 | void | 540 | void |
| 186 | xfs_trans_init( | 541 | xfs_trans_init( |
| 187 | xfs_mount_t *mp) | 542 | struct xfs_mount *mp) |
| 188 | { | 543 | { |
| 189 | xfs_trans_reservations_t *resp; | 544 | struct xfs_trans_reservations *resp = &mp->m_reservations; |
| 190 | 545 | ||
| 191 | resp = &(mp->m_reservations); | ||
| 192 | resp->tr_write = xfs_calc_write_reservation(mp); | 546 | resp->tr_write = xfs_calc_write_reservation(mp); |
| 193 | resp->tr_itruncate = xfs_calc_itruncate_reservation(mp); | 547 | resp->tr_itruncate = xfs_calc_itruncate_reservation(mp); |
| 194 | resp->tr_rename = xfs_calc_rename_reservation(mp); | 548 | resp->tr_rename = xfs_calc_rename_reservation(mp); |
