diff options
| -rw-r--r-- | fs/ext4/ext4.h | 4 | ||||
| -rw-r--r-- | fs/ext4/namei.c | 115 |
2 files changed, 75 insertions, 44 deletions
diff --git a/fs/ext4/ext4.h b/fs/ext4/ext4.h index 268636af7f5c..cf840146ce81 100644 --- a/fs/ext4/ext4.h +++ b/fs/ext4/ext4.h | |||
| @@ -2415,6 +2415,10 @@ extern void ext4_unwritten_wait(struct inode *inode); | |||
| 2415 | extern const struct inode_operations ext4_dir_inode_operations; | 2415 | extern const struct inode_operations ext4_dir_inode_operations; |
| 2416 | extern const struct inode_operations ext4_special_inode_operations; | 2416 | extern const struct inode_operations ext4_special_inode_operations; |
| 2417 | extern struct dentry *ext4_get_parent(struct dentry *child); | 2417 | extern struct dentry *ext4_get_parent(struct dentry *child); |
| 2418 | extern struct ext4_dir_entry_2 *ext4_init_dot_dotdot(struct inode *inode, | ||
| 2419 | struct ext4_dir_entry_2 *de, | ||
| 2420 | int blocksize, int csum_size, | ||
| 2421 | unsigned int parent_ino, int dotdot_real_len); | ||
| 2418 | 2422 | ||
| 2419 | /* symlink.c */ | 2423 | /* symlink.c */ |
| 2420 | extern const struct inode_operations ext4_symlink_inode_operations; | 2424 | extern const struct inode_operations ext4_symlink_inode_operations; |
diff --git a/fs/ext4/namei.c b/fs/ext4/namei.c index 88e9a2c7e328..edb9f10c1455 100644 --- a/fs/ext4/namei.c +++ b/fs/ext4/namei.c | |||
| @@ -2230,21 +2230,87 @@ retry: | |||
| 2230 | return err; | 2230 | return err; |
| 2231 | } | 2231 | } |
| 2232 | 2232 | ||
| 2233 | static int ext4_mkdir(struct inode *dir, struct dentry *dentry, umode_t mode) | 2233 | struct ext4_dir_entry_2 *ext4_init_dot_dotdot(struct inode *inode, |
| 2234 | struct ext4_dir_entry_2 *de, | ||
| 2235 | int blocksize, int csum_size, | ||
| 2236 | unsigned int parent_ino, int dotdot_real_len) | ||
| 2237 | { | ||
| 2238 | de->inode = cpu_to_le32(inode->i_ino); | ||
| 2239 | de->name_len = 1; | ||
| 2240 | de->rec_len = ext4_rec_len_to_disk(EXT4_DIR_REC_LEN(de->name_len), | ||
| 2241 | blocksize); | ||
| 2242 | strcpy(de->name, "."); | ||
| 2243 | ext4_set_de_type(inode->i_sb, de, S_IFDIR); | ||
| 2244 | |||
| 2245 | de = ext4_next_entry(de, blocksize); | ||
| 2246 | de->inode = cpu_to_le32(parent_ino); | ||
| 2247 | de->name_len = 2; | ||
| 2248 | if (!dotdot_real_len) | ||
| 2249 | de->rec_len = ext4_rec_len_to_disk(blocksize - | ||
| 2250 | (csum_size + EXT4_DIR_REC_LEN(1)), | ||
| 2251 | blocksize); | ||
| 2252 | else | ||
| 2253 | de->rec_len = ext4_rec_len_to_disk( | ||
| 2254 | EXT4_DIR_REC_LEN(de->name_len), blocksize); | ||
| 2255 | strcpy(de->name, ".."); | ||
| 2256 | ext4_set_de_type(inode->i_sb, de, S_IFDIR); | ||
| 2257 | |||
| 2258 | return ext4_next_entry(de, blocksize); | ||
| 2259 | } | ||
| 2260 | |||
| 2261 | static int ext4_init_new_dir(handle_t *handle, struct inode *dir, | ||
| 2262 | struct inode *inode) | ||
| 2234 | { | 2263 | { |
| 2235 | handle_t *handle; | ||
| 2236 | struct inode *inode; | ||
| 2237 | struct buffer_head *dir_block = NULL; | 2264 | struct buffer_head *dir_block = NULL; |
| 2238 | struct ext4_dir_entry_2 *de; | 2265 | struct ext4_dir_entry_2 *de; |
| 2239 | struct ext4_dir_entry_tail *t; | 2266 | struct ext4_dir_entry_tail *t; |
| 2240 | unsigned int blocksize = dir->i_sb->s_blocksize; | 2267 | unsigned int blocksize = dir->i_sb->s_blocksize; |
| 2241 | int csum_size = 0; | 2268 | int csum_size = 0; |
| 2242 | int err, retries = 0; | 2269 | int err; |
| 2243 | 2270 | ||
| 2244 | if (EXT4_HAS_RO_COMPAT_FEATURE(dir->i_sb, | 2271 | if (EXT4_HAS_RO_COMPAT_FEATURE(dir->i_sb, |
| 2245 | EXT4_FEATURE_RO_COMPAT_METADATA_CSUM)) | 2272 | EXT4_FEATURE_RO_COMPAT_METADATA_CSUM)) |
| 2246 | csum_size = sizeof(struct ext4_dir_entry_tail); | 2273 | csum_size = sizeof(struct ext4_dir_entry_tail); |
| 2247 | 2274 | ||
| 2275 | inode->i_size = EXT4_I(inode)->i_disksize = blocksize; | ||
| 2276 | dir_block = ext4_bread(handle, inode, 0, 1, &err); | ||
| 2277 | if (!(dir_block = ext4_bread(handle, inode, 0, 1, &err))) { | ||
| 2278 | if (!err) { | ||
| 2279 | err = -EIO; | ||
| 2280 | ext4_error(inode->i_sb, | ||
| 2281 | "Directory hole detected on inode %lu\n", | ||
| 2282 | inode->i_ino); | ||
| 2283 | } | ||
| 2284 | goto out; | ||
| 2285 | } | ||
| 2286 | BUFFER_TRACE(dir_block, "get_write_access"); | ||
| 2287 | err = ext4_journal_get_write_access(handle, dir_block); | ||
| 2288 | if (err) | ||
| 2289 | goto out; | ||
| 2290 | de = (struct ext4_dir_entry_2 *)dir_block->b_data; | ||
| 2291 | ext4_init_dot_dotdot(inode, de, blocksize, csum_size, dir->i_ino, 0); | ||
| 2292 | set_nlink(inode, 2); | ||
| 2293 | if (csum_size) { | ||
| 2294 | t = EXT4_DIRENT_TAIL(dir_block->b_data, blocksize); | ||
| 2295 | initialize_dirent_tail(t, blocksize); | ||
| 2296 | } | ||
| 2297 | |||
| 2298 | BUFFER_TRACE(dir_block, "call ext4_handle_dirty_metadata"); | ||
| 2299 | err = ext4_handle_dirty_dirent_node(handle, inode, dir_block); | ||
| 2300 | if (err) | ||
| 2301 | goto out; | ||
| 2302 | set_buffer_verified(dir_block); | ||
| 2303 | out: | ||
| 2304 | brelse(dir_block); | ||
| 2305 | return err; | ||
| 2306 | } | ||
| 2307 | |||
| 2308 | static int ext4_mkdir(struct inode *dir, struct dentry *dentry, umode_t mode) | ||
| 2309 | { | ||
| 2310 | handle_t *handle; | ||
| 2311 | struct inode *inode; | ||
| 2312 | int err, retries = 0; | ||
| 2313 | |||
| 2248 | if (EXT4_DIR_LINK_MAX(dir)) | 2314 | if (EXT4_DIR_LINK_MAX(dir)) |
| 2249 | return -EMLINK; | 2315 | return -EMLINK; |
| 2250 | 2316 | ||
| @@ -2268,47 +2334,9 @@ retry: | |||
| 2268 | 2334 | ||
| 2269 | inode->i_op = &ext4_dir_inode_operations; | 2335 | inode->i_op = &ext4_dir_inode_operations; |
| 2270 | inode->i_fop = &ext4_dir_operations; | 2336 | inode->i_fop = &ext4_dir_operations; |
| 2271 | inode->i_size = EXT4_I(inode)->i_disksize = inode->i_sb->s_blocksize; | 2337 | err = ext4_init_new_dir(handle, dir, inode); |
| 2272 | if (!(dir_block = ext4_bread(handle, inode, 0, 1, &err))) { | ||
| 2273 | if (!err) { | ||
| 2274 | err = -EIO; | ||
| 2275 | ext4_error(inode->i_sb, | ||
| 2276 | "Directory hole detected on inode %lu\n", | ||
| 2277 | inode->i_ino); | ||
| 2278 | } | ||
| 2279 | goto out_clear_inode; | ||
| 2280 | } | ||
| 2281 | BUFFER_TRACE(dir_block, "get_write_access"); | ||
| 2282 | err = ext4_journal_get_write_access(handle, dir_block); | ||
| 2283 | if (err) | 2338 | if (err) |
| 2284 | goto out_clear_inode; | 2339 | goto out_clear_inode; |
| 2285 | de = (struct ext4_dir_entry_2 *) dir_block->b_data; | ||
| 2286 | de->inode = cpu_to_le32(inode->i_ino); | ||
| 2287 | de->name_len = 1; | ||
| 2288 | de->rec_len = ext4_rec_len_to_disk(EXT4_DIR_REC_LEN(de->name_len), | ||
| 2289 | blocksize); | ||
| 2290 | strcpy(de->name, "."); | ||
| 2291 | ext4_set_de_type(dir->i_sb, de, S_IFDIR); | ||
| 2292 | de = ext4_next_entry(de, blocksize); | ||
| 2293 | de->inode = cpu_to_le32(dir->i_ino); | ||
| 2294 | de->rec_len = ext4_rec_len_to_disk(blocksize - | ||
| 2295 | (csum_size + EXT4_DIR_REC_LEN(1)), | ||
| 2296 | blocksize); | ||
| 2297 | de->name_len = 2; | ||
| 2298 | strcpy(de->name, ".."); | ||
| 2299 | ext4_set_de_type(dir->i_sb, de, S_IFDIR); | ||
| 2300 | set_nlink(inode, 2); | ||
| 2301 | |||
| 2302 | if (csum_size) { | ||
| 2303 | t = EXT4_DIRENT_TAIL(dir_block->b_data, blocksize); | ||
| 2304 | initialize_dirent_tail(t, blocksize); | ||
| 2305 | } | ||
| 2306 | |||
| 2307 | BUFFER_TRACE(dir_block, "call ext4_handle_dirty_metadata"); | ||
| 2308 | err = ext4_handle_dirty_dirent_node(handle, inode, dir_block); | ||
| 2309 | if (err) | ||
| 2310 | goto out_clear_inode; | ||
| 2311 | set_buffer_verified(dir_block); | ||
| 2312 | err = ext4_mark_inode_dirty(handle, inode); | 2340 | err = ext4_mark_inode_dirty(handle, inode); |
| 2313 | if (!err) | 2341 | if (!err) |
| 2314 | err = ext4_add_entry(handle, dentry, inode); | 2342 | err = ext4_add_entry(handle, dentry, inode); |
| @@ -2328,7 +2356,6 @@ out_clear_inode: | |||
| 2328 | unlock_new_inode(inode); | 2356 | unlock_new_inode(inode); |
| 2329 | d_instantiate(dentry, inode); | 2357 | d_instantiate(dentry, inode); |
| 2330 | out_stop: | 2358 | out_stop: |
| 2331 | brelse(dir_block); | ||
| 2332 | ext4_journal_stop(handle); | 2359 | ext4_journal_stop(handle); |
| 2333 | if (err == -ENOSPC && ext4_should_retry_alloc(dir->i_sb, &retries)) | 2360 | if (err == -ENOSPC && ext4_should_retry_alloc(dir->i_sb, &retries)) |
| 2334 | goto retry; | 2361 | goto retry; |
