2005-04-16 15:20:36 -07:00
/*
* Copyright ( C ) International Business Machines Corp . , 2000 - 2004
*
* This program is free software ; you can redistribute it and / or modify
* it under the terms of the GNU General Public License as published by
2006-10-02 09:55:27 -05:00
* the Free Software Foundation ; either version 2 of the License , or
2005-04-16 15:20:36 -07:00
* ( at your option ) any later version .
2006-10-02 09:55:27 -05:00
*
2005-04-16 15:20:36 -07:00
* This program is distributed in the hope that it will 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 .
*
* You should have received a copy of the GNU General Public License
2006-10-02 09:55:27 -05:00
* along with this program ; if not , write to the Free Software
2005-04-16 15:20:36 -07:00
* Foundation , Inc . , 59 Temple Place , Suite 330 , Boston , MA 02111 - 1307 USA
*/
# include <linux/fs.h>
# include <linux/quotaops.h>
# include "jfs_incore.h"
2005-05-04 15:29:35 -05:00
# include "jfs_inode.h"
2005-04-16 15:20:36 -07:00
# include "jfs_filsys.h"
# include "jfs_imap.h"
# include "jfs_dinode.h"
# include "jfs_debug.h"
2006-02-09 09:09:16 -06:00
void jfs_set_inode_flags ( struct inode * inode )
{
unsigned int flags = JFS_IP ( inode ) - > mode2 ;
2014-04-14 09:39:01 +02:00
unsigned int new_fl = 0 ;
2006-02-09 09:09:16 -06:00
if ( flags & JFS_IMMUTABLE_FL )
2014-04-14 09:39:01 +02:00
new_fl | = S_IMMUTABLE ;
2006-02-09 09:09:16 -06:00
if ( flags & JFS_APPEND_FL )
2014-04-14 09:39:01 +02:00
new_fl | = S_APPEND ;
2006-02-09 09:09:16 -06:00
if ( flags & JFS_NOATIME_FL )
2014-04-14 09:39:01 +02:00
new_fl | = S_NOATIME ;
2006-02-09 09:09:16 -06:00
if ( flags & JFS_DIRSYNC_FL )
2014-04-14 09:39:01 +02:00
new_fl | = S_DIRSYNC ;
2006-02-09 09:09:16 -06:00
if ( flags & JFS_SYNC_FL )
2014-04-14 09:39:01 +02:00
new_fl | = S_SYNC ;
inode_set_flags ( inode , new_fl , S_IMMUTABLE | S_APPEND | S_NOATIME |
S_DIRSYNC | S_SYNC ) ;
2006-02-09 09:09:16 -06:00
}
2007-04-25 09:36:20 -05:00
void jfs_get_inode_flags ( struct jfs_inode_info * jfs_ip )
{
unsigned int flags = jfs_ip - > vfs_inode . i_flags ;
jfs_ip - > mode2 & = ~ ( JFS_IMMUTABLE_FL | JFS_APPEND_FL | JFS_NOATIME_FL |
JFS_DIRSYNC_FL | JFS_SYNC_FL ) ;
if ( flags & S_IMMUTABLE )
jfs_ip - > mode2 | = JFS_IMMUTABLE_FL ;
if ( flags & S_APPEND )
jfs_ip - > mode2 | = JFS_APPEND_FL ;
if ( flags & S_NOATIME )
jfs_ip - > mode2 | = JFS_NOATIME_FL ;
if ( flags & S_DIRSYNC )
jfs_ip - > mode2 | = JFS_DIRSYNC_FL ;
if ( flags & S_SYNC )
jfs_ip - > mode2 | = JFS_SYNC_FL ;
}
2005-04-16 15:20:36 -07:00
/*
* NAME : ialloc ( )
*
* FUNCTION : Allocate a new inode
*
*/
struct inode * ialloc ( struct inode * parent , umode_t mode )
{
struct super_block * sb = parent - > i_sb ;
struct inode * inode ;
struct jfs_inode_info * jfs_inode ;
int rc ;
inode = new_inode ( sb ) ;
if ( ! inode ) {
jfs_warn ( " ialloc: new_inode returned NULL! " ) ;
2008-12-30 22:08:37 -06:00
rc = - ENOMEM ;
goto fail ;
2005-04-16 15:20:36 -07:00
}
jfs_inode = JFS_IP ( inode ) ;
rc = diAlloc ( parent , S_ISDIR ( mode ) , inode ) ;
if ( rc ) {
jfs_warn ( " ialloc: diAlloc returned %d! " , rc ) ;
2006-09-14 09:22:38 -05:00
if ( rc = = - EIO )
make_bad_inode ( inode ) ;
2008-12-30 22:08:37 -06:00
goto fail_put ;
}
if ( insert_inode_locked ( inode ) < 0 ) {
rc = - EINVAL ;
2013-09-06 21:49:56 -05:00
goto fail_put ;
2005-04-16 15:20:36 -07:00
}
2010-03-04 17:30:58 +03:00
inode_init_owner ( inode , parent , mode ) ;
2006-03-09 13:59:30 -06:00
/*
* New inodes need to save sane values on disk when
* uid & gid mount options are used
*/
jfs_inode - > saved_uid = inode - > i_uid ;
jfs_inode - > saved_gid = inode - > i_gid ;
2005-04-16 15:20:36 -07:00
/*
* Allocate inode to quota .
*/
2015-07-15 13:53:19 -05:00
rc = dquot_initialize ( inode ) ;
if ( rc )
goto fail_drop ;
2010-03-03 09:05:01 -05:00
rc = dquot_alloc_inode ( inode ) ;
if ( rc )
2008-12-30 22:08:37 -06:00
goto fail_drop ;
2005-04-16 15:20:36 -07:00
2006-02-09 09:09:16 -06:00
/* inherit flags from parent */
jfs_inode - > mode2 = JFS_IP ( parent ) - > mode2 & JFS_FL_INHERIT ;
if ( S_ISDIR ( mode ) ) {
jfs_inode - > mode2 | = IDIRECTORY ;
jfs_inode - > mode2 & = ~ JFS_DIRSYNC_FL ;
}
2006-02-10 08:11:53 -06:00
else {
2006-02-09 09:09:16 -06:00
jfs_inode - > mode2 | = INLINEEA | ISPARSE ;
2006-02-10 08:11:53 -06:00
if ( S_ISLNK ( mode ) )
jfs_inode - > mode2 & = ~ ( JFS_IMMUTABLE_FL | JFS_APPEND_FL ) ;
}
2010-03-04 17:30:58 +03:00
jfs_inode - > mode2 | = inode - > i_mode ;
2006-02-09 09:09:16 -06:00
2005-04-16 15:20:36 -07:00
inode - > i_blocks = 0 ;
inode - > i_mtime = inode - > i_atime = inode - > i_ctime = CURRENT_TIME ;
jfs_inode - > otime = inode - > i_ctime . tv_sec ;
inode - > i_generation = JFS_SBI ( sb ) - > gengen + + ;
jfs_inode - > cflag = 0 ;
/* Zero remaining fields */
memset ( & jfs_inode - > acl , 0 , sizeof ( dxd_t ) ) ;
memset ( & jfs_inode - > ea , 0 , sizeof ( dxd_t ) ) ;
jfs_inode - > next_index = 0 ;
jfs_inode - > acltype = 0 ;
jfs_inode - > btorder = 0 ;
jfs_inode - > btindex = 0 ;
jfs_inode - > bxflag = 0 ;
jfs_inode - > blid = 0 ;
jfs_inode - > atlhead = 0 ;
jfs_inode - > atltail = 0 ;
jfs_inode - > xtlid = 0 ;
2006-02-09 09:09:16 -06:00
jfs_set_inode_flags ( inode ) ;
2005-04-16 15:20:36 -07:00
2016-03-30 05:23:16 -07:00
jfs_info ( " ialloc returns inode = 0x%p " , inode ) ;
2005-04-16 15:20:36 -07:00
return inode ;
2008-12-30 22:08:37 -06:00
fail_drop :
2010-03-03 09:05:05 -05:00
dquot_drop ( inode ) ;
2008-12-30 22:08:37 -06:00
inode - > i_flags | = S_NOQUOTA ;
2011-10-28 14:13:28 +02:00
clear_nlink ( inode ) ;
2008-12-30 22:08:37 -06:00
unlock_new_inode ( inode ) ;
fail_put :
iput ( inode ) ;
fail :
return ERR_PTR ( rc ) ;
2005-04-16 15:20:36 -07:00
}