2005-04-17 02:20:36 +04:00
/*
* linux / fs / hfs / dir . c
*
* Copyright ( C ) 1995 - 1997 Paul H . Hargrove
* ( C ) 2003 Ardis Technologies < roman @ ardistech . com >
* This file may be distributed under the terms of the GNU General Public License .
*
* This file contains directory - related functions independent of which
* scheme is being used to represent forks .
*
* Based on the minix file system code , ( C ) 1991 , 1992 by Linus Torvalds
*/
# include "hfs_fs.h"
# include "btree.h"
/*
* hfs_lookup ( )
*/
static struct dentry * hfs_lookup ( struct inode * dir , struct dentry * dentry ,
2012-06-11 01:13:09 +04:00
unsigned int flags )
2005-04-17 02:20:36 +04:00
{
hfs_cat_rec rec ;
struct hfs_find_data fd ;
struct inode * inode = NULL ;
int res ;
2013-05-01 02:27:52 +04:00
res = hfs_find_init ( HFS_SB ( dir - > i_sb ) - > cat_tree , & fd ) ;
if ( res )
return ERR_PTR ( res ) ;
2005-09-07 02:18:49 +04:00
hfs_cat_build_key ( dir - > i_sb , fd . search_key , dir - > i_ino , & dentry - > d_name ) ;
2005-04-17 02:20:36 +04:00
res = hfs_brec_read ( & fd , & rec , sizeof ( rec ) ) ;
if ( res ) {
2018-05-01 02:51:43 +03:00
if ( res ! = - ENOENT )
inode = ERR_PTR ( res ) ;
} else {
inode = hfs_iget ( dir - > i_sb , & fd . search_key - > cat , & rec ) ;
if ( ! inode )
inode = ERR_PTR ( - EACCES ) ;
2005-04-17 02:20:36 +04:00
}
hfs_find_exit ( & fd ) ;
2018-05-01 02:51:43 +03:00
return d_splice_alias ( inode , dentry ) ;
2005-04-17 02:20:36 +04:00
}
/*
* hfs_readdir
*/
2013-05-22 22:29:35 +04:00
static int hfs_readdir ( struct file * file , struct dir_context * ctx )
2005-04-17 02:20:36 +04:00
{
2013-05-22 22:29:35 +04:00
struct inode * inode = file_inode ( file ) ;
2005-04-17 02:20:36 +04:00
struct super_block * sb = inode - > i_sb ;
int len , err ;
2005-09-07 02:18:49 +04:00
char strbuf [ HFS_MAX_NAMELEN ] ;
2005-04-17 02:20:36 +04:00
union hfs_cat_rec entry ;
struct hfs_find_data fd ;
struct hfs_readdir_data * rd ;
u16 type ;
2013-05-22 22:29:35 +04:00
if ( ctx - > pos > = inode - > i_size )
2005-04-17 02:20:36 +04:00
return 0 ;
2013-05-01 02:27:52 +04:00
err = hfs_find_init ( HFS_SB ( sb ) - > cat_tree , & fd ) ;
if ( err )
return err ;
2005-09-07 02:18:49 +04:00
hfs_cat_build_key ( sb , fd . search_key , inode - > i_ino , NULL ) ;
2005-04-17 02:20:36 +04:00
err = hfs_brec_find ( & fd ) ;
if ( err )
goto out ;
2013-05-22 22:29:35 +04:00
if ( ctx - > pos = = 0 ) {
2005-04-17 02:20:36 +04:00
/* This is completely artificial... */
2013-05-22 22:29:35 +04:00
if ( ! dir_emit_dot ( file , ctx ) )
2005-04-17 02:20:36 +04:00
goto out ;
2013-05-22 22:29:35 +04:00
ctx - > pos = 1 ;
}
if ( ctx - > pos = = 1 ) {
2009-12-15 04:57:37 +03:00
if ( fd . entrylength > sizeof ( entry ) | | fd . entrylength < 0 ) {
err = - EIO ;
goto out ;
}
2005-04-17 02:20:36 +04:00
hfs_bnode_read ( fd . bnode , & entry , fd . entryoffset , fd . entrylength ) ;
if ( entry . type ! = HFS_CDR_THD ) {
2013-05-01 02:27:55 +04:00
pr_err ( " bad catalog folder thread \n " ) ;
2005-04-17 02:20:36 +04:00
err = - EIO ;
goto out ;
}
//if (fd.entrylength < HFS_MIN_THREAD_SZ) {
2013-05-01 02:27:55 +04:00
// pr_err("truncated catalog thread\n");
2005-04-17 02:20:36 +04:00
// err = -EIO;
// goto out;
//}
2013-05-22 22:29:35 +04:00
if ( ! dir_emit ( ctx , " .. " , 2 ,
2005-04-17 02:20:36 +04:00
be32_to_cpu ( entry . thread . ParID ) , DT_DIR ) )
goto out ;
2013-05-22 22:29:35 +04:00
ctx - > pos = 2 ;
2005-04-17 02:20:36 +04:00
}
2013-05-22 22:29:35 +04:00
if ( ctx - > pos > = inode - > i_size )
goto out ;
err = hfs_brec_goto ( & fd , ctx - > pos - 1 ) ;
if ( err )
goto out ;
2005-04-17 02:20:36 +04:00
for ( ; ; ) {
if ( be32_to_cpu ( fd . key - > cat . ParID ) ! = inode - > i_ino ) {
2013-05-01 02:27:55 +04:00
pr_err ( " walked past end of dir \n " ) ;
2005-04-17 02:20:36 +04:00
err = - EIO ;
goto out ;
}
2009-12-15 04:57:37 +03:00
if ( fd . entrylength > sizeof ( entry ) | | fd . entrylength < 0 ) {
err = - EIO ;
goto out ;
}
2005-04-17 02:20:36 +04:00
hfs_bnode_read ( fd . bnode , & entry , fd . entryoffset , fd . entrylength ) ;
type = entry . type ;
2005-09-07 02:18:49 +04:00
len = hfs_mac2asc ( sb , strbuf , & fd . key - > cat . CName ) ;
2005-04-17 02:20:36 +04:00
if ( type = = HFS_CDR_DIR ) {
if ( fd . entrylength < sizeof ( struct hfs_cat_dir ) ) {
2013-05-01 02:27:55 +04:00
pr_err ( " small dir entry \n " ) ;
2005-04-17 02:20:36 +04:00
err = - EIO ;
goto out ;
}
2013-05-22 22:29:35 +04:00
if ( ! dir_emit ( ctx , strbuf , len ,
2005-04-17 02:20:36 +04:00
be32_to_cpu ( entry . dir . DirID ) , DT_DIR ) )
break ;
} else if ( type = = HFS_CDR_FIL ) {
if ( fd . entrylength < sizeof ( struct hfs_cat_file ) ) {
2013-05-01 02:27:55 +04:00
pr_err ( " small file entry \n " ) ;
2005-04-17 02:20:36 +04:00
err = - EIO ;
goto out ;
}
2013-05-22 22:29:35 +04:00
if ( ! dir_emit ( ctx , strbuf , len ,
2005-04-17 02:20:36 +04:00
be32_to_cpu ( entry . file . FlNum ) , DT_REG ) )
break ;
} else {
2013-05-01 02:27:55 +04:00
pr_err ( " bad catalog entry type %d \n " , type ) ;
2005-04-17 02:20:36 +04:00
err = - EIO ;
goto out ;
}
2013-05-22 22:29:35 +04:00
ctx - > pos + + ;
if ( ctx - > pos > = inode - > i_size )
2005-04-17 02:20:36 +04:00
goto out ;
err = hfs_brec_goto ( & fd , 1 ) ;
if ( err )
goto out ;
}
2013-05-22 22:29:35 +04:00
rd = file - > private_data ;
2005-04-17 02:20:36 +04:00
if ( ! rd ) {
rd = kmalloc ( sizeof ( struct hfs_readdir_data ) , GFP_KERNEL ) ;
if ( ! rd ) {
err = - ENOMEM ;
goto out ;
}
2013-05-22 22:29:35 +04:00
file - > private_data = rd ;
rd - > file = file ;
2016-05-13 03:13:50 +03:00
spin_lock ( & HFS_I ( inode ) - > open_dir_lock ) ;
2005-04-17 02:20:36 +04:00
list_add ( & rd - > list , & HFS_I ( inode ) - > open_dir_list ) ;
2016-05-13 03:13:50 +03:00
spin_unlock ( & HFS_I ( inode ) - > open_dir_lock ) ;
2005-04-17 02:20:36 +04:00
}
2016-05-13 03:13:50 +03:00
/*
* Can be done after the list insertion ; exclusion with
* hfs_delete_cat ( ) is provided by directory lock .
*/
2017-01-18 14:13:20 +03:00
memcpy ( & rd - > key , & fd . key - > cat , sizeof ( struct hfs_cat_key ) ) ;
2005-04-17 02:20:36 +04:00
out :
hfs_find_exit ( & fd ) ;
return err ;
}
static int hfs_dir_release ( struct inode * inode , struct file * file )
{
struct hfs_readdir_data * rd = file - > private_data ;
if ( rd ) {
2016-05-13 03:13:50 +03:00
spin_lock ( & HFS_I ( inode ) - > open_dir_lock ) ;
2005-04-17 02:20:36 +04:00
list_del ( & rd - > list ) ;
2016-05-13 03:13:50 +03:00
spin_unlock ( & HFS_I ( inode ) - > open_dir_lock ) ;
2005-04-17 02:20:36 +04:00
kfree ( rd ) ;
}
return 0 ;
}
/*
* hfs_create ( )
*
* This is the create ( ) entry in the inode_operations structure for
* regular HFS directories . The purpose is to create a new file in
* a directory and return a corresponding inode , given the inode for
* the directory and the name ( and its length ) of the new file .
*/
2023-01-13 14:49:13 +03:00
static int hfs_create ( struct mnt_idmap * idmap , struct inode * dir ,
2021-01-21 16:19:43 +03:00
struct dentry * dentry , umode_t mode , bool excl )
2005-04-17 02:20:36 +04:00
{
struct inode * inode ;
int res ;
inode = hfs_new_inode ( dir , & dentry - > d_name , mode ) ;
if ( ! inode )
2015-04-16 22:46:53 +03:00
return - ENOMEM ;
2005-04-17 02:20:36 +04:00
res = hfs_cat_create ( inode - > i_ino , dir , & dentry - > d_name , inode ) ;
if ( res ) {
2011-10-28 16:13:28 +04:00
clear_nlink ( inode ) ;
2005-04-17 02:20:36 +04:00
hfs_delete_inode ( inode ) ;
iput ( inode ) ;
return res ;
}
d_instantiate ( dentry , inode ) ;
mark_inode_dirty ( inode ) ;
return 0 ;
}
/*
* hfs_mkdir ( )
*
* This is the mkdir ( ) entry in the inode_operations structure for
* regular HFS directories . The purpose is to create a new directory
* in a directory , given the inode for the parent directory and the
* name ( and its length ) of the new directory .
*/
2023-01-13 14:49:15 +03:00
static int hfs_mkdir ( struct mnt_idmap * idmap , struct inode * dir ,
2021-01-21 16:19:43 +03:00
struct dentry * dentry , umode_t mode )
2005-04-17 02:20:36 +04:00
{
struct inode * inode ;
int res ;
inode = hfs_new_inode ( dir , & dentry - > d_name , S_IFDIR | mode ) ;
if ( ! inode )
2015-04-16 22:46:53 +03:00
return - ENOMEM ;
2005-04-17 02:20:36 +04:00
res = hfs_cat_create ( inode - > i_ino , dir , & dentry - > d_name , inode ) ;
if ( res ) {
2011-10-28 16:13:28 +04:00
clear_nlink ( inode ) ;
2005-04-17 02:20:36 +04:00
hfs_delete_inode ( inode ) ;
iput ( inode ) ;
return res ;
}
d_instantiate ( dentry , inode ) ;
mark_inode_dirty ( inode ) ;
return 0 ;
}
/*
2011-03-03 07:46:51 +03:00
* hfs_remove ( )
2005-04-17 02:20:36 +04:00
*
2011-03-03 07:46:51 +03:00
* This serves as both unlink ( ) and rmdir ( ) in the inode_operations
* structure for regular HFS directories . The purpose is to delete
* an existing child , given the inode for the parent directory and
* the name ( and its length ) of the existing directory .
2005-04-17 02:20:36 +04:00
*
2011-03-03 07:46:51 +03:00
* HFS does not have hardlinks , so both rmdir and unlink set the
* link count to 0. The only difference is the emptiness check .
2005-04-17 02:20:36 +04:00
*/
2011-03-03 07:46:51 +03:00
static int hfs_remove ( struct inode * dir , struct dentry * dentry )
2005-04-17 02:20:36 +04:00
{
2015-03-18 01:25:59 +03:00
struct inode * inode = d_inode ( dentry ) ;
2005-04-17 02:20:36 +04:00
int res ;
2011-03-03 07:46:51 +03:00
if ( S_ISDIR ( inode - > i_mode ) & & inode - > i_size ! = 2 )
2005-04-17 02:20:36 +04:00
return - ENOTEMPTY ;
res = hfs_cat_delete ( inode - > i_ino , dir , & dentry - > d_name ) ;
if ( res )
return res ;
2006-10-01 10:29:06 +04:00
clear_nlink ( inode ) ;
2023-07-05 22:01:13 +03:00
inode_set_ctime_current ( inode ) ;
2005-04-17 02:20:36 +04:00
hfs_delete_inode ( inode ) ;
mark_inode_dirty ( inode ) ;
return 0 ;
}
/*
* hfs_rename ( )
*
* This is the rename ( ) entry in the inode_operations structure for
* regular HFS directories . The purpose is to rename an existing
* file or directory , given the inode for the current directory and
* the name ( and its length ) of the existing file / directory and the
* inode for the new directory and the name ( and its length ) of the
* new file / directory .
* XXX : how do you handle must_be dir ?
*/
2023-01-13 14:49:17 +03:00
static int hfs_rename ( struct mnt_idmap * idmap , struct inode * old_dir ,
2021-01-21 16:19:43 +03:00
struct dentry * old_dentry , struct inode * new_dir ,
struct dentry * new_dentry , unsigned int flags )
2005-04-17 02:20:36 +04:00
{
int res ;
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 12:03:57 +03:00
if ( flags & ~ RENAME_NOREPLACE )
return - EINVAL ;
2005-04-17 02:20:36 +04:00
/* Unlink destination if it already exists */
2015-03-18 01:25:59 +03:00
if ( d_really_is_positive ( new_dentry ) ) {
2011-03-03 07:46:51 +03:00
res = hfs_remove ( new_dir , new_dentry ) ;
2005-04-17 02:20:36 +04:00
if ( res )
return res ;
}
2015-03-18 01:25:59 +03:00
res = hfs_cat_move ( d_inode ( old_dentry ) - > i_ino ,
2005-04-17 02:20:36 +04:00
old_dir , & old_dentry - > d_name ,
new_dir , & new_dentry - > d_name ) ;
if ( ! res )
2005-09-07 02:18:49 +04:00
hfs_cat_build_key ( old_dir - > i_sb ,
2015-03-18 01:25:59 +03:00
( btree_key * ) & HFS_I ( d_inode ( old_dentry ) ) - > cat_key ,
2005-04-17 02:20:36 +04:00
new_dir - > i_ino , & new_dentry - > d_name ) ;
return res ;
}
2006-03-28 13:56:42 +04:00
const struct file_operations hfs_dir_operations = {
2005-04-17 02:20:36 +04:00
. read = generic_read_dir ,
2016-05-13 03:13:50 +03:00
. iterate_shared = hfs_readdir ,
2005-04-17 02:20:36 +04:00
. llseek = generic_file_llseek ,
. release = hfs_dir_release ,
} ;
2007-02-12 11:55:39 +03:00
const struct inode_operations hfs_dir_inode_operations = {
2005-04-17 02:20:36 +04:00
. create = hfs_create ,
. lookup = hfs_lookup ,
2011-03-03 07:46:51 +03:00
. unlink = hfs_remove ,
2005-04-17 02:20:36 +04:00
. mkdir = hfs_mkdir ,
2011-03-03 07:46:51 +03:00
. rmdir = hfs_remove ,
2005-04-17 02:20:36 +04:00
. rename = hfs_rename ,
. setattr = hfs_inode_setattr ,
} ;