|  | // SPDX-License-Identifier: GPL-2.0 | 
|  | /* | 
|  | * Copyright (c) 2000-2005 Silicon Graphics, Inc. | 
|  | * All Rights Reserved. | 
|  | */ | 
|  |  | 
|  | #include "libxfs_priv.h" | 
|  | #include "libxfs.h" | 
|  | #include "libxfs_io.h" | 
|  | #include "init.h" | 
|  | #include "xfs_fs.h" | 
|  | #include "xfs_shared.h" | 
|  | #include "xfs_format.h" | 
|  | #include "xfs_log_format.h" | 
|  | #include "xfs_trans_resv.h" | 
|  | #include "xfs_mount.h" | 
|  | #include "xfs_defer.h" | 
|  | #include "xfs_inode_buf.h" | 
|  | #include "xfs_inode_fork.h" | 
|  | #include "xfs_inode.h" | 
|  | #include "xfs_trans.h" | 
|  | #include "xfs_bmap.h" | 
|  | #include "xfs_bmap_btree.h" | 
|  | #include "xfs_trans_space.h" | 
|  | #include "xfs_ialloc.h" | 
|  | #include "xfs_alloc.h" | 
|  | #include "xfs_bit.h" | 
|  | #include "xfs_da_format.h" | 
|  | #include "xfs_da_btree.h" | 
|  | #include "xfs_dir2_priv.h" | 
|  |  | 
|  | /* | 
|  | * Calculate the worst case log unit reservation for a given superblock | 
|  | * configuration. Copied and munged from the kernel code, and assumes a | 
|  | * worse case header usage (maximum log buffer sizes) | 
|  | */ | 
|  | int | 
|  | xfs_log_calc_unit_res( | 
|  | struct xfs_mount	*mp, | 
|  | int			unit_bytes) | 
|  | { | 
|  | int			iclog_space; | 
|  | int			iclog_header_size; | 
|  | int			iclog_size; | 
|  | uint			num_headers; | 
|  |  | 
|  | if (xfs_sb_version_haslogv2(&mp->m_sb)) { | 
|  | iclog_size = XLOG_MAX_RECORD_BSIZE; | 
|  | iclog_header_size = BBTOB(iclog_size / XLOG_HEADER_CYCLE_SIZE); | 
|  | } else { | 
|  | iclog_size = XLOG_BIG_RECORD_BSIZE; | 
|  | iclog_header_size = BBSIZE; | 
|  | } | 
|  |  | 
|  | /* | 
|  | * Permanent reservations have up to 'cnt'-1 active log operations | 
|  | * in the log.  A unit in this case is the amount of space for one | 
|  | * of these log operations.  Normal reservations have a cnt of 1 | 
|  | * and their unit amount is the total amount of space required. | 
|  | * | 
|  | * The following lines of code account for non-transaction data | 
|  | * which occupy space in the on-disk log. | 
|  | * | 
|  | * Normal form of a transaction is: | 
|  | * <oph><trans-hdr><start-oph><reg1-oph><reg1><reg2-oph>...<commit-oph> | 
|  | * and then there are LR hdrs, split-recs and roundoff at end of syncs. | 
|  | * | 
|  | * We need to account for all the leadup data and trailer data | 
|  | * around the transaction data. | 
|  | * And then we need to account for the worst case in terms of using | 
|  | * more space. | 
|  | * The worst case will happen if: | 
|  | * - the placement of the transaction happens to be such that the | 
|  | *   roundoff is at its maximum | 
|  | * - the transaction data is synced before the commit record is synced | 
|  | *   i.e. <transaction-data><roundoff> | <commit-rec><roundoff> | 
|  | *   Therefore the commit record is in its own Log Record. | 
|  | *   This can happen as the commit record is called with its | 
|  | *   own region to xlog_write(). | 
|  | *   This then means that in the worst case, roundoff can happen for | 
|  | *   the commit-rec as well. | 
|  | *   The commit-rec is smaller than padding in this scenario and so it is | 
|  | *   not added separately. | 
|  | */ | 
|  |  | 
|  | /* for trans header */ | 
|  | unit_bytes += sizeof(xlog_op_header_t); | 
|  | unit_bytes += sizeof(xfs_trans_header_t); | 
|  |  | 
|  | /* for start-rec */ | 
|  | unit_bytes += sizeof(xlog_op_header_t); | 
|  |  | 
|  | /* | 
|  | * for LR headers - the space for data in an iclog is the size minus | 
|  | * the space used for the headers. If we use the iclog size, then we | 
|  | * undercalculate the number of headers required. | 
|  | * | 
|  | * Furthermore - the addition of op headers for split-recs might | 
|  | * increase the space required enough to require more log and op | 
|  | * headers, so take that into account too. | 
|  | * | 
|  | * IMPORTANT: This reservation makes the assumption that if this | 
|  | * transaction is the first in an iclog and hence has the LR headers | 
|  | * accounted to it, then the remaining space in the iclog is | 
|  | * exclusively for this transaction.  i.e. if the transaction is larger | 
|  | * than the iclog, it will be the only thing in that iclog. | 
|  | * Fundamentally, this means we must pass the entire log vector to | 
|  | * xlog_write to guarantee this. | 
|  | */ | 
|  | iclog_space = iclog_size - iclog_header_size; | 
|  | num_headers = howmany(unit_bytes, iclog_space); | 
|  |  | 
|  | /* for split-recs - ophdrs added when data split over LRs */ | 
|  | unit_bytes += sizeof(xlog_op_header_t) * num_headers; | 
|  |  | 
|  | /* add extra header reservations if we overrun */ | 
|  | while (!num_headers || | 
|  | howmany(unit_bytes, iclog_space) > num_headers) { | 
|  | unit_bytes += sizeof(xlog_op_header_t); | 
|  | num_headers++; | 
|  | } | 
|  | unit_bytes += iclog_header_size * num_headers; | 
|  |  | 
|  | /* for commit-rec LR header - note: padding will subsume the ophdr */ | 
|  | unit_bytes += iclog_header_size; | 
|  |  | 
|  | /* for roundoff padding for transaction data and one for commit record */ | 
|  | if (xfs_sb_version_haslogv2(&mp->m_sb) && mp->m_sb.sb_logsunit > 1) { | 
|  | /* log su roundoff */ | 
|  | unit_bytes += 2 * mp->m_sb.sb_logsunit; | 
|  | } else { | 
|  | /* BB roundoff */ | 
|  | unit_bytes += 2 * BBSIZE; | 
|  | } | 
|  |  | 
|  | return unit_bytes; | 
|  | } | 
|  |  | 
|  | struct timespec64 | 
|  | current_time(struct inode *inode) | 
|  | { | 
|  | struct timespec64	tv; | 
|  | struct timeval		stv; | 
|  |  | 
|  | gettimeofday(&stv, (struct timezone *)0); | 
|  | tv.tv_sec = stv.tv_sec; | 
|  | tv.tv_nsec = stv.tv_usec * 1000; | 
|  |  | 
|  | return tv; | 
|  | } | 
|  |  | 
|  | STATIC uint16_t | 
|  | xfs_flags2diflags( | 
|  | struct xfs_inode	*ip, | 
|  | unsigned int		xflags) | 
|  | { | 
|  | /* can't set PREALLOC this way, just preserve it */ | 
|  | uint16_t		di_flags = | 
|  | (ip->i_d.di_flags & XFS_DIFLAG_PREALLOC); | 
|  |  | 
|  | if (xflags & FS_XFLAG_IMMUTABLE) | 
|  | di_flags |= XFS_DIFLAG_IMMUTABLE; | 
|  | if (xflags & FS_XFLAG_APPEND) | 
|  | di_flags |= XFS_DIFLAG_APPEND; | 
|  | if (xflags & FS_XFLAG_SYNC) | 
|  | di_flags |= XFS_DIFLAG_SYNC; | 
|  | if (xflags & FS_XFLAG_NOATIME) | 
|  | di_flags |= XFS_DIFLAG_NOATIME; | 
|  | if (xflags & FS_XFLAG_NODUMP) | 
|  | di_flags |= XFS_DIFLAG_NODUMP; | 
|  | if (xflags & FS_XFLAG_NODEFRAG) | 
|  | di_flags |= XFS_DIFLAG_NODEFRAG; | 
|  | if (xflags & FS_XFLAG_FILESTREAM) | 
|  | di_flags |= XFS_DIFLAG_FILESTREAM; | 
|  | if (S_ISDIR(VFS_I(ip)->i_mode)) { | 
|  | if (xflags & FS_XFLAG_RTINHERIT) | 
|  | di_flags |= XFS_DIFLAG_RTINHERIT; | 
|  | if (xflags & FS_XFLAG_NOSYMLINKS) | 
|  | di_flags |= XFS_DIFLAG_NOSYMLINKS; | 
|  | if (xflags & FS_XFLAG_EXTSZINHERIT) | 
|  | di_flags |= XFS_DIFLAG_EXTSZINHERIT; | 
|  | if (xflags & FS_XFLAG_PROJINHERIT) | 
|  | di_flags |= XFS_DIFLAG_PROJINHERIT; | 
|  | } else if (S_ISREG(VFS_I(ip)->i_mode)) { | 
|  | if (xflags & FS_XFLAG_REALTIME) | 
|  | di_flags |= XFS_DIFLAG_REALTIME; | 
|  | if (xflags & FS_XFLAG_EXTSIZE) | 
|  | di_flags |= XFS_DIFLAG_EXTSIZE; | 
|  | } | 
|  |  | 
|  | return di_flags; | 
|  | } | 
|  |  | 
|  | STATIC uint64_t | 
|  | xfs_flags2diflags2( | 
|  | struct xfs_inode	*ip, | 
|  | unsigned int		xflags) | 
|  | { | 
|  | uint64_t		di_flags2 = | 
|  | (ip->i_d.di_flags2 & (XFS_DIFLAG2_REFLINK | | 
|  | XFS_DIFLAG2_BIGTIME)); | 
|  |  | 
|  | if (xflags & FS_XFLAG_DAX) | 
|  | di_flags2 |= XFS_DIFLAG2_DAX; | 
|  | if (xflags & FS_XFLAG_COWEXTSIZE) | 
|  | di_flags2 |= XFS_DIFLAG2_COWEXTSIZE; | 
|  |  | 
|  | return di_flags2; | 
|  | } | 
|  |  | 
|  | /* Propagate di_flags from a parent inode to a child inode. */ | 
|  | static void | 
|  | xfs_inode_propagate_flags( | 
|  | struct xfs_inode	*ip, | 
|  | const struct xfs_inode	*pip) | 
|  | { | 
|  | unsigned int		di_flags = 0; | 
|  | umode_t			mode = VFS_I(ip)->i_mode; | 
|  |  | 
|  | if ((mode & S_IFMT) == S_IFDIR) { | 
|  | if (pip->i_d.di_flags & XFS_DIFLAG_RTINHERIT) | 
|  | di_flags |= XFS_DIFLAG_RTINHERIT; | 
|  | if (pip->i_d.di_flags & XFS_DIFLAG_EXTSZINHERIT) { | 
|  | di_flags |= XFS_DIFLAG_EXTSZINHERIT; | 
|  | ip->i_d.di_extsize = pip->i_d.di_extsize; | 
|  | } | 
|  | } else { | 
|  | if ((pip->i_d.di_flags & XFS_DIFLAG_RTINHERIT) && | 
|  | xfs_sb_version_hasrealtime(&ip->i_mount->m_sb)) | 
|  | di_flags |= XFS_DIFLAG_REALTIME; | 
|  | if (pip->i_d.di_flags & XFS_DIFLAG_EXTSZINHERIT) { | 
|  | di_flags |= XFS_DIFLAG_EXTSIZE; | 
|  | ip->i_d.di_extsize = pip->i_d.di_extsize; | 
|  | } | 
|  | } | 
|  | if (pip->i_d.di_flags & XFS_DIFLAG_PROJINHERIT) | 
|  | di_flags |= XFS_DIFLAG_PROJINHERIT; | 
|  | ip->i_d.di_flags |= di_flags; | 
|  | } | 
|  |  | 
|  | /* | 
|  | * Allocate an inode on disk and return a copy of its in-core version. | 
|  | * Set mode, nlink, and rdev appropriately within the inode. | 
|  | * The uid and gid for the inode are set according to the contents of | 
|  | * the given cred structure. | 
|  | * | 
|  | * This was once shared with the kernel, but has diverged to the point | 
|  | * where it's no longer worth the hassle of maintaining common code. | 
|  | */ | 
|  | static int | 
|  | libxfs_ialloc( | 
|  | xfs_trans_t	*tp, | 
|  | xfs_inode_t	*pip, | 
|  | mode_t		mode, | 
|  | nlink_t		nlink, | 
|  | xfs_dev_t	rdev, | 
|  | struct cred	*cr, | 
|  | struct fsxattr	*fsx, | 
|  | xfs_buf_t	**ialloc_context, | 
|  | xfs_inode_t	**ipp) | 
|  | { | 
|  | xfs_ino_t	ino; | 
|  | xfs_inode_t	*ip; | 
|  | uint		flags; | 
|  | int		error; | 
|  |  | 
|  | /* | 
|  | * Call the space management code to pick | 
|  | * the on-disk inode to be allocated. | 
|  | */ | 
|  | error = xfs_dialloc(tp, pip ? pip->i_ino : 0, mode, | 
|  | ialloc_context, &ino); | 
|  | if (error != 0) | 
|  | return error; | 
|  | if (*ialloc_context || ino == NULLFSINO) { | 
|  | *ipp = NULL; | 
|  | return 0; | 
|  | } | 
|  | ASSERT(*ialloc_context == NULL); | 
|  |  | 
|  | error = libxfs_iget(tp->t_mountp, tp, ino, 0, &ip); | 
|  | if (error != 0) | 
|  | return error; | 
|  | ASSERT(ip != NULL); | 
|  |  | 
|  | VFS_I(ip)->i_mode = mode; | 
|  | set_nlink(VFS_I(ip), nlink); | 
|  | i_uid_write(VFS_I(ip), cr->cr_uid); | 
|  | i_gid_write(VFS_I(ip), cr->cr_gid); | 
|  | ip->i_d.di_projid = pip ? 0 : fsx->fsx_projid; | 
|  | xfs_trans_ichgtime(tp, ip, XFS_ICHGTIME_CHG | XFS_ICHGTIME_MOD); | 
|  |  | 
|  | if (pip && (VFS_I(pip)->i_mode & S_ISGID)) { | 
|  | VFS_I(ip)->i_gid = VFS_I(pip)->i_gid; | 
|  | if ((VFS_I(pip)->i_mode & S_ISGID) && (mode & S_IFMT) == S_IFDIR) | 
|  | VFS_I(ip)->i_mode |= S_ISGID; | 
|  | } | 
|  |  | 
|  | ip->i_d.di_size = 0; | 
|  | ip->i_df.if_nextents = 0; | 
|  | ASSERT(ip->i_d.di_nblocks == 0); | 
|  | ip->i_d.di_extsize = pip ? 0 : fsx->fsx_extsize; | 
|  | ip->i_d.di_dmevmask = 0; | 
|  | ip->i_d.di_dmstate = 0; | 
|  | ip->i_d.di_flags = pip ? 0 : xfs_flags2diflags(ip, fsx->fsx_xflags); | 
|  |  | 
|  | if (xfs_sb_version_has_v3inode(&ip->i_mount->m_sb)) { | 
|  | ASSERT(ip->i_d.di_ino == ino); | 
|  | ASSERT(uuid_equal(&ip->i_d.di_uuid, &mp->m_sb.sb_meta_uuid)); | 
|  | VFS_I(ip)->i_version = 1; | 
|  | ip->i_d.di_flags2 = pip ? ip->i_mount->m_ino_geo.new_diflags2 : | 
|  | xfs_flags2diflags2(ip, fsx->fsx_xflags); | 
|  | ip->i_d.di_crtime.tv_sec = (int32_t)VFS_I(ip)->i_mtime.tv_sec; | 
|  | ip->i_d.di_crtime.tv_nsec = (int32_t)VFS_I(ip)->i_mtime.tv_nsec; | 
|  | ip->i_d.di_cowextsize = pip ? 0 : fsx->fsx_cowextsize; | 
|  | } | 
|  |  | 
|  | flags = XFS_ILOG_CORE; | 
|  | switch (mode & S_IFMT) { | 
|  | case S_IFIFO: | 
|  | case S_IFSOCK: | 
|  | /* doesn't make sense to set an rdev for these */ | 
|  | rdev = 0; | 
|  | /* FALLTHROUGH */ | 
|  | case S_IFCHR: | 
|  | case S_IFBLK: | 
|  | ip->i_df.if_format = XFS_DINODE_FMT_DEV; | 
|  | flags |= XFS_ILOG_DEV; | 
|  | VFS_I(ip)->i_rdev = rdev; | 
|  | break; | 
|  | case S_IFREG: | 
|  | case S_IFDIR: | 
|  | if (pip && (pip->i_d.di_flags & XFS_DIFLAG_ANY)) | 
|  | xfs_inode_propagate_flags(ip, pip); | 
|  | /* FALLTHROUGH */ | 
|  | case S_IFLNK: | 
|  | ip->i_df.if_format = XFS_DINODE_FMT_EXTENTS; | 
|  | ip->i_df.if_flags = XFS_IFEXTENTS; | 
|  | ip->i_df.if_bytes = 0; | 
|  | ip->i_df.if_u1.if_root = NULL; | 
|  | break; | 
|  | default: | 
|  | ASSERT(0); | 
|  | } | 
|  |  | 
|  | /* | 
|  | * Log the new values stuffed into the inode. | 
|  | */ | 
|  | xfs_trans_ijoin(tp, ip, 0); | 
|  | xfs_trans_log_inode(tp, ip, flags); | 
|  | *ipp = ip; | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | /* | 
|  | * Writes a modified inode's changes out to the inode's on disk home. | 
|  | * Originally based on xfs_iflush_int() from xfs_inode.c in the kernel. | 
|  | */ | 
|  | int | 
|  | libxfs_iflush_int( | 
|  | xfs_inode_t			*ip, | 
|  | xfs_buf_t			*bp) | 
|  | { | 
|  | struct xfs_inode_log_item	*iip; | 
|  | xfs_dinode_t			*dip; | 
|  | xfs_mount_t			*mp; | 
|  |  | 
|  | ASSERT(ip->i_df.if_format != XFS_DINODE_FMT_BTREE || | 
|  | ip->i_df.if_nextents > ip->i_df.if_ext_max); | 
|  |  | 
|  | iip = ip->i_itemp; | 
|  | mp = ip->i_mount; | 
|  |  | 
|  | /* set *dip = inode's place in the buffer */ | 
|  | dip = xfs_buf_offset(bp, ip->i_imap.im_boffset); | 
|  |  | 
|  | ASSERT(ip->i_d.di_magic == XFS_DINODE_MAGIC); | 
|  | if (XFS_ISREG(ip)) { | 
|  | ASSERT( (ip->i_df.if_format == XFS_DINODE_FMT_EXTENTS) || | 
|  | (ip->i_df.if_format == XFS_DINODE_FMT_BTREE) ); | 
|  | } else if (XFS_ISDIR(ip)) { | 
|  | ASSERT( (ip->i_df.if_format == XFS_DINODE_FMT_EXTENTS) || | 
|  | (ip->i_df.if_format == XFS_DINODE_FMT_BTREE)   || | 
|  | (ip->i_df.if_format == XFS_DINODE_FMT_LOCAL) ); | 
|  | } | 
|  | ASSERT(ip->i_df.if_nextents+ip->i_afp->if_nextents <= ip->i_d.di_nblocks); | 
|  | ASSERT(ip->i_d.di_forkoff <= mp->m_sb.sb_inodesize); | 
|  |  | 
|  | /* bump the change count on v3 inodes */ | 
|  | if (xfs_sb_version_has_v3inode(&mp->m_sb)) | 
|  | VFS_I(ip)->i_version++; | 
|  |  | 
|  | /* | 
|  | * If there are inline format data / attr forks attached to this inode, | 
|  | * make sure they are not corrupt. | 
|  | */ | 
|  | if (ip->i_df.if_format == XFS_DINODE_FMT_LOCAL && | 
|  | xfs_ifork_verify_local_data(ip)) | 
|  | return -EFSCORRUPTED; | 
|  | if (ip->i_afp && ip->i_afp->if_format == XFS_DINODE_FMT_LOCAL && | 
|  | xfs_ifork_verify_local_attr(ip)) | 
|  | return -EFSCORRUPTED; | 
|  |  | 
|  | /* | 
|  | * Copy the dirty parts of the inode into the on-disk | 
|  | * inode.  We always copy out the core of the inode, | 
|  | * because if the inode is dirty at all the core must | 
|  | * be. | 
|  | */ | 
|  | xfs_inode_to_disk(ip, dip, iip->ili_item.li_lsn); | 
|  |  | 
|  | xfs_iflush_fork(ip, dip, iip, XFS_DATA_FORK); | 
|  | if (XFS_IFORK_Q(ip)) | 
|  | xfs_iflush_fork(ip, dip, iip, XFS_ATTR_FORK); | 
|  |  | 
|  | /* generate the checksum. */ | 
|  | xfs_dinode_calc_crc(mp, dip); | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | int | 
|  | libxfs_mod_incore_sb( | 
|  | struct xfs_mount *mp, | 
|  | int		field, | 
|  | int64_t		delta, | 
|  | int		rsvd) | 
|  | { | 
|  | long long	lcounter;	/* long counter for 64 bit fields */ | 
|  |  | 
|  | switch (field) { | 
|  | case XFS_TRANS_SB_FDBLOCKS: | 
|  | lcounter = (long long)mp->m_sb.sb_fdblocks; | 
|  | lcounter += delta; | 
|  | if (lcounter < 0) | 
|  | return -ENOSPC; | 
|  | mp->m_sb.sb_fdblocks = lcounter; | 
|  | return 0; | 
|  | default: | 
|  | ASSERT(0); | 
|  | return -EINVAL; | 
|  | } | 
|  | } | 
|  |  | 
|  | /* | 
|  | * This routine allocates disk space for the given file. | 
|  | * Originally derived from xfs_alloc_file_space(). | 
|  | */ | 
|  | int | 
|  | libxfs_alloc_file_space( | 
|  | xfs_inode_t	*ip, | 
|  | xfs_off_t	offset, | 
|  | xfs_off_t	len, | 
|  | int		alloc_type, | 
|  | int		attr_flags) | 
|  | { | 
|  | xfs_mount_t	*mp; | 
|  | xfs_off_t	count; | 
|  | xfs_filblks_t	datablocks; | 
|  | xfs_filblks_t	allocated_fsb; | 
|  | xfs_filblks_t	allocatesize_fsb; | 
|  | xfs_bmbt_irec_t *imapp; | 
|  | xfs_bmbt_irec_t imaps[1]; | 
|  | int		reccount; | 
|  | uint		resblks; | 
|  | xfs_fileoff_t	startoffset_fsb; | 
|  | xfs_trans_t	*tp; | 
|  | int		xfs_bmapi_flags; | 
|  | int		error; | 
|  |  | 
|  | if (len <= 0) | 
|  | return -EINVAL; | 
|  |  | 
|  | count = len; | 
|  | error = 0; | 
|  | imapp = &imaps[0]; | 
|  | reccount = 1; | 
|  | xfs_bmapi_flags = alloc_type ? XFS_BMAPI_PREALLOC : 0; | 
|  | mp = ip->i_mount; | 
|  | startoffset_fsb = XFS_B_TO_FSBT(mp, offset); | 
|  | allocatesize_fsb = XFS_B_TO_FSB(mp, count); | 
|  |  | 
|  | /* allocate file space until done or until there is an error */ | 
|  | while (allocatesize_fsb && !error) { | 
|  | datablocks = allocatesize_fsb; | 
|  |  | 
|  | resblks = (uint)XFS_DIOSTRAT_SPACE_RES(mp, datablocks); | 
|  | error = xfs_trans_alloc(mp, &M_RES(mp)->tr_write, resblks, | 
|  | 0, 0, &tp); | 
|  | /* | 
|  | * Check for running out of space | 
|  | */ | 
|  | if (error) { | 
|  | ASSERT(error == -ENOSPC); | 
|  | break; | 
|  | } | 
|  | xfs_trans_ijoin(tp, ip, 0); | 
|  |  | 
|  | error = xfs_bmapi_write(tp, ip, startoffset_fsb, allocatesize_fsb, | 
|  | xfs_bmapi_flags, 0, imapp, &reccount); | 
|  |  | 
|  | if (error) | 
|  | goto error0; | 
|  |  | 
|  | /* | 
|  | * Complete the transaction | 
|  | */ | 
|  | error = xfs_trans_commit(tp); | 
|  | if (error) | 
|  | break; | 
|  |  | 
|  | allocated_fsb = imapp->br_blockcount; | 
|  | if (reccount == 0) | 
|  | return -ENOSPC; | 
|  |  | 
|  | startoffset_fsb += allocated_fsb; | 
|  | allocatesize_fsb -= allocated_fsb; | 
|  | } | 
|  | return error; | 
|  |  | 
|  | error0:	/* Cancel bmap, cancel trans */ | 
|  | xfs_trans_cancel(tp); | 
|  | return error; | 
|  | } | 
|  |  | 
|  | /* | 
|  | * Wrapper around call to libxfs_ialloc. Takes care of committing and | 
|  | * allocating a new transaction as needed. | 
|  | * | 
|  | * Originally there were two copies of this code - one in mkfs, the | 
|  | * other in repair - now there is just the one. | 
|  | */ | 
|  | int | 
|  | libxfs_inode_alloc( | 
|  | xfs_trans_t	**tp, | 
|  | xfs_inode_t	*pip, | 
|  | mode_t		mode, | 
|  | nlink_t		nlink, | 
|  | xfs_dev_t	rdev, | 
|  | struct cred	*cr, | 
|  | struct fsxattr	*fsx, | 
|  | xfs_inode_t	**ipp) | 
|  | { | 
|  | xfs_buf_t	*ialloc_context; | 
|  | xfs_inode_t	*ip; | 
|  | int		error; | 
|  |  | 
|  | ialloc_context = (xfs_buf_t *)0; | 
|  | error = libxfs_ialloc(*tp, pip, mode, nlink, rdev, cr, fsx, | 
|  | &ialloc_context, &ip); | 
|  | if (error) { | 
|  | *ipp = NULL; | 
|  | return error; | 
|  | } | 
|  | if (!ialloc_context && !ip) { | 
|  | *ipp = NULL; | 
|  | return -ENOSPC; | 
|  | } | 
|  |  | 
|  | if (ialloc_context) { | 
|  |  | 
|  | xfs_trans_bhold(*tp, ialloc_context); | 
|  |  | 
|  | error = xfs_trans_roll(tp); | 
|  | if (error) { | 
|  | fprintf(stderr, _("%s: cannot duplicate transaction: %s\n"), | 
|  | progname, strerror(error)); | 
|  | exit(1); | 
|  | } | 
|  | xfs_trans_bjoin(*tp, ialloc_context); | 
|  | error = libxfs_ialloc(*tp, pip, mode, nlink, rdev, cr, | 
|  | fsx, &ialloc_context, &ip); | 
|  | if (!ip) | 
|  | error = -ENOSPC; | 
|  | if (error) | 
|  | return error; | 
|  | } | 
|  |  | 
|  | *ipp = ip; | 
|  | return error; | 
|  | } | 
|  |  | 
|  | void | 
|  | cmn_err(int level, char *fmt, ...) | 
|  | { | 
|  | va_list	ap; | 
|  |  | 
|  | va_start(ap, fmt); | 
|  | vfprintf(stderr, fmt, ap); | 
|  | fputs("\n", stderr); | 
|  | va_end(ap); | 
|  | } | 
|  |  | 
|  | /* | 
|  | * Warnings specifically for verifier errors.  Differentiate CRC vs. invalid | 
|  | * values, and omit the stack trace unless the error level is tuned high. | 
|  | */ | 
|  | void | 
|  | xfs_verifier_error( | 
|  | struct xfs_buf		*bp, | 
|  | int			error, | 
|  | xfs_failaddr_t		failaddr) | 
|  | { | 
|  | xfs_buf_ioerror(bp, error); | 
|  |  | 
|  | xfs_alert(NULL, "Metadata %s detected at %p, %s block 0x%llx/0x%x", | 
|  | bp->b_error == -EFSBADCRC ? "CRC error" : "corruption", | 
|  | failaddr ? failaddr : __return_address, | 
|  | bp->b_ops->name, bp->b_bn, BBTOB(bp->b_length)); | 
|  | } | 
|  |  | 
|  | /* | 
|  | * Warnings for inode corruption problems.  Don't bother with the stack | 
|  | * trace unless the error level is turned up high. | 
|  | */ | 
|  | void | 
|  | xfs_inode_verifier_error( | 
|  | struct xfs_inode	*ip, | 
|  | int			error, | 
|  | const char		*name, | 
|  | void			*buf, | 
|  | size_t			bufsz, | 
|  | xfs_failaddr_t		failaddr) | 
|  | { | 
|  | xfs_alert(NULL, "Metadata %s detected at %p, inode 0x%llx %s", | 
|  | error == -EFSBADCRC ? "CRC error" : "corruption", | 
|  | failaddr ? failaddr : __return_address, | 
|  | ip->i_ino, name); | 
|  | } | 
|  |  | 
|  | /* | 
|  | * Complain about the kinds of metadata corruption that we can't detect from a | 
|  | * verifier, such as incorrect inter-block relationship data.  Does not set | 
|  | * bp->b_error. | 
|  | */ | 
|  | void | 
|  | xfs_buf_corruption_error( | 
|  | struct xfs_buf		*bp, | 
|  | xfs_failaddr_t		fa) | 
|  | { | 
|  | xfs_alert(NULL, "Metadata corruption detected at %p, %s block 0x%llx", | 
|  | fa, bp->b_ops->name, bp->b_bn); | 
|  | } | 
|  |  | 
|  | /* | 
|  | * This is called from I/O verifiers on v5 superblock filesystems. In the | 
|  | * kernel, it validates the metadata LSN parameter against the current LSN of | 
|  | * the active log. We don't have an active log in userspace so this kind of | 
|  | * validation is not required. Therefore, this function always returns true in | 
|  | * userspace. | 
|  | * | 
|  | * xfs_repair piggybacks off this mechanism to help track the largest metadata | 
|  | * LSN in use on a filesystem. Keep a record of the largest LSN seen such that | 
|  | * repair can validate it against the state of the log. | 
|  | */ | 
|  | xfs_lsn_t	libxfs_max_lsn = 0; | 
|  | static pthread_mutex_t	libxfs_max_lsn_lock = PTHREAD_MUTEX_INITIALIZER; | 
|  |  | 
|  | bool | 
|  | xfs_log_check_lsn( | 
|  | struct xfs_mount	*mp, | 
|  | xfs_lsn_t		lsn) | 
|  | { | 
|  | int			cycle = CYCLE_LSN(lsn); | 
|  | int			block = BLOCK_LSN(lsn); | 
|  | int			max_cycle; | 
|  | int			max_block; | 
|  |  | 
|  | if (lsn == NULLCOMMITLSN) | 
|  | return true; | 
|  |  | 
|  | pthread_mutex_lock(&libxfs_max_lsn_lock); | 
|  |  | 
|  | max_cycle = CYCLE_LSN(libxfs_max_lsn); | 
|  | max_block = BLOCK_LSN(libxfs_max_lsn); | 
|  |  | 
|  | if ((cycle > max_cycle) || | 
|  | (cycle == max_cycle && block > max_block)) | 
|  | libxfs_max_lsn = lsn; | 
|  |  | 
|  | pthread_mutex_unlock(&libxfs_max_lsn_lock); | 
|  |  | 
|  | return true; | 
|  | } | 
|  |  | 
|  | void | 
|  | xfs_log_item_init( | 
|  | struct xfs_mount	*mp, | 
|  | struct xfs_log_item	*item, | 
|  | int			type) | 
|  | { | 
|  | item->li_mountp = mp; | 
|  | item->li_type = type; | 
|  |  | 
|  | INIT_LIST_HEAD(&item->li_trans); | 
|  | INIT_LIST_HEAD(&item->li_bio_list); | 
|  | } | 
|  |  | 
|  | static struct xfs_buftarg * | 
|  | xfs_find_bdev_for_inode( | 
|  | struct xfs_inode	*ip) | 
|  | { | 
|  | struct xfs_mount	*mp = ip->i_mount; | 
|  |  | 
|  | if (XFS_IS_REALTIME_INODE(ip)) | 
|  | return mp->m_rtdev_targp; | 
|  | return mp->m_ddev_targp; | 
|  | } | 
|  |  | 
|  | static xfs_daddr_t | 
|  | xfs_fsb_to_db(struct xfs_inode *ip, xfs_fsblock_t fsb) | 
|  | { | 
|  | if (XFS_IS_REALTIME_INODE(ip)) | 
|  | return XFS_FSB_TO_BB(ip->i_mount, fsb); | 
|  | return XFS_FSB_TO_DADDR(ip->i_mount, (fsb)); | 
|  | } | 
|  |  | 
|  | int | 
|  | libxfs_zero_extent( | 
|  | struct xfs_inode *ip, | 
|  | xfs_fsblock_t	start_fsb, | 
|  | xfs_off_t	count_fsb) | 
|  | { | 
|  | xfs_daddr_t	sector = xfs_fsb_to_db(ip, start_fsb); | 
|  | ssize_t		size = XFS_FSB_TO_BB(ip->i_mount, count_fsb); | 
|  |  | 
|  | return libxfs_device_zero(xfs_find_bdev_for_inode(ip), sector, size); | 
|  | } | 
|  |  | 
|  | unsigned int | 
|  | hweight8(unsigned int w) | 
|  | { | 
|  | unsigned int res = w - ((w >> 1) & 0x55); | 
|  | res = (res & 0x33) + ((res >> 2) & 0x33); | 
|  | return (res + (res >> 4)) & 0x0F; | 
|  | } | 
|  |  | 
|  | unsigned int | 
|  | hweight32(unsigned int w) | 
|  | { | 
|  | unsigned int res = w - ((w >> 1) & 0x55555555); | 
|  | res = (res & 0x33333333) + ((res >> 2) & 0x33333333); | 
|  | res = (res + (res >> 4)) & 0x0F0F0F0F; | 
|  | res = res + (res >> 8); | 
|  | return (res + (res >> 16)) & 0x000000FF; | 
|  | } | 
|  |  | 
|  | unsigned int | 
|  | hweight64(__u64 w) | 
|  | { | 
|  | return hweight32((unsigned int)w) + | 
|  | hweight32((unsigned int)(w >> 32)); | 
|  | } | 
|  |  | 
|  | /* xfs_health.c */ | 
|  |  | 
|  | /* Mark a per-fs metadata healed. */ | 
|  | void | 
|  | xfs_fs_mark_healthy( | 
|  | struct xfs_mount	*mp, | 
|  | unsigned int		mask) | 
|  | { | 
|  | ASSERT(!(mask & ~XFS_SICK_FS_PRIMARY)); | 
|  | trace_xfs_fs_mark_healthy(mp, mask); | 
|  |  | 
|  | spin_lock(&mp->m_sb_lock); | 
|  | mp->m_fs_sick &= ~mask; | 
|  | mp->m_fs_checked |= mask; | 
|  | spin_unlock(&mp->m_sb_lock); | 
|  | } | 
|  |  | 
|  | void xfs_ag_geom_health(struct xfs_perag *pag, struct xfs_ag_geometry *ageo) { } |