2005-04-16 15:20:36 -07:00
/*
2005-11-02 14:58:39 +11:00
* Copyright ( c ) 2000 - 2001 , 2005 Silicon Graphics , Inc .
* All Rights Reserved .
2005-04-16 15:20:36 -07:00
*
2005-11-02 14:58:39 +11:00
* This program is free software ; you can redistribute it and / or
* modify it under the terms of the GNU General Public License as
2005-04-16 15:20:36 -07:00
* published by the Free Software Foundation .
*
2005-11-02 14:58:39 +11:00
* 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 .
2005-04-16 15:20:36 -07:00
*
2005-11-02 14:58:39 +11:00
* 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
2005-04-16 15:20:36 -07:00
*/
# include "xfs.h"
2005-11-02 14:38:42 +11:00
# include "xfs_fs.h"
2013-10-23 10:51:50 +11:00
# include "xfs_format.h"
2013-10-23 10:50:10 +11:00
# include "xfs_log_format.h"
# include "xfs_trans_resv.h"
2005-04-16 15:20:36 -07:00
# include "xfs_mount.h"
2013-10-15 09:17:51 +11:00
# include "xfs_da_format.h"
2005-11-02 14:38:42 +11:00
# include "xfs_da_btree.h"
2005-04-16 15:20:36 -07:00
# include "xfs_inode.h"
2013-10-23 10:50:10 +11:00
# include "xfs_trans.h"
2005-11-02 14:38:42 +11:00
# include "xfs_inode_item.h"
2005-04-16 15:20:36 -07:00
# include "xfs_bmap.h"
2013-08-12 20:49:37 +10:00
# include "xfs_dir2.h"
2011-07-13 13:43:48 +02:00
# include "xfs_dir2_priv.h"
2005-04-16 15:20:36 -07:00
# include "xfs_error.h"
2009-12-14 23:14:59 +00:00
# include "xfs_trace.h"
2005-04-16 15:20:36 -07:00
2013-08-12 20:50:09 +10:00
struct xfs_name xfs_name_dotdot = { ( unsigned char * ) " .. " , 2 , XFS_DIR3_FT_DIR } ;
2014-12-04 09:43:17 +11:00
/*
* @ mode , if set , indicates that the type field needs to be set up .
* This uses the transformation from file mode to DT_ * as defined in linux / fs . h
* for file type specification . This will be propagated into the directory
* structure if appropriate for the given operation and filesystem config .
*/
const unsigned char xfs_mode_to_ftype [ S_IFMT > > S_SHIFT ] = {
[ 0 ] = XFS_DIR3_FT_UNKNOWN ,
[ S_IFREG > > S_SHIFT ] = XFS_DIR3_FT_REG_FILE ,
[ S_IFDIR > > S_SHIFT ] = XFS_DIR3_FT_DIR ,
[ S_IFCHR > > S_SHIFT ] = XFS_DIR3_FT_CHRDEV ,
[ S_IFBLK > > S_SHIFT ] = XFS_DIR3_FT_BLKDEV ,
[ S_IFIFO > > S_SHIFT ] = XFS_DIR3_FT_FIFO ,
[ S_IFSOCK > > S_SHIFT ] = XFS_DIR3_FT_SOCK ,
[ S_IFLNK > > S_SHIFT ] = XFS_DIR3_FT_SYMLINK ,
} ;
2005-04-16 15:20:36 -07:00
2008-05-21 16:58:55 +10:00
/*
* ASCII case - insensitive ( ie . A - Z ) support for directories that was
* used in IRIX .
*/
STATIC xfs_dahash_t
xfs_ascii_ci_hashname (
struct xfs_name * name )
{
xfs_dahash_t hash ;
int i ;
for ( i = 0 , hash = 0 ; i < name - > len ; i + + )
hash = tolower ( name - > name [ i ] ) ^ rol32 ( hash , 7 ) ;
return hash ;
}
STATIC enum xfs_dacmp
xfs_ascii_ci_compname (
struct xfs_da_args * args ,
2010-01-20 10:47:17 +11:00
const unsigned char * name ,
int len )
2008-05-21 16:58:55 +10:00
{
enum xfs_dacmp result ;
int i ;
if ( args - > namelen ! = len )
return XFS_CMP_DIFFERENT ;
result = XFS_CMP_EXACT ;
for ( i = 0 ; i < len ; i + + ) {
if ( args - > name [ i ] = = name [ i ] )
continue ;
if ( tolower ( args - > name [ i ] ) ! = tolower ( name [ i ] ) )
return XFS_CMP_DIFFERENT ;
result = XFS_CMP_CASE ;
}
return result ;
}
static struct xfs_nameops xfs_ascii_ci_nameops = {
. hashname = xfs_ascii_ci_hashname ,
. compname = xfs_ascii_ci_compname ,
} ;
2014-06-06 15:01:58 +10:00
int
xfs_da_mount (
struct xfs_mount * mp )
2005-04-16 15:20:36 -07:00
{
2014-06-06 15:01:58 +10:00
struct xfs_da_geometry * dageo ;
int nodehdr_size ;
2013-08-26 14:13:30 +10:00
2014-05-20 07:46:55 +10:00
ASSERT ( mp - > m_sb . sb_versionnum & XFS_SB_VERSION_DIRV2BIT ) ;
2005-04-16 15:20:36 -07:00
ASSERT ( ( 1 < < ( mp - > m_sb . sb_blocklog + mp - > m_sb . sb_dirblklog ) ) < =
XFS_MAX_BLOCKSIZE ) ;
2013-10-29 22:11:51 +11:00
mp - > m_dir_inode_ops = xfs_dir_get_ops ( mp , NULL ) ;
mp - > m_nondir_inode_ops = xfs_nondir_get_ops ( mp , NULL ) ;
2013-10-30 09:15:02 +11:00
nodehdr_size = mp - > m_dir_inode_ops - > node_hdr_size ;
2014-06-06 15:01:58 +10:00
mp - > m_dir_geo = kmem_zalloc ( sizeof ( struct xfs_da_geometry ) ,
KM_SLEEP | KM_MAYFAIL ) ;
mp - > m_attr_geo = kmem_zalloc ( sizeof ( struct xfs_da_geometry ) ,
KM_SLEEP | KM_MAYFAIL ) ;
if ( ! mp - > m_dir_geo | | ! mp - > m_attr_geo ) {
kmem_free ( mp - > m_dir_geo ) ;
kmem_free ( mp - > m_attr_geo ) ;
2014-06-25 14:58:08 +10:00
return - ENOMEM ;
2014-06-06 15:01:58 +10:00
}
/* set up directory geometry */
dageo = mp - > m_dir_geo ;
dageo - > blklog = mp - > m_sb . sb_blocklog + mp - > m_sb . sb_dirblklog ;
dageo - > fsblog = mp - > m_sb . sb_blocklog ;
dageo - > blksize = 1 < < dageo - > blklog ;
dageo - > fsbcount = 1 < < mp - > m_sb . sb_dirblklog ;
2014-06-06 15:08:18 +10:00
/*
* Now we ' ve set up the block conversion variables , we can calculate the
* segment block constants using the geometry structure .
*/
dageo - > datablk = xfs_dir2_byte_to_da ( dageo , XFS_DIR2_DATA_OFFSET ) ;
dageo - > leafblk = xfs_dir2_byte_to_da ( dageo , XFS_DIR2_LEAF_OFFSET ) ;
dageo - > freeblk = xfs_dir2_byte_to_da ( dageo , XFS_DIR2_FREE_OFFSET ) ;
2014-06-06 15:01:58 +10:00
dageo - > node_ents = ( dageo - > blksize - nodehdr_size ) /
2013-08-26 14:13:30 +10:00
( uint ) sizeof ( xfs_da_node_entry_t ) ;
2014-06-06 15:01:58 +10:00
dageo - > magicpct = ( dageo - > blksize * 37 ) / 100 ;
/* set up attribute geometry - single fsb only */
dageo = mp - > m_attr_geo ;
dageo - > blklog = mp - > m_sb . sb_blocklog ;
dageo - > fsblog = mp - > m_sb . sb_blocklog ;
dageo - > blksize = 1 < < dageo - > blklog ;
dageo - > fsbcount = 1 ;
dageo - > node_ents = ( dageo - > blksize - nodehdr_size ) /
2013-08-26 14:13:30 +10:00
( uint ) sizeof ( xfs_da_node_entry_t ) ;
2014-06-06 15:01:58 +10:00
dageo - > magicpct = ( dageo - > blksize * 37 ) / 100 ;
2013-08-26 14:13:30 +10:00
2008-05-21 16:58:55 +10:00
if ( xfs_sb_version_hasasciici ( & mp - > m_sb ) )
mp - > m_dirnameops = & xfs_ascii_ci_nameops ;
else
mp - > m_dirnameops = & xfs_default_nameops ;
2013-10-29 22:11:46 +11:00
2014-06-06 15:01:58 +10:00
return 0 ;
}
void
xfs_da_unmount (
struct xfs_mount * mp )
{
kmem_free ( mp - > m_dir_geo ) ;
kmem_free ( mp - > m_attr_geo ) ;
2005-04-16 15:20:36 -07:00
}
/*
* Return 1 if directory contains only " . " and " .. " .
*/
2006-06-20 13:04:51 +10:00
int
xfs_dir_isempty (
xfs_inode_t * dp )
2005-04-16 15:20:36 -07:00
{
2011-07-08 14:35:13 +02:00
xfs_dir2_sf_hdr_t * sfp ;
2005-04-16 15:20:36 -07:00
2011-07-26 02:31:30 -04:00
ASSERT ( S_ISDIR ( dp - > i_d . di_mode ) ) ;
2006-06-20 13:04:51 +10:00
if ( dp - > i_d . di_size = = 0 ) /* might happen during shutdown. */
2005-04-16 15:20:36 -07:00
return 1 ;
if ( dp - > i_d . di_size > XFS_IFORK_DSIZE ( dp ) )
return 0 ;
2011-07-08 14:35:13 +02:00
sfp = ( xfs_dir2_sf_hdr_t * ) dp - > i_df . if_u1 . if_data ;
return ! sfp - > count ;
2005-04-16 15:20:36 -07:00
}
2006-06-20 13:04:51 +10:00
/*
* Validate a given inode number .
*/
int
xfs_dir_ino_validate (
xfs_mount_t * mp ,
xfs_ino_t ino )
{
xfs_agblock_t agblkno ;
xfs_agino_t agino ;
xfs_agnumber_t agno ;
int ino_ok ;
int ioff ;
agno = XFS_INO_TO_AGNO ( mp , ino ) ;
agblkno = XFS_INO_TO_AGBNO ( mp , ino ) ;
ioff = XFS_INO_TO_OFFSET ( mp , ino ) ;
agino = XFS_OFFBNO_TO_AGINO ( mp , agblkno , ioff ) ;
ino_ok =
agno < mp - > m_sb . sb_agcount & &
agblkno < mp - > m_sb . sb_agblocks & &
agblkno ! = 0 & &
ioff < ( 1 < < mp - > m_sb . sb_inopblog ) & &
XFS_AGINO_TO_INO ( mp , agno , agino ) = = ino ;
if ( unlikely ( XFS_TEST_ERROR ( ! ino_ok , mp , XFS_ERRTAG_DIR_INO_VALIDATE ,
XFS_RANDOM_DIR_INO_VALIDATE ) ) ) {
2011-03-07 10:05:35 +11:00
xfs_warn ( mp , " Invalid inode number 0x%Lx " ,
2006-06-20 13:04:51 +10:00
( unsigned long long ) ino ) ;
XFS_ERROR_REPORT ( " xfs_dir_ino_validate " , XFS_ERRLEVEL_LOW , mp ) ;
2014-06-25 14:58:08 +10:00
return - EFSCORRUPTED ;
2006-06-20 13:04:51 +10:00
}
return 0 ;
}
2005-04-16 15:20:36 -07:00
/*
* Initialize a directory with its " . " and " .. " entries .
*/
2006-06-20 13:04:51 +10:00
int
xfs_dir_init (
xfs_trans_t * tp ,
xfs_inode_t * dp ,
xfs_inode_t * pdp )
2005-04-16 15:20:36 -07:00
{
2014-02-27 16:51:26 +11:00
struct xfs_da_args * args ;
2006-06-20 13:04:51 +10:00
int error ;
2005-04-16 15:20:36 -07:00
2011-07-26 02:31:30 -04:00
ASSERT ( S_ISDIR ( dp - > i_d . di_mode ) ) ;
2014-02-27 16:51:26 +11:00
error = xfs_dir_ino_validate ( tp - > t_mountp , pdp - > i_ino ) ;
if ( error )
2005-04-16 15:20:36 -07:00
return error ;
2014-02-27 16:51:26 +11:00
args = kmem_zalloc ( sizeof ( * args ) , KM_SLEEP | KM_NOFS ) ;
if ( ! args )
2014-06-25 14:58:08 +10:00
return - ENOMEM ;
2014-02-27 16:51:26 +11:00
2014-06-06 15:01:58 +10:00
args - > geo = dp - > i_mount - > m_dir_geo ;
2014-02-27 16:51:26 +11:00
args - > dp = dp ;
args - > trans = tp ;
error = xfs_dir2_sf_create ( args , pdp - > i_ino ) ;
kmem_free ( args ) ;
return error ;
2005-04-16 15:20:36 -07:00
}
/*
2014-09-09 11:58:07 +10:00
* Enter a name in a directory , or check for available space .
* If inum is 0 , only the available space test is performed .
2005-04-16 15:20:36 -07:00
*/
2006-06-20 13:04:51 +10:00
int
xfs_dir_createname (
xfs_trans_t * tp ,
xfs_inode_t * dp ,
2008-04-10 12:22:07 +10:00
struct xfs_name * name ,
2005-04-16 15:20:36 -07:00
xfs_ino_t inum , /* new entry inode number */
xfs_fsblock_t * first , /* bmap's firstblock */
xfs_bmap_free_t * flist , /* bmap's freeblock list */
xfs_extlen_t total ) /* bmap's total block count */
{
2014-02-27 16:51:26 +11:00
struct xfs_da_args * args ;
2006-06-20 13:04:51 +10:00
int rval ;
2005-04-16 15:20:36 -07:00
int v ; /* type-checking value */
2011-07-26 02:31:30 -04:00
ASSERT ( S_ISDIR ( dp - > i_d . di_mode ) ) ;
2014-09-09 11:58:07 +10:00
if ( inum ) {
rval = xfs_dir_ino_validate ( tp - > t_mountp , inum ) ;
if ( rval )
return rval ;
XFS_STATS_INC ( xs_dir_create ) ;
}
2006-06-20 13:04:51 +10:00
2014-02-27 16:51:26 +11:00
args = kmem_zalloc ( sizeof ( * args ) , KM_SLEEP | KM_NOFS ) ;
if ( ! args )
2014-06-25 14:58:08 +10:00
return - ENOMEM ;
2014-02-27 16:51:26 +11:00
2014-06-06 15:01:58 +10:00
args - > geo = dp - > i_mount - > m_dir_geo ;
2014-02-27 16:51:26 +11:00
args - > name = name - > name ;
args - > namelen = name - > len ;
args - > filetype = name - > type ;
args - > hashval = dp - > i_mount - > m_dirnameops - > hashname ( name ) ;
args - > inumber = inum ;
args - > dp = dp ;
args - > firstblock = first ;
args - > flist = flist ;
args - > total = total ;
args - > whichfork = XFS_DATA_FORK ;
args - > trans = tp ;
args - > op_flags = XFS_DA_OP_ADDNAME | XFS_DA_OP_OKNOENT ;
2014-09-09 11:58:07 +10:00
if ( ! inum )
args - > op_flags | = XFS_DA_OP_JUSTCHECK ;
2014-02-27 16:51:26 +11:00
if ( dp - > i_d . di_format = = XFS_DINODE_FMT_LOCAL ) {
rval = xfs_dir2_sf_addname ( args ) ;
goto out_free ;
}
2014-06-06 15:20:32 +10:00
rval = xfs_dir2_isblock ( args , & v ) ;
2014-02-27 16:51:26 +11:00
if ( rval )
goto out_free ;
if ( v ) {
rval = xfs_dir2_block_addname ( args ) ;
goto out_free ;
}
2014-06-06 15:20:32 +10:00
rval = xfs_dir2_isleaf ( args , & v ) ;
2014-02-27 16:51:26 +11:00
if ( rval )
goto out_free ;
if ( v )
rval = xfs_dir2_leaf_addname ( args ) ;
2005-04-16 15:20:36 -07:00
else
2014-02-27 16:51:26 +11:00
rval = xfs_dir2_node_addname ( args ) ;
out_free :
kmem_free ( args ) ;
2005-04-16 15:20:36 -07:00
return rval ;
}
2008-05-21 16:58:22 +10:00
/*
* If doing a CI lookup and case - insensitive match , dup actual name into
* args . value . Return EEXIST for success ( ie . name found ) or an error .
*/
int
xfs_dir_cilookup_result (
struct xfs_da_args * args ,
2010-01-20 10:47:25 +11:00
const unsigned char * name ,
2008-05-21 16:58:22 +10:00
int len )
{
if ( args - > cmpresult = = XFS_CMP_DIFFERENT )
2014-06-25 14:58:08 +10:00
return - ENOENT ;
2008-05-21 16:58:22 +10:00
if ( args - > cmpresult ! = XFS_CMP_CASE | |
! ( args - > op_flags & XFS_DA_OP_CILOOKUP ) )
2014-06-25 14:58:08 +10:00
return - EEXIST ;
2008-05-21 16:58:22 +10:00
2009-07-18 18:14:57 -04:00
args - > value = kmem_alloc ( len , KM_NOFS | KM_MAYFAIL ) ;
2008-05-21 16:58:22 +10:00
if ( ! args - > value )
2014-06-25 14:58:08 +10:00
return - ENOMEM ;
2008-05-21 16:58:22 +10:00
memcpy ( args - > value , name , len ) ;
args - > valuelen = len ;
2014-06-25 14:58:08 +10:00
return - EEXIST ;
2008-05-21 16:58:22 +10:00
}
2005-04-16 15:20:36 -07:00
/*
* Lookup a name in a directory , give back the inode number .
2008-05-21 16:58:22 +10:00
* If ci_name is not NULL , returns the actual name in ci_name if it differs
* to name , or ci_name - > name is set to NULL for an exact match .
2005-04-16 15:20:36 -07:00
*/
2008-05-21 16:58:22 +10:00
2006-06-20 13:04:51 +10:00
int
xfs_dir_lookup (
xfs_trans_t * tp ,
xfs_inode_t * dp ,
2008-04-10 12:22:07 +10:00
struct xfs_name * name ,
2008-05-21 16:58:22 +10:00
xfs_ino_t * inum , /* out: inode number */
struct xfs_name * ci_name ) /* out: actual name if CI match */
2005-04-16 15:20:36 -07:00
{
2014-02-27 16:51:26 +11:00
struct xfs_da_args * args ;
2006-06-20 13:04:51 +10:00
int rval ;
2005-04-16 15:20:36 -07:00
int v ; /* type-checking value */
2011-07-26 02:31:30 -04:00
ASSERT ( S_ISDIR ( dp - > i_d . di_mode ) ) ;
2005-04-16 15:20:36 -07:00
XFS_STATS_INC ( xs_dir_lookup ) ;
2014-02-27 16:51:26 +11:00
/*
* We need to use KM_NOFS here so that lockdep will not throw false
* positive deadlock warnings on a non - transactional lookup path . It is
* safe to recurse into inode recalim in that case , but lockdep can ' t
* easily be taught about it . Hence KM_NOFS avoids having to add more
* lockdep Doing this avoids having to add a bunch of lockdep class
* annotations into the reclaim path for the ilock .
*/
args = kmem_zalloc ( sizeof ( * args ) , KM_SLEEP | KM_NOFS ) ;
2014-06-06 15:01:58 +10:00
args - > geo = dp - > i_mount - > m_dir_geo ;
2014-02-27 16:51:26 +11:00
args - > name = name - > name ;
args - > namelen = name - > len ;
args - > filetype = name - > type ;
args - > hashval = dp - > i_mount - > m_dirnameops - > hashname ( name ) ;
args - > dp = dp ;
args - > whichfork = XFS_DATA_FORK ;
args - > trans = tp ;
args - > op_flags = XFS_DA_OP_OKNOENT ;
2008-05-21 16:58:22 +10:00
if ( ci_name )
2014-02-27 16:51:26 +11:00
args - > op_flags | = XFS_DA_OP_CILOOKUP ;
2006-06-20 13:04:51 +10:00
2014-02-27 16:51:26 +11:00
if ( dp - > i_d . di_format = = XFS_DINODE_FMT_LOCAL ) {
rval = xfs_dir2_sf_lookup ( args ) ;
goto out_check_rval ;
}
2014-06-06 15:20:32 +10:00
rval = xfs_dir2_isblock ( args , & v ) ;
2014-02-27 16:51:26 +11:00
if ( rval )
goto out_free ;
if ( v ) {
rval = xfs_dir2_block_lookup ( args ) ;
goto out_check_rval ;
}
2014-06-06 15:20:32 +10:00
rval = xfs_dir2_isleaf ( args , & v ) ;
2014-02-27 16:51:26 +11:00
if ( rval )
goto out_free ;
if ( v )
rval = xfs_dir2_leaf_lookup ( args ) ;
2005-04-16 15:20:36 -07:00
else
2014-02-27 16:51:26 +11:00
rval = xfs_dir2_node_lookup ( args ) ;
out_check_rval :
2014-06-25 14:58:08 +10:00
if ( rval = = - EEXIST )
2005-04-16 15:20:36 -07:00
rval = 0 ;
2008-05-21 16:58:22 +10:00
if ( ! rval ) {
2014-02-27 16:51:26 +11:00
* inum = args - > inumber ;
2008-05-21 16:58:22 +10:00
if ( ci_name ) {
2014-02-27 16:51:26 +11:00
ci_name - > name = args - > value ;
ci_name - > len = args - > valuelen ;
2008-05-21 16:58:22 +10:00
}
}
2014-02-27 16:51:26 +11:00
out_free :
kmem_free ( args ) ;
2005-04-16 15:20:36 -07:00
return rval ;
}
/*
* Remove an entry from a directory .
*/
2006-06-20 13:04:51 +10:00
int
xfs_dir_removename (
xfs_trans_t * tp ,
xfs_inode_t * dp ,
2008-04-10 12:22:07 +10:00
struct xfs_name * name ,
2006-06-20 13:04:51 +10:00
xfs_ino_t ino ,
2005-04-16 15:20:36 -07:00
xfs_fsblock_t * first , /* bmap's firstblock */
xfs_bmap_free_t * flist , /* bmap's freeblock list */
xfs_extlen_t total ) /* bmap's total block count */
{
2014-02-27 16:51:26 +11:00
struct xfs_da_args * args ;
2006-06-20 13:04:51 +10:00
int rval ;
2005-04-16 15:20:36 -07:00
int v ; /* type-checking value */
2011-07-26 02:31:30 -04:00
ASSERT ( S_ISDIR ( dp - > i_d . di_mode ) ) ;
2005-04-16 15:20:36 -07:00
XFS_STATS_INC ( xs_dir_remove ) ;
2006-06-20 13:04:51 +10:00
2014-02-27 16:51:26 +11:00
args = kmem_zalloc ( sizeof ( * args ) , KM_SLEEP | KM_NOFS ) ;
if ( ! args )
2014-06-25 14:58:08 +10:00
return - ENOMEM ;
2014-02-27 16:51:26 +11:00
2014-06-06 15:01:58 +10:00
args - > geo = dp - > i_mount - > m_dir_geo ;
2014-02-27 16:51:26 +11:00
args - > name = name - > name ;
args - > namelen = name - > len ;
args - > filetype = name - > type ;
args - > hashval = dp - > i_mount - > m_dirnameops - > hashname ( name ) ;
args - > inumber = ino ;
args - > dp = dp ;
args - > firstblock = first ;
args - > flist = flist ;
args - > total = total ;
args - > whichfork = XFS_DATA_FORK ;
args - > trans = tp ;
if ( dp - > i_d . di_format = = XFS_DINODE_FMT_LOCAL ) {
rval = xfs_dir2_sf_removename ( args ) ;
goto out_free ;
}
2014-06-06 15:20:32 +10:00
rval = xfs_dir2_isblock ( args , & v ) ;
2014-02-27 16:51:26 +11:00
if ( rval )
goto out_free ;
if ( v ) {
rval = xfs_dir2_block_removename ( args ) ;
goto out_free ;
}
2014-06-06 15:20:32 +10:00
rval = xfs_dir2_isleaf ( args , & v ) ;
2014-02-27 16:51:26 +11:00
if ( rval )
goto out_free ;
if ( v )
rval = xfs_dir2_leaf_removename ( args ) ;
2005-04-16 15:20:36 -07:00
else
2014-02-27 16:51:26 +11:00
rval = xfs_dir2_node_removename ( args ) ;
out_free :
kmem_free ( args ) ;
2005-04-16 15:20:36 -07:00
return rval ;
}
/*
* Replace the inode number of a directory entry .
*/
2006-06-20 13:04:51 +10:00
int
xfs_dir_replace (
xfs_trans_t * tp ,
xfs_inode_t * dp ,
2008-04-10 12:22:07 +10:00
struct xfs_name * name , /* name of entry to replace */
2005-04-16 15:20:36 -07:00
xfs_ino_t inum , /* new inode number */
xfs_fsblock_t * first , /* bmap's firstblock */
xfs_bmap_free_t * flist , /* bmap's freeblock list */
xfs_extlen_t total ) /* bmap's total block count */
{
2014-02-27 16:51:26 +11:00
struct xfs_da_args * args ;
2006-06-20 13:04:51 +10:00
int rval ;
2005-04-16 15:20:36 -07:00
int v ; /* type-checking value */
2011-07-26 02:31:30 -04:00
ASSERT ( S_ISDIR ( dp - > i_d . di_mode ) ) ;
2005-04-16 15:20:36 -07:00
2014-02-27 16:51:26 +11:00
rval = xfs_dir_ino_validate ( tp - > t_mountp , inum ) ;
if ( rval )
2005-04-16 15:20:36 -07:00
return rval ;
2006-06-20 13:04:51 +10:00
2014-02-27 16:51:26 +11:00
args = kmem_zalloc ( sizeof ( * args ) , KM_SLEEP | KM_NOFS ) ;
if ( ! args )
2014-06-25 14:58:08 +10:00
return - ENOMEM ;
2014-02-27 16:51:26 +11:00
2014-06-06 15:01:58 +10:00
args - > geo = dp - > i_mount - > m_dir_geo ;
2014-02-27 16:51:26 +11:00
args - > name = name - > name ;
args - > namelen = name - > len ;
args - > filetype = name - > type ;
args - > hashval = dp - > i_mount - > m_dirnameops - > hashname ( name ) ;
args - > inumber = inum ;
args - > dp = dp ;
args - > firstblock = first ;
args - > flist = flist ;
args - > total = total ;
args - > whichfork = XFS_DATA_FORK ;
args - > trans = tp ;
if ( dp - > i_d . di_format = = XFS_DINODE_FMT_LOCAL ) {
rval = xfs_dir2_sf_replace ( args ) ;
goto out_free ;
}
2014-06-06 15:20:32 +10:00
rval = xfs_dir2_isblock ( args , & v ) ;
2014-02-27 16:51:26 +11:00
if ( rval )
goto out_free ;
if ( v ) {
rval = xfs_dir2_block_replace ( args ) ;
goto out_free ;
}
2014-06-06 15:20:32 +10:00
rval = xfs_dir2_isleaf ( args , & v ) ;
2014-02-27 16:51:26 +11:00
if ( rval )
goto out_free ;
if ( v )
rval = xfs_dir2_leaf_replace ( args ) ;
2005-04-16 15:20:36 -07:00
else
2014-02-27 16:51:26 +11:00
rval = xfs_dir2_node_replace ( args ) ;
out_free :
kmem_free ( args ) ;
2005-04-16 15:20:36 -07:00
return rval ;
}
/*
* See if this entry can be added to the directory without allocating space .
*/
2006-06-20 13:04:51 +10:00
int
xfs_dir_canenter (
xfs_trans_t * tp ,
xfs_inode_t * dp ,
2014-09-09 11:57:52 +10:00
struct xfs_name * name ) /* name of entry to add */
2005-04-16 15:20:36 -07:00
{
2014-09-09 11:58:07 +10:00
return xfs_dir_createname ( tp , dp , name , 0 , NULL , NULL , 0 ) ;
2005-04-16 15:20:36 -07:00
}
/*
* Utility routines .
*/
/*
* Add a block to the directory .
2011-07-13 13:43:49 +02:00
*
* This routine is for data and free blocks , not leaf / node blocks which are
* handled by xfs_da_grow_inode .
2005-04-16 15:20:36 -07:00
*/
2006-06-20 13:04:51 +10:00
int
2005-04-16 15:20:36 -07:00
xfs_dir2_grow_inode (
2011-07-13 13:43:49 +02:00
struct xfs_da_args * args ,
int space , /* v2 dir's space XFS_DIR2_xxx_SPACE */
xfs_dir2_db_t * dbp ) /* out: block number added */
2005-04-16 15:20:36 -07:00
{
2011-07-13 13:43:49 +02:00
struct xfs_inode * dp = args - > dp ;
struct xfs_mount * mp = dp - > i_mount ;
xfs_fileoff_t bno ; /* directory offset of new block */
int count ; /* count of filesystem blocks */
int error ;
2005-04-16 15:20:36 -07:00
2009-12-14 23:14:59 +00:00
trace_xfs_dir2_grow_inode ( args , space ) ;
2005-04-16 15:20:36 -07:00
/*
* Set lowest possible block in the space requested .
*/
bno = XFS_B_TO_FSBT ( mp , space * XFS_DIR2_SPACE_SIZE ) ;
2014-06-06 15:14:11 +10:00
count = args - > geo - > fsbcount ;
2011-07-13 13:43:49 +02:00
error = xfs_da_grow_inode_int ( args , & bno , count ) ;
if ( error )
2005-04-16 15:20:36 -07:00
return error ;
2008-10-30 17:38:12 +11:00
2014-06-06 15:07:53 +10:00
* dbp = xfs_dir2_da_to_db ( args - > geo , ( xfs_dablk_t ) bno ) ;
2008-10-30 17:38:12 +11:00
2005-04-16 15:20:36 -07:00
/*
* Update file ' s size if this is the data space and it grew .
*/
if ( space = = XFS_DIR2_DATA_SPACE ) {
xfs_fsize_t size ; /* directory file (data) size */
size = XFS_FSB_TO_B ( mp , bno + count ) ;
if ( size > dp - > i_d . di_size ) {
dp - > i_d . di_size = size ;
2011-07-13 13:43:49 +02:00
xfs_trans_log_inode ( args - > trans , dp , XFS_ILOG_CORE ) ;
2005-04-16 15:20:36 -07:00
}
}
return 0 ;
}
/*
* See if the directory is a single - block form directory .
*/
2006-06-20 13:04:51 +10:00
int
2005-04-16 15:20:36 -07:00
xfs_dir2_isblock (
2014-06-06 15:20:32 +10:00
struct xfs_da_args * args ,
int * vp ) /* out: 1 is block, 0 is not block */
2005-04-16 15:20:36 -07:00
{
2014-06-06 15:20:32 +10:00
xfs_fileoff_t last ; /* last file offset */
int rval ;
2005-04-16 15:20:36 -07:00
2014-06-06 15:20:32 +10:00
if ( ( rval = xfs_bmap_last_offset ( args - > dp , & last , XFS_DATA_FORK ) ) )
2005-04-16 15:20:36 -07:00
return rval ;
2014-06-06 15:20:32 +10:00
rval = XFS_FSB_TO_B ( args - > dp - > i_mount , last ) = = args - > geo - > blksize ;
ASSERT ( rval = = 0 | | args - > dp - > i_d . di_size = = args - > geo - > blksize ) ;
2005-04-16 15:20:36 -07:00
* vp = rval ;
return 0 ;
}
/*
* See if the directory is a single - leaf form directory .
*/
2006-06-20 13:04:51 +10:00
int
2005-04-16 15:20:36 -07:00
xfs_dir2_isleaf (
2014-06-06 15:20:32 +10:00
struct xfs_da_args * args ,
int * vp ) /* out: 1 is block, 0 is not block */
2005-04-16 15:20:36 -07:00
{
2014-06-06 15:20:32 +10:00
xfs_fileoff_t last ; /* last file offset */
int rval ;
2005-04-16 15:20:36 -07:00
2014-06-06 15:20:32 +10:00
if ( ( rval = xfs_bmap_last_offset ( args - > dp , & last , XFS_DATA_FORK ) ) )
2005-04-16 15:20:36 -07:00
return rval ;
2014-06-06 15:20:32 +10:00
* vp = last = = args - > geo - > leafblk + args - > geo - > fsbcount ;
2005-04-16 15:20:36 -07:00
return 0 ;
}
/*
* Remove the given block from the directory .
* This routine is used for data and free blocks , leaf / node are done
* by xfs_da_shrink_inode .
*/
int
xfs_dir2_shrink_inode (
2006-06-20 13:04:51 +10:00
xfs_da_args_t * args ,
xfs_dir2_db_t db ,
2012-06-22 18:50:14 +10:00
struct xfs_buf * bp )
2005-04-16 15:20:36 -07:00
{
xfs_fileoff_t bno ; /* directory file offset */
xfs_dablk_t da ; /* directory file offset */
int done ; /* bunmap is finished */
2006-06-20 13:04:51 +10:00
xfs_inode_t * dp ;
int error ;
xfs_mount_t * mp ;
xfs_trans_t * tp ;
2005-04-16 15:20:36 -07:00
2009-12-14 23:14:59 +00:00
trace_xfs_dir2_shrink_inode ( args , db ) ;
2005-04-16 15:20:36 -07:00
dp = args - > dp ;
mp = dp - > i_mount ;
tp = args - > trans ;
2014-06-06 15:07:53 +10:00
da = xfs_dir2_db_to_da ( args - > geo , db ) ;
2005-04-16 15:20:36 -07:00
/*
* Unmap the fsblock ( s ) .
*/
2014-06-06 15:14:11 +10:00
if ( ( error = xfs_bunmapi ( tp , dp , da , args - > geo - > fsbcount ,
2005-04-16 15:20:36 -07:00
XFS_BMAPI_METADATA , 0 , args - > firstblock , args - > flist ,
2010-06-23 18:11:15 +10:00
& done ) ) ) {
2005-04-16 15:20:36 -07:00
/*
* ENOSPC actually can happen if we ' re in a removename with
* no space reservation , and the resulting block removal
* would cause a bmap btree split or conversion from extents
* to btree . This can only happen for un - fragmented
* directory blocks , since you need to be punching out
* the middle of an extent .
* In this case we need to leave the block in the file ,
* and not binval it .
* So the block has to be in a consistent empty state
* and appropriately logged .
* We don ' t free up the buffer , the caller can tell it
* hasn ' t happened since it got an error back .
*/
return error ;
}
ASSERT ( done ) ;
/*
* Invalidate the buffer from the transaction .
*/
2012-06-22 18:50:14 +10:00
xfs_trans_binval ( tp , bp ) ;
2005-04-16 15:20:36 -07:00
/*
* If it ' s not a data block , we ' re done .
*/
2014-06-06 15:08:18 +10:00
if ( db > = xfs_dir2_byte_to_db ( args - > geo , XFS_DIR2_LEAF_OFFSET ) )
2005-04-16 15:20:36 -07:00
return 0 ;
/*
* If the block isn ' t the last one in the directory , we ' re done .
*/
2014-06-06 15:06:53 +10:00
if ( dp - > i_d . di_size > xfs_dir2_db_off_to_byte ( args - > geo , db + 1 , 0 ) )
2005-04-16 15:20:36 -07:00
return 0 ;
bno = da ;
if ( ( error = xfs_bmap_last_before ( tp , dp , & bno , XFS_DATA_FORK ) ) ) {
/*
* This can ' t really happen unless there ' s kernel corruption .
*/
return error ;
}
2014-06-06 15:11:18 +10:00
if ( db = = args - > geo - > datablk )
2005-04-16 15:20:36 -07:00
ASSERT ( bno = = 0 ) ;
else
ASSERT ( bno > 0 ) ;
/*
* Set the size to the new last block .
*/
dp - > i_d . di_size = XFS_FSB_TO_B ( mp , bno ) ;
xfs_trans_log_inode ( tp , dp , XFS_ILOG_CORE ) ;
return 0 ;
}