2005-04-16 15:20:36 -07:00
/*
* namei . c
*
* Copyright ( c ) 1999 Al Smith
*
* Portions derived from work ( c ) 1995 , 1996 Christian Vogelgsang .
*/
# include <linux/buffer_head.h>
# include <linux/string.h>
2007-10-21 16:42:09 -07:00
# include <linux/exportfs.h>
2008-02-23 15:23:51 -08:00
# include "efs.h"
2007-10-21 16:42:09 -07:00
2005-04-16 15:20:36 -07:00
2014-08-08 14:19:39 -07:00
static efs_ino_t efs_find_entry ( struct inode * inode , const char * name , int len )
{
2005-04-16 15:20:36 -07:00
struct buffer_head * bh ;
int slot , namelen ;
char * nameptr ;
struct efs_dir * dirblock ;
struct efs_dentry * dirslot ;
efs_ino_t inodenum ;
efs_block_t block ;
if ( inode - > i_size & ( EFS_DIRBSIZE - 1 ) )
2014-06-04 16:12:12 -07:00
pr_warn ( " %s(): directory size not a multiple of EFS_DIRBSIZE \n " ,
__func__ ) ;
2005-04-16 15:20:36 -07:00
for ( block = 0 ; block < inode - > i_blocks ; block + + ) {
bh = sb_bread ( inode - > i_sb , efs_bmap ( inode , block ) ) ;
if ( ! bh ) {
2014-06-04 16:12:12 -07:00
pr_err ( " %s(): failed to read dir block %d \n " ,
__func__ , block ) ;
2005-04-16 15:20:36 -07:00
return 0 ;
}
dirblock = ( struct efs_dir * ) bh - > b_data ;
if ( be16_to_cpu ( dirblock - > magic ) ! = EFS_DIRBLK_MAGIC ) {
2014-06-04 16:12:12 -07:00
pr_err ( " %s(): invalid directory block \n " , __func__ ) ;
2005-04-16 15:20:36 -07:00
brelse ( bh ) ;
2014-08-08 14:19:39 -07:00
return 0 ;
2005-04-16 15:20:36 -07:00
}
2014-08-08 14:19:39 -07:00
for ( slot = 0 ; slot < dirblock - > slots ; slot + + ) {
2005-04-16 15:20:36 -07:00
dirslot = ( struct efs_dentry * ) ( ( ( char * ) bh - > b_data ) + EFS_SLOTAT ( dirblock , slot ) ) ;
namelen = dirslot - > namelen ;
nameptr = dirslot - > name ;
if ( ( namelen = = len ) & & ( ! memcmp ( name , nameptr , len ) ) ) {
inodenum = be32_to_cpu ( dirslot - > inode ) ;
brelse ( bh ) ;
2014-08-08 14:19:39 -07:00
return inodenum ;
2005-04-16 15:20:36 -07:00
}
}
brelse ( bh ) ;
}
2014-08-08 14:19:39 -07:00
return 0 ;
2005-04-16 15:20:36 -07:00
}
2012-06-10 17:13:09 -04:00
struct dentry * efs_lookup ( struct inode * dir , struct dentry * dentry , unsigned int flags )
{
2005-04-16 15:20:36 -07:00
efs_ino_t inodenum ;
2011-07-08 21:20:11 -04:00
struct inode * inode = NULL ;
2005-04-16 15:20:36 -07:00
inodenum = efs_find_entry ( dir , dentry - > d_name . name , dentry - > d_name . len ) ;
2011-07-08 21:20:11 -04:00
if ( inodenum )
2008-02-07 00:15:34 -08:00
inode = efs_iget ( dir - > i_sb , inodenum ) ;
2005-04-16 15:20:36 -07:00
2008-08-11 11:33:57 -04:00
return d_splice_alias ( inode , dentry ) ;
2005-04-16 15:20:36 -07:00
}
2007-10-21 16:42:09 -07:00
static struct inode * efs_nfs_get_inode ( struct super_block * sb , u64 ino ,
u32 generation )
2007-07-17 04:04:29 -07:00
{
struct inode * inode ;
if ( ino = = 0 )
return ERR_PTR ( - ESTALE ) ;
2008-02-07 00:15:34 -08:00
inode = efs_iget ( sb , ino ) ;
if ( IS_ERR ( inode ) )
return ERR_CAST ( inode ) ;
2007-07-17 04:04:29 -07:00
2008-02-07 00:15:34 -08:00
if ( generation & & inode - > i_generation ! = generation ) {
2007-10-21 16:42:09 -07:00
iput ( inode ) ;
return ERR_PTR ( - ESTALE ) ;
2007-07-17 04:04:29 -07:00
}
2007-10-21 16:42:09 -07:00
return inode ;
}
2007-07-17 04:04:29 -07:00
2007-10-21 16:42:09 -07:00
struct dentry * efs_fh_to_dentry ( struct super_block * sb , struct fid * fid ,
int fh_len , int fh_type )
{
return generic_fh_to_dentry ( sb , fid , fh_len , fh_type ,
efs_nfs_get_inode ) ;
}
struct dentry * efs_fh_to_parent ( struct super_block * sb , struct fid * fid ,
int fh_len , int fh_type )
{
return generic_fh_to_parent ( sb , fid , fh_len , fh_type ,
efs_nfs_get_inode ) ;
2007-07-17 04:04:29 -07:00
}
2005-04-16 15:20:36 -07:00
struct dentry * efs_get_parent ( struct dentry * child )
{
2008-08-11 15:49:04 +02:00
struct dentry * parent = ERR_PTR ( - ENOENT ) ;
2005-04-16 15:20:36 -07:00
efs_ino_t ino ;
2015-03-17 22:25:59 +00:00
ino = efs_find_entry ( d_inode ( child ) , " .. " , 2 ) ;
2008-08-11 15:49:04 +02:00
if ( ino )
2016-04-10 01:33:30 -04:00
parent = d_obtain_alias ( efs_iget ( child - > d_sb , ino ) ) ;
2005-04-16 15:20:36 -07:00
2008-08-11 15:49:04 +02:00
return parent ;
2005-04-16 15:20:36 -07:00
}