blob: 33dee52d27d1884fefad1c3e9d8831dd06a7e453 [file] [log] [blame]
/*
* Copyright (c) 2000-2001,2004-2005 Silicon Graphics, Inc.
* All Rights Reserved.
*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU General Public License as
* published by the Free Software Foundation.
*
* 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.
*
* 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
*/
/*
* maxtrres.c
*
* Compute the maximum transaction reservation for a legal combination
* of sector size, block size, inode size, directory version, and
* directory block size.
*/
#include <xfs/libxfs.h>
#include "xfs_mkfs.h"
static void
max_attrset_trans_res_adjust(
xfs_mount_t *mp)
{
int local;
int size;
int nblks;
int res;
/*
* Determine space the maximal sized attribute will use,
* to calculate the largest reservation size needed.
*/
size = libxfs_attr_leaf_newentsize(MAXNAMELEN, 64 * 1024,
mp->m_sb.sb_blocksize, &local);
ASSERT(!local);
nblks = XFS_DAENTER_SPACE_RES(mp, XFS_ATTR_FORK);
nblks += XFS_B_TO_FSB(mp, size);
nblks += XFS_NEXTENTADD_SPACE_RES(mp, size, XFS_ATTR_FORK);
res = XFS_ATTRSET_LOG_RES(mp, nblks);
#if 0
printf("size = %d nblks = %d res = %d\n", size, nblks, res);
#endif
mp->m_reservations.tr_attrset = res;
}
static int
max_trans_res_by_mount(
xfs_mount_t *mp)
{
uint *p;
int rval;
xfs_trans_reservations_t *tr = &mp->m_reservations;
for (rval = 0, p = (uint *)tr; p < (uint *)(tr + 1); p++) {
if ((int)*p > rval)
rval = (int)*p;
}
return rval;
}
int
max_trans_res(
int dirversion,
int sectorlog,
int blocklog,
int inodelog,
int dirblocklog)
{
xfs_sb_t *sbp;
xfs_mount_t mount;
int maxres, maxfsb;
memset(&mount, 0, sizeof(mount));
sbp = &mount.m_sb;
sbp->sb_magicnum = XFS_SB_MAGIC;
sbp->sb_sectlog = sectorlog;
sbp->sb_sectsize = 1 << sbp->sb_sectlog;
sbp->sb_blocklog = blocklog;
sbp->sb_blocksize = 1 << blocklog;
sbp->sb_agblocks = XFS_AG_MIN_BYTES / (1 << blocklog);
sbp->sb_inodelog = inodelog;
sbp->sb_inopblog = blocklog - inodelog;
sbp->sb_inodesize = 1 << inodelog;
sbp->sb_inopblock = 1 << (blocklog - inodelog);
sbp->sb_dirblklog = dirblocklog - blocklog;
sbp->sb_versionnum = XFS_SB_VERSION_4 |
(dirversion == 2 ? XFS_SB_VERSION_DIRV2BIT : 0);
libxfs_mount(&mount, sbp, 0,0,0,0);
max_attrset_trans_res_adjust(&mount);
maxres = max_trans_res_by_mount(&mount);
maxfsb = XFS_B_TO_FSB(&mount, maxres);
libxfs_umount(&mount);
#if 0
printf("#define\tMAXTRRES_S%d_B%d_I%d_D%d_V%d\t%lld\n",
sectorlog, blocklog, inodelog, dirblocklog, dirversion, maxfsb);
#endif
return maxfsb;
}