2005-04-16 15:20:36 -07:00
/*
* linux / fs / affs / namei . c
*
* ( c ) 1996 Hans - Joachim Widmaier - Rewritten
*
* ( C ) 1993 Ray Burr - Modified for Amiga FFS filesystem .
*
* ( C ) 1991 Linus Torvalds - minix filesystem
*/
# include "affs.h"
2017-02-27 14:27:49 -08:00
# include <linux/exportfs.h>
2005-04-16 15:20:36 -07:00
typedef int ( * toupper_t ) ( int ) ;
/* Simple toupper() for DOS\1 */
static int
affs_toupper ( int ch )
{
return ch > = ' a ' & & ch < = ' z ' ? ch - = ( ' a ' - ' A ' ) : ch ;
}
/* International toupper() for DOS\3 ("international") */
static int
affs_intl_toupper ( int ch )
{
return ( ch > = ' a ' & & ch < = ' z ' ) | | ( ch > = 0xE0
& & ch < = 0xFE & & ch ! = 0xF7 ) ?
ch - ( ' a ' - ' A ' ) : ch ;
}
static inline toupper_t
affs_get_toupper ( struct super_block * sb )
{
2015-04-16 12:48:24 -07:00
return affs_test_opt ( AFFS_SB ( sb ) - > s_flags , SF_INTL ) ?
affs_intl_toupper : affs_toupper ;
2005-04-16 15:20:36 -07:00
}
/*
* Note : the dentry argument is the parent dentry .
*/
static inline int
2016-06-10 07:51:30 -07:00
__affs_hash_dentry ( const struct dentry * dentry , struct qstr * qstr , toupper_t toupper , bool notruncate )
2005-04-16 15:20:36 -07:00
{
const u8 * name = qstr - > name ;
unsigned long hash ;
2015-02-17 13:46:20 -08:00
int retval ;
u32 len ;
2005-04-16 15:20:36 -07:00
2015-02-17 13:46:20 -08:00
retval = affs_check_name ( qstr - > name , qstr - > len , notruncate ) ;
if ( retval )
return retval ;
2005-04-16 15:20:36 -07:00
2016-06-10 07:51:30 -07:00
hash = init_name_hash ( dentry ) ;
2015-02-17 13:46:23 -08:00
len = min ( qstr - > len , AFFSNAMEMAX ) ;
2015-02-17 13:46:20 -08:00
for ( ; len > 0 ; name + + , len - - )
2005-04-16 15:20:36 -07:00
hash = partial_name_hash ( toupper ( * name ) , hash ) ;
qstr - > hash = end_name_hash ( hash ) ;
return 0 ;
}
static int
2013-05-21 15:22:44 -07:00
affs_hash_dentry ( const struct dentry * dentry , struct qstr * qstr )
2005-04-16 15:20:36 -07:00
{
2016-06-10 07:51:30 -07:00
return __affs_hash_dentry ( dentry , qstr , affs_toupper ,
2014-04-07 15:39:01 -07:00
affs_nofilenametruncate ( dentry ) ) ;
2005-04-16 15:20:36 -07:00
}
2014-04-07 15:39:01 -07:00
2005-04-16 15:20:36 -07:00
static int
2013-05-21 15:22:44 -07:00
affs_intl_hash_dentry ( const struct dentry * dentry , struct qstr * qstr )
2005-04-16 15:20:36 -07:00
{
2016-06-10 07:51:30 -07:00
return __affs_hash_dentry ( dentry , qstr , affs_intl_toupper ,
2014-04-07 15:39:01 -07:00
affs_nofilenametruncate ( dentry ) ) ;
2005-04-16 15:20:36 -07:00
}
2011-01-07 17:49:27 +11:00
static inline int __affs_compare_dentry ( unsigned int len ,
2014-04-07 15:39:01 -07:00
const char * str , const struct qstr * name , toupper_t toupper ,
bool notruncate )
2005-04-16 15:20:36 -07:00
{
2011-01-07 17:49:27 +11:00
const u8 * aname = str ;
const u8 * bname = name - > name ;
2005-04-16 15:20:36 -07:00
2011-01-07 17:49:27 +11:00
/*
* ' str ' is the name of an already existing dentry , so the name
* must be valid . ' name ' must be validated first .
2005-04-16 15:20:36 -07:00
*/
2014-04-07 15:39:01 -07:00
if ( affs_check_name ( name - > name , name - > len , notruncate ) )
2005-04-16 15:20:36 -07:00
return 1 ;
2011-01-07 17:49:27 +11:00
/*
* If the names are longer than the allowed 30 chars ,
2005-04-16 15:20:36 -07:00
* the excess is ignored , so their length may differ .
*/
2015-02-17 13:46:23 -08:00
if ( len > = AFFSNAMEMAX ) {
if ( name - > len < AFFSNAMEMAX )
2005-04-16 15:20:36 -07:00
return 1 ;
2015-02-17 13:46:23 -08:00
len = AFFSNAMEMAX ;
2011-01-07 17:49:27 +11:00
} else if ( len ! = name - > len )
2005-04-16 15:20:36 -07:00
return 1 ;
for ( ; len > 0 ; len - - )
if ( toupper ( * aname + + ) ! = toupper ( * bname + + ) )
return 1 ;
return 0 ;
}
static int
2016-07-31 16:37:25 -04:00
affs_compare_dentry ( const struct dentry * dentry ,
2011-01-07 17:49:27 +11:00
unsigned int len , const char * str , const struct qstr * name )
2005-04-16 15:20:36 -07:00
{
2014-04-07 15:39:01 -07:00
return __affs_compare_dentry ( len , str , name , affs_toupper ,
2016-07-29 18:22:49 -04:00
affs_nofilenametruncate ( dentry ) ) ;
2005-04-16 15:20:36 -07:00
}
2014-04-07 15:39:01 -07:00
2005-04-16 15:20:36 -07:00
static int
2016-07-31 16:37:25 -04:00
affs_intl_compare_dentry ( const struct dentry * dentry ,
2011-01-07 17:49:27 +11:00
unsigned int len , const char * str , const struct qstr * name )
2005-04-16 15:20:36 -07:00
{
2014-04-07 15:39:01 -07:00
return __affs_compare_dentry ( len , str , name , affs_intl_toupper ,
2016-07-29 18:22:49 -04:00
affs_nofilenametruncate ( dentry ) ) ;
2014-04-07 15:39:01 -07:00
2005-04-16 15:20:36 -07:00
}
/*
* NOTE ! unlike strncmp , affs_match returns 1 for success , 0 for failure .
*/
static inline int
affs_match ( struct dentry * dentry , const u8 * name2 , toupper_t toupper )
{
const u8 * name = dentry - > d_name . name ;
int len = dentry - > d_name . len ;
2015-02-17 13:46:23 -08:00
if ( len > = AFFSNAMEMAX ) {
if ( * name2 < AFFSNAMEMAX )
2005-04-16 15:20:36 -07:00
return 0 ;
2015-02-17 13:46:23 -08:00
len = AFFSNAMEMAX ;
2005-04-16 15:20:36 -07:00
} else if ( len ! = * name2 )
return 0 ;
for ( name2 + + ; len > 0 ; len - - )
if ( toupper ( * name + + ) ! = toupper ( * name2 + + ) )
return 0 ;
return 1 ;
}
int
affs_hash_name ( struct super_block * sb , const u8 * name , unsigned int len )
{
toupper_t toupper = affs_get_toupper ( sb ) ;
2015-02-17 13:46:20 -08:00
u32 hash ;
2005-04-16 15:20:36 -07:00
2015-02-17 13:46:23 -08:00
hash = len = min ( len , AFFSNAMEMAX ) ;
2005-04-16 15:20:36 -07:00
for ( ; len > 0 ; len - - )
hash = ( hash * 13 + toupper ( * name + + ) ) & 0x7ff ;
return hash % AFFS_SB ( sb ) - > s_hashsize ;
}
static struct buffer_head *
affs_find_entry ( struct inode * dir , struct dentry * dentry )
{
struct super_block * sb = dir - > i_sb ;
struct buffer_head * bh ;
toupper_t toupper = affs_get_toupper ( sb ) ;
u32 key ;
2014-10-21 20:11:25 -04:00
pr_debug ( " %s( \" %pd \" ) \n " , __func__ , dentry ) ;
2005-04-16 15:20:36 -07:00
bh = affs_bread ( sb , dir - > i_ino ) ;
if ( ! bh )
return ERR_PTR ( - EIO ) ;
key = be32_to_cpu ( AFFS_HEAD ( bh ) - > table [ affs_hash_name ( sb , dentry - > d_name . name , dentry - > d_name . len ) ] ) ;
for ( ; ; ) {
affs_brelse ( bh ) ;
if ( key = = 0 )
return NULL ;
bh = affs_bread ( sb , key ) ;
if ( ! bh )
return ERR_PTR ( - EIO ) ;
if ( affs_match ( dentry , AFFS_TAIL ( sb , bh ) - > name , toupper ) )
return bh ;
key = be32_to_cpu ( AFFS_TAIL ( sb , bh ) - > hash_chain ) ;
}
}
struct dentry *
2012-06-10 17:13:09 -04:00
affs_lookup ( struct inode * dir , struct dentry * dentry , unsigned int flags )
2005-04-16 15:20:36 -07:00
{
struct super_block * sb = dir - > i_sb ;
struct buffer_head * bh ;
struct inode * inode = NULL ;
2014-10-21 20:11:25 -04:00
pr_debug ( " %s( \" %pd \" ) \n " , __func__ , dentry ) ;
2005-04-16 15:20:36 -07:00
affs_lock_dir ( dir ) ;
bh = affs_find_entry ( dir , dentry ) ;
affs_unlock_dir ( dir ) ;
2008-02-07 00:15:29 -08:00
if ( IS_ERR ( bh ) )
2008-02-07 00:15:26 -08:00
return ERR_CAST ( bh ) ;
2005-04-16 15:20:36 -07:00
if ( bh ) {
u32 ino = bh - > b_blocknr ;
/* store the real header ino in d_fsdata for faster lookups */
dentry - > d_fsdata = ( void * ) ( long ) ino ;
switch ( be32_to_cpu ( AFFS_TAIL ( sb , bh ) - > stype ) ) {
//link to dirs disabled
//case ST_LINKDIR:
case ST_LINKFILE :
ino = be32_to_cpu ( AFFS_TAIL ( sb , bh ) - > original ) ;
}
affs_brelse ( bh ) ;
2008-02-07 00:15:29 -08:00
inode = affs_iget ( sb , ino ) ;
if ( IS_ERR ( inode ) )
2010-05-26 14:44:23 -07:00
return ERR_CAST ( inode ) ;
2005-04-16 15:20:36 -07:00
}
d_add ( dentry , inode ) ;
return NULL ;
}
int
affs_unlink ( struct inode * dir , struct dentry * dentry )
{
2015-02-17 13:46:10 -08:00
pr_debug ( " %s(dir=%lu, %lu \" %pd \" ) \n " , __func__ , dir - > i_ino ,
2015-03-17 22:25:59 +00:00
d_inode ( dentry ) - > i_ino , dentry ) ;
2005-04-16 15:20:36 -07:00
return affs_remove_header ( dentry ) ;
}
int
2012-06-10 18:05:36 -04:00
affs_create ( struct inode * dir , struct dentry * dentry , umode_t mode , bool excl )
2005-04-16 15:20:36 -07:00
{
struct super_block * sb = dir - > i_sb ;
struct inode * inode ;
int error ;
2014-10-21 20:11:25 -04:00
pr_debug ( " %s(%lu, \" %pd \" ,0%ho) \n " ,
__func__ , dir - > i_ino , dentry , mode ) ;
2005-04-16 15:20:36 -07:00
inode = affs_new_inode ( dir ) ;
if ( ! inode )
return - ENOSPC ;
inode - > i_mode = mode ;
2017-02-27 14:27:54 -08:00
affs_mode_to_prot ( inode ) ;
2005-04-16 15:20:36 -07:00
mark_inode_dirty ( inode ) ;
inode - > i_op = & affs_file_inode_operations ;
inode - > i_fop = & affs_file_operations ;
2015-04-16 12:48:24 -07:00
inode - > i_mapping - > a_ops = affs_test_opt ( AFFS_SB ( sb ) - > s_flags , SF_OFS ) ?
2015-04-16 12:48:15 -07:00
& affs_aops_ofs : & affs_aops ;
2005-04-16 15:20:36 -07:00
error = affs_add_entry ( dir , inode , dentry , ST_FILE ) ;
if ( error ) {
2011-10-28 14:13:28 +02:00
clear_nlink ( inode ) ;
2005-04-16 15:20:36 -07:00
iput ( inode ) ;
return error ;
}
return 0 ;
}
int
2011-07-26 01:41:39 -04:00
affs_mkdir ( struct inode * dir , struct dentry * dentry , umode_t mode )
2005-04-16 15:20:36 -07:00
{
struct inode * inode ;
int error ;
2014-10-21 20:11:25 -04:00
pr_debug ( " %s(%lu, \" %pd \" ,0%ho) \n " ,
__func__ , dir - > i_ino , dentry , mode ) ;
2005-04-16 15:20:36 -07:00
inode = affs_new_inode ( dir ) ;
if ( ! inode )
return - ENOSPC ;
inode - > i_mode = S_IFDIR | mode ;
2017-02-27 14:27:54 -08:00
affs_mode_to_prot ( inode ) ;
2005-04-16 15:20:36 -07:00
inode - > i_op = & affs_dir_inode_operations ;
inode - > i_fop = & affs_dir_operations ;
error = affs_add_entry ( dir , inode , dentry , ST_USERDIR ) ;
if ( error ) {
2011-10-28 14:13:28 +02:00
clear_nlink ( inode ) ;
2005-04-16 15:20:36 -07:00
mark_inode_dirty ( inode ) ;
iput ( inode ) ;
return error ;
}
return 0 ;
}
int
affs_rmdir ( struct inode * dir , struct dentry * dentry )
{
2015-02-17 13:46:10 -08:00
pr_debug ( " %s(dir=%lu, %lu \" %pd \" ) \n " , __func__ , dir - > i_ino ,
2015-03-17 22:25:59 +00:00
d_inode ( dentry ) - > i_ino , dentry ) ;
2005-04-16 15:20:36 -07:00
return affs_remove_header ( dentry ) ;
}
int
affs_symlink ( struct inode * dir , struct dentry * dentry , const char * symname )
{
struct super_block * sb = dir - > i_sb ;
struct buffer_head * bh ;
struct inode * inode ;
char * p ;
int i , maxlen , error ;
char c , lc ;
2014-10-21 20:11:25 -04:00
pr_debug ( " %s(%lu, \" %pd \" -> \" %s \" ) \n " ,
__func__ , dir - > i_ino , dentry , symname ) ;
2005-04-16 15:20:36 -07:00
maxlen = AFFS_SB ( sb ) - > s_hashsize * sizeof ( u32 ) - 1 ;
inode = affs_new_inode ( dir ) ;
if ( ! inode )
return - ENOSPC ;
inode - > i_op = & affs_symlink_inode_operations ;
2015-11-17 01:07:57 -05:00
inode_nohighmem ( inode ) ;
2005-04-16 15:20:36 -07:00
inode - > i_data . a_ops = & affs_symlink_aops ;
inode - > i_mode = S_IFLNK | 0777 ;
2017-02-27 14:27:54 -08:00
affs_mode_to_prot ( inode ) ;
2005-04-16 15:20:36 -07:00
error = - EIO ;
bh = affs_bread ( sb , inode - > i_ino ) ;
if ( ! bh )
goto err ;
i = 0 ;
p = ( char * ) AFFS_HEAD ( bh ) - > table ;
lc = ' / ' ;
if ( * symname = = ' / ' ) {
2010-01-24 00:04:07 -05:00
struct affs_sb_info * sbi = AFFS_SB ( sb ) ;
2005-04-16 15:20:36 -07:00
while ( * symname = = ' / ' )
symname + + ;
2010-01-24 00:04:07 -05:00
spin_lock ( & sbi - > symlink_lock ) ;
while ( sbi - > s_volume [ i ] ) /* Cannot overflow */
* p + + = sbi - > s_volume [ i + + ] ;
spin_unlock ( & sbi - > symlink_lock ) ;
2005-04-16 15:20:36 -07:00
}
while ( i < maxlen & & ( c = * symname + + ) ) {
if ( c = = ' . ' & & lc = = ' / ' & & * symname = = ' . ' & & symname [ 1 ] = = ' / ' ) {
* p + + = ' / ' ;
i + + ;
symname + = 2 ;
lc = ' / ' ;
} else if ( c = = ' . ' & & lc = = ' / ' & & * symname = = ' / ' ) {
symname + + ;
lc = ' / ' ;
} else {
* p + + = c ;
lc = c ;
i + + ;
}
if ( lc = = ' / ' )
while ( * symname = = ' / ' )
symname + + ;
}
* p = 0 ;
mark_buffer_dirty_inode ( bh , inode ) ;
affs_brelse ( bh ) ;
mark_inode_dirty ( inode ) ;
error = affs_add_entry ( dir , inode , dentry , ST_SOFTLINK ) ;
if ( error )
goto err ;
return 0 ;
err :
2011-10-28 14:13:28 +02:00
clear_nlink ( inode ) ;
2005-04-16 15:20:36 -07:00
mark_inode_dirty ( inode ) ;
iput ( inode ) ;
return error ;
}
int
affs_link ( struct dentry * old_dentry , struct inode * dir , struct dentry * dentry )
{
2015-03-17 22:25:59 +00:00
struct inode * inode = d_inode ( old_dentry ) ;
2005-04-16 15:20:36 -07:00
2015-02-17 13:46:10 -08:00
pr_debug ( " %s(%lu, %lu, \" %pd \" ) \n " , __func__ , inode - > i_ino , dir - > i_ino ,
2014-10-21 20:11:25 -04:00
dentry ) ;
2005-04-16 15:20:36 -07:00
return affs_add_entry ( dir , inode , dentry , ST_LINKFILE ) ;
}
int
affs_rename ( struct inode * old_dir , struct dentry * old_dentry ,
fs: support RENAME_NOREPLACE for local filesystems
This is trivial to do:
- add flags argument to foo_rename()
- check if flags doesn't have any other than RENAME_NOREPLACE
- assign foo_rename() to .rename2 instead of .rename
Filesystems converted:
affs, bfs, exofs, ext2, hfs, hfsplus, jffs2, jfs, logfs, minix, msdos,
nilfs2, omfs, reiserfs, sysvfs, ubifs, udf, ufs, vfat.
Signed-off-by: Miklos Szeredi <mszeredi@redhat.com>
Acked-by: Boaz Harrosh <ooo@electrozaur.com>
Acked-by: Richard Weinberger <richard@nod.at>
Acked-by: Bob Copeland <me@bobcopeland.com>
Acked-by: Jan Kara <jack@suse.cz>
Cc: Theodore Ts'o <tytso@mit.edu>
Cc: Jaegeuk Kim <jaegeuk@kernel.org>
Cc: OGAWA Hirofumi <hirofumi@mail.parknet.co.jp>
Cc: Mikulas Patocka <mpatocka@redhat.com>
Cc: David Woodhouse <dwmw2@infradead.org>
Cc: Dave Kleikamp <shaggy@kernel.org>
Cc: Ryusuke Konishi <konishi.ryusuke@lab.ntt.co.jp>
Cc: Christoph Hellwig <hch@infradead.org>
2016-09-27 11:03:57 +02:00
struct inode * new_dir , struct dentry * new_dentry ,
unsigned int flags )
2005-04-16 15:20:36 -07:00
{
struct super_block * sb = old_dir - > i_sb ;
struct buffer_head * bh = NULL ;
int retval ;
fs: support RENAME_NOREPLACE for local filesystems
This is trivial to do:
- add flags argument to foo_rename()
- check if flags doesn't have any other than RENAME_NOREPLACE
- assign foo_rename() to .rename2 instead of .rename
Filesystems converted:
affs, bfs, exofs, ext2, hfs, hfsplus, jffs2, jfs, logfs, minix, msdos,
nilfs2, omfs, reiserfs, sysvfs, ubifs, udf, ufs, vfat.
Signed-off-by: Miklos Szeredi <mszeredi@redhat.com>
Acked-by: Boaz Harrosh <ooo@electrozaur.com>
Acked-by: Richard Weinberger <richard@nod.at>
Acked-by: Bob Copeland <me@bobcopeland.com>
Acked-by: Jan Kara <jack@suse.cz>
Cc: Theodore Ts'o <tytso@mit.edu>
Cc: Jaegeuk Kim <jaegeuk@kernel.org>
Cc: OGAWA Hirofumi <hirofumi@mail.parknet.co.jp>
Cc: Mikulas Patocka <mpatocka@redhat.com>
Cc: David Woodhouse <dwmw2@infradead.org>
Cc: Dave Kleikamp <shaggy@kernel.org>
Cc: Ryusuke Konishi <konishi.ryusuke@lab.ntt.co.jp>
Cc: Christoph Hellwig <hch@infradead.org>
2016-09-27 11:03:57 +02:00
if ( flags & ~ RENAME_NOREPLACE )
return - EINVAL ;
2015-02-17 13:46:10 -08:00
pr_debug ( " %s(old=%lu, \" %pd \" to new=%lu, \" %pd \" ) \n " , __func__ ,
old_dir - > i_ino , old_dentry , new_dir - > i_ino , new_dentry ) ;
2005-04-16 15:20:36 -07:00
2014-04-07 15:39:01 -07:00
retval = affs_check_name ( new_dentry - > d_name . name ,
new_dentry - > d_name . len ,
affs_nofilenametruncate ( old_dentry ) ) ;
2005-04-16 15:20:36 -07:00
if ( retval )
return retval ;
/* Unlink destination if it already exists */
2015-03-17 22:25:59 +00:00
if ( d_really_is_positive ( new_dentry ) ) {
2005-04-16 15:20:36 -07:00
retval = affs_remove_header ( new_dentry ) ;
if ( retval )
return retval ;
}
2015-03-17 22:25:59 +00:00
bh = affs_bread ( sb , d_inode ( old_dentry ) - > i_ino ) ;
2005-04-16 15:20:36 -07:00
if ( ! bh )
2006-05-25 18:44:23 -07:00
return - EIO ;
2005-04-16 15:20:36 -07:00
/* Remove header from its parent directory. */
affs_lock_dir ( old_dir ) ;
retval = affs_remove_hash ( old_dir , bh ) ;
affs_unlock_dir ( old_dir ) ;
if ( retval )
goto done ;
/* And insert it into the new directory with the new name. */
affs_copy_name ( AFFS_TAIL ( sb , bh ) - > name , new_dentry ) ;
affs_fix_checksum ( sb , bh ) ;
affs_lock_dir ( new_dir ) ;
retval = affs_insert_hash ( new_dir , bh ) ;
affs_unlock_dir ( new_dir ) ;
/* TODO: move it back to old_dir, if error? */
done :
mark_buffer_dirty_inode ( bh , retval ? old_dir : new_dir ) ;
affs_brelse ( bh ) ;
return retval ;
}
2017-02-27 14:27:49 -08:00
2017-02-27 14:28:00 -08:00
static struct dentry * affs_get_parent ( struct dentry * child )
{
struct inode * parent ;
struct buffer_head * bh ;
bh = affs_bread ( child - > d_sb , d_inode ( child ) - > i_ino ) ;
if ( ! bh )
return ERR_PTR ( - EIO ) ;
parent = affs_iget ( child - > d_sb ,
be32_to_cpu ( AFFS_TAIL ( child - > d_sb , bh ) - > parent ) ) ;
brelse ( bh ) ;
if ( IS_ERR ( parent ) )
return ERR_CAST ( parent ) ;
return d_obtain_alias ( parent ) ;
}
2017-02-27 14:27:49 -08:00
static struct inode * affs_nfs_get_inode ( struct super_block * sb , u64 ino ,
u32 generation )
{
struct inode * inode ;
if ( ! affs_validblock ( sb , ino ) )
return ERR_PTR ( - ESTALE ) ;
inode = affs_iget ( sb , ino ) ;
if ( IS_ERR ( inode ) )
return ERR_CAST ( inode ) ;
if ( generation & & inode - > i_generation ! = generation ) {
iput ( inode ) ;
return ERR_PTR ( - ESTALE ) ;
}
return inode ;
}
static struct dentry * affs_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 ,
affs_nfs_get_inode ) ;
}
static struct dentry * affs_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 ,
affs_nfs_get_inode ) ;
}
const struct export_operations affs_export_ops = {
. fh_to_dentry = affs_fh_to_dentry ,
. fh_to_parent = affs_fh_to_parent ,
2017-02-27 14:28:00 -08:00
. get_parent = affs_get_parent ,
2017-02-27 14:27:49 -08:00
} ;
2017-02-27 14:27:57 -08:00
const struct dentry_operations affs_dentry_operations = {
. d_hash = affs_hash_dentry ,
. d_compare = affs_compare_dentry ,
} ;
const struct dentry_operations affs_intl_dentry_operations = {
. d_hash = affs_intl_hash_dentry ,
. d_compare = affs_intl_compare_dentry ,
} ;