diff options
Diffstat (limited to 'fs/fat/dir.c')
-rw-r--r-- | fs/fat/dir.c | 13 |
1 files changed, 11 insertions, 2 deletions
diff --git a/fs/fat/dir.c b/fs/fat/dir.c index 1bda2ab6745b..814ad2c2ba80 100644 --- a/fs/fat/dir.c +++ b/fs/fat/dir.c | |||
@@ -1100,8 +1100,11 @@ static int fat_zeroed_cluster(struct inode *dir, sector_t blknr, int nr_used, | |||
1100 | err = -ENOMEM; | 1100 | err = -ENOMEM; |
1101 | goto error; | 1101 | goto error; |
1102 | } | 1102 | } |
1103 | /* Avoid race with userspace read via bdev */ | ||
1104 | lock_buffer(bhs[n]); | ||
1103 | memset(bhs[n]->b_data, 0, sb->s_blocksize); | 1105 | memset(bhs[n]->b_data, 0, sb->s_blocksize); |
1104 | set_buffer_uptodate(bhs[n]); | 1106 | set_buffer_uptodate(bhs[n]); |
1107 | unlock_buffer(bhs[n]); | ||
1105 | mark_buffer_dirty_inode(bhs[n], dir); | 1108 | mark_buffer_dirty_inode(bhs[n], dir); |
1106 | 1109 | ||
1107 | n++; | 1110 | n++; |
@@ -1158,6 +1161,8 @@ int fat_alloc_new_dir(struct inode *dir, struct timespec64 *ts) | |||
1158 | fat_time_unix2fat(sbi, ts, &time, &date, &time_cs); | 1161 | fat_time_unix2fat(sbi, ts, &time, &date, &time_cs); |
1159 | 1162 | ||
1160 | de = (struct msdos_dir_entry *)bhs[0]->b_data; | 1163 | de = (struct msdos_dir_entry *)bhs[0]->b_data; |
1164 | /* Avoid race with userspace read via bdev */ | ||
1165 | lock_buffer(bhs[0]); | ||
1161 | /* filling the new directory slots ("." and ".." entries) */ | 1166 | /* filling the new directory slots ("." and ".." entries) */ |
1162 | memcpy(de[0].name, MSDOS_DOT, MSDOS_NAME); | 1167 | memcpy(de[0].name, MSDOS_DOT, MSDOS_NAME); |
1163 | memcpy(de[1].name, MSDOS_DOTDOT, MSDOS_NAME); | 1168 | memcpy(de[1].name, MSDOS_DOTDOT, MSDOS_NAME); |
@@ -1180,6 +1185,7 @@ int fat_alloc_new_dir(struct inode *dir, struct timespec64 *ts) | |||
1180 | de[0].size = de[1].size = 0; | 1185 | de[0].size = de[1].size = 0; |
1181 | memset(de + 2, 0, sb->s_blocksize - 2 * sizeof(*de)); | 1186 | memset(de + 2, 0, sb->s_blocksize - 2 * sizeof(*de)); |
1182 | set_buffer_uptodate(bhs[0]); | 1187 | set_buffer_uptodate(bhs[0]); |
1188 | unlock_buffer(bhs[0]); | ||
1183 | mark_buffer_dirty_inode(bhs[0], dir); | 1189 | mark_buffer_dirty_inode(bhs[0], dir); |
1184 | 1190 | ||
1185 | err = fat_zeroed_cluster(dir, blknr, 1, bhs, MAX_BUF_PER_PAGE); | 1191 | err = fat_zeroed_cluster(dir, blknr, 1, bhs, MAX_BUF_PER_PAGE); |
@@ -1237,11 +1243,14 @@ static int fat_add_new_entries(struct inode *dir, void *slots, int nr_slots, | |||
1237 | 1243 | ||
1238 | /* fill the directory entry */ | 1244 | /* fill the directory entry */ |
1239 | copy = min(size, sb->s_blocksize); | 1245 | copy = min(size, sb->s_blocksize); |
1246 | /* Avoid race with userspace read via bdev */ | ||
1247 | lock_buffer(bhs[n]); | ||
1240 | memcpy(bhs[n]->b_data, slots, copy); | 1248 | memcpy(bhs[n]->b_data, slots, copy); |
1241 | slots += copy; | ||
1242 | size -= copy; | ||
1243 | set_buffer_uptodate(bhs[n]); | 1249 | set_buffer_uptodate(bhs[n]); |
1250 | unlock_buffer(bhs[n]); | ||
1244 | mark_buffer_dirty_inode(bhs[n], dir); | 1251 | mark_buffer_dirty_inode(bhs[n], dir); |
1252 | slots += copy; | ||
1253 | size -= copy; | ||
1245 | if (!size) | 1254 | if (!size) |
1246 | break; | 1255 | break; |
1247 | n++; | 1256 | n++; |