-
Notifications
You must be signed in to change notification settings - Fork 3
/
Copy pathaeon_inode.h
318 lines (270 loc) · 7.88 KB
/
aeon_inode.h
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
#ifndef __AEON_INODE_H
#define __AEON_INODE_H
#define AEON_INODE_SIZE (1 << AEON_I_SHIFT)
#define AEON_INODE_CSIZE (AEON_INODE_SIZE - CHECKSUM_SIZE)
#define AEON_I_NUM_PER_PAGE ((AEON_DEF_BLOCK_SIZE_4K / AEON_INODE_SIZE) * \
AEON_PAGES_FOR_INODE)
/*
* extent tree's header referred from inode
*/
#ifdef CONFIG_AEON_FS_COMPRESSION
#define PI_MAX_INTERNAL_EXTENT 2
#define PI_MAX_EXTERNAL_EXTENT 9
#else
#define PI_MAX_INTERNAL_EXTENT 5
#define PI_MAX_EXTERNAL_EXTENT 4
#endif
struct icache {
u32 ino;
u64 addr;
int independent;
struct icache *head;
struct list_head imem_list;
};
struct i_valid_list {
u32 parent_ino;
u64 addr;
struct i_valid_child_list *ivcl;
struct list_head i_valid_list;
};
struct i_valid_child_list {
u32 ino;
u64 addr;
u32 parent_ino;
struct list_head i_valid_child_list;
};
struct inode_map {
struct mutex inode_table_mutex;
struct rb_root inode_inuse_tree;
unsigned long num_range_node_inode;
struct aeon_range_node *first_inode_range;
struct icache *im;
void *i_table_addr;
};
struct aeon_extent_middle_header {
__le16 em_entries;
__le16 pad0;
__le32 pad1;
__le64 em_next_addr;
} __attribute((__packed__));
struct aeon_extent_header {
__le32 eh_entries; /* 2^32 extents */
__le32 eh_blocks;
__le64 eh_cur_block_addr;
__le64 eh_first_block_addr;
__le64 eh_prev_extent;
#ifdef CONFIG_AEON_FS_COMPRESSION
#endif
} __attribute((__packed__));
struct aeon_extent {
__le16 ex_index;
__le64 ex_block; /*TODO: store an address */
__le16 ex_length;
__le32 ex_offset;
#ifdef CONFIG_AEON_FS_COMPRESSION
__le32 ex_original_length;
__le32 ex_original_offset;
__le16 ex_compressed;
#endif
} __attribute((__packed__));
enum aeon_new_inode_type {
TYPE_CREATE = 0,
TYPE_MKNOD,
TYPE_SYMLINK,
TYPE_MKDIR
};
/*
* Structure of an inode in AEON on pmem.
*/
struct aeon_inode {
/* first 40 bytes */
u8 valid; /* Is this inode valid? */
u8 compressed; /* Is this file compressed? */
u8 deleted; /* Is this inode deleted? */
u8 i_new; /* Is this inode new? */
/* 4 */
__le32 i_flags; /* Inode flags */
#ifdef CONFIG_AEON_FS_COMPRESSION
__le64 i_original_size;
#endif
__le64 i_size; /* Size of data in bytes */
__le32 i_ctime; /* Inode modification time */
__le32 i_mtime; /* Inode tree Modification time */
__le32 i_atime; /* Access time */
__le16 i_mode; /* File mode */
__le64 i_links_count; /* Links count */
__le64 i_xattr; /* Extended attribute block */
/* second 40 bytes */
__le32 i_uid; /* Owner Uid */
__le32 i_gid; /* Group Id */
__le32 i_generation; /* File version (for NFS) */
__le32 i_create_time; /* Create time */
__le32 aeon_ino; /* aeon inode number */
__le32 parent_ino; /* parent inode number */
__le64 i_pinode_addr; /* parent inode address offset */
__le64 i_dentry_addr; /* A related dentry address offset */
__le64 i_inode_addr; /* inode itself address offset */
__le64 i_next_inode_block;
__le64 i_dentry_table_block;
__le64 i_block; /* exist extent or point extent block */
__le64 i_blocks; /* block counts */
__le64 sym_block; /* for symbolic link */
struct {
__le32 rdev; /* major/minor # */
} dev; /* device inode */
struct aeon_extent_header aeh;
struct aeon_extent ae[PI_MAX_INTERNAL_EXTENT];
__le16 i_exblocks;
__le32 csum; /* CRC32 checksum */
} __attribute((__packed__));
/*
* Structure of an inode in AEON on DRAM.
*/
struct aeon_inode_info_header {
/* Map from file offsets to write log entries. */
struct aeon_dentry_info *de_info;
struct rb_root rb_tree; /* RB tree for directory or extent*/
#ifdef CONFIG_AEON_FS_COMPRESSION
struct rb_root rb_ctree;
#endif
struct rw_semaphore dax_sem;
struct rw_semaphore xattr_sem;
struct mutex truncate_mutex;
int num_vmas;
u64 pi_addr;
u8 i_blk_type;
rwlock_t i_meta_lock;
};
struct aeon_inode_info {
struct aeon_inode_info_header header;
struct inode vfs_inode;
};
static inline struct aeon_inode_info *AEON_I(struct inode *inode)
{
return container_of(inode, struct aeon_inode_info, vfs_inode);
}
#include "aeon_super.h"
static inline
u64 aeon_get_reserved_inode_addr(struct super_block *sb, u64 inode_number)
{
return AEON_HEAD(sb) + AEON_SB_SIZE +
(inode_number % 32 - 1) * AEON_INODE_SIZE;
}
static inline
struct aeon_inode *aeon_get_reserved_inode(struct super_block *sb, u64 ino)
{
return (struct aeon_inode *)aeon_get_reserved_inode_addr(sb, ino);
}
static inline
struct aeon_inode *aeon_get_reserved_inode_ino(struct super_block *sb, u64 ino)
{
if (ino == 0)
return NULL;
return (struct aeon_inode *)aeon_get_reserved_inode_addr(sb, ino);
}
static inline
struct aeon_inode *aeon_get_inode(struct super_block *sb,
struct aeon_inode_info_header *sih)
{
struct aeon_inode fake_pi;
void *addr;
int rc;
addr = (void *)sih->pi_addr;
rc = memcpy_mcsafe(&fake_pi, addr, sizeof(struct aeon_inode));
if (rc) {
aeon_err(sb, "%s: ERROR\n", __func__);
return NULL;
}
return (struct aeon_inode *)addr;
}
static inline
struct icache *aeon_alloc_icache(struct super_block *sb)
{
return (struct icache *)aeon_alloc_inode_node(sb);
}
static inline
void aeon_free_icache(struct icache *im)
{
aeon_free_inode_node((struct aeon_range_node *)im);
}
static inline struct inode_map *aeon_alloc_inode_maps(struct super_block *sb)
{
#ifdef CONFIG_AEON_FS_PERCPU_INODEMAP
return alloc_percpu(struct inode_map);
#else
struct aeon_sb_info *sbi = AEON_SB(sb);
return kcalloc(sbi->cpus, sizeof(struct inode_map), GFP_KERNEL);
#endif
}
static inline struct inode_map *aeon_get_inode_map(struct super_block *sb,
int cpu_id)
{
struct aeon_sb_info *sbi = AEON_SB(sb);
#ifdef CONFIG_AEON_FS_PERCPU_INODEMAP
return per_cpu_ptr(sbi->inode_maps, cpu_id);
#else
return &sbi->inode_maps[cpu_id];
#endif
}
static inline void aeon_free_inode_maps(struct super_block *sb)
{
struct aeon_sb_info *sbi = AEON_SB(sb);
#ifdef CONFIG_AEON_FS_PERCPU_INODEMAP
free_percpu(sbi->inode_maps);
#else
kfree(sbi->inode_maps);
#endif
}
/*
* This function only is called from ioctl.c for the purpose of
* file system test so far.
*/
static inline
struct aeon_inode *aeon_get_pinode(struct super_block *sb,
struct aeon_inode_info_header *sih)
{
struct aeon_sb_info *sbi = AEON_SB(sb);
struct aeon_inode *pi;
u64 addr;
pi = aeon_get_inode(sb, sih);
addr = (u64)sbi->virt_addr + le64_to_cpu(pi->i_pinode_addr);
return (struct aeon_inode *)addr;
}
static inline int is_persisted_inode(struct aeon_inode *pi)
{
__le32 temp;
temp = cpu_to_le32(crc32_le(SEED,
(unsigned char *)pi,
AEON_INODE_CSIZE));
if (temp != pi->csum)
return 0;
return 1;
}
static inline void aeon_update_inode_csum(struct aeon_inode *pi)
{
pi->csum = cpu_to_le32(crc32_le(SEED,
(unsigned char *)pi,
AEON_INODE_CSIZE));
}
int aeon_init_inode_inuse_list(struct super_block *sb);
int aeon_get_inode_address(struct super_block *sb,
u32 ino, u64 *pi_addr, struct aeon_dentry *de);
u32 aeon_inode_by_name(struct inode *dir, struct qstr *entry);
void aeon_init_file(struct aeon_inode *pi,
struct aeon_extent_header *aeh);
void aeon_set_file_ops(struct inode *inode);
struct inode *aeon_new_vfs_inode(enum aeon_new_inode_type type,
struct inode *dir, struct aeon_mdata *am);
int aeon_new_aeon_inode(struct super_block *sb, struct aeon_mdata *am);
void aeon_set_inode_flags(struct inode *inode, struct aeon_inode *pi,
unsigned int flags);
struct inode *aeon_iget(struct super_block *sb, u32 ino);
int aeon_free_inode_resource(struct super_block *sb, struct aeon_inode *pi,
struct aeon_inode_info_header *sih);
int aeon_free_dram_resource(struct super_block *sb,
struct aeon_inode_info_header *sih);
void aeon_destroy_icache(struct inode_map *inode_map);
int aeon_update_time(struct inode *inode, struct timespec64 *time, int flags);
int aeon_truncate_blocks(struct inode *inode, loff_t offset);
int aeon_setattr(struct dentry *dentry, struct iattr *iattr);
#endif