2005-04-16 15:20:36 -07:00
/*
* ialloc . c
*
* PURPOSE
* Inode allocation handling routines for the OSTA - UDF ( tm ) filesystem .
*
* COPYRIGHT
* This file is distributed under the terms of the GNU General Public
* License ( GPL ) . Copies of the GPL can be obtained from :
* ftp : //prep.ai.mit.edu/pub/gnu/GPL
* Each contributing author retains all rights to their own work .
*
* ( C ) 1998 - 2001 Ben Fennema
*
* HISTORY
*
* 02 / 24 / 99 blf Created .
*
*/
# include "udfdecl.h"
# include <linux/fs.h>
# include <linux/quotaops.h>
# include <linux/udf_fs.h>
# include <linux/sched.h>
# include <linux/slab.h>
# include "udf_i.h"
# include "udf_sb.h"
2007-07-19 01:47:43 -07:00
void udf_free_inode ( struct inode * inode )
2005-04-16 15:20:36 -07:00
{
struct super_block * sb = inode - > i_sb ;
struct udf_sb_info * sbi = UDF_SB ( sb ) ;
/*
* Note : we must free any quota before locking the superblock ,
* as writing the quota to disk may need the lock as well .
*/
DQUOT_FREE_INODE ( inode ) ;
DQUOT_DROP ( inode ) ;
clear_inode ( inode ) ;
2006-03-23 03:00:44 -08:00
mutex_lock ( & sbi - > s_alloc_mutex ) ;
2005-04-16 15:20:36 -07:00
if ( sbi - > s_lvidbh ) {
if ( S_ISDIR ( inode - > i_mode ) )
UDF_SB_LVIDIU ( sb ) - > numDirs =
2007-07-19 01:47:43 -07:00
cpu_to_le32 ( le32_to_cpu ( UDF_SB_LVIDIU ( sb ) - > numDirs )
- 1 ) ;
2005-04-16 15:20:36 -07:00
else
UDF_SB_LVIDIU ( sb ) - > numFiles =
2007-07-19 01:47:43 -07:00
cpu_to_le32 ( le32_to_cpu ( UDF_SB_LVIDIU ( sb ) - > numFiles )
- 1 ) ;
2007-07-15 23:39:43 -07:00
2005-04-16 15:20:36 -07:00
mark_buffer_dirty ( sbi - > s_lvidbh ) ;
}
2006-03-23 03:00:44 -08:00
mutex_unlock ( & sbi - > s_alloc_mutex ) ;
2005-04-16 15:20:36 -07:00
udf_free_blocks ( sb , NULL , UDF_I_LOCATION ( inode ) , 0 , 1 ) ;
}
2007-07-19 01:47:43 -07:00
struct inode * udf_new_inode ( struct inode * dir , int mode , int * err )
2005-04-16 15:20:36 -07:00
{
struct super_block * sb = dir - > i_sb ;
struct udf_sb_info * sbi = UDF_SB ( sb ) ;
2007-07-19 01:47:43 -07:00
struct inode * inode ;
2005-04-16 15:20:36 -07:00
int block ;
uint32_t start = UDF_I_LOCATION ( dir ) . logicalBlockNum ;
inode = new_inode ( sb ) ;
2007-07-19 01:47:43 -07:00
if ( ! inode ) {
2005-04-16 15:20:36 -07:00
* err = - ENOMEM ;
return NULL ;
}
* err = - ENOSPC ;
2006-08-05 12:15:17 -07:00
UDF_I_UNIQUE ( inode ) = 0 ;
UDF_I_LENEXTENTS ( inode ) = 0 ;
UDF_I_NEXT_ALLOC_BLOCK ( inode ) = 0 ;
UDF_I_NEXT_ALLOC_GOAL ( inode ) = 0 ;
UDF_I_STRAT4096 ( inode ) = 0 ;
2007-07-19 01:47:43 -07:00
block =
udf_new_block ( dir - > i_sb , NULL ,
UDF_I_LOCATION ( dir ) . partitionReferenceNum , start ,
err ) ;
if ( * err ) {
2005-04-16 15:20:36 -07:00
iput ( inode ) ;
return NULL ;
}
2006-03-23 03:00:44 -08:00
mutex_lock ( & sbi - > s_alloc_mutex ) ;
2007-07-19 01:47:43 -07:00
if ( UDF_SB_LVIDBH ( sb ) ) {
2005-04-16 15:20:36 -07:00
struct logicalVolHeaderDesc * lvhd ;
uint64_t uniqueID ;
2007-07-19 01:47:43 -07:00
lvhd =
( struct logicalVolHeaderDesc * ) ( UDF_SB_LVID ( sb ) - >
logicalVolContentsUse ) ;
2005-04-16 15:20:36 -07:00
if ( S_ISDIR ( mode ) )
UDF_SB_LVIDIU ( sb ) - > numDirs =
2007-07-19 01:47:43 -07:00
cpu_to_le32 ( le32_to_cpu ( UDF_SB_LVIDIU ( sb ) - > numDirs )
+ 1 ) ;
2005-04-16 15:20:36 -07:00
else
UDF_SB_LVIDIU ( sb ) - > numFiles =
2007-07-19 01:47:43 -07:00
cpu_to_le32 ( le32_to_cpu ( UDF_SB_LVIDIU ( sb ) - > numFiles )
+ 1 ) ;
2005-04-16 15:20:36 -07:00
UDF_I_UNIQUE ( inode ) = uniqueID = le64_to_cpu ( lvhd - > uniqueID ) ;
if ( ! ( + + uniqueID & 0x00000000FFFFFFFFUL ) )
uniqueID + = 16 ;
lvhd - > uniqueID = cpu_to_le64 ( uniqueID ) ;
mark_buffer_dirty ( UDF_SB_LVIDBH ( sb ) ) ;
}
inode - > i_mode = mode ;
inode - > i_uid = current - > fsuid ;
2007-07-19 01:47:43 -07:00
if ( dir - > i_mode & S_ISGID ) {
2005-04-16 15:20:36 -07:00
inode - > i_gid = dir - > i_gid ;
if ( S_ISDIR ( mode ) )
mode | = S_ISGID ;
2007-07-19 01:47:43 -07:00
} else
2005-04-16 15:20:36 -07:00
inode - > i_gid = current - > fsgid ;
UDF_I_LOCATION ( inode ) . logicalBlockNum = block ;
2007-07-19 01:47:43 -07:00
UDF_I_LOCATION ( inode ) . partitionReferenceNum =
UDF_I_LOCATION ( dir ) . partitionReferenceNum ;
2005-04-16 15:20:36 -07:00
inode - > i_ino = udf_get_lb_pblock ( sb , UDF_I_LOCATION ( inode ) , 0 ) ;
inode - > i_blocks = 0 ;
UDF_I_LENEATTR ( inode ) = 0 ;
UDF_I_LENALLOC ( inode ) = 0 ;
UDF_I_USE ( inode ) = 0 ;
2007-07-19 01:47:43 -07:00
if ( UDF_QUERY_FLAG ( inode - > i_sb , UDF_FLAG_USE_EXTENDED_FE ) ) {
2005-04-16 15:20:36 -07:00
UDF_I_EFE ( inode ) = 1 ;
UDF_UPDATE_UDFREV ( inode - > i_sb , UDF_VERS_USE_EXTENDED_FE ) ;
2007-07-19 01:47:43 -07:00
UDF_I_DATA ( inode ) =
kzalloc ( inode - > i_sb - > s_blocksize -
sizeof ( struct extendedFileEntry ) , GFP_KERNEL ) ;
} else {
2005-04-16 15:20:36 -07:00
UDF_I_EFE ( inode ) = 0 ;
2007-07-19 01:47:43 -07:00
UDF_I_DATA ( inode ) =
kzalloc ( inode - > i_sb - > s_blocksize - sizeof ( struct fileEntry ) ,
GFP_KERNEL ) ;
2005-04-16 15:20:36 -07:00
}
2007-07-19 01:47:43 -07:00
if ( ! UDF_I_DATA ( inode ) ) {
2007-07-15 23:39:43 -07:00
iput ( inode ) ;
* err = - ENOMEM ;
mutex_unlock ( & sbi - > s_alloc_mutex ) ;
return NULL ;
}
2005-04-16 15:20:36 -07:00
if ( UDF_QUERY_FLAG ( inode - > i_sb , UDF_FLAG_USE_AD_IN_ICB ) )
UDF_I_ALLOCTYPE ( inode ) = ICBTAG_FLAG_AD_IN_ICB ;
else if ( UDF_QUERY_FLAG ( inode - > i_sb , UDF_FLAG_USE_SHORT_AD ) )
UDF_I_ALLOCTYPE ( inode ) = ICBTAG_FLAG_AD_SHORT ;
else
UDF_I_ALLOCTYPE ( inode ) = ICBTAG_FLAG_AD_LONG ;
inode - > i_mtime = inode - > i_atime = inode - > i_ctime =
2007-07-19 01:47:43 -07:00
UDF_I_CRTIME ( inode ) = current_fs_time ( inode - > i_sb ) ;
2005-04-16 15:20:36 -07:00
insert_inode_hash ( inode ) ;
mark_inode_dirty ( inode ) ;
2006-03-23 03:00:44 -08:00
mutex_unlock ( & sbi - > s_alloc_mutex ) ;
2005-04-16 15:20:36 -07:00
2007-07-19 01:47:43 -07:00
if ( DQUOT_ALLOC_INODE ( inode ) ) {
2005-04-16 15:20:36 -07:00
DQUOT_DROP ( inode ) ;
inode - > i_flags | = S_NOQUOTA ;
inode - > i_nlink = 0 ;
iput ( inode ) ;
* err = - EDQUOT ;
return NULL ;
}
* err = 0 ;
return inode ;
}