Browse Source

FIX: ext4_frename does not check whether the target file exist or not.

pull/7/head
ngkaho1234 9 years ago
parent
commit
9f0a68086c
  1. 63
      lwext4/ext4.c

63
lwext4/ext4.c

@ -180,7 +180,7 @@ static int ext4_has_children(bool *has_children, struct ext4_inode_ref *enode)
static int ext4_link(struct ext4_mountpoint *mp, struct ext4_inode_ref *parent, static int ext4_link(struct ext4_mountpoint *mp, struct ext4_inode_ref *parent,
struct ext4_inode_ref *child, const char *name, struct ext4_inode_ref *child, const char *name,
uint32_t name_len) uint32_t name_len, bool rename)
{ {
/* Check maximum name length */ /* Check maximum name length */
if (name_len > EXT4_DIRECTORY_FILENAME_LEN) if (name_len > EXT4_DIRECTORY_FILENAME_LEN)
@ -196,7 +196,7 @@ static int ext4_link(struct ext4_mountpoint *mp, struct ext4_inode_ref *parent,
*/ */
if (ext4_inode_is_type(&mp->fs.sb, child->inode, if (ext4_inode_is_type(&mp->fs.sb, child->inode,
EXT4_INODE_MODE_DIRECTORY) && EXT4_INODE_MODE_DIRECTORY) &&
ext4_inode_get_links_count(child->inode) == 0) { !rename) {
rc = ext4_dir_add_entry(child, ".", strlen("."), child); rc = ext4_dir_add_entry(child, ".", strlen("."), child);
if (rc != EOK) { if (rc != EOK) {
ext4_dir_remove_entry(parent, name, strlen(name)); ext4_dir_remove_entry(parent, name, strlen(name));
@ -231,6 +231,10 @@ static int ext4_link(struct ext4_mountpoint *mp, struct ext4_inode_ref *parent,
child->dirty = true; child->dirty = true;
parent->dirty = true; parent->dirty = true;
} else { } else {
/*
* In case we want to rename a directory,
* we reset the original '..' pointer.
*/
if (ext4_inode_is_type(&mp->fs.sb, child->inode, if (ext4_inode_is_type(&mp->fs.sb, child->inode,
EXT4_INODE_MODE_DIRECTORY)) { EXT4_INODE_MODE_DIRECTORY)) {
int has_flag_index = int has_flag_index =
@ -263,7 +267,8 @@ static int ext4_link(struct ext4_mountpoint *mp, struct ext4_inode_ref *parent,
ext4_fs_inode_links_count_inc(parent); ext4_fs_inode_links_count_inc(parent);
parent->dirty = true; parent->dirty = true;
} else { }
if (!rename) {
ext4_fs_inode_links_count_inc(child); ext4_fs_inode_links_count_inc(child);
child->dirty = true; child->dirty = true;
} }
@ -632,6 +637,9 @@ static int ext4_generic_open2(ext4_file *f, const char *path, int flags,
r = ext4_dir_find_entry(&result, &ref, path, len); r = ext4_dir_find_entry(&result, &ref, path, len);
if (r != EOK) { if (r != EOK) {
/*Destroy last result*/
ext4_dir_destroy_result(&ref, &result);
if (r != ENOENT) if (r != ENOENT)
break; break;
@ -647,11 +655,9 @@ static int ext4_generic_open2(ext4_file *f, const char *path, int flags,
if (r != EOK) if (r != EOK)
break; break;
/*Destroy last result*/
ext4_dir_destroy_result(&ref, &result);
/*Link with root dir.*/ /*Link with root dir.*/
r = ext4_link(mp, &ref, &child_ref, path, len); r = ext4_link(mp, &ref, &child_ref, path, len, false);
if (r != EOK) { if (r != EOK) {
/*Fail. Free new inode.*/ /*Fail. Free new inode.*/
ext4_fs_free_inode(&child_ref); ext4_fs_free_inode(&child_ref);
@ -758,7 +764,8 @@ static int ext4_generic_open(ext4_file *f, const char *path, const char *flags,
} }
static int __ext4_create_hardlink(const char *path, static int __ext4_create_hardlink(const char *path,
struct ext4_inode_ref *child_ref) struct ext4_inode_ref *child_ref,
bool rename)
{ {
bool is_goal = false; bool is_goal = false;
uint8_t inode_type = EXT4_DIRENTRY_DIR; uint8_t inode_type = EXT4_DIRENTRY_DIR;
@ -799,14 +806,19 @@ static int __ext4_create_hardlink(const char *path,
r = ext4_dir_find_entry(&result, &ref, path, len); r = ext4_dir_find_entry(&result, &ref, path, len);
if (r != EOK) { if (r != EOK) {
/*Destroy last result*/
ext4_dir_destroy_result(&ref, &result);
if (r != ENOENT || !is_goal) if (r != ENOENT || !is_goal)
break; break;
/*Link with root dir.*/
r = ext4_link(mp, &ref, child_ref, path, len, rename);
break;
} else if (r == EOK && is_goal) {
/*Destroy last result*/ /*Destroy last result*/
ext4_dir_destroy_result(&ref, &result); ext4_dir_destroy_result(&ref, &result);
r = EEXIST;
/*Link with root dir.*/
r = ext4_link(mp, &ref, child_ref, path, len);
break; break;
} }
@ -850,10 +862,10 @@ static int __ext4_create_hardlink(const char *path,
return r; return r;
} }
static int __ext4_remove_hardlink(const char *path, static int ext4_remove_orig_reference(const char *path,
uint32_t name_off, uint32_t name_off,
struct ext4_inode_ref *parent_ref, struct ext4_inode_ref *parent_ref,
struct ext4_inode_ref *child_ref) struct ext4_inode_ref *child_ref)
{ {
bool is_goal; bool is_goal;
int r; int r;
@ -868,11 +880,16 @@ static int __ext4_remove_hardlink(const char *path,
len = ext4_path_check(path, &is_goal); len = ext4_path_check(path, &is_goal);
/*Unlink from parent*/ /* Remove entry from parent directory */
r = ext4_unlink(mp, parent_ref, child_ref, path, len); r = ext4_dir_remove_entry(parent_ref, path, len);
if (r != EOK) if (r != EOK)
goto Finish; goto Finish;
if (ext4_inode_is_type(&mp->fs.sb, child_ref->inode,
EXT4_INODE_MODE_DIRECTORY)) {
ext4_fs_inode_links_count_dec(parent_ref);
parent_ref->dirty = true;
}
Finish: Finish:
return r; return r;
} }
@ -913,7 +930,14 @@ int ext4_flink(const char *path, const char *hardlink_path)
child_loaded = true; child_loaded = true;
r = __ext4_create_hardlink(hardlink_path, &child_ref); /* Creating hardlink for directory is not allowed. */
if (ext4_inode_is_type(&mp->fs.sb, child_ref.inode,
EXT4_INODE_MODE_DIRECTORY)) {
r = EINVAL;
goto Finish;
}
r = __ext4_create_hardlink(hardlink_path, &child_ref, false);
Finish: Finish:
if (child_loaded) if (child_loaded)
@ -962,11 +986,12 @@ int ext4_frename(const char *path, const char *new_path)
child_loaded = true; child_loaded = true;
r = __ext4_create_hardlink(new_path, &child_ref); r = __ext4_create_hardlink(new_path, &child_ref, true);
if (r != EOK) if (r != EOK)
goto Finish; goto Finish;
r = __ext4_remove_hardlink(path, name_off, &parent_ref, &child_ref); r = ext4_remove_orig_reference(path, name_off,
&parent_ref, &child_ref);
if (r != EOK) if (r != EOK)
goto Finish; goto Finish;

Loading…
Cancel
Save