Commit d97e0606 authored by Yuezhang Mo's avatar Yuezhang Mo Committed by Namjae Jeon

exfat: convert exfat_init_ext_entry() to use dentry cache

Before this conversion, in exfat_init_ext_entry(), to init
the dentries in a dentry set, the sync times is equals the
dentry number if 'dirsync' or 'sync' is enabled.
That affects not only performance but also device life.

After this conversion, only needs to be synchronized once if
'dirsync' or 'sync' is enabled.
Signed-off-by: default avatarYuezhang Mo <Yuezhang.Mo@sony.com>
Reviewed-by: default avatarAndy Wu <Andy.Wu@sony.com>
Reviewed-by: default avatarAoyama Wataru <wataru.aoyama@sony.com>
Reviewed-by: default avatarSungjong Seo <sj1557.seo@samsung.com>
Signed-off-by: default avatarNamjae Jeon <linkinjeon@kernel.org>
parent 4e1aa22f
...@@ -532,46 +532,27 @@ static void exfat_free_benign_secondary_clusters(struct inode *inode, ...@@ -532,46 +532,27 @@ static void exfat_free_benign_secondary_clusters(struct inode *inode,
exfat_free_cluster(inode, &dir); exfat_free_cluster(inode, &dir);
} }
int exfat_init_ext_entry(struct inode *inode, struct exfat_chain *p_dir, void exfat_init_ext_entry(struct exfat_entry_set_cache *es, int num_entries,
int entry, int num_entries, struct exfat_uni_name *p_uniname) struct exfat_uni_name *p_uniname)
{ {
struct super_block *sb = inode->i_sb;
int i; int i;
unsigned short *uniname = p_uniname->name; unsigned short *uniname = p_uniname->name;
struct exfat_dentry *ep; struct exfat_dentry *ep;
struct buffer_head *bh;
int sync = IS_DIRSYNC(inode);
ep = exfat_get_dentry(sb, p_dir, entry, &bh);
if (!ep)
return -EIO;
ep = exfat_get_dentry_cached(es, ES_IDX_FILE);
ep->dentry.file.num_ext = (unsigned char)(num_entries - 1); ep->dentry.file.num_ext = (unsigned char)(num_entries - 1);
exfat_update_bh(bh, sync);
brelse(bh);
ep = exfat_get_dentry(sb, p_dir, entry + 1, &bh);
if (!ep)
return -EIO;
ep = exfat_get_dentry_cached(es, ES_IDX_STREAM);
ep->dentry.stream.name_len = p_uniname->name_len; ep->dentry.stream.name_len = p_uniname->name_len;
ep->dentry.stream.name_hash = cpu_to_le16(p_uniname->name_hash); ep->dentry.stream.name_hash = cpu_to_le16(p_uniname->name_hash);
exfat_update_bh(bh, sync);
brelse(bh);
for (i = EXFAT_FIRST_CLUSTER; i < num_entries; i++) {
ep = exfat_get_dentry(sb, p_dir, entry + i, &bh);
if (!ep)
return -EIO;
for (i = ES_IDX_FIRST_FILENAME; i < num_entries; i++) {
ep = exfat_get_dentry_cached(es, i);
exfat_init_name_entry(ep, uniname); exfat_init_name_entry(ep, uniname);
exfat_update_bh(bh, sync);
brelse(bh);
uniname += EXFAT_FILE_NAME_LEN; uniname += EXFAT_FILE_NAME_LEN;
} }
exfat_update_dir_chksum(inode, p_dir, entry); exfat_update_dir_chksum_with_entry_set(es);
return 0;
} }
void exfat_remove_entries(struct inode *inode, struct exfat_entry_set_cache *es, void exfat_remove_entries(struct inode *inode, struct exfat_entry_set_cache *es,
......
...@@ -483,8 +483,8 @@ unsigned int exfat_get_entry_type(struct exfat_dentry *p_entry); ...@@ -483,8 +483,8 @@ unsigned int exfat_get_entry_type(struct exfat_dentry *p_entry);
void exfat_init_dir_entry(struct exfat_entry_set_cache *es, void exfat_init_dir_entry(struct exfat_entry_set_cache *es,
unsigned int type, unsigned int start_clu, unsigned int type, unsigned int start_clu,
unsigned long long size, struct timespec64 *ts); unsigned long long size, struct timespec64 *ts);
int exfat_init_ext_entry(struct inode *inode, struct exfat_chain *p_dir, void exfat_init_ext_entry(struct exfat_entry_set_cache *es, int num_entries,
int entry, int num_entries, struct exfat_uni_name *p_uniname); struct exfat_uni_name *p_uniname);
void exfat_remove_entries(struct inode *inode, struct exfat_entry_set_cache *es, void exfat_remove_entries(struct inode *inode, struct exfat_entry_set_cache *es,
int order); int order);
int exfat_update_dir_chksum(struct inode *inode, struct exfat_chain *p_dir, int exfat_update_dir_chksum(struct inode *inode, struct exfat_chain *p_dir,
......
...@@ -539,15 +539,12 @@ static int exfat_add_entry(struct inode *inode, const char *path, ...@@ -539,15 +539,12 @@ static int exfat_add_entry(struct inode *inode, const char *path,
goto out; goto out;
exfat_init_dir_entry(&es, type, start_clu, clu_size, &ts); exfat_init_dir_entry(&es, type, start_clu, clu_size, &ts);
exfat_init_ext_entry(&es, num_entries, &uniname);
ret = exfat_put_dentry_set(&es, IS_DIRSYNC(inode)); ret = exfat_put_dentry_set(&es, IS_DIRSYNC(inode));
if (ret) if (ret)
goto out; goto out;
ret = exfat_init_ext_entry(inode, p_dir, dentry, num_entries, &uniname);
if (ret)
goto out;
info->dir = *p_dir; info->dir = *p_dir;
info->entry = dentry; info->entry = dentry;
info->flags = ALLOC_NO_FAT_CHAIN; info->flags = ALLOC_NO_FAT_CHAIN;
...@@ -1018,8 +1015,7 @@ static int exfat_rename_file(struct inode *inode, struct exfat_chain *p_dir, ...@@ -1018,8 +1015,7 @@ static int exfat_rename_file(struct inode *inode, struct exfat_chain *p_dir,
int ret, num_new_entries; int ret, num_new_entries;
struct exfat_dentry *epold, *epnew; struct exfat_dentry *epold, *epnew;
struct super_block *sb = inode->i_sb; struct super_block *sb = inode->i_sb;
struct buffer_head *new_bh; struct exfat_entry_set_cache old_es, new_es;
struct exfat_entry_set_cache old_es;
int sync = IS_DIRSYNC(inode); int sync = IS_DIRSYNC(inode);
num_new_entries = exfat_calc_num_entries(p_uniname); num_new_entries = exfat_calc_num_entries(p_uniname);
...@@ -1044,33 +1040,25 @@ static int exfat_rename_file(struct inode *inode, struct exfat_chain *p_dir, ...@@ -1044,33 +1040,25 @@ static int exfat_rename_file(struct inode *inode, struct exfat_chain *p_dir,
goto put_old_es; goto put_old_es;
} }
epnew = exfat_get_dentry(sb, p_dir, newentry, &new_bh); ret = exfat_get_empty_dentry_set(&new_es, sb, p_dir, newentry,
if (!epnew) { num_new_entries);
ret = -EIO; if (ret)
goto put_old_es; goto put_old_es;
}
epnew = exfat_get_dentry_cached(&new_es, ES_IDX_FILE);
*epnew = *epold; *epnew = *epold;
if (exfat_get_entry_type(epnew) == TYPE_FILE) { if (exfat_get_entry_type(epnew) == TYPE_FILE) {
epnew->dentry.file.attr |= cpu_to_le16(EXFAT_ATTR_ARCHIVE); epnew->dentry.file.attr |= cpu_to_le16(EXFAT_ATTR_ARCHIVE);
ei->attr |= EXFAT_ATTR_ARCHIVE; ei->attr |= EXFAT_ATTR_ARCHIVE;
} }
exfat_update_bh(new_bh, sync);
brelse(new_bh);
epold = exfat_get_dentry_cached(&old_es, ES_IDX_STREAM); epold = exfat_get_dentry_cached(&old_es, ES_IDX_STREAM);
epnew = exfat_get_dentry(sb, p_dir, newentry + 1, &new_bh); epnew = exfat_get_dentry_cached(&new_es, ES_IDX_STREAM);
if (!epnew) {
ret = -EIO;
goto put_old_es;
}
*epnew = *epold; *epnew = *epold;
exfat_update_bh(new_bh, sync);
brelse(new_bh);
ret = exfat_init_ext_entry(inode, p_dir, newentry, exfat_init_ext_entry(&new_es, num_new_entries, p_uniname);
num_new_entries, p_uniname);
ret = exfat_put_dentry_set(&new_es, sync);
if (ret) if (ret)
goto put_old_es; goto put_old_es;
...@@ -1084,11 +1072,7 @@ static int exfat_rename_file(struct inode *inode, struct exfat_chain *p_dir, ...@@ -1084,11 +1072,7 @@ static int exfat_rename_file(struct inode *inode, struct exfat_chain *p_dir,
} }
exfat_remove_entries(inode, &old_es, ES_IDX_FIRST_FILENAME + 1); exfat_remove_entries(inode, &old_es, ES_IDX_FIRST_FILENAME + 1);
exfat_init_ext_entry(&old_es, num_new_entries, p_uniname);
ret = exfat_init_ext_entry(inode, p_dir, oldentry,
num_new_entries, p_uniname);
if (ret)
goto put_old_es;
} }
return exfat_put_dentry_set(&old_es, sync); return exfat_put_dentry_set(&old_es, sync);
...@@ -1104,8 +1088,7 @@ static int exfat_move_file(struct inode *inode, struct exfat_chain *p_olddir, ...@@ -1104,8 +1088,7 @@ static int exfat_move_file(struct inode *inode, struct exfat_chain *p_olddir,
int ret, newentry, num_new_entries; int ret, newentry, num_new_entries;
struct exfat_dentry *epmov, *epnew; struct exfat_dentry *epmov, *epnew;
struct super_block *sb = inode->i_sb; struct super_block *sb = inode->i_sb;
struct buffer_head *new_bh; struct exfat_entry_set_cache mov_es, new_es;
struct exfat_entry_set_cache mov_es;
num_new_entries = exfat_calc_num_entries(p_uniname); num_new_entries = exfat_calc_num_entries(p_uniname);
if (num_new_entries < 0) if (num_new_entries < 0)
...@@ -1120,43 +1103,35 @@ static int exfat_move_file(struct inode *inode, struct exfat_chain *p_olddir, ...@@ -1120,43 +1103,35 @@ static int exfat_move_file(struct inode *inode, struct exfat_chain *p_olddir,
if (ret) if (ret)
return -EIO; return -EIO;
epmov = exfat_get_dentry_cached(&mov_es, ES_IDX_FILE); ret = exfat_get_empty_dentry_set(&new_es, sb, p_newdir, newentry,
epnew = exfat_get_dentry(sb, p_newdir, newentry, &new_bh); num_new_entries);
if (!epnew) { if (ret)
ret = -EIO;
goto put_mov_es; goto put_mov_es;
}
epmov = exfat_get_dentry_cached(&mov_es, ES_IDX_FILE);
epnew = exfat_get_dentry_cached(&new_es, ES_IDX_FILE);
*epnew = *epmov; *epnew = *epmov;
if (exfat_get_entry_type(epnew) == TYPE_FILE) { if (exfat_get_entry_type(epnew) == TYPE_FILE) {
epnew->dentry.file.attr |= cpu_to_le16(EXFAT_ATTR_ARCHIVE); epnew->dentry.file.attr |= cpu_to_le16(EXFAT_ATTR_ARCHIVE);
ei->attr |= EXFAT_ATTR_ARCHIVE; ei->attr |= EXFAT_ATTR_ARCHIVE;
} }
exfat_update_bh(new_bh, IS_DIRSYNC(inode));
brelse(new_bh);
epmov = exfat_get_dentry_cached(&mov_es, ES_IDX_STREAM); epmov = exfat_get_dentry_cached(&mov_es, ES_IDX_STREAM);
epnew = exfat_get_dentry(sb, p_newdir, newentry + 1, &new_bh); epnew = exfat_get_dentry_cached(&new_es, ES_IDX_STREAM);
if (!epnew) {
ret = -EIO;
goto put_mov_es;
}
*epnew = *epmov; *epnew = *epmov;
exfat_update_bh(new_bh, IS_DIRSYNC(inode));
brelse(new_bh);
ret = exfat_init_ext_entry(inode, p_newdir, newentry, num_new_entries,
p_uniname);
if (ret)
return ret;
exfat_init_ext_entry(&new_es, num_new_entries, p_uniname);
exfat_remove_entries(inode, &mov_es, ES_IDX_FILE); exfat_remove_entries(inode, &mov_es, ES_IDX_FILE);
exfat_chain_set(&ei->dir, p_newdir->dir, p_newdir->size, exfat_chain_set(&ei->dir, p_newdir->dir, p_newdir->size,
p_newdir->flags); p_newdir->flags);
ei->entry = newentry; ei->entry = newentry;
ret = exfat_put_dentry_set(&new_es, IS_DIRSYNC(inode));
if (ret)
goto put_mov_es;
return exfat_put_dentry_set(&mov_es, IS_DIRSYNC(inode)); return exfat_put_dentry_set(&mov_es, IS_DIRSYNC(inode));
put_mov_es: put_mov_es:
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment