mirror of
https://github.com/torvalds/linux.git
synced 2026-04-18 06:44:00 -04:00
Changes the type of ntfs_inode::mft_no from unsigned long to u64 to safely handle the full 48-bit range without truncation risk, especially in preparation for broader VFS inode number type (i_ino:u64) and to improve consistency with ntfs driver practices. Signed-off-by: Namjae Jeon <linkinjeon@kernel.org>
359 lines
12 KiB
C
359 lines
12 KiB
C
/* SPDX-License-Identifier: GPL-2.0-or-later */
|
|
/*
|
|
* Defines for inode structures NTFS Linux kernel driver.
|
|
*
|
|
* Copyright (c) 2001-2007 Anton Altaparmakov
|
|
* Copyright (c) 2002 Richard Russon
|
|
* Copyright (c) 2025 LG Electronics Co., Ltd.
|
|
*/
|
|
|
|
#ifndef _LINUX_NTFS_INODE_H
|
|
#define _LINUX_NTFS_INODE_H
|
|
|
|
#include "debug.h"
|
|
|
|
enum ntfs_inode_mutex_lock_class {
|
|
NTFS_INODE_MUTEX_PARENT,
|
|
NTFS_INODE_MUTEX_NORMAL,
|
|
NTFS_INODE_MUTEX_NORMAL_CHILD,
|
|
NTFS_INODE_MUTEX_PARENT_2,
|
|
NTFS_INODE_MUTEX_NORMAL_2,
|
|
NTFS_EXTEND_MUTEX_PARENT,
|
|
NTFS_EA_MUTEX_NORMAL
|
|
};
|
|
|
|
/*
|
|
* The NTFS in-memory inode structure. It is just used as an extension to the
|
|
* fields already provided in the VFS inode.
|
|
* @size_lock: Lock serializing access to inode sizes.
|
|
* @state: NTFS specific flags describing this inode.
|
|
* @flags: Flags describing the file. (Copy from STANDARD_INFORMATION).
|
|
* @mft_no: Number of the mft record / inode.
|
|
* @seq_no: Sequence number of the mft record.
|
|
* @count: Inode reference count for book keeping.
|
|
* @vol: Pointer to the ntfs volume of this inode.
|
|
*
|
|
* If NInoAttr() is true, the below fields describe the attribute which
|
|
* this fake inode belongs to. The actual inode of this attribute is
|
|
* pointed to by base_ntfs_ino and nr_extents is always set to -1 (see
|
|
* below). For real inodes, we also set the type (AT_DATA for files and
|
|
* AT_INDEX_ALLOCATION for directories), with the name = NULL and
|
|
* name_len = 0 for files and name = I30 (global constant) and
|
|
* name_len = 4 for directories.
|
|
* @type: Attribute type of this fake inode.
|
|
* @name: Attribute name of this fake inode.
|
|
* @name_len: Attribute name length of this fake inode.
|
|
* @runlist: If state has the NI_NonResident bit set, the runlist of
|
|
* the unnamed data attribute (if a file) or of the index allocation
|
|
* attribute (directory) or of the attribute described by the fake inode
|
|
* (if NInoAttr()). If runlist.rl is NULL, the runlist has not been read
|
|
* in yet or has been unmapped. If NI_NonResident is clear, the attribute
|
|
* is resident (file and fake inode) or there is no $I30 index allocation
|
|
* attribute (small directory). In the latter case runlist.rl is always
|
|
* NULL.
|
|
* @data_size: Copy from the attribute record.
|
|
* @initialized_size: Copy from the attribute record.
|
|
* @allocated_size: Copy from the attribute record.
|
|
* @i_crtime: File Creation time.
|
|
* @mrec: MFT record
|
|
* @mrec_lock: Lock for serializing access to the mft record belonging to
|
|
* this inode.
|
|
* @folio: The folio containing the mft record of the inode.
|
|
* @folio_ofs: Offset into the folio at which the mft record begins.
|
|
* @mft_lcn: Number containing the mft record.
|
|
* @mft_lcn_count: Number of clusters per mft record.
|
|
*
|
|
* Attribute list support (only for use by the attribute lookup
|
|
* functions). Setup during read_inode for all inodes with attribute
|
|
* lists. Only valid if NI_AttrList is set in state.
|
|
* @attr_list_size: Length of attribute list value in bytes.
|
|
* @attr_list: Attribute list value itself.
|
|
*
|
|
* It is a directory, $MFT, or an index inode.
|
|
* @block_size: Size of an index block.
|
|
* @vcn_size: Size of a vcn in this index.
|
|
* @collation_rule: The collation rule for the index.
|
|
* @block_size_bits: Log2 of the above.
|
|
* @vcn_size_bits: Log2 of the above.
|
|
*
|
|
* It is a compressed/sparse file/attribute inode.
|
|
* @size: Copy of compressed_size from $DATA.
|
|
* @block_size: Size of a compression block (cb).
|
|
* @block_size_bits: Log2 of the size of a cb.
|
|
* @block_clusters: Number of clusters per cb.
|
|
* @extent_lock: Lock for accessing/modifying the below.
|
|
* @nr_extents: For a base mft record, the number of attached extent inodes
|
|
* (0 if none), for extent records and for fake inodes describing an
|
|
* attribute this is -1.
|
|
*
|
|
* This union is only used if nr_extents != 0.
|
|
* @extent_ntfs_inos: For nr_extents > 0, array of the ntfs inodes of
|
|
* the extent mft records belonging to this base inode which have been
|
|
* loaded.
|
|
* @base_ntfs_ino: For nr_extents == -1, the ntfs inode of the base mft
|
|
* record. For fake inodes, the real (base) inode to which the attribute
|
|
* belongs.
|
|
* @i_dealloc_clusters: delayed allocated clusters.
|
|
* @target: symlink buffer.
|
|
*/
|
|
struct ntfs_inode {
|
|
rwlock_t size_lock;
|
|
unsigned long state;
|
|
__le32 flags;
|
|
u64 mft_no;
|
|
u16 seq_no;
|
|
atomic_t count;
|
|
struct ntfs_volume *vol;
|
|
__le32 type;
|
|
__le16 *name;
|
|
u32 name_len;
|
|
struct runlist runlist;
|
|
s64 data_size;
|
|
s64 initialized_size;
|
|
s64 allocated_size;
|
|
struct timespec64 i_crtime;
|
|
void *mrec;
|
|
struct mutex mrec_lock;
|
|
struct folio *folio;
|
|
int folio_ofs;
|
|
s64 mft_lcn[2];
|
|
unsigned int mft_lcn_count;
|
|
u32 attr_list_size;
|
|
u8 *attr_list;
|
|
union {
|
|
struct {
|
|
u32 block_size;
|
|
u32 vcn_size;
|
|
__le32 collation_rule;
|
|
u8 block_size_bits;
|
|
u8 vcn_size_bits;
|
|
} index;
|
|
struct {
|
|
s64 size;
|
|
u32 block_size;
|
|
u8 block_size_bits;
|
|
u8 block_clusters;
|
|
} compressed;
|
|
} itype;
|
|
struct mutex extent_lock;
|
|
s32 nr_extents;
|
|
union {
|
|
struct ntfs_inode **extent_ntfs_inos;
|
|
struct ntfs_inode *base_ntfs_ino;
|
|
} ext;
|
|
unsigned int i_dealloc_clusters;
|
|
char *target;
|
|
};
|
|
|
|
/*
|
|
* Defined bits for the state field in the ntfs_inode structure.
|
|
* (f) = files only, (d) = directories only, (a) = attributes/fake inodes only
|
|
*
|
|
* NI_Dirty Mft record needs to be written to disk.
|
|
* NI_AttrListDirty Mft record contains an attribute list.
|
|
* NI_AttrList Mft record contains an attribute list.
|
|
* NI_AttrListNonResident Attribute list is non-resident. Implies
|
|
* NI_AttrList is set.
|
|
* NI_Attr 1: Fake inode for attribute i/o.
|
|
* 0: Real inode or extent inode.
|
|
* NI_MstProtected Attribute is protected by MST fixups.
|
|
* NI_NonResident Unnamed data attr is non-resident (f)
|
|
* Attribute is non-resident (a).
|
|
* NI_IndexAllocPresent $I30 index alloc attr is present (d).
|
|
* NI_Compressed Unnamed data attr is compressed (f).
|
|
* Create compressed files by default (d).
|
|
* Attribute is compressed (a).
|
|
* NI_Encrypted Unnamed data attr is encrypted (f).
|
|
* Create encrypted files by default (d).
|
|
* Attribute is encrypted (a).
|
|
* NI_Sparse Unnamed data attr is sparse (f).
|
|
* Create sparse files by default (d).
|
|
* Attribute is sparse (a).
|
|
* NI_SparseDisabled May not create sparse regions.
|
|
* NI_FullyMapped Runlist is fully mapped.
|
|
* NI_FileNameDirty FILE_NAME attributes need to be updated.
|
|
* NI_BeingDeleted ntfs inode is being delated.
|
|
* NI_BeingCreated ntfs inode is being created.
|
|
* NI_HasEA ntfs inode has EA attribute.
|
|
* NI_RunlistDirty runlist need to be updated.
|
|
*/
|
|
enum {
|
|
NI_Dirty,
|
|
NI_AttrListDirty,
|
|
NI_AttrList,
|
|
NI_AttrListNonResident,
|
|
NI_Attr,
|
|
NI_MstProtected,
|
|
NI_NonResident,
|
|
NI_IndexAllocPresent,
|
|
NI_Compressed,
|
|
NI_Encrypted,
|
|
NI_Sparse,
|
|
NI_SparseDisabled,
|
|
NI_FullyMapped,
|
|
NI_FileNameDirty,
|
|
NI_BeingDeleted,
|
|
NI_BeingCreated,
|
|
NI_HasEA,
|
|
NI_RunlistDirty,
|
|
};
|
|
|
|
/*
|
|
* NOTE: We should be adding dirty mft records to a list somewhere and they
|
|
* should be independent of the (ntfs/vfs) inode structure so that an inode can
|
|
* be removed but the record can be left dirty for syncing later.
|
|
*/
|
|
|
|
/*
|
|
* Macro tricks to expand the NInoFoo(), NInoSetFoo(), and NInoClearFoo()
|
|
* functions.
|
|
*/
|
|
#define NINO_FNS(flag) \
|
|
static inline int NIno##flag(struct ntfs_inode *ni) \
|
|
{ \
|
|
return test_bit(NI_##flag, &(ni)->state); \
|
|
} \
|
|
static inline void NInoSet##flag(struct ntfs_inode *ni) \
|
|
{ \
|
|
set_bit(NI_##flag, &(ni)->state); \
|
|
} \
|
|
static inline void NInoClear##flag(struct ntfs_inode *ni) \
|
|
{ \
|
|
clear_bit(NI_##flag, &(ni)->state); \
|
|
}
|
|
|
|
/*
|
|
* As above for NInoTestSetFoo() and NInoTestClearFoo().
|
|
*/
|
|
#define TAS_NINO_FNS(flag) \
|
|
static inline int NInoTestSet##flag(struct ntfs_inode *ni) \
|
|
{ \
|
|
return test_and_set_bit(NI_##flag, &(ni)->state); \
|
|
} \
|
|
static inline int NInoTestClear##flag(struct ntfs_inode *ni) \
|
|
{ \
|
|
return test_and_clear_bit(NI_##flag, &(ni)->state); \
|
|
}
|
|
|
|
/* Emit the ntfs inode bitops functions. */
|
|
NINO_FNS(Dirty)
|
|
TAS_NINO_FNS(Dirty)
|
|
NINO_FNS(AttrList)
|
|
NINO_FNS(AttrListDirty)
|
|
NINO_FNS(AttrListNonResident)
|
|
NINO_FNS(Attr)
|
|
NINO_FNS(MstProtected)
|
|
NINO_FNS(NonResident)
|
|
NINO_FNS(IndexAllocPresent)
|
|
NINO_FNS(Compressed)
|
|
NINO_FNS(Encrypted)
|
|
NINO_FNS(Sparse)
|
|
NINO_FNS(SparseDisabled)
|
|
NINO_FNS(FullyMapped)
|
|
NINO_FNS(FileNameDirty)
|
|
TAS_NINO_FNS(FileNameDirty)
|
|
NINO_FNS(BeingDeleted)
|
|
NINO_FNS(HasEA)
|
|
NINO_FNS(RunlistDirty)
|
|
|
|
/*
|
|
* The full structure containing a ntfs_inode and a vfs struct inode. Used for
|
|
* all real and fake inodes but not for extent inodes which lack the vfs struct
|
|
* inode.
|
|
*/
|
|
struct big_ntfs_inode {
|
|
struct ntfs_inode ntfs_inode;
|
|
struct inode vfs_inode; /* The vfs inode structure. */
|
|
};
|
|
|
|
/*
|
|
* NTFS_I - return the ntfs inode given a vfs inode
|
|
* @inode: VFS inode
|
|
*
|
|
* NTFS_I() returns the ntfs inode associated with the VFS @inode.
|
|
*/
|
|
static inline struct ntfs_inode *NTFS_I(struct inode *inode)
|
|
{
|
|
return &container_of(inode, struct big_ntfs_inode, vfs_inode)->ntfs_inode;
|
|
}
|
|
|
|
static inline struct inode *VFS_I(struct ntfs_inode *ni)
|
|
{
|
|
return &container_of(ni, struct big_ntfs_inode, ntfs_inode)->vfs_inode;
|
|
}
|
|
|
|
/*
|
|
* ntfs_attr - ntfs in memory attribute structure
|
|
*
|
|
* This structure exists only to provide a small structure for the
|
|
* ntfs_{attr_}iget()/ntfs_test_inode()/ntfs_init_locked_inode() mechanism.
|
|
*
|
|
* NOTE: Elements are ordered by size to make the structure as compact as
|
|
* possible on all architectures.
|
|
*/
|
|
struct ntfs_attr {
|
|
u64 mft_no;
|
|
__le16 *name;
|
|
u32 name_len;
|
|
__le32 type;
|
|
unsigned long state;
|
|
};
|
|
|
|
int ntfs_test_inode(struct inode *vi, void *data);
|
|
struct inode *ntfs_iget(struct super_block *sb, u64 mft_no);
|
|
struct inode *ntfs_attr_iget(struct inode *base_vi, __le32 type,
|
|
__le16 *name, u32 name_len);
|
|
struct inode *ntfs_index_iget(struct inode *base_vi, __le16 *name,
|
|
u32 name_len);
|
|
struct inode *ntfs_alloc_big_inode(struct super_block *sb);
|
|
void ntfs_free_big_inode(struct inode *inode);
|
|
int ntfs_drop_big_inode(struct inode *inode);
|
|
void ntfs_evict_big_inode(struct inode *vi);
|
|
void __ntfs_init_inode(struct super_block *sb, struct ntfs_inode *ni);
|
|
|
|
static inline void ntfs_init_big_inode(struct inode *vi)
|
|
{
|
|
struct ntfs_inode *ni = NTFS_I(vi);
|
|
|
|
ntfs_debug("Entering.");
|
|
__ntfs_init_inode(vi->i_sb, ni);
|
|
ni->mft_no = vi->i_ino;
|
|
}
|
|
|
|
struct ntfs_inode *ntfs_new_extent_inode(struct super_block *sb, u64 mft_no);
|
|
void ntfs_clear_extent_inode(struct ntfs_inode *ni);
|
|
int ntfs_read_inode_mount(struct inode *vi);
|
|
int ntfs_show_options(struct seq_file *sf, struct dentry *root);
|
|
int ntfs_truncate_vfs(struct inode *vi, loff_t new_size, loff_t i_size);
|
|
|
|
int ntfs_setattr(struct mnt_idmap *idmap, struct dentry *dentry,
|
|
struct iattr *attr);
|
|
int ntfs_getattr(struct mnt_idmap *idmap, const struct path *path,
|
|
struct kstat *stat, unsigned int request_mask,
|
|
unsigned int query_flags);
|
|
|
|
int ntfs_get_block_mft_record(struct ntfs_inode *mft_ni, struct ntfs_inode *ni);
|
|
int __ntfs_write_inode(struct inode *vi, int sync);
|
|
int ntfs_inode_attach_all_extents(struct ntfs_inode *ni);
|
|
int ntfs_inode_add_attrlist(struct ntfs_inode *ni);
|
|
void ntfs_destroy_ext_inode(struct ntfs_inode *ni);
|
|
int ntfs_inode_free_space(struct ntfs_inode *ni, int size);
|
|
s64 ntfs_inode_attr_pread(struct inode *vi, s64 pos, s64 count, u8 *buf);
|
|
s64 ntfs_inode_attr_pwrite(struct inode *vi, s64 pos, s64 count, u8 *buf,
|
|
bool sync);
|
|
int ntfs_inode_close(struct ntfs_inode *ni);
|
|
|
|
static inline void ntfs_commit_inode(struct inode *vi)
|
|
{
|
|
__ntfs_write_inode(vi, 1);
|
|
}
|
|
|
|
int ntfs_inode_sync_filename(struct ntfs_inode *ni);
|
|
int ntfs_extend_initialized_size(struct inode *vi, const loff_t offset,
|
|
const loff_t new_size, bool bsync);
|
|
void ntfs_set_vfs_operations(struct inode *inode, mode_t mode, dev_t dev);
|
|
struct folio *ntfs_get_locked_folio(struct address_space *mapping,
|
|
pgoff_t index, pgoff_t end_index, struct file_ra_state *ra);
|
|
|
|
#endif /* _LINUX_NTFS_INODE_H */
|