diff options
author | Changman Lee <cm224.lee@samsung.com> | 2013-01-22 19:40:23 -0500 |
---|---|---|
committer | Jaegeuk Kim <jaegeuk.kim@samsung.com> | 2013-02-11 17:15:00 -0500 |
commit | 7d79e75f6420fc13cfe72554b3ea5afad24c8625 (patch) | |
tree | 4832f0c8bde80c7d9d806b83d59f133d232b2ac5 /fs/f2fs | |
parent | 836dc9e3fbbab0c30aa6e664417225f5c1fb1c39 (diff) |
f2fs: save device node number into f2fs_inode
This patch stores inode->i_rdev into on-disk inode structure.
Alun reported that:
aspire tmp # mount -t f2fs /dev/sdb mnt
aspire tmp # mknod mnt/sda1 b 8 1
aspire tmp # mknod mnt/null c 1 3
aspire tmp # mknod mnt/console c 5 1
aspire tmp # ls -l mnt
total 2
crw-r--r-- 1 root root 5, 1 Jan 22 18:44 console
crw-r--r-- 1 root root 1, 3 Jan 22 18:44 null
brw-r--r-- 1 root root 8, 1 Jan 22 18:44 sda1
aspire tmp # umount mnt
aspire tmp # mount -t f2fs /dev/sdb mnt
aspire tmp # ls -l mnt
total 2
crw-r--r-- 1 root root 0, 0 Jan 22 18:44 console
crw-r--r-- 1 root root 0, 0 Jan 22 18:44 null
brw-r--r-- 1 root root 0, 0 Jan 22 18:44 sda1
In this report, f2fs lost the major/minor numbers of device files after umount.
The reason was revealed that f2fs does not store the inode->i_rdev to the
on-disk inode data structure.
So, as the other file systems do, f2fs also stores i_rdev into the i_addr fields
in on-disk inode structure without any on-disk layout changes.
Note that, this bug is limited to device files made by mknod().
Reported-and-Tested-by: Alun Jones <alun.linux@ty-penguin.org.uk>
Signed-off-by: Changman Lee <cm224.lee@samsung.com>
Signed-off-by: Jaegeuk Kim <jaegeuk.kim@samsung.com>
Diffstat (limited to 'fs/f2fs')
-rw-r--r-- | fs/f2fs/inode.c | 18 |
1 files changed, 18 insertions, 0 deletions
diff --git a/fs/f2fs/inode.c b/fs/f2fs/inode.c index 794241777322..340a21be5a76 100644 --- a/fs/f2fs/inode.c +++ b/fs/f2fs/inode.c | |||
@@ -100,6 +100,10 @@ static int do_read_inode(struct inode *inode) | |||
100 | inode->i_ctime.tv_nsec = le32_to_cpu(ri->i_ctime_nsec); | 100 | inode->i_ctime.tv_nsec = le32_to_cpu(ri->i_ctime_nsec); |
101 | inode->i_mtime.tv_nsec = le32_to_cpu(ri->i_mtime_nsec); | 101 | inode->i_mtime.tv_nsec = le32_to_cpu(ri->i_mtime_nsec); |
102 | inode->i_generation = le32_to_cpu(ri->i_generation); | 102 | inode->i_generation = le32_to_cpu(ri->i_generation); |
103 | if (ri->i_addr[0]) | ||
104 | inode->i_rdev = old_decode_dev(le32_to_cpu(ri->i_addr[0])); | ||
105 | else | ||
106 | inode->i_rdev = new_decode_dev(le32_to_cpu(ri->i_addr[1])); | ||
103 | 107 | ||
104 | fi->i_current_depth = le32_to_cpu(ri->i_current_depth); | 108 | fi->i_current_depth = le32_to_cpu(ri->i_current_depth); |
105 | fi->i_xattr_nid = le32_to_cpu(ri->i_xattr_nid); | 109 | fi->i_xattr_nid = le32_to_cpu(ri->i_xattr_nid); |
@@ -203,6 +207,20 @@ void update_inode(struct inode *inode, struct page *node_page) | |||
203 | ri->i_flags = cpu_to_le32(F2FS_I(inode)->i_flags); | 207 | ri->i_flags = cpu_to_le32(F2FS_I(inode)->i_flags); |
204 | ri->i_pino = cpu_to_le32(F2FS_I(inode)->i_pino); | 208 | ri->i_pino = cpu_to_le32(F2FS_I(inode)->i_pino); |
205 | ri->i_generation = cpu_to_le32(inode->i_generation); | 209 | ri->i_generation = cpu_to_le32(inode->i_generation); |
210 | |||
211 | if (S_ISCHR(inode->i_mode) || S_ISBLK(inode->i_mode)) { | ||
212 | if (old_valid_dev(inode->i_rdev)) { | ||
213 | ri->i_addr[0] = | ||
214 | cpu_to_le32(old_encode_dev(inode->i_rdev)); | ||
215 | ri->i_addr[1] = 0; | ||
216 | } else { | ||
217 | ri->i_addr[0] = 0; | ||
218 | ri->i_addr[1] = | ||
219 | cpu_to_le32(new_encode_dev(inode->i_rdev)); | ||
220 | ri->i_addr[2] = 0; | ||
221 | } | ||
222 | } | ||
223 | |||
206 | set_cold_node(inode, node_page); | 224 | set_cold_node(inode, node_page); |
207 | set_page_dirty(node_page); | 225 | set_page_dirty(node_page); |
208 | } | 226 | } |