blob: 67942b97fac6d6ed1b68badf012bac705306729d [file]
/* 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 */