diff options
Diffstat (limited to 'fs/fat')
-rw-r--r-- | fs/fat/fat.h | 20 | ||||
-rw-r--r-- | fs/fat/file.c | 32 | ||||
-rw-r--r-- | fs/fat/inode.c | 19 |
3 files changed, 40 insertions, 31 deletions
diff --git a/fs/fat/fat.h b/fs/fat/fat.h index 2b8e94c3eef4..3b4753a024e3 100644 --- a/fs/fat/fat.h +++ b/fs/fat/fat.h | |||
@@ -117,14 +117,32 @@ static inline struct msdos_inode_info *MSDOS_I(struct inode *inode) | |||
117 | return container_of(inode, struct msdos_inode_info, vfs_inode); | 117 | return container_of(inode, struct msdos_inode_info, vfs_inode); |
118 | } | 118 | } |
119 | 119 | ||
120 | /* Convert attribute bits and a mask to the UNIX mode. */ | ||
121 | static inline mode_t fat_make_mode(struct msdos_sb_info *sbi, | ||
122 | u8 attrs, mode_t mode) | ||
123 | { | ||
124 | if (attrs & ATTR_RO) | ||
125 | mode &= ~S_IWUGO; | ||
126 | |||
127 | if (attrs & ATTR_DIR) | ||
128 | return (mode & ~sbi->options.fs_dmask) | S_IFDIR; | ||
129 | else | ||
130 | return (mode & ~sbi->options.fs_fmask) | S_IFREG; | ||
131 | } | ||
132 | |||
120 | /* Return the FAT attribute byte for this inode */ | 133 | /* Return the FAT attribute byte for this inode */ |
121 | static inline u8 fat_attr(struct inode *inode) | 134 | static inline u8 fat_make_attrs(struct inode *inode) |
122 | { | 135 | { |
123 | return ((inode->i_mode & S_IWUGO) ? ATTR_NONE : ATTR_RO) | | 136 | return ((inode->i_mode & S_IWUGO) ? ATTR_NONE : ATTR_RO) | |
124 | (S_ISDIR(inode->i_mode) ? ATTR_DIR : ATTR_NONE) | | 137 | (S_ISDIR(inode->i_mode) ? ATTR_DIR : ATTR_NONE) | |
125 | MSDOS_I(inode)->i_attrs; | 138 | MSDOS_I(inode)->i_attrs; |
126 | } | 139 | } |
127 | 140 | ||
141 | static inline void fat_save_attrs(struct inode *inode, u8 attrs) | ||
142 | { | ||
143 | MSDOS_I(inode)->i_attrs = attrs & ATTR_UNUSED; | ||
144 | } | ||
145 | |||
128 | static inline unsigned char fat_checksum(const __u8 *name) | 146 | static inline unsigned char fat_checksum(const __u8 *name) |
129 | { | 147 | { |
130 | unsigned char s = name[0]; | 148 | unsigned char s = name[0]; |
diff --git a/fs/fat/file.c b/fs/fat/file.c index b21973f266a1..f5a7e907a8fa 100644 --- a/fs/fat/file.c +++ b/fs/fat/file.c | |||
@@ -27,13 +27,7 @@ int fat_generic_ioctl(struct inode *inode, struct file *filp, | |||
27 | switch (cmd) { | 27 | switch (cmd) { |
28 | case FAT_IOCTL_GET_ATTRIBUTES: | 28 | case FAT_IOCTL_GET_ATTRIBUTES: |
29 | { | 29 | { |
30 | u32 attr; | 30 | u32 attr = fat_make_attrs(inode); |
31 | |||
32 | if (inode->i_ino == MSDOS_ROOT_INO) | ||
33 | attr = ATTR_DIR; | ||
34 | else | ||
35 | attr = fat_attr(inode); | ||
36 | |||
37 | return put_user(attr, user_attr); | 31 | return put_user(attr, user_attr); |
38 | } | 32 | } |
39 | case FAT_IOCTL_SET_ATTRIBUTES: | 33 | case FAT_IOCTL_SET_ATTRIBUTES: |
@@ -62,20 +56,16 @@ int fat_generic_ioctl(struct inode *inode, struct file *filp, | |||
62 | /* Merge in ATTR_VOLUME and ATTR_DIR */ | 56 | /* Merge in ATTR_VOLUME and ATTR_DIR */ |
63 | attr |= (MSDOS_I(inode)->i_attrs & ATTR_VOLUME) | | 57 | attr |= (MSDOS_I(inode)->i_attrs & ATTR_VOLUME) | |
64 | (is_dir ? ATTR_DIR : 0); | 58 | (is_dir ? ATTR_DIR : 0); |
65 | oldattr = fat_attr(inode); | 59 | oldattr = fat_make_attrs(inode); |
66 | 60 | ||
67 | /* Equivalent to a chmod() */ | 61 | /* Equivalent to a chmod() */ |
68 | ia.ia_valid = ATTR_MODE | ATTR_CTIME; | 62 | ia.ia_valid = ATTR_MODE | ATTR_CTIME; |
69 | ia.ia_ctime = current_fs_time(inode->i_sb); | 63 | ia.ia_ctime = current_fs_time(inode->i_sb); |
70 | if (is_dir) { | 64 | if (is_dir) |
71 | ia.ia_mode = MSDOS_MKMODE(attr, | 65 | ia.ia_mode = fat_make_mode(sbi, attr, S_IRWXUGO); |
72 | S_IRWXUGO & ~sbi->options.fs_dmask) | 66 | else { |
73 | | S_IFDIR; | 67 | ia.ia_mode = fat_make_mode(sbi, attr, |
74 | } else { | 68 | S_IRUGO | S_IWUGO | (inode->i_mode & S_IXUGO)); |
75 | ia.ia_mode = MSDOS_MKMODE(attr, | ||
76 | (S_IRUGO | S_IWUGO | (inode->i_mode & S_IXUGO)) | ||
77 | & ~sbi->options.fs_fmask) | ||
78 | | S_IFREG; | ||
79 | } | 69 | } |
80 | 70 | ||
81 | /* The root directory has no attributes */ | 71 | /* The root directory has no attributes */ |
@@ -115,7 +105,7 @@ int fat_generic_ioctl(struct inode *inode, struct file *filp, | |||
115 | inode->i_flags &= S_IMMUTABLE; | 105 | inode->i_flags &= S_IMMUTABLE; |
116 | } | 106 | } |
117 | 107 | ||
118 | MSDOS_I(inode)->i_attrs = attr & ATTR_UNUSED; | 108 | fat_save_attrs(inode, attr); |
119 | mark_inode_dirty(inode); | 109 | mark_inode_dirty(inode); |
120 | up: | 110 | up: |
121 | mnt_drop_write(filp->f_path.mnt); | 111 | mnt_drop_write(filp->f_path.mnt); |
@@ -274,7 +264,7 @@ static int fat_sanitize_mode(const struct msdos_sb_info *sbi, | |||
274 | 264 | ||
275 | /* | 265 | /* |
276 | * Note, the basic check is already done by a caller of | 266 | * Note, the basic check is already done by a caller of |
277 | * (attr->ia_mode & ~MSDOS_VALID_MODE) | 267 | * (attr->ia_mode & ~FAT_VALID_MODE) |
278 | */ | 268 | */ |
279 | 269 | ||
280 | if (S_ISREG(inode->i_mode)) | 270 | if (S_ISREG(inode->i_mode)) |
@@ -314,6 +304,8 @@ static int fat_allow_set_time(struct msdos_sb_info *sbi, struct inode *inode) | |||
314 | } | 304 | } |
315 | 305 | ||
316 | #define TIMES_SET_FLAGS (ATTR_MTIME_SET | ATTR_ATIME_SET | ATTR_TIMES_SET) | 306 | #define TIMES_SET_FLAGS (ATTR_MTIME_SET | ATTR_ATIME_SET | ATTR_TIMES_SET) |
307 | /* valid file mode bits */ | ||
308 | #define FAT_VALID_MODE (S_IFREG | S_IFDIR | S_IRWXUGO) | ||
317 | 309 | ||
318 | int fat_setattr(struct dentry *dentry, struct iattr *attr) | 310 | int fat_setattr(struct dentry *dentry, struct iattr *attr) |
319 | { | 311 | { |
@@ -356,7 +348,7 @@ int fat_setattr(struct dentry *dentry, struct iattr *attr) | |||
356 | ((attr->ia_valid & ATTR_GID) && | 348 | ((attr->ia_valid & ATTR_GID) && |
357 | (attr->ia_gid != sbi->options.fs_gid)) || | 349 | (attr->ia_gid != sbi->options.fs_gid)) || |
358 | ((attr->ia_valid & ATTR_MODE) && | 350 | ((attr->ia_valid & ATTR_MODE) && |
359 | (attr->ia_mode & ~MSDOS_VALID_MODE))) | 351 | (attr->ia_mode & ~FAT_VALID_MODE))) |
360 | error = -EPERM; | 352 | error = -EPERM; |
361 | 353 | ||
362 | if (error) { | 354 | if (error) { |
diff --git a/fs/fat/inode.c b/fs/fat/inode.c index 8e1b75c63c7f..7aaa21cf019a 100644 --- a/fs/fat/inode.c +++ b/fs/fat/inode.c | |||
@@ -337,8 +337,7 @@ static int fat_fill_inode(struct inode *inode, struct msdos_dir_entry *de) | |||
337 | 337 | ||
338 | if ((de->attr & ATTR_DIR) && !IS_FREE(de->name)) { | 338 | if ((de->attr & ATTR_DIR) && !IS_FREE(de->name)) { |
339 | inode->i_generation &= ~1; | 339 | inode->i_generation &= ~1; |
340 | inode->i_mode = MSDOS_MKMODE(de->attr, | 340 | inode->i_mode = fat_make_mode(sbi, de->attr, S_IRWXUGO); |
341 | S_IRWXUGO & ~sbi->options.fs_dmask) | S_IFDIR; | ||
342 | inode->i_op = sbi->dir_ops; | 341 | inode->i_op = sbi->dir_ops; |
343 | inode->i_fop = &fat_dir_operations; | 342 | inode->i_fop = &fat_dir_operations; |
344 | 343 | ||
@@ -355,10 +354,9 @@ static int fat_fill_inode(struct inode *inode, struct msdos_dir_entry *de) | |||
355 | inode->i_nlink = fat_subdirs(inode); | 354 | inode->i_nlink = fat_subdirs(inode); |
356 | } else { /* not a directory */ | 355 | } else { /* not a directory */ |
357 | inode->i_generation |= 1; | 356 | inode->i_generation |= 1; |
358 | inode->i_mode = MSDOS_MKMODE(de->attr, | 357 | inode->i_mode = fat_make_mode(sbi, de->attr, |
359 | ((sbi->options.showexec && !is_exec(de->name + 8)) | 358 | ((sbi->options.showexec && !is_exec(de->name + 8)) |
360 | ? S_IRUGO|S_IWUGO : S_IRWXUGO) | 359 | ? S_IRUGO|S_IWUGO : S_IRWXUGO)); |
361 | & ~sbi->options.fs_fmask) | S_IFREG; | ||
362 | MSDOS_I(inode)->i_start = le16_to_cpu(de->start); | 360 | MSDOS_I(inode)->i_start = le16_to_cpu(de->start); |
363 | if (sbi->fat_bits == 32) | 361 | if (sbi->fat_bits == 32) |
364 | MSDOS_I(inode)->i_start |= (le16_to_cpu(de->starthi) << 16); | 362 | MSDOS_I(inode)->i_start |= (le16_to_cpu(de->starthi) << 16); |
@@ -374,7 +372,8 @@ static int fat_fill_inode(struct inode *inode, struct msdos_dir_entry *de) | |||
374 | if (sbi->options.sys_immutable) | 372 | if (sbi->options.sys_immutable) |
375 | inode->i_flags |= S_IMMUTABLE; | 373 | inode->i_flags |= S_IMMUTABLE; |
376 | } | 374 | } |
377 | MSDOS_I(inode)->i_attrs = de->attr & ATTR_UNUSED; | 375 | fat_save_attrs(inode, de->attr); |
376 | |||
378 | inode->i_blocks = ((inode->i_size + (sbi->cluster_size - 1)) | 377 | inode->i_blocks = ((inode->i_size + (sbi->cluster_size - 1)) |
379 | & ~((loff_t)sbi->cluster_size - 1)) >> 9; | 378 | & ~((loff_t)sbi->cluster_size - 1)) >> 9; |
380 | 379 | ||
@@ -569,7 +568,7 @@ retry: | |||
569 | raw_entry->size = 0; | 568 | raw_entry->size = 0; |
570 | else | 569 | else |
571 | raw_entry->size = cpu_to_le32(inode->i_size); | 570 | raw_entry->size = cpu_to_le32(inode->i_size); |
572 | raw_entry->attr = fat_attr(inode); | 571 | raw_entry->attr = fat_make_attrs(inode); |
573 | raw_entry->start = cpu_to_le16(MSDOS_I(inode)->i_logstart); | 572 | raw_entry->start = cpu_to_le16(MSDOS_I(inode)->i_logstart); |
574 | raw_entry->starthi = cpu_to_le16(MSDOS_I(inode)->i_logstart >> 16); | 573 | raw_entry->starthi = cpu_to_le16(MSDOS_I(inode)->i_logstart >> 16); |
575 | fat_time_unix2fat(sbi, &inode->i_mtime, &raw_entry->time, | 574 | fat_time_unix2fat(sbi, &inode->i_mtime, &raw_entry->time, |
@@ -1105,7 +1104,7 @@ static int fat_read_root(struct inode *inode) | |||
1105 | inode->i_gid = sbi->options.fs_gid; | 1104 | inode->i_gid = sbi->options.fs_gid; |
1106 | inode->i_version++; | 1105 | inode->i_version++; |
1107 | inode->i_generation = 0; | 1106 | inode->i_generation = 0; |
1108 | inode->i_mode = (S_IRWXUGO & ~sbi->options.fs_dmask) | S_IFDIR; | 1107 | inode->i_mode = fat_make_mode(sbi, ATTR_DIR, S_IRWXUGO); |
1109 | inode->i_op = sbi->dir_ops; | 1108 | inode->i_op = sbi->dir_ops; |
1110 | inode->i_fop = &fat_dir_operations; | 1109 | inode->i_fop = &fat_dir_operations; |
1111 | if (sbi->fat_bits == 32) { | 1110 | if (sbi->fat_bits == 32) { |
@@ -1122,7 +1121,7 @@ static int fat_read_root(struct inode *inode) | |||
1122 | MSDOS_I(inode)->i_logstart = 0; | 1121 | MSDOS_I(inode)->i_logstart = 0; |
1123 | MSDOS_I(inode)->mmu_private = inode->i_size; | 1122 | MSDOS_I(inode)->mmu_private = inode->i_size; |
1124 | 1123 | ||
1125 | MSDOS_I(inode)->i_attrs = ATTR_NONE; | 1124 | fat_save_attrs(inode, ATTR_DIR); |
1126 | inode->i_mtime.tv_sec = inode->i_atime.tv_sec = inode->i_ctime.tv_sec = 0; | 1125 | inode->i_mtime.tv_sec = inode->i_atime.tv_sec = inode->i_ctime.tv_sec = 0; |
1127 | inode->i_mtime.tv_nsec = inode->i_atime.tv_nsec = inode->i_ctime.tv_nsec = 0; | 1126 | inode->i_mtime.tv_nsec = inode->i_atime.tv_nsec = inode->i_ctime.tv_nsec = 0; |
1128 | inode->i_nlink = fat_subdirs(inode)+2; | 1127 | inode->i_nlink = fat_subdirs(inode)+2; |