-
Notifications
You must be signed in to change notification settings - Fork 6
Expand file tree
/
Copy pathinode.h
More file actions
386 lines (363 loc) · 12.4 KB
/
inode.h
File metadata and controls
386 lines (363 loc) · 12.4 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
/* 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 <linux/version.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;
unsigned long 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;
#if LINUX_VERSION_CODE >= KERNEL_VERSION(6, 6, 0)
struct folio *folio;
int folio_ofs;
#else
struct page *page;
int page_ofs;
#endif
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 {
unsigned long 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, unsigned long 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,
unsigned long 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);
#if LINUX_VERSION_CODE >= KERNEL_VERSION(5, 12, 0)
#if LINUX_VERSION_CODE >= KERNEL_VERSION(6, 3, 0)
int ntfs_setattr(struct mnt_idmap *idmap, struct dentry *dentry,
struct iattr *attr);
#else
int ntfs_setattr(struct user_namespace *mnt_userns,
struct dentry *dentry, struct iattr *attr);
#endif
#else
int ntfs_setattr(struct dentry *dentry, struct iattr *attr);
#endif
#if LINUX_VERSION_CODE >= KERNEL_VERSION(5, 12, 0)
#if LINUX_VERSION_CODE >= KERNEL_VERSION(6, 3, 0)
int ntfs_getattr(struct mnt_idmap *idmap, const struct path *path,
struct kstat *stat, unsigned int request_mask,
unsigned int query_flags);
#else
int ntfs_getattr(struct user_namespace *mnt_uerns, const struct path *path,
struct kstat *stat, unsigned int request_mask,
unsigned int query_flags);
#endif
#else
int ntfs_getattr(const struct path *path, struct kstat *stat,
unsigned int request_mask, unsigned int query_flags);
#endif
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 */