-
Notifications
You must be signed in to change notification settings - Fork 10
/
Copy pathsuper.c
206 lines (172 loc) · 5.58 KB
/
super.c
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
#include "HUST_fs.h"
#include "constants.h"
struct file_system_type HUST_fs_type = {
.owner = THIS_MODULE,
.name = "HUST_fs",
.mount = HUST_fs_mount,
.kill_sb = HUST_fs_kill_superblock, /* unmount */
};
const struct file_operations HUST_fs_file_ops = {
.owner = THIS_MODULE,
.llseek = generic_file_llseek,
.mmap = generic_file_mmap,
.fsync = generic_file_fsync,
.read_iter = generic_file_read_iter,
.write_iter = generic_file_write_iter,
};
const struct file_operations HUST_fs_dir_ops = {
.owner = THIS_MODULE,
.iterate = HUST_fs_iterate,
};
const struct inode_operations HUST_fs_inode_ops = {
.lookup = HUST_fs_lookup,
.mkdir = HUST_fs_mkdir,
.create = HUST_fs_create,
.unlink = HUST_fs_unlink,
};
const struct super_operations HUST_fs_super_ops = {
.evict_inode = HUST_evict_inode,
.write_inode = HUST_write_inode,
};
const struct address_space_operations HUST_fs_aops = {
.readpage = HUST_fs_readpage,
.writepage = HUST_fs_writepage,
.write_begin = HUST_fs_write_begin,
.write_end = generic_write_end,
};
int save_super(struct super_block* sb)
{
struct HUST_fs_super_block *disk_sb = sb->s_fs_info;
struct buffer_head* bh;
bh = sb_bread(sb, 1);
printk(KERN_ERR "In save bmap\n");
memcpy(bh->b_data, disk_sb, HUST_BLOCKSIZE);
map_bh(bh, sb, 1);
brelse(bh);
return 0;
}
int HUST_fs_fill_super(struct super_block *sb, void *data, int silent)
{
int ret = -EPERM;
struct buffer_head *bh;
bh = sb_bread(sb, 1);
BUG_ON(!bh);
struct HUST_fs_super_block *sb_disk;
sb_disk = (struct HUST_fs_super_block *)bh->b_data;
printk(KERN_INFO "HUST_fs: version num is %lu\n", sb_disk->version);
printk(KERN_INFO "HUST_fs: magic num is %lu\n", sb_disk->magic);
printk(KERN_INFO "HUST_fs: block_size num is %lu\n",
sb_disk->block_size);
printk(KERN_INFO "HUST_fs: inodes_count num is %lu\n",
sb_disk->inodes_count);
printk(KERN_INFO "HUST_fs: free_blocks num is %lu\n",
sb_disk->free_blocks);
printk(KERN_INFO "HUST_fs: blocks_count num is %lu\n",
sb_disk->blocks_count);
if (sb_disk->magic != MAGIC_NUM) {
printk(KERN_ERR "Magic number not match!\n");
goto release;
}
struct inode *root_inode;
if (sb_disk->block_size != 4096) {
printk(KERN_ERR "HUST_fs expects a blocksize of %d\n", 4096);
ret = -EFAULT;
goto release;
}
//fill vfs super block
sb->s_magic = sb_disk->magic;
sb->s_fs_info = sb_disk;
sb->s_maxbytes = HUST_BLOCKSIZE * HUST_N_BLOCKS; /* Max file size */
sb->s_op = &HUST_fs_super_ops;
//-----------test get inode-----
struct HUST_inode raw_root_node;
if (HUST_fs_get_inode(sb, HUST_ROOT_INODE_NUM, &raw_root_node) != -1) {
printk(KERN_INFO "Get inode sucessfully!\n");
printk(KERN_INFO "root blocks is %llu and block[0] is %llu\n",
raw_root_node.blocks, raw_root_node.block[0]);
}
//-----------end test-----------
root_inode = new_inode(sb);
if (!root_inode)
return -ENOMEM;
/* Our root inode. It doesn't contain useful information for now.
* Note that i_ino must not be 0, since valid inode numbers start at
* 1. */
inode_init_owner(root_inode, NULL, S_IFDIR |
S_IRUSR | S_IXUSR |
S_IRGRP | S_IXGRP | S_IROTH | S_IXOTH);
root_inode->i_sb = sb;
root_inode->i_ino = HUST_ROOT_INODE_NUM;
root_inode->i_atime = root_inode->i_mtime = root_inode->i_ctime =
current_time(root_inode);
root_inode->i_mode = raw_root_node.mode;
root_inode->i_size = raw_root_node.dir_children_count;
//root_inode->i_private = HUST_fs_get_inode(sb, HUST_ROOT_INODE_NUM);
/* Doesn't really matter. Since this is a directory, it "should"
* have a link count of 2. See btrfs, though, where directories
* always have a link count of 1. That appears to work, even though
* it created a number of bug reports in other tools. :-) Just
* search the web for that topic. */
inc_nlink(root_inode);
/* What can you do with our inode? */
root_inode->i_op = &HUST_fs_inode_ops;
root_inode->i_fop = &HUST_fs_dir_ops;
/* Make a struct dentry from our inode and store it in our
* superblock. */
sb->s_root = d_make_root(root_inode);
if (!sb->s_root)
return -ENOMEM;
release:
brelse(bh);
return 0;
}
struct dentry *HUST_fs_mount(struct file_system_type *fs_type, int flags,
const char *dev_name, void *data)
{
struct dentry *ret;
/* This is a generic mount function that accepts a callback. */
ret = mount_bdev(fs_type, flags, dev_name, data, HUST_fs_fill_super);
/* Use IS_ERR to find out if this pointer is a valid pointer to data
* or if it indicates an error condition. */
if (IS_ERR(ret))
printk(KERN_ERR "Error mounting HUST_fs\n");
else
printk(KERN_INFO "HUST_fs is succesfully mounted on [%s]\n",
dev_name);
return ret;
}
void HUST_fs_kill_superblock(struct super_block *s)
{
/* We don't do anything here, but it's important to call
* kill_block_super because it frees some internal resources. */
kill_block_super(s);
printk(KERN_INFO
"HUST_fs superblock is destroyed. Unmount succesful.\n");
}
/* Called when the module is loaded. */
int HUST_fs_init(void)
{
int ret;
ret = register_filesystem(&HUST_fs_type);
if (ret == 0)
printk(KERN_INFO "Sucessfully registered HUST_fs\n");
else
printk(KERN_ERR "Failed to register HUST_fs. Error: [%d]\n",
ret);
return ret;
}
/* Called when the module is unloaded. */
void HUST_fs_exit(void)
{
int ret;
ret = unregister_filesystem(&HUST_fs_type);
if (ret == 0)
printk(KERN_INFO "Sucessfully unregistered HUST_fs\n");
else
printk(KERN_ERR "Failed to unregister HUST_fs. Error: [%d]\n",
ret);
}
module_init(HUST_fs_init);
module_exit(HUST_fs_exit);
MODULE_LICENSE("MIT");
MODULE_AUTHOR("cv");