| /* 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 */ |