Merge tag 'exfat-for-5.8-rc1' of git://git.kernel.org/pub/scm/linux/kernel/git/linkinjeon/exfat
Pull exfat update from Namjae Jeon: "Bug fixes: - Fix memory leak on mount failure with iocharset= option - Fix incorrect update of stream entry - Fix cluster range validation error Clean-ups: - Remove unused code and unneeded assignment - Rename variables in exfat structure as specification - Reorganize boot sector analysis code - Simplify exfat_utf8_d_hash and exfat_utf8_d_cmp() - Optimize exfat entry cache functions - Improve wording of EXFAT_DEFAULT_IOCHARSET config option New Feature: - Add boot region verification" * tag 'exfat-for-5.8-rc1' of git://git.kernel.org/pub/scm/linux/kernel/git/linkinjeon/exfat: exfat: Fix potential use after free in exfat_load_upcase_table() exfat: fix range validation error in alloc and free cluster exfat: fix incorrect update of stream entry in __exfat_truncate() exfat: fix memory leak in exfat_parse_param() exfat: remove unnecessary reassignment of p_uniname->name_len exfat: standardize checksum calculation exfat: add boot region verification exfat: separate the boot sector analysis exfat: redefine PBR as boot_sector exfat: optimize dir-cache exfat: replace 'time_ms' with 'time_cs' exfat: remove the assignment of 0 to bool variable exfat: Remove unused functions exfat_high_surrogate() and exfat_low_surrogate() exfat: Simplify exfat_utf8_d_hash() for code points above U+FFFF exfat: Improve wording of EXFAT_DEFAULT_IOCHARSET config option exfat: Use a more common logging style exfat: Simplify exfat_utf8_d_cmp() for code points above U+FFFF
This commit is contained in:
@@ -16,6 +16,7 @@ config EXFAT_DEFAULT_IOCHARSET
|
|||||||
depends on EXFAT_FS
|
depends on EXFAT_FS
|
||||||
help
|
help
|
||||||
Set this to the default input/output character set to use for
|
Set this to the default input/output character set to use for
|
||||||
converting between the encoding is used for user visible filename and
|
converting between the encoding that is used for user visible
|
||||||
UTF-16 character that exfat filesystem use, and can be overridden with
|
filenames and the UTF-16 character encoding that the exFAT
|
||||||
the "iocharset" mount option for exFAT filesystems.
|
filesystem uses. This can be overridden with the "iocharset" mount
|
||||||
|
option for the exFAT filesystems.
|
||||||
|
@@ -58,8 +58,7 @@ static int exfat_allocate_bitmap(struct super_block *sb,
|
|||||||
need_map_size = ((EXFAT_DATA_CLUSTER_COUNT(sbi) - 1) / BITS_PER_BYTE)
|
need_map_size = ((EXFAT_DATA_CLUSTER_COUNT(sbi) - 1) / BITS_PER_BYTE)
|
||||||
+ 1;
|
+ 1;
|
||||||
if (need_map_size != map_size) {
|
if (need_map_size != map_size) {
|
||||||
exfat_msg(sb, KERN_ERR,
|
exfat_err(sb, "bogus allocation bitmap size(need : %u, cur : %lld)",
|
||||||
"bogus allocation bitmap size(need : %u, cur : %lld)",
|
|
||||||
need_map_size, map_size);
|
need_map_size, map_size);
|
||||||
/*
|
/*
|
||||||
* Only allowed when bogus allocation
|
* Only allowed when bogus allocation
|
||||||
@@ -192,8 +191,7 @@ void exfat_clear_bitmap(struct inode *inode, unsigned int clu)
|
|||||||
(1 << sbi->sect_per_clus_bits), GFP_NOFS, 0);
|
(1 << sbi->sect_per_clus_bits), GFP_NOFS, 0);
|
||||||
|
|
||||||
if (ret_discard == -EOPNOTSUPP) {
|
if (ret_discard == -EOPNOTSUPP) {
|
||||||
exfat_msg(sb, KERN_ERR,
|
exfat_err(sb, "discard not supported by device, disabling");
|
||||||
"discard not supported by device, disabling");
|
|
||||||
opts->discard = 0;
|
opts->discard = 0;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
202
fs/exfat/dir.c
202
fs/exfat/dir.c
@@ -32,35 +32,30 @@ static void exfat_get_uniname_from_ext_entry(struct super_block *sb,
|
|||||||
struct exfat_chain *p_dir, int entry, unsigned short *uniname)
|
struct exfat_chain *p_dir, int entry, unsigned short *uniname)
|
||||||
{
|
{
|
||||||
int i;
|
int i;
|
||||||
struct exfat_dentry *ep;
|
|
||||||
struct exfat_entry_set_cache *es;
|
struct exfat_entry_set_cache *es;
|
||||||
|
|
||||||
es = exfat_get_dentry_set(sb, p_dir, entry, ES_ALL_ENTRIES, &ep);
|
es = exfat_get_dentry_set(sb, p_dir, entry, ES_ALL_ENTRIES);
|
||||||
if (!es)
|
if (!es)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
if (es->num_entries < 3)
|
|
||||||
goto free_es;
|
|
||||||
|
|
||||||
ep += 2;
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* First entry : file entry
|
* First entry : file entry
|
||||||
* Second entry : stream-extension entry
|
* Second entry : stream-extension entry
|
||||||
* Third entry : first file-name entry
|
* Third entry : first file-name entry
|
||||||
* So, the index of first file-name dentry should start from 2.
|
* So, the index of first file-name dentry should start from 2.
|
||||||
*/
|
*/
|
||||||
for (i = 2; i < es->num_entries; i++, ep++) {
|
for (i = 2; i < es->num_entries; i++) {
|
||||||
|
struct exfat_dentry *ep = exfat_get_dentry_cached(es, i);
|
||||||
|
|
||||||
/* end of name entry */
|
/* end of name entry */
|
||||||
if (exfat_get_entry_type(ep) != TYPE_EXTEND)
|
if (exfat_get_entry_type(ep) != TYPE_EXTEND)
|
||||||
goto free_es;
|
break;
|
||||||
|
|
||||||
exfat_extract_uni_name(ep, uniname);
|
exfat_extract_uni_name(ep, uniname);
|
||||||
uniname += EXFAT_FILE_NAME_LEN;
|
uniname += EXFAT_FILE_NAME_LEN;
|
||||||
}
|
}
|
||||||
|
|
||||||
free_es:
|
exfat_free_dentry_set(es, false);
|
||||||
kfree(es);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* read a directory entry from the opened directory */
|
/* read a directory entry from the opened directory */
|
||||||
@@ -137,12 +132,12 @@ static int exfat_readdir(struct inode *inode, struct exfat_dir_entry *dir_entry)
|
|||||||
ep->dentry.file.create_tz,
|
ep->dentry.file.create_tz,
|
||||||
ep->dentry.file.create_time,
|
ep->dentry.file.create_time,
|
||||||
ep->dentry.file.create_date,
|
ep->dentry.file.create_date,
|
||||||
ep->dentry.file.create_time_ms);
|
ep->dentry.file.create_time_cs);
|
||||||
exfat_get_entry_time(sbi, &dir_entry->mtime,
|
exfat_get_entry_time(sbi, &dir_entry->mtime,
|
||||||
ep->dentry.file.modify_tz,
|
ep->dentry.file.modify_tz,
|
||||||
ep->dentry.file.modify_time,
|
ep->dentry.file.modify_time,
|
||||||
ep->dentry.file.modify_date,
|
ep->dentry.file.modify_date,
|
||||||
ep->dentry.file.modify_time_ms);
|
ep->dentry.file.modify_time_cs);
|
||||||
exfat_get_entry_time(sbi, &dir_entry->atime,
|
exfat_get_entry_time(sbi, &dir_entry->atime,
|
||||||
ep->dentry.file.access_tz,
|
ep->dentry.file.access_tz,
|
||||||
ep->dentry.file.access_time,
|
ep->dentry.file.access_time,
|
||||||
@@ -461,12 +456,12 @@ int exfat_init_dir_entry(struct inode *inode, struct exfat_chain *p_dir,
|
|||||||
&ep->dentry.file.create_tz,
|
&ep->dentry.file.create_tz,
|
||||||
&ep->dentry.file.create_time,
|
&ep->dentry.file.create_time,
|
||||||
&ep->dentry.file.create_date,
|
&ep->dentry.file.create_date,
|
||||||
&ep->dentry.file.create_time_ms);
|
&ep->dentry.file.create_time_cs);
|
||||||
exfat_set_entry_time(sbi, &ts,
|
exfat_set_entry_time(sbi, &ts,
|
||||||
&ep->dentry.file.modify_tz,
|
&ep->dentry.file.modify_tz,
|
||||||
&ep->dentry.file.modify_time,
|
&ep->dentry.file.modify_time,
|
||||||
&ep->dentry.file.modify_date,
|
&ep->dentry.file.modify_date,
|
||||||
&ep->dentry.file.modify_time_ms);
|
&ep->dentry.file.modify_time_cs);
|
||||||
exfat_set_entry_time(sbi, &ts,
|
exfat_set_entry_time(sbi, &ts,
|
||||||
&ep->dentry.file.access_tz,
|
&ep->dentry.file.access_tz,
|
||||||
&ep->dentry.file.access_time,
|
&ep->dentry.file.access_time,
|
||||||
@@ -496,7 +491,7 @@ int exfat_update_dir_chksum(struct inode *inode, struct exfat_chain *p_dir,
|
|||||||
int ret = 0;
|
int ret = 0;
|
||||||
int i, num_entries;
|
int i, num_entries;
|
||||||
sector_t sector;
|
sector_t sector;
|
||||||
unsigned short chksum;
|
u16 chksum;
|
||||||
struct exfat_dentry *ep, *fep;
|
struct exfat_dentry *ep, *fep;
|
||||||
struct buffer_head *fbh, *bh;
|
struct buffer_head *fbh, *bh;
|
||||||
|
|
||||||
@@ -505,7 +500,7 @@ int exfat_update_dir_chksum(struct inode *inode, struct exfat_chain *p_dir,
|
|||||||
return -EIO;
|
return -EIO;
|
||||||
|
|
||||||
num_entries = fep->dentry.file.num_ext + 1;
|
num_entries = fep->dentry.file.num_ext + 1;
|
||||||
chksum = exfat_calc_chksum_2byte(fep, DENTRY_SIZE, 0, CS_DIR_ENTRY);
|
chksum = exfat_calc_chksum16(fep, DENTRY_SIZE, 0, CS_DIR_ENTRY);
|
||||||
|
|
||||||
for (i = 1; i < num_entries; i++) {
|
for (i = 1; i < num_entries; i++) {
|
||||||
ep = exfat_get_dentry(sb, p_dir, entry + i, &bh, NULL);
|
ep = exfat_get_dentry(sb, p_dir, entry + i, &bh, NULL);
|
||||||
@@ -513,7 +508,7 @@ int exfat_update_dir_chksum(struct inode *inode, struct exfat_chain *p_dir,
|
|||||||
ret = -EIO;
|
ret = -EIO;
|
||||||
goto release_fbh;
|
goto release_fbh;
|
||||||
}
|
}
|
||||||
chksum = exfat_calc_chksum_2byte(ep, DENTRY_SIZE, chksum,
|
chksum = exfat_calc_chksum16(ep, DENTRY_SIZE, chksum,
|
||||||
CS_DEFAULT);
|
CS_DEFAULT);
|
||||||
brelse(bh);
|
brelse(bh);
|
||||||
}
|
}
|
||||||
@@ -590,62 +585,33 @@ int exfat_remove_entries(struct inode *inode, struct exfat_chain *p_dir,
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
int exfat_update_dir_chksum_with_entry_set(struct super_block *sb,
|
void exfat_update_dir_chksum_with_entry_set(struct exfat_entry_set_cache *es)
|
||||||
struct exfat_entry_set_cache *es, int sync)
|
|
||||||
{
|
{
|
||||||
struct exfat_sb_info *sbi = EXFAT_SB(sb);
|
int chksum_type = CS_DIR_ENTRY, i;
|
||||||
struct buffer_head *bh;
|
|
||||||
sector_t sec = es->sector;
|
|
||||||
unsigned int off = es->offset;
|
|
||||||
int chksum_type = CS_DIR_ENTRY, i, num_entries = es->num_entries;
|
|
||||||
unsigned int buf_off = (off - es->offset);
|
|
||||||
unsigned int remaining_byte_in_sector, copy_entries, clu;
|
|
||||||
unsigned short chksum = 0;
|
unsigned short chksum = 0;
|
||||||
|
struct exfat_dentry *ep;
|
||||||
|
|
||||||
for (i = 0; i < num_entries; i++) {
|
for (i = 0; i < es->num_entries; i++) {
|
||||||
chksum = exfat_calc_chksum_2byte(&es->entries[i], DENTRY_SIZE,
|
ep = exfat_get_dentry_cached(es, i);
|
||||||
chksum, chksum_type);
|
chksum = exfat_calc_chksum16(ep, DENTRY_SIZE, chksum,
|
||||||
|
chksum_type);
|
||||||
chksum_type = CS_DEFAULT;
|
chksum_type = CS_DEFAULT;
|
||||||
}
|
}
|
||||||
|
ep = exfat_get_dentry_cached(es, 0);
|
||||||
|
ep->dentry.file.checksum = cpu_to_le16(chksum);
|
||||||
|
es->modified = true;
|
||||||
|
}
|
||||||
|
|
||||||
es->entries[0].dentry.file.checksum = cpu_to_le16(chksum);
|
void exfat_free_dentry_set(struct exfat_entry_set_cache *es, int sync)
|
||||||
|
{
|
||||||
|
int i;
|
||||||
|
|
||||||
while (num_entries) {
|
for (i = 0; i < es->num_bh; i++) {
|
||||||
/* write per sector base */
|
if (es->modified)
|
||||||
remaining_byte_in_sector = (1 << sb->s_blocksize_bits) - off;
|
exfat_update_bh(es->sb, es->bh[i], sync);
|
||||||
copy_entries = min_t(int,
|
brelse(es->bh[i]);
|
||||||
EXFAT_B_TO_DEN(remaining_byte_in_sector),
|
|
||||||
num_entries);
|
|
||||||
bh = sb_bread(sb, sec);
|
|
||||||
if (!bh)
|
|
||||||
goto err_out;
|
|
||||||
memcpy(bh->b_data + off,
|
|
||||||
(unsigned char *)&es->entries[0] + buf_off,
|
|
||||||
EXFAT_DEN_TO_B(copy_entries));
|
|
||||||
exfat_update_bh(sb, bh, sync);
|
|
||||||
brelse(bh);
|
|
||||||
num_entries -= copy_entries;
|
|
||||||
|
|
||||||
if (num_entries) {
|
|
||||||
/* get next sector */
|
|
||||||
if (exfat_is_last_sector_in_cluster(sbi, sec)) {
|
|
||||||
clu = exfat_sector_to_cluster(sbi, sec);
|
|
||||||
if (es->alloc_flag == ALLOC_NO_FAT_CHAIN)
|
|
||||||
clu++;
|
|
||||||
else if (exfat_get_next_cluster(sb, &clu))
|
|
||||||
goto err_out;
|
|
||||||
sec = exfat_cluster_to_sector(sbi, clu);
|
|
||||||
} else {
|
|
||||||
sec++;
|
|
||||||
}
|
}
|
||||||
off = 0;
|
kfree(es);
|
||||||
buf_off += EXFAT_DEN_TO_B(copy_entries);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
err_out:
|
|
||||||
return -EIO;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static int exfat_walk_fat_chain(struct super_block *sb,
|
static int exfat_walk_fat_chain(struct super_block *sb,
|
||||||
@@ -720,8 +686,7 @@ static int exfat_dir_readahead(struct super_block *sb, sector_t sec)
|
|||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
if (sec < sbi->data_start_sector) {
|
if (sec < sbi->data_start_sector) {
|
||||||
exfat_msg(sb, KERN_ERR,
|
exfat_err(sb, "requested sector is invalid(sect:%llu, root:%llu)",
|
||||||
"requested sector is invalid(sect:%llu, root:%llu)",
|
|
||||||
(unsigned long long)sec, sbi->data_start_sector);
|
(unsigned long long)sec, sbi->data_start_sector);
|
||||||
return -EIO;
|
return -EIO;
|
||||||
}
|
}
|
||||||
@@ -750,7 +715,7 @@ struct exfat_dentry *exfat_get_dentry(struct super_block *sb,
|
|||||||
sector_t sec;
|
sector_t sec;
|
||||||
|
|
||||||
if (p_dir->dir == DIR_DELETED) {
|
if (p_dir->dir == DIR_DELETED) {
|
||||||
exfat_msg(sb, KERN_ERR, "abnormal access to deleted dentry\n");
|
exfat_err(sb, "abnormal access to deleted dentry");
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -821,39 +786,45 @@ static bool exfat_validate_entry(unsigned int type,
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
struct exfat_dentry *exfat_get_dentry_cached(
|
||||||
|
struct exfat_entry_set_cache *es, int num)
|
||||||
|
{
|
||||||
|
int off = es->start_off + num * DENTRY_SIZE;
|
||||||
|
struct buffer_head *bh = es->bh[EXFAT_B_TO_BLK(off, es->sb)];
|
||||||
|
char *p = bh->b_data + EXFAT_BLK_OFFSET(off, es->sb);
|
||||||
|
|
||||||
|
return (struct exfat_dentry *)p;
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Returns a set of dentries for a file or dir.
|
* Returns a set of dentries for a file or dir.
|
||||||
*
|
*
|
||||||
* Note that this is a copy (dump) of dentries so that user should
|
* Note It provides a direct pointer to bh->data via exfat_get_dentry_cached().
|
||||||
* call write_entry_set() to apply changes made in this entry set
|
* User should call exfat_get_dentry_set() after setting 'modified' to apply
|
||||||
* to the real device.
|
* changes made in this entry set to the real device.
|
||||||
*
|
*
|
||||||
* in:
|
* in:
|
||||||
* sb+p_dir+entry: indicates a file/dir
|
* sb+p_dir+entry: indicates a file/dir
|
||||||
* type: specifies how many dentries should be included.
|
* type: specifies how many dentries should be included.
|
||||||
* out:
|
|
||||||
* file_ep: will point the first dentry(= file dentry) on success
|
|
||||||
* return:
|
* return:
|
||||||
* pointer of entry set on success,
|
* pointer of entry set on success,
|
||||||
* NULL on failure.
|
* NULL on failure.
|
||||||
*/
|
*/
|
||||||
struct exfat_entry_set_cache *exfat_get_dentry_set(struct super_block *sb,
|
struct exfat_entry_set_cache *exfat_get_dentry_set(struct super_block *sb,
|
||||||
struct exfat_chain *p_dir, int entry, unsigned int type,
|
struct exfat_chain *p_dir, int entry, unsigned int type)
|
||||||
struct exfat_dentry **file_ep)
|
|
||||||
{
|
{
|
||||||
int ret;
|
int ret, i, num_bh;
|
||||||
unsigned int off, byte_offset, clu = 0;
|
unsigned int off, byte_offset, clu = 0;
|
||||||
unsigned int entry_type;
|
|
||||||
sector_t sec;
|
sector_t sec;
|
||||||
struct exfat_sb_info *sbi = EXFAT_SB(sb);
|
struct exfat_sb_info *sbi = EXFAT_SB(sb);
|
||||||
struct exfat_entry_set_cache *es;
|
struct exfat_entry_set_cache *es;
|
||||||
struct exfat_dentry *ep, *pos;
|
struct exfat_dentry *ep;
|
||||||
unsigned char num_entries;
|
int num_entries;
|
||||||
enum exfat_validate_dentry_mode mode = ES_MODE_STARTED;
|
enum exfat_validate_dentry_mode mode = ES_MODE_STARTED;
|
||||||
struct buffer_head *bh;
|
struct buffer_head *bh;
|
||||||
|
|
||||||
if (p_dir->dir == DIR_DELETED) {
|
if (p_dir->dir == DIR_DELETED) {
|
||||||
exfat_msg(sb, KERN_ERR, "access to deleted dentry\n");
|
exfat_err(sb, "access to deleted dentry");
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -862,11 +833,18 @@ struct exfat_entry_set_cache *exfat_get_dentry_set(struct super_block *sb,
|
|||||||
if (ret)
|
if (ret)
|
||||||
return NULL;
|
return NULL;
|
||||||
|
|
||||||
|
es = kzalloc(sizeof(*es), GFP_KERNEL);
|
||||||
|
if (!es)
|
||||||
|
return NULL;
|
||||||
|
es->sb = sb;
|
||||||
|
es->modified = false;
|
||||||
|
|
||||||
/* byte offset in cluster */
|
/* byte offset in cluster */
|
||||||
byte_offset = EXFAT_CLU_OFFSET(byte_offset, sbi);
|
byte_offset = EXFAT_CLU_OFFSET(byte_offset, sbi);
|
||||||
|
|
||||||
/* byte offset in sector */
|
/* byte offset in sector */
|
||||||
off = EXFAT_BLK_OFFSET(byte_offset, sb);
|
off = EXFAT_BLK_OFFSET(byte_offset, sb);
|
||||||
|
es->start_off = off;
|
||||||
|
|
||||||
/* sector offset in cluster */
|
/* sector offset in cluster */
|
||||||
sec = EXFAT_B_TO_BLK(byte_offset, sb);
|
sec = EXFAT_B_TO_BLK(byte_offset, sb);
|
||||||
@@ -874,42 +852,22 @@ struct exfat_entry_set_cache *exfat_get_dentry_set(struct super_block *sb,
|
|||||||
|
|
||||||
bh = sb_bread(sb, sec);
|
bh = sb_bread(sb, sec);
|
||||||
if (!bh)
|
if (!bh)
|
||||||
return NULL;
|
goto free_es;
|
||||||
|
es->bh[es->num_bh++] = bh;
|
||||||
|
|
||||||
ep = (struct exfat_dentry *)(bh->b_data + off);
|
ep = exfat_get_dentry_cached(es, 0);
|
||||||
entry_type = exfat_get_entry_type(ep);
|
|
||||||
|
|
||||||
if (entry_type != TYPE_FILE && entry_type != TYPE_DIR)
|
|
||||||
goto release_bh;
|
|
||||||
|
|
||||||
num_entries = type == ES_ALL_ENTRIES ?
|
|
||||||
ep->dentry.file.num_ext + 1 : type;
|
|
||||||
es = kmalloc(struct_size(es, entries, num_entries), GFP_KERNEL);
|
|
||||||
if (!es)
|
|
||||||
goto release_bh;
|
|
||||||
|
|
||||||
es->num_entries = num_entries;
|
|
||||||
es->sector = sec;
|
|
||||||
es->offset = off;
|
|
||||||
es->alloc_flag = p_dir->flags;
|
|
||||||
|
|
||||||
pos = &es->entries[0];
|
|
||||||
|
|
||||||
while (num_entries) {
|
|
||||||
if (!exfat_validate_entry(exfat_get_entry_type(ep), &mode))
|
if (!exfat_validate_entry(exfat_get_entry_type(ep), &mode))
|
||||||
goto free_es;
|
goto free_es;
|
||||||
|
|
||||||
/* copy dentry */
|
num_entries = type == ES_ALL_ENTRIES ?
|
||||||
memcpy(pos, ep, sizeof(struct exfat_dentry));
|
ep->dentry.file.num_ext + 1 : type;
|
||||||
|
es->num_entries = num_entries;
|
||||||
|
|
||||||
if (--num_entries == 0)
|
num_bh = EXFAT_B_TO_BLK_ROUND_UP(off + num_entries * DENTRY_SIZE, sb);
|
||||||
break;
|
for (i = 1; i < num_bh; i++) {
|
||||||
|
|
||||||
if (((off + DENTRY_SIZE) & (sb->s_blocksize - 1)) <
|
|
||||||
(off & (sb->s_blocksize - 1))) {
|
|
||||||
/* get the next sector */
|
/* get the next sector */
|
||||||
if (exfat_is_last_sector_in_cluster(sbi, sec)) {
|
if (exfat_is_last_sector_in_cluster(sbi, sec)) {
|
||||||
if (es->alloc_flag == ALLOC_NO_FAT_CHAIN)
|
if (p_dir->flags == ALLOC_NO_FAT_CHAIN)
|
||||||
clu++;
|
clu++;
|
||||||
else if (exfat_get_next_cluster(sb, &clu))
|
else if (exfat_get_next_cluster(sb, &clu))
|
||||||
goto free_es;
|
goto free_es;
|
||||||
@@ -918,28 +876,22 @@ struct exfat_entry_set_cache *exfat_get_dentry_set(struct super_block *sb,
|
|||||||
sec++;
|
sec++;
|
||||||
}
|
}
|
||||||
|
|
||||||
brelse(bh);
|
|
||||||
bh = sb_bread(sb, sec);
|
bh = sb_bread(sb, sec);
|
||||||
if (!bh)
|
if (!bh)
|
||||||
goto free_es;
|
goto free_es;
|
||||||
off = 0;
|
es->bh[es->num_bh++] = bh;
|
||||||
ep = (struct exfat_dentry *)bh->b_data;
|
|
||||||
} else {
|
|
||||||
ep++;
|
|
||||||
off += DENTRY_SIZE;
|
|
||||||
}
|
|
||||||
pos++;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (file_ep)
|
/* validiate cached dentries */
|
||||||
*file_ep = &es->entries[0];
|
for (i = 1; i < num_entries; i++) {
|
||||||
brelse(bh);
|
ep = exfat_get_dentry_cached(es, i);
|
||||||
|
if (!exfat_validate_entry(exfat_get_entry_type(ep), &mode))
|
||||||
|
goto free_es;
|
||||||
|
}
|
||||||
return es;
|
return es;
|
||||||
|
|
||||||
free_es:
|
free_es:
|
||||||
kfree(es);
|
exfat_free_dentry_set(es, false);
|
||||||
release_bh:
|
|
||||||
brelse(bh);
|
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -1048,7 +1000,7 @@ rewind:
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (entry_type == TYPE_STREAM) {
|
if (entry_type == TYPE_STREAM) {
|
||||||
unsigned short name_hash;
|
u16 name_hash;
|
||||||
|
|
||||||
if (step != DIRENT_STEP_STRM) {
|
if (step != DIRENT_STEP_STRM) {
|
||||||
step = DIRENT_STEP_FILE;
|
step = DIRENT_STEP_FILE;
|
||||||
|
@@ -71,10 +71,8 @@ enum {
|
|||||||
#define MAX_NAME_LENGTH 255 /* max len of file name excluding NULL */
|
#define MAX_NAME_LENGTH 255 /* max len of file name excluding NULL */
|
||||||
#define MAX_VFSNAME_BUF_SIZE ((MAX_NAME_LENGTH + 1) * MAX_CHARSET_SIZE)
|
#define MAX_VFSNAME_BUF_SIZE ((MAX_NAME_LENGTH + 1) * MAX_CHARSET_SIZE)
|
||||||
|
|
||||||
#define FAT_CACHE_SIZE 128
|
/* Enough size to hold 256 dentry (even 512 Byte sector) */
|
||||||
#define FAT_CACHE_HASH_SIZE 64
|
#define DIR_CACHE_SIZE (256*sizeof(struct exfat_dentry)/512+1)
|
||||||
#define BUF_CACHE_SIZE 256
|
|
||||||
#define BUF_CACHE_HASH_SIZE 64
|
|
||||||
|
|
||||||
#define EXFAT_HINT_NONE -1
|
#define EXFAT_HINT_NONE -1
|
||||||
#define EXFAT_MIN_SUBDIR 2
|
#define EXFAT_MIN_SUBDIR 2
|
||||||
@@ -139,7 +137,7 @@ struct exfat_dentry_namebuf {
|
|||||||
struct exfat_uni_name {
|
struct exfat_uni_name {
|
||||||
/* +3 for null and for converting */
|
/* +3 for null and for converting */
|
||||||
unsigned short name[MAX_NAME_LENGTH + 3];
|
unsigned short name[MAX_NAME_LENGTH + 3];
|
||||||
unsigned short name_hash;
|
u16 name_hash;
|
||||||
unsigned char name_len;
|
unsigned char name_len;
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -170,14 +168,12 @@ struct exfat_hint {
|
|||||||
};
|
};
|
||||||
|
|
||||||
struct exfat_entry_set_cache {
|
struct exfat_entry_set_cache {
|
||||||
/* sector number that contains file_entry */
|
struct super_block *sb;
|
||||||
sector_t sector;
|
bool modified;
|
||||||
/* byte offset in the sector */
|
unsigned int start_off;
|
||||||
unsigned int offset;
|
int num_bh;
|
||||||
/* flag in stream entry. 01 for cluster chain, 03 for contig. */
|
struct buffer_head *bh[DIR_CACHE_SIZE];
|
||||||
int alloc_flag;
|
|
||||||
unsigned int num_entries;
|
unsigned int num_entries;
|
||||||
struct exfat_dentry entries[];
|
|
||||||
};
|
};
|
||||||
|
|
||||||
struct exfat_dir_entry {
|
struct exfat_dir_entry {
|
||||||
@@ -231,7 +227,7 @@ struct exfat_sb_info {
|
|||||||
unsigned int root_dir; /* root dir cluster */
|
unsigned int root_dir; /* root dir cluster */
|
||||||
unsigned int dentries_per_clu; /* num of dentries per cluster */
|
unsigned int dentries_per_clu; /* num of dentries per cluster */
|
||||||
unsigned int vol_flag; /* volume dirty flag */
|
unsigned int vol_flag; /* volume dirty flag */
|
||||||
struct buffer_head *pbr_bh; /* buffer_head of PBR sector */
|
struct buffer_head *boot_bh; /* buffer_head of BOOT sector */
|
||||||
|
|
||||||
unsigned int map_clu; /* allocation bitmap start cluster */
|
unsigned int map_clu; /* allocation bitmap start cluster */
|
||||||
unsigned int map_sectors; /* num of allocation bitmap sectors */
|
unsigned int map_sectors; /* num of allocation bitmap sectors */
|
||||||
@@ -451,8 +447,7 @@ int exfat_remove_entries(struct inode *inode, struct exfat_chain *p_dir,
|
|||||||
int entry, int order, int num_entries);
|
int entry, int order, int num_entries);
|
||||||
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,
|
||||||
int entry);
|
int entry);
|
||||||
int exfat_update_dir_chksum_with_entry_set(struct super_block *sb,
|
void exfat_update_dir_chksum_with_entry_set(struct exfat_entry_set_cache *es);
|
||||||
struct exfat_entry_set_cache *es, int sync);
|
|
||||||
int exfat_calc_num_entries(struct exfat_uni_name *p_uniname);
|
int exfat_calc_num_entries(struct exfat_uni_name *p_uniname);
|
||||||
int exfat_find_dir_entry(struct super_block *sb, struct exfat_inode_info *ei,
|
int exfat_find_dir_entry(struct super_block *sb, struct exfat_inode_info *ei,
|
||||||
struct exfat_chain *p_dir, struct exfat_uni_name *p_uniname,
|
struct exfat_chain *p_dir, struct exfat_uni_name *p_uniname,
|
||||||
@@ -463,9 +458,11 @@ int exfat_find_location(struct super_block *sb, struct exfat_chain *p_dir,
|
|||||||
struct exfat_dentry *exfat_get_dentry(struct super_block *sb,
|
struct exfat_dentry *exfat_get_dentry(struct super_block *sb,
|
||||||
struct exfat_chain *p_dir, int entry, struct buffer_head **bh,
|
struct exfat_chain *p_dir, int entry, struct buffer_head **bh,
|
||||||
sector_t *sector);
|
sector_t *sector);
|
||||||
|
struct exfat_dentry *exfat_get_dentry_cached(struct exfat_entry_set_cache *es,
|
||||||
|
int num);
|
||||||
struct exfat_entry_set_cache *exfat_get_dentry_set(struct super_block *sb,
|
struct exfat_entry_set_cache *exfat_get_dentry_set(struct super_block *sb,
|
||||||
struct exfat_chain *p_dir, int entry, unsigned int type,
|
struct exfat_chain *p_dir, int entry, unsigned int type);
|
||||||
struct exfat_dentry **file_ep);
|
void exfat_free_dentry_set(struct exfat_entry_set_cache *es, int sync);
|
||||||
int exfat_count_dir_entries(struct super_block *sb, struct exfat_chain *p_dir);
|
int exfat_count_dir_entries(struct super_block *sb, struct exfat_chain *p_dir);
|
||||||
|
|
||||||
/* inode.c */
|
/* inode.c */
|
||||||
@@ -492,8 +489,6 @@ int exfat_nls_to_utf16(struct super_block *sb,
|
|||||||
struct exfat_uni_name *uniname, int *p_lossy);
|
struct exfat_uni_name *uniname, int *p_lossy);
|
||||||
int exfat_create_upcase_table(struct super_block *sb);
|
int exfat_create_upcase_table(struct super_block *sb);
|
||||||
void exfat_free_upcase_table(struct exfat_sb_info *sbi);
|
void exfat_free_upcase_table(struct exfat_sb_info *sbi);
|
||||||
unsigned short exfat_high_surrogate(unicode_t u);
|
|
||||||
unsigned short exfat_low_surrogate(unicode_t u);
|
|
||||||
|
|
||||||
/* exfat/misc.c */
|
/* exfat/misc.c */
|
||||||
void __exfat_fs_error(struct super_block *sb, int report, const char *fmt, ...)
|
void __exfat_fs_error(struct super_block *sb, int report, const char *fmt, ...)
|
||||||
@@ -505,13 +500,20 @@ void __exfat_fs_error(struct super_block *sb, int report, const char *fmt, ...)
|
|||||||
fmt, ## args)
|
fmt, ## args)
|
||||||
void exfat_msg(struct super_block *sb, const char *lv, const char *fmt, ...)
|
void exfat_msg(struct super_block *sb, const char *lv, const char *fmt, ...)
|
||||||
__printf(3, 4) __cold;
|
__printf(3, 4) __cold;
|
||||||
|
#define exfat_err(sb, fmt, ...) \
|
||||||
|
exfat_msg(sb, KERN_ERR, fmt, ##__VA_ARGS__)
|
||||||
|
#define exfat_warn(sb, fmt, ...) \
|
||||||
|
exfat_msg(sb, KERN_WARNING, fmt, ##__VA_ARGS__)
|
||||||
|
#define exfat_info(sb, fmt, ...) \
|
||||||
|
exfat_msg(sb, KERN_INFO, fmt, ##__VA_ARGS__)
|
||||||
|
|
||||||
void exfat_get_entry_time(struct exfat_sb_info *sbi, struct timespec64 *ts,
|
void exfat_get_entry_time(struct exfat_sb_info *sbi, struct timespec64 *ts,
|
||||||
u8 tz, __le16 time, __le16 date, u8 time_ms);
|
u8 tz, __le16 time, __le16 date, u8 time_cs);
|
||||||
void exfat_truncate_atime(struct timespec64 *ts);
|
void exfat_truncate_atime(struct timespec64 *ts);
|
||||||
void exfat_set_entry_time(struct exfat_sb_info *sbi, struct timespec64 *ts,
|
void exfat_set_entry_time(struct exfat_sb_info *sbi, struct timespec64 *ts,
|
||||||
u8 *tz, __le16 *time, __le16 *date, u8 *time_ms);
|
u8 *tz, __le16 *time, __le16 *date, u8 *time_cs);
|
||||||
unsigned short exfat_calc_chksum_2byte(void *data, int len,
|
u16 exfat_calc_chksum16(void *data, int len, u16 chksum, int type);
|
||||||
unsigned short chksum, int type);
|
u32 exfat_calc_chksum32(void *data, int len, u32 chksum, int type);
|
||||||
void exfat_update_bh(struct super_block *sb, struct buffer_head *bh, int sync);
|
void exfat_update_bh(struct super_block *sb, struct buffer_head *bh, int sync);
|
||||||
void exfat_chain_set(struct exfat_chain *ec, unsigned int dir,
|
void exfat_chain_set(struct exfat_chain *ec, unsigned int dir,
|
||||||
unsigned int size, unsigned char flags);
|
unsigned int size, unsigned char flags);
|
||||||
|
@@ -8,12 +8,15 @@
|
|||||||
|
|
||||||
#include <linux/types.h>
|
#include <linux/types.h>
|
||||||
|
|
||||||
#define PBR_SIGNATURE 0xAA55
|
#define BOOT_SIGNATURE 0xAA55
|
||||||
|
#define EXBOOT_SIGNATURE 0xAA550000
|
||||||
|
#define STR_EXFAT "EXFAT " /* size should be 8 */
|
||||||
|
|
||||||
#define EXFAT_MAX_FILE_LEN 255
|
#define EXFAT_MAX_FILE_LEN 255
|
||||||
|
|
||||||
#define VOL_CLEAN 0x0000
|
#define VOL_CLEAN 0x0000
|
||||||
#define VOL_DIRTY 0x0002
|
#define VOL_DIRTY 0x0002
|
||||||
|
#define ERR_MEDIUM 0x0004
|
||||||
|
|
||||||
#define EXFAT_EOF_CLUSTER 0xFFFFFFFFu
|
#define EXFAT_EOF_CLUSTER 0xFFFFFFFFu
|
||||||
#define EXFAT_BAD_CLUSTER 0xFFFFFFF7u
|
#define EXFAT_BAD_CLUSTER 0xFFFFFFF7u
|
||||||
@@ -55,7 +58,7 @@
|
|||||||
|
|
||||||
/* checksum types */
|
/* checksum types */
|
||||||
#define CS_DIR_ENTRY 0
|
#define CS_DIR_ENTRY 0
|
||||||
#define CS_PBR_SECTOR 1
|
#define CS_BOOT_SECTOR 1
|
||||||
#define CS_DEFAULT 2
|
#define CS_DEFAULT 2
|
||||||
|
|
||||||
/* file attributes */
|
/* file attributes */
|
||||||
@@ -69,22 +72,18 @@
|
|||||||
#define ATTR_RWMASK (ATTR_HIDDEN | ATTR_SYSTEM | ATTR_VOLUME | \
|
#define ATTR_RWMASK (ATTR_HIDDEN | ATTR_SYSTEM | ATTR_VOLUME | \
|
||||||
ATTR_SUBDIR | ATTR_ARCHIVE)
|
ATTR_SUBDIR | ATTR_ARCHIVE)
|
||||||
|
|
||||||
#define PBR64_JUMP_BOOT_LEN 3
|
#define BOOTSEC_JUMP_BOOT_LEN 3
|
||||||
#define PBR64_OEM_NAME_LEN 8
|
#define BOOTSEC_FS_NAME_LEN 8
|
||||||
#define PBR64_RESERVED_LEN 53
|
#define BOOTSEC_OLDBPB_LEN 53
|
||||||
|
|
||||||
#define EXFAT_FILE_NAME_LEN 15
|
#define EXFAT_FILE_NAME_LEN 15
|
||||||
|
|
||||||
/* EXFAT BIOS parameter block (64 bytes) */
|
/* EXFAT: Main and Backup Boot Sector (512 bytes) */
|
||||||
struct bpb64 {
|
struct boot_sector {
|
||||||
__u8 jmp_boot[PBR64_JUMP_BOOT_LEN];
|
__u8 jmp_boot[BOOTSEC_JUMP_BOOT_LEN];
|
||||||
__u8 oem_name[PBR64_OEM_NAME_LEN];
|
__u8 fs_name[BOOTSEC_FS_NAME_LEN];
|
||||||
__u8 res_zero[PBR64_RESERVED_LEN];
|
__u8 must_be_zero[BOOTSEC_OLDBPB_LEN];
|
||||||
} __packed;
|
__le64 partition_offset;
|
||||||
|
|
||||||
/* EXFAT EXTEND BIOS parameter block (56 bytes) */
|
|
||||||
struct bsx64 {
|
|
||||||
__le64 vol_offset;
|
|
||||||
__le64 vol_length;
|
__le64 vol_length;
|
||||||
__le32 fat_offset;
|
__le32 fat_offset;
|
||||||
__le32 fat_length;
|
__le32 fat_length;
|
||||||
@@ -92,32 +91,14 @@ struct bsx64 {
|
|||||||
__le32 clu_count;
|
__le32 clu_count;
|
||||||
__le32 root_cluster;
|
__le32 root_cluster;
|
||||||
__le32 vol_serial;
|
__le32 vol_serial;
|
||||||
__u8 fs_version[2];
|
__u8 fs_revision[2];
|
||||||
__le16 vol_flags;
|
__le16 vol_flags;
|
||||||
__u8 sect_size_bits;
|
__u8 sect_size_bits;
|
||||||
__u8 sect_per_clus_bits;
|
__u8 sect_per_clus_bits;
|
||||||
__u8 num_fats;
|
__u8 num_fats;
|
||||||
__u8 phy_drv_no;
|
__u8 drv_sel;
|
||||||
__u8 perc_in_use;
|
__u8 percent_in_use;
|
||||||
__u8 reserved2[7];
|
__u8 reserved[7];
|
||||||
} __packed;
|
|
||||||
|
|
||||||
/* EXFAT PBR[BPB+BSX] (120 bytes) */
|
|
||||||
struct pbr64 {
|
|
||||||
struct bpb64 bpb;
|
|
||||||
struct bsx64 bsx;
|
|
||||||
} __packed;
|
|
||||||
|
|
||||||
/* Common PBR[Partition Boot Record] (512 bytes) */
|
|
||||||
struct pbr {
|
|
||||||
union {
|
|
||||||
__u8 raw[64];
|
|
||||||
struct bpb64 f64;
|
|
||||||
} bpb;
|
|
||||||
union {
|
|
||||||
__u8 raw[56];
|
|
||||||
struct bsx64 f64;
|
|
||||||
} bsx;
|
|
||||||
__u8 boot_code[390];
|
__u8 boot_code[390];
|
||||||
__le16 signature;
|
__le16 signature;
|
||||||
} __packed;
|
} __packed;
|
||||||
@@ -136,8 +117,8 @@ struct exfat_dentry {
|
|||||||
__le16 modify_date;
|
__le16 modify_date;
|
||||||
__le16 access_time;
|
__le16 access_time;
|
||||||
__le16 access_date;
|
__le16 access_date;
|
||||||
__u8 create_time_ms;
|
__u8 create_time_cs;
|
||||||
__u8 modify_time_ms;
|
__u8 modify_time_cs;
|
||||||
__u8 create_tz;
|
__u8 create_tz;
|
||||||
__u8 modify_tz;
|
__u8 modify_tz;
|
||||||
__u8 access_tz;
|
__u8 access_tz;
|
||||||
|
@@ -169,9 +169,8 @@ int exfat_free_cluster(struct inode *inode, struct exfat_chain *p_chain)
|
|||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
/* check cluster validation */
|
/* check cluster validation */
|
||||||
if (p_chain->dir < 2 && p_chain->dir >= sbi->num_clusters) {
|
if (!is_valid_cluster(sbi, p_chain->dir)) {
|
||||||
exfat_msg(sb, KERN_ERR, "invalid start cluster (%u)",
|
exfat_err(sb, "invalid start cluster (%u)", p_chain->dir);
|
||||||
p_chain->dir);
|
|
||||||
return -EIO;
|
return -EIO;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -305,8 +304,7 @@ int exfat_zeroed_cluster(struct inode *dir, unsigned int clu)
|
|||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
release_bhs:
|
release_bhs:
|
||||||
exfat_msg(sb, KERN_ERR, "failed zeroed sect %llu\n",
|
exfat_err(sb, "failed zeroed sect %llu\n", (unsigned long long)blknr);
|
||||||
(unsigned long long)blknr);
|
|
||||||
for (i = 0; i < n; i++)
|
for (i = 0; i < n; i++)
|
||||||
bforget(bhs[i]);
|
bforget(bhs[i]);
|
||||||
return err;
|
return err;
|
||||||
@@ -337,8 +335,7 @@ int exfat_alloc_cluster(struct inode *inode, unsigned int num_alloc,
|
|||||||
/* find new cluster */
|
/* find new cluster */
|
||||||
if (hint_clu == EXFAT_EOF_CLUSTER) {
|
if (hint_clu == EXFAT_EOF_CLUSTER) {
|
||||||
if (sbi->clu_srch_ptr < EXFAT_FIRST_CLUSTER) {
|
if (sbi->clu_srch_ptr < EXFAT_FIRST_CLUSTER) {
|
||||||
exfat_msg(sb, KERN_ERR,
|
exfat_err(sb, "sbi->clu_srch_ptr is invalid (%u)\n",
|
||||||
"sbi->clu_srch_ptr is invalid (%u)\n",
|
|
||||||
sbi->clu_srch_ptr);
|
sbi->clu_srch_ptr);
|
||||||
sbi->clu_srch_ptr = EXFAT_FIRST_CLUSTER;
|
sbi->clu_srch_ptr = EXFAT_FIRST_CLUSTER;
|
||||||
}
|
}
|
||||||
@@ -349,8 +346,8 @@ int exfat_alloc_cluster(struct inode *inode, unsigned int num_alloc,
|
|||||||
}
|
}
|
||||||
|
|
||||||
/* check cluster validation */
|
/* check cluster validation */
|
||||||
if (hint_clu < EXFAT_FIRST_CLUSTER && hint_clu >= sbi->num_clusters) {
|
if (!is_valid_cluster(sbi, hint_clu)) {
|
||||||
exfat_msg(sb, KERN_ERR, "hint_cluster is invalid (%u)\n",
|
exfat_err(sb, "hint_cluster is invalid (%u)",
|
||||||
hint_clu);
|
hint_clu);
|
||||||
hint_clu = EXFAT_FIRST_CLUSTER;
|
hint_clu = EXFAT_FIRST_CLUSTER;
|
||||||
if (p_chain->flags == ALLOC_NO_FAT_CHAIN) {
|
if (p_chain->flags == ALLOC_NO_FAT_CHAIN) {
|
||||||
|
@@ -96,11 +96,9 @@ int __exfat_truncate(struct inode *inode, loff_t new_size)
|
|||||||
unsigned int num_clusters_new, num_clusters_phys;
|
unsigned int num_clusters_new, num_clusters_phys;
|
||||||
unsigned int last_clu = EXFAT_FREE_CLUSTER;
|
unsigned int last_clu = EXFAT_FREE_CLUSTER;
|
||||||
struct exfat_chain clu;
|
struct exfat_chain clu;
|
||||||
struct exfat_dentry *ep, *ep2;
|
|
||||||
struct super_block *sb = inode->i_sb;
|
struct super_block *sb = inode->i_sb;
|
||||||
struct exfat_sb_info *sbi = EXFAT_SB(sb);
|
struct exfat_sb_info *sbi = EXFAT_SB(sb);
|
||||||
struct exfat_inode_info *ei = EXFAT_I(inode);
|
struct exfat_inode_info *ei = EXFAT_I(inode);
|
||||||
struct exfat_entry_set_cache *es = NULL;
|
|
||||||
int evict = (ei->dir.dir == DIR_DELETED) ? 1 : 0;
|
int evict = (ei->dir.dir == DIR_DELETED) ? 1 : 0;
|
||||||
|
|
||||||
/* check if the given file ID is opened */
|
/* check if the given file ID is opened */
|
||||||
@@ -153,28 +151,31 @@ int __exfat_truncate(struct inode *inode, loff_t new_size)
|
|||||||
/* update the directory entry */
|
/* update the directory entry */
|
||||||
if (!evict) {
|
if (!evict) {
|
||||||
struct timespec64 ts;
|
struct timespec64 ts;
|
||||||
|
struct exfat_dentry *ep, *ep2;
|
||||||
|
struct exfat_entry_set_cache *es;
|
||||||
|
|
||||||
es = exfat_get_dentry_set(sb, &(ei->dir), ei->entry,
|
es = exfat_get_dentry_set(sb, &(ei->dir), ei->entry,
|
||||||
ES_ALL_ENTRIES, &ep);
|
ES_ALL_ENTRIES);
|
||||||
if (!es)
|
if (!es)
|
||||||
return -EIO;
|
return -EIO;
|
||||||
ep2 = ep + 1;
|
ep = exfat_get_dentry_cached(es, 0);
|
||||||
|
ep2 = exfat_get_dentry_cached(es, 1);
|
||||||
|
|
||||||
ts = current_time(inode);
|
ts = current_time(inode);
|
||||||
exfat_set_entry_time(sbi, &ts,
|
exfat_set_entry_time(sbi, &ts,
|
||||||
&ep->dentry.file.modify_tz,
|
&ep->dentry.file.modify_tz,
|
||||||
&ep->dentry.file.modify_time,
|
&ep->dentry.file.modify_time,
|
||||||
&ep->dentry.file.modify_date,
|
&ep->dentry.file.modify_date,
|
||||||
&ep->dentry.file.modify_time_ms);
|
&ep->dentry.file.modify_time_cs);
|
||||||
ep->dentry.file.attr = cpu_to_le16(ei->attr);
|
ep->dentry.file.attr = cpu_to_le16(ei->attr);
|
||||||
|
|
||||||
/* File size should be zero if there is no cluster allocated */
|
/* File size should be zero if there is no cluster allocated */
|
||||||
if (ei->start_clu == EXFAT_EOF_CLUSTER) {
|
if (ei->start_clu == EXFAT_EOF_CLUSTER) {
|
||||||
ep->dentry.stream.valid_size = 0;
|
ep2->dentry.stream.valid_size = 0;
|
||||||
ep->dentry.stream.size = 0;
|
ep2->dentry.stream.size = 0;
|
||||||
} else {
|
} else {
|
||||||
ep->dentry.stream.valid_size = cpu_to_le64(new_size);
|
ep2->dentry.stream.valid_size = cpu_to_le64(new_size);
|
||||||
ep->dentry.stream.size = ep->dentry.stream.valid_size;
|
ep2->dentry.stream.size = ep->dentry.stream.valid_size;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (new_size == 0) {
|
if (new_size == 0) {
|
||||||
@@ -185,10 +186,8 @@ int __exfat_truncate(struct inode *inode, loff_t new_size)
|
|||||||
ep2->dentry.stream.start_clu = EXFAT_FREE_CLUSTER;
|
ep2->dentry.stream.start_clu = EXFAT_FREE_CLUSTER;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (exfat_update_dir_chksum_with_entry_set(sb, es,
|
exfat_update_dir_chksum_with_entry_set(es);
|
||||||
inode_needs_sync(inode)))
|
exfat_free_dentry_set(es, inode_needs_sync(inode));
|
||||||
return -EIO;
|
|
||||||
kfree(es);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* cut off from the FAT chain */
|
/* cut off from the FAT chain */
|
||||||
|
@@ -19,7 +19,6 @@
|
|||||||
|
|
||||||
static int __exfat_write_inode(struct inode *inode, int sync)
|
static int __exfat_write_inode(struct inode *inode, int sync)
|
||||||
{
|
{
|
||||||
int ret = -EIO;
|
|
||||||
unsigned long long on_disk_size;
|
unsigned long long on_disk_size;
|
||||||
struct exfat_dentry *ep, *ep2;
|
struct exfat_dentry *ep, *ep2;
|
||||||
struct exfat_entry_set_cache *es = NULL;
|
struct exfat_entry_set_cache *es = NULL;
|
||||||
@@ -43,11 +42,11 @@ static int __exfat_write_inode(struct inode *inode, int sync)
|
|||||||
exfat_set_vol_flags(sb, VOL_DIRTY);
|
exfat_set_vol_flags(sb, VOL_DIRTY);
|
||||||
|
|
||||||
/* get the directory entry of given file or directory */
|
/* get the directory entry of given file or directory */
|
||||||
es = exfat_get_dentry_set(sb, &(ei->dir), ei->entry, ES_ALL_ENTRIES,
|
es = exfat_get_dentry_set(sb, &(ei->dir), ei->entry, ES_ALL_ENTRIES);
|
||||||
&ep);
|
|
||||||
if (!es)
|
if (!es)
|
||||||
return -EIO;
|
return -EIO;
|
||||||
ep2 = ep + 1;
|
ep = exfat_get_dentry_cached(es, 0);
|
||||||
|
ep2 = exfat_get_dentry_cached(es, 1);
|
||||||
|
|
||||||
ep->dentry.file.attr = cpu_to_le16(exfat_make_attr(inode));
|
ep->dentry.file.attr = cpu_to_le16(exfat_make_attr(inode));
|
||||||
|
|
||||||
@@ -56,12 +55,12 @@ static int __exfat_write_inode(struct inode *inode, int sync)
|
|||||||
&ep->dentry.file.create_tz,
|
&ep->dentry.file.create_tz,
|
||||||
&ep->dentry.file.create_time,
|
&ep->dentry.file.create_time,
|
||||||
&ep->dentry.file.create_date,
|
&ep->dentry.file.create_date,
|
||||||
&ep->dentry.file.create_time_ms);
|
&ep->dentry.file.create_time_cs);
|
||||||
exfat_set_entry_time(sbi, &inode->i_mtime,
|
exfat_set_entry_time(sbi, &inode->i_mtime,
|
||||||
&ep->dentry.file.modify_tz,
|
&ep->dentry.file.modify_tz,
|
||||||
&ep->dentry.file.modify_time,
|
&ep->dentry.file.modify_time,
|
||||||
&ep->dentry.file.modify_date,
|
&ep->dentry.file.modify_date,
|
||||||
&ep->dentry.file.modify_time_ms);
|
&ep->dentry.file.modify_time_cs);
|
||||||
exfat_set_entry_time(sbi, &inode->i_atime,
|
exfat_set_entry_time(sbi, &inode->i_atime,
|
||||||
&ep->dentry.file.access_tz,
|
&ep->dentry.file.access_tz,
|
||||||
&ep->dentry.file.access_time,
|
&ep->dentry.file.access_time,
|
||||||
@@ -77,9 +76,9 @@ static int __exfat_write_inode(struct inode *inode, int sync)
|
|||||||
ep2->dentry.stream.valid_size = cpu_to_le64(on_disk_size);
|
ep2->dentry.stream.valid_size = cpu_to_le64(on_disk_size);
|
||||||
ep2->dentry.stream.size = ep2->dentry.stream.valid_size;
|
ep2->dentry.stream.size = ep2->dentry.stream.valid_size;
|
||||||
|
|
||||||
ret = exfat_update_dir_chksum_with_entry_set(sb, es, sync);
|
exfat_update_dir_chksum_with_entry_set(es);
|
||||||
kfree(es);
|
exfat_free_dentry_set(es, sync);
|
||||||
return ret;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
int exfat_write_inode(struct inode *inode, struct writeback_control *wbc)
|
int exfat_write_inode(struct inode *inode, struct writeback_control *wbc)
|
||||||
@@ -110,8 +109,6 @@ static int exfat_map_cluster(struct inode *inode, unsigned int clu_offset,
|
|||||||
int ret, modified = false;
|
int ret, modified = false;
|
||||||
unsigned int last_clu;
|
unsigned int last_clu;
|
||||||
struct exfat_chain new_clu;
|
struct exfat_chain new_clu;
|
||||||
struct exfat_dentry *ep;
|
|
||||||
struct exfat_entry_set_cache *es = NULL;
|
|
||||||
struct super_block *sb = inode->i_sb;
|
struct super_block *sb = inode->i_sb;
|
||||||
struct exfat_sb_info *sbi = EXFAT_SB(sb);
|
struct exfat_sb_info *sbi = EXFAT_SB(sb);
|
||||||
struct exfat_inode_info *ei = EXFAT_I(inode);
|
struct exfat_inode_info *ei = EXFAT_I(inode);
|
||||||
@@ -222,34 +219,28 @@ static int exfat_map_cluster(struct inode *inode, unsigned int clu_offset,
|
|||||||
num_clusters += num_to_be_allocated;
|
num_clusters += num_to_be_allocated;
|
||||||
*clu = new_clu.dir;
|
*clu = new_clu.dir;
|
||||||
|
|
||||||
if (ei->dir.dir != DIR_DELETED) {
|
if (ei->dir.dir != DIR_DELETED && modified) {
|
||||||
|
struct exfat_dentry *ep;
|
||||||
|
struct exfat_entry_set_cache *es;
|
||||||
|
|
||||||
es = exfat_get_dentry_set(sb, &(ei->dir), ei->entry,
|
es = exfat_get_dentry_set(sb, &(ei->dir), ei->entry,
|
||||||
ES_ALL_ENTRIES, &ep);
|
ES_ALL_ENTRIES);
|
||||||
if (!es)
|
if (!es)
|
||||||
return -EIO;
|
return -EIO;
|
||||||
/* get stream entry */
|
/* get stream entry */
|
||||||
ep++;
|
ep = exfat_get_dentry_cached(es, 1);
|
||||||
|
|
||||||
/* update directory entry */
|
/* update directory entry */
|
||||||
if (modified) {
|
|
||||||
if (ep->dentry.stream.flags != ei->flags)
|
|
||||||
ep->dentry.stream.flags = ei->flags;
|
ep->dentry.stream.flags = ei->flags;
|
||||||
|
|
||||||
if (le32_to_cpu(ep->dentry.stream.start_clu) !=
|
|
||||||
ei->start_clu)
|
|
||||||
ep->dentry.stream.start_clu =
|
ep->dentry.stream.start_clu =
|
||||||
cpu_to_le32(ei->start_clu);
|
cpu_to_le32(ei->start_clu);
|
||||||
|
|
||||||
ep->dentry.stream.valid_size =
|
ep->dentry.stream.valid_size =
|
||||||
cpu_to_le64(i_size_read(inode));
|
cpu_to_le64(i_size_read(inode));
|
||||||
ep->dentry.stream.size =
|
ep->dentry.stream.size =
|
||||||
ep->dentry.stream.valid_size;
|
ep->dentry.stream.valid_size;
|
||||||
}
|
|
||||||
|
|
||||||
if (exfat_update_dir_chksum_with_entry_set(sb, es,
|
exfat_update_dir_chksum_with_entry_set(es);
|
||||||
inode_needs_sync(inode)))
|
exfat_free_dentry_set(es, inode_needs_sync(inode));
|
||||||
return -EIO;
|
|
||||||
kfree(es);
|
|
||||||
|
|
||||||
} /* end of if != DIR_DELETED */
|
} /* end of if != DIR_DELETED */
|
||||||
|
|
||||||
|
@@ -32,7 +32,7 @@ void __exfat_fs_error(struct super_block *sb, int report, const char *fmt, ...)
|
|||||||
va_start(args, fmt);
|
va_start(args, fmt);
|
||||||
vaf.fmt = fmt;
|
vaf.fmt = fmt;
|
||||||
vaf.va = &args;
|
vaf.va = &args;
|
||||||
exfat_msg(sb, KERN_ERR, "error, %pV\n", &vaf);
|
exfat_err(sb, "error, %pV", &vaf);
|
||||||
va_end(args);
|
va_end(args);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -41,7 +41,7 @@ void __exfat_fs_error(struct super_block *sb, int report, const char *fmt, ...)
|
|||||||
sb->s_id);
|
sb->s_id);
|
||||||
} else if (opts->errors == EXFAT_ERRORS_RO && !sb_rdonly(sb)) {
|
} else if (opts->errors == EXFAT_ERRORS_RO && !sb_rdonly(sb)) {
|
||||||
sb->s_flags |= SB_RDONLY;
|
sb->s_flags |= SB_RDONLY;
|
||||||
exfat_msg(sb, KERN_ERR, "Filesystem has been set read-only");
|
exfat_err(sb, "Filesystem has been set read-only");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -75,7 +75,7 @@ static void exfat_adjust_tz(struct timespec64 *ts, u8 tz_off)
|
|||||||
|
|
||||||
/* Convert a EXFAT time/date pair to a UNIX date (seconds since 1 1 70). */
|
/* Convert a EXFAT time/date pair to a UNIX date (seconds since 1 1 70). */
|
||||||
void exfat_get_entry_time(struct exfat_sb_info *sbi, struct timespec64 *ts,
|
void exfat_get_entry_time(struct exfat_sb_info *sbi, struct timespec64 *ts,
|
||||||
u8 tz, __le16 time, __le16 date, u8 time_ms)
|
u8 tz, __le16 time, __le16 date, u8 time_cs)
|
||||||
{
|
{
|
||||||
u16 t = le16_to_cpu(time);
|
u16 t = le16_to_cpu(time);
|
||||||
u16 d = le16_to_cpu(date);
|
u16 d = le16_to_cpu(date);
|
||||||
@@ -84,10 +84,10 @@ void exfat_get_entry_time(struct exfat_sb_info *sbi, struct timespec64 *ts,
|
|||||||
t >> 11, (t >> 5) & 0x003F, (t & 0x001F) << 1);
|
t >> 11, (t >> 5) & 0x003F, (t & 0x001F) << 1);
|
||||||
|
|
||||||
|
|
||||||
/* time_ms field represent 0 ~ 199(1990 ms) */
|
/* time_cs field represent 0 ~ 199cs(1990 ms) */
|
||||||
if (time_ms) {
|
if (time_cs) {
|
||||||
ts->tv_sec += time_ms / 100;
|
ts->tv_sec += time_cs / 100;
|
||||||
ts->tv_nsec = (time_ms % 100) * 10 * NSEC_PER_MSEC;
|
ts->tv_nsec = (time_cs % 100) * 10 * NSEC_PER_MSEC;
|
||||||
} else
|
} else
|
||||||
ts->tv_nsec = 0;
|
ts->tv_nsec = 0;
|
||||||
|
|
||||||
@@ -101,7 +101,7 @@ void exfat_get_entry_time(struct exfat_sb_info *sbi, struct timespec64 *ts,
|
|||||||
|
|
||||||
/* Convert linear UNIX date to a EXFAT time/date pair. */
|
/* Convert linear UNIX date to a EXFAT time/date pair. */
|
||||||
void exfat_set_entry_time(struct exfat_sb_info *sbi, struct timespec64 *ts,
|
void exfat_set_entry_time(struct exfat_sb_info *sbi, struct timespec64 *ts,
|
||||||
u8 *tz, __le16 *time, __le16 *date, u8 *time_ms)
|
u8 *tz, __le16 *time, __le16 *date, u8 *time_cs)
|
||||||
{
|
{
|
||||||
struct tm tm;
|
struct tm tm;
|
||||||
u16 t, d;
|
u16 t, d;
|
||||||
@@ -113,9 +113,9 @@ void exfat_set_entry_time(struct exfat_sb_info *sbi, struct timespec64 *ts,
|
|||||||
*time = cpu_to_le16(t);
|
*time = cpu_to_le16(t);
|
||||||
*date = cpu_to_le16(d);
|
*date = cpu_to_le16(d);
|
||||||
|
|
||||||
/* time_ms field represent 0 ~ 199(1990 ms) */
|
/* time_cs field represent 0 ~ 199cs(1990 ms) */
|
||||||
if (time_ms)
|
if (time_cs)
|
||||||
*time_ms = (tm.tm_sec & 1) * 100 +
|
*time_cs = (tm.tm_sec & 1) * 100 +
|
||||||
ts->tv_nsec / (10 * NSEC_PER_MSEC);
|
ts->tv_nsec / (10 * NSEC_PER_MSEC);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@@ -136,17 +136,29 @@ void exfat_truncate_atime(struct timespec64 *ts)
|
|||||||
ts->tv_nsec = 0;
|
ts->tv_nsec = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
unsigned short exfat_calc_chksum_2byte(void *data, int len,
|
u16 exfat_calc_chksum16(void *data, int len, u16 chksum, int type)
|
||||||
unsigned short chksum, int type)
|
|
||||||
{
|
{
|
||||||
int i;
|
int i;
|
||||||
unsigned char *c = (unsigned char *)data;
|
u8 *c = (u8 *)data;
|
||||||
|
|
||||||
for (i = 0; i < len; i++, c++) {
|
for (i = 0; i < len; i++, c++) {
|
||||||
if (((i == 2) || (i == 3)) && (type == CS_DIR_ENTRY))
|
if (unlikely(type == CS_DIR_ENTRY && (i == 2 || i == 3)))
|
||||||
continue;
|
continue;
|
||||||
chksum = (((chksum & 1) << 15) | ((chksum & 0xFFFE) >> 1)) +
|
chksum = ((chksum << 15) | (chksum >> 1)) + *c;
|
||||||
(unsigned short)*c;
|
}
|
||||||
|
return chksum;
|
||||||
|
}
|
||||||
|
|
||||||
|
u32 exfat_calc_chksum32(void *data, int len, u32 chksum, int type)
|
||||||
|
{
|
||||||
|
int i;
|
||||||
|
u8 *c = (u8 *)data;
|
||||||
|
|
||||||
|
for (i = 0; i < len; i++, c++) {
|
||||||
|
if (unlikely(type == CS_BOOT_SECTOR &&
|
||||||
|
(i == 106 || i == 107 || i == 112)))
|
||||||
|
continue;
|
||||||
|
chksum = ((chksum << 31) | (chksum >> 1)) + *c;
|
||||||
}
|
}
|
||||||
return chksum;
|
return chksum;
|
||||||
}
|
}
|
||||||
|
@@ -147,16 +147,10 @@ static int exfat_utf8_d_hash(const struct dentry *dentry, struct qstr *qstr)
|
|||||||
return charlen;
|
return charlen;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Convert to UTF-16: code points above U+FFFF are encoded as
|
|
||||||
* surrogate pairs.
|
|
||||||
* exfat_toupper() works only for code points up to the U+FFFF.
|
* exfat_toupper() works only for code points up to the U+FFFF.
|
||||||
*/
|
*/
|
||||||
if (u > 0xFFFF) {
|
hash = partial_name_hash(u <= 0xFFFF ? exfat_toupper(sb, u) : u,
|
||||||
hash = partial_name_hash(exfat_high_surrogate(u), hash);
|
hash);
|
||||||
hash = partial_name_hash(exfat_low_surrogate(u), hash);
|
|
||||||
} else {
|
|
||||||
hash = partial_name_hash(exfat_toupper(sb, u), hash);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
qstr->hash = end_name_hash(hash);
|
qstr->hash = end_name_hash(hash);
|
||||||
@@ -185,13 +179,8 @@ static int exfat_utf8_d_cmp(const struct dentry *dentry, unsigned int len,
|
|||||||
if (u_a <= 0xFFFF && u_b <= 0xFFFF) {
|
if (u_a <= 0xFFFF && u_b <= 0xFFFF) {
|
||||||
if (exfat_toupper(sb, u_a) != exfat_toupper(sb, u_b))
|
if (exfat_toupper(sb, u_a) != exfat_toupper(sb, u_b))
|
||||||
return 1;
|
return 1;
|
||||||
} else if (u_a > 0xFFFF && u_b > 0xFFFF) {
|
|
||||||
if (exfat_low_surrogate(u_a) !=
|
|
||||||
exfat_low_surrogate(u_b) ||
|
|
||||||
exfat_high_surrogate(u_a) !=
|
|
||||||
exfat_high_surrogate(u_b))
|
|
||||||
return 1;
|
|
||||||
} else {
|
} else {
|
||||||
|
if (u_a != u_b)
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -611,8 +600,6 @@ static int exfat_find(struct inode *dir, struct qstr *qname,
|
|||||||
int ret, dentry, num_entries, count;
|
int ret, dentry, num_entries, count;
|
||||||
struct exfat_chain cdir;
|
struct exfat_chain cdir;
|
||||||
struct exfat_uni_name uni_name;
|
struct exfat_uni_name uni_name;
|
||||||
struct exfat_dentry *ep, *ep2;
|
|
||||||
struct exfat_entry_set_cache *es = NULL;
|
|
||||||
struct super_block *sb = dir->i_sb;
|
struct super_block *sb = dir->i_sb;
|
||||||
struct exfat_sb_info *sbi = EXFAT_SB(sb);
|
struct exfat_sb_info *sbi = EXFAT_SB(sb);
|
||||||
struct exfat_inode_info *ei = EXFAT_I(dir);
|
struct exfat_inode_info *ei = EXFAT_I(dir);
|
||||||
@@ -671,10 +658,14 @@ static int exfat_find(struct inode *dir, struct qstr *qname,
|
|||||||
|
|
||||||
info->num_subdirs = count;
|
info->num_subdirs = count;
|
||||||
} else {
|
} else {
|
||||||
es = exfat_get_dentry_set(sb, &cdir, dentry, ES_2_ENTRIES, &ep);
|
struct exfat_dentry *ep, *ep2;
|
||||||
|
struct exfat_entry_set_cache *es;
|
||||||
|
|
||||||
|
es = exfat_get_dentry_set(sb, &cdir, dentry, ES_2_ENTRIES);
|
||||||
if (!es)
|
if (!es)
|
||||||
return -EIO;
|
return -EIO;
|
||||||
ep2 = ep + 1;
|
ep = exfat_get_dentry_cached(es, 0);
|
||||||
|
ep2 = exfat_get_dentry_cached(es, 1);
|
||||||
|
|
||||||
info->type = exfat_get_entry_type(ep);
|
info->type = exfat_get_entry_type(ep);
|
||||||
info->attr = le16_to_cpu(ep->dentry.file.attr);
|
info->attr = le16_to_cpu(ep->dentry.file.attr);
|
||||||
@@ -692,7 +683,7 @@ static int exfat_find(struct inode *dir, struct qstr *qname,
|
|||||||
exfat_fs_error(sb,
|
exfat_fs_error(sb,
|
||||||
"non-zero size file starts with zero cluster (size : %llu, p_dir : %u, entry : 0x%08x)",
|
"non-zero size file starts with zero cluster (size : %llu, p_dir : %u, entry : 0x%08x)",
|
||||||
i_size_read(dir), ei->dir.dir, ei->entry);
|
i_size_read(dir), ei->dir.dir, ei->entry);
|
||||||
kfree(es);
|
exfat_free_dentry_set(es, false);
|
||||||
return -EIO;
|
return -EIO;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -700,18 +691,18 @@ static int exfat_find(struct inode *dir, struct qstr *qname,
|
|||||||
ep->dentry.file.create_tz,
|
ep->dentry.file.create_tz,
|
||||||
ep->dentry.file.create_time,
|
ep->dentry.file.create_time,
|
||||||
ep->dentry.file.create_date,
|
ep->dentry.file.create_date,
|
||||||
ep->dentry.file.create_time_ms);
|
ep->dentry.file.create_time_cs);
|
||||||
exfat_get_entry_time(sbi, &info->mtime,
|
exfat_get_entry_time(sbi, &info->mtime,
|
||||||
ep->dentry.file.modify_tz,
|
ep->dentry.file.modify_tz,
|
||||||
ep->dentry.file.modify_time,
|
ep->dentry.file.modify_time,
|
||||||
ep->dentry.file.modify_date,
|
ep->dentry.file.modify_date,
|
||||||
ep->dentry.file.modify_time_ms);
|
ep->dentry.file.modify_time_cs);
|
||||||
exfat_get_entry_time(sbi, &info->atime,
|
exfat_get_entry_time(sbi, &info->atime,
|
||||||
ep->dentry.file.access_tz,
|
ep->dentry.file.access_tz,
|
||||||
ep->dentry.file.access_time,
|
ep->dentry.file.access_time,
|
||||||
ep->dentry.file.access_date,
|
ep->dentry.file.access_date,
|
||||||
0);
|
0);
|
||||||
kfree(es);
|
exfat_free_dentry_set(es, false);
|
||||||
|
|
||||||
if (info->type == TYPE_DIR) {
|
if (info->type == TYPE_DIR) {
|
||||||
exfat_chain_set(&cdir, info->start_clu,
|
exfat_chain_set(&cdir, info->start_clu,
|
||||||
@@ -778,8 +769,8 @@ static struct dentry *exfat_lookup(struct inode *dir, struct dentry *dentry,
|
|||||||
if (d_unhashed(alias)) {
|
if (d_unhashed(alias)) {
|
||||||
WARN_ON(alias->d_name.hash_len !=
|
WARN_ON(alias->d_name.hash_len !=
|
||||||
dentry->d_name.hash_len);
|
dentry->d_name.hash_len);
|
||||||
exfat_msg(sb, KERN_INFO,
|
exfat_info(sb, "rehashed a dentry(%p) in read lookup",
|
||||||
"rehashed a dentry(%p) in read lookup", alias);
|
alias);
|
||||||
d_drop(dentry);
|
d_drop(dentry);
|
||||||
d_rehash(alias);
|
d_rehash(alias);
|
||||||
} else if (!S_ISDIR(i_mode)) {
|
} else if (!S_ISDIR(i_mode)) {
|
||||||
@@ -824,7 +815,7 @@ static int exfat_unlink(struct inode *dir, struct dentry *dentry)
|
|||||||
exfat_chain_dup(&cdir, &ei->dir);
|
exfat_chain_dup(&cdir, &ei->dir);
|
||||||
entry = ei->entry;
|
entry = ei->entry;
|
||||||
if (ei->dir.dir == DIR_DELETED) {
|
if (ei->dir.dir == DIR_DELETED) {
|
||||||
exfat_msg(sb, KERN_ERR, "abnormal access to deleted dentry");
|
exfat_err(sb, "abnormal access to deleted dentry");
|
||||||
err = -ENOENT;
|
err = -ENOENT;
|
||||||
goto unlock;
|
goto unlock;
|
||||||
}
|
}
|
||||||
@@ -979,7 +970,7 @@ static int exfat_rmdir(struct inode *dir, struct dentry *dentry)
|
|||||||
entry = ei->entry;
|
entry = ei->entry;
|
||||||
|
|
||||||
if (ei->dir.dir == DIR_DELETED) {
|
if (ei->dir.dir == DIR_DELETED) {
|
||||||
exfat_msg(sb, KERN_ERR, "abnormal access to deleted dentry");
|
exfat_err(sb, "abnormal access to deleted dentry");
|
||||||
err = -ENOENT;
|
err = -ENOENT;
|
||||||
goto unlock;
|
goto unlock;
|
||||||
}
|
}
|
||||||
@@ -991,8 +982,7 @@ static int exfat_rmdir(struct inode *dir, struct dentry *dentry)
|
|||||||
err = exfat_check_dir_empty(sb, &clu_to_free);
|
err = exfat_check_dir_empty(sb, &clu_to_free);
|
||||||
if (err) {
|
if (err) {
|
||||||
if (err == -EIO)
|
if (err == -EIO)
|
||||||
exfat_msg(sb, KERN_ERR,
|
exfat_err(sb, "failed to exfat_check_dir_empty : err(%d)",
|
||||||
"failed to exfat_check_dir_empty : err(%d)",
|
|
||||||
err);
|
err);
|
||||||
goto unlock;
|
goto unlock;
|
||||||
}
|
}
|
||||||
@@ -1014,9 +1004,7 @@ static int exfat_rmdir(struct inode *dir, struct dentry *dentry)
|
|||||||
|
|
||||||
err = exfat_remove_entries(dir, &cdir, entry, 0, num_entries);
|
err = exfat_remove_entries(dir, &cdir, entry, 0, num_entries);
|
||||||
if (err) {
|
if (err) {
|
||||||
exfat_msg(sb, KERN_ERR,
|
exfat_err(sb, "failed to exfat_remove_entries : err(%d)", err);
|
||||||
"failed to exfat_remove_entries : err(%d)",
|
|
||||||
err);
|
|
||||||
goto unlock;
|
goto unlock;
|
||||||
}
|
}
|
||||||
ei->dir.dir = DIR_DELETED;
|
ei->dir.dir = DIR_DELETED;
|
||||||
@@ -1245,8 +1233,7 @@ static int __exfat_rename(struct inode *old_parent_inode,
|
|||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
|
|
||||||
if (ei->dir.dir == DIR_DELETED) {
|
if (ei->dir.dir == DIR_DELETED) {
|
||||||
exfat_msg(sb, KERN_ERR,
|
exfat_err(sb, "abnormal access to deleted source dentry");
|
||||||
"abnormal access to deleted source dentry");
|
|
||||||
return -ENOENT;
|
return -ENOENT;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -1268,8 +1255,7 @@ static int __exfat_rename(struct inode *old_parent_inode,
|
|||||||
new_ei = EXFAT_I(new_inode);
|
new_ei = EXFAT_I(new_inode);
|
||||||
|
|
||||||
if (new_ei->dir.dir == DIR_DELETED) {
|
if (new_ei->dir.dir == DIR_DELETED) {
|
||||||
exfat_msg(sb, KERN_ERR,
|
exfat_err(sb, "abnormal access to deleted target dentry");
|
||||||
"abnormal access to deleted target dentry");
|
|
||||||
goto out;
|
goto out;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -1431,8 +1417,7 @@ static int exfat_rename(struct inode *old_dir, struct dentry *old_dentry,
|
|||||||
if (S_ISDIR(new_inode->i_mode))
|
if (S_ISDIR(new_inode->i_mode))
|
||||||
drop_nlink(new_inode);
|
drop_nlink(new_inode);
|
||||||
} else {
|
} else {
|
||||||
exfat_msg(sb, KERN_WARNING,
|
exfat_warn(sb, "abnormal access to an inode dropped");
|
||||||
"abnormal access to an inode dropped");
|
|
||||||
WARN_ON(new_inode->i_nlink == 0);
|
WARN_ON(new_inode->i_nlink == 0);
|
||||||
}
|
}
|
||||||
new_inode->i_ctime = EXFAT_I(new_inode)->i_crtime =
|
new_inode->i_ctime = EXFAT_I(new_inode)->i_crtime =
|
||||||
|
@@ -503,21 +503,17 @@ static int exfat_utf8_to_utf16(struct super_block *sb,
|
|||||||
unilen = utf8s_to_utf16s(p_cstring, len, UTF16_HOST_ENDIAN,
|
unilen = utf8s_to_utf16s(p_cstring, len, UTF16_HOST_ENDIAN,
|
||||||
(wchar_t *)uniname, MAX_NAME_LENGTH + 2);
|
(wchar_t *)uniname, MAX_NAME_LENGTH + 2);
|
||||||
if (unilen < 0) {
|
if (unilen < 0) {
|
||||||
exfat_msg(sb, KERN_ERR,
|
exfat_err(sb, "failed to %s (err : %d) nls len : %d",
|
||||||
"failed to %s (err : %d) nls len : %d",
|
|
||||||
__func__, unilen, len);
|
__func__, unilen, len);
|
||||||
return unilen;
|
return unilen;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (unilen > MAX_NAME_LENGTH) {
|
if (unilen > MAX_NAME_LENGTH) {
|
||||||
exfat_msg(sb, KERN_ERR,
|
exfat_err(sb, "failed to %s (estr:ENAMETOOLONG) nls len : %d, unilen : %d > %d",
|
||||||
"failed to %s (estr:ENAMETOOLONG) nls len : %d, unilen : %d > %d",
|
|
||||||
__func__, len, unilen, MAX_NAME_LENGTH);
|
__func__, len, unilen, MAX_NAME_LENGTH);
|
||||||
return -ENAMETOOLONG;
|
return -ENAMETOOLONG;
|
||||||
}
|
}
|
||||||
|
|
||||||
p_uniname->name_len = unilen & 0xFF;
|
|
||||||
|
|
||||||
for (i = 0; i < unilen; i++) {
|
for (i = 0; i < unilen; i++) {
|
||||||
if (*uniname < 0x0020 ||
|
if (*uniname < 0x0020 ||
|
||||||
exfat_wstrchr(bad_uni_chars, *uniname))
|
exfat_wstrchr(bad_uni_chars, *uniname))
|
||||||
@@ -529,7 +525,7 @@ static int exfat_utf8_to_utf16(struct super_block *sb,
|
|||||||
|
|
||||||
*uniname = '\0';
|
*uniname = '\0';
|
||||||
p_uniname->name_len = unilen;
|
p_uniname->name_len = unilen;
|
||||||
p_uniname->name_hash = exfat_calc_chksum_2byte(upname, unilen << 1, 0,
|
p_uniname->name_hash = exfat_calc_chksum16(upname, unilen << 1, 0,
|
||||||
CS_DEFAULT);
|
CS_DEFAULT);
|
||||||
|
|
||||||
if (p_lossy)
|
if (p_lossy)
|
||||||
@@ -537,22 +533,9 @@ static int exfat_utf8_to_utf16(struct super_block *sb,
|
|||||||
return unilen;
|
return unilen;
|
||||||
}
|
}
|
||||||
|
|
||||||
#define PLANE_SIZE 0x00010000
|
|
||||||
#define SURROGATE_MASK 0xfffff800
|
#define SURROGATE_MASK 0xfffff800
|
||||||
#define SURROGATE_PAIR 0x0000d800
|
#define SURROGATE_PAIR 0x0000d800
|
||||||
#define SURROGATE_LOW 0x00000400
|
#define SURROGATE_LOW 0x00000400
|
||||||
#define SURROGATE_BITS 0x000003ff
|
|
||||||
|
|
||||||
unsigned short exfat_high_surrogate(unicode_t u)
|
|
||||||
{
|
|
||||||
return ((u - PLANE_SIZE) >> 10) + SURROGATE_PAIR;
|
|
||||||
}
|
|
||||||
|
|
||||||
unsigned short exfat_low_surrogate(unicode_t u)
|
|
||||||
{
|
|
||||||
return ((u - PLANE_SIZE) & SURROGATE_BITS) | SURROGATE_PAIR |
|
|
||||||
SURROGATE_LOW;
|
|
||||||
}
|
|
||||||
|
|
||||||
static int __exfat_utf16_to_nls(struct super_block *sb,
|
static int __exfat_utf16_to_nls(struct super_block *sb,
|
||||||
struct exfat_uni_name *p_uniname, unsigned char *p_cstring,
|
struct exfat_uni_name *p_uniname, unsigned char *p_cstring,
|
||||||
@@ -638,7 +621,7 @@ static int exfat_nls_to_ucs2(struct super_block *sb,
|
|||||||
|
|
||||||
*uniname = '\0';
|
*uniname = '\0';
|
||||||
p_uniname->name_len = unilen;
|
p_uniname->name_len = unilen;
|
||||||
p_uniname->name_hash = exfat_calc_chksum_2byte(upname, unilen << 1, 0,
|
p_uniname->name_hash = exfat_calc_chksum16(upname, unilen << 1, 0,
|
||||||
CS_DEFAULT);
|
CS_DEFAULT);
|
||||||
|
|
||||||
if (p_lossy)
|
if (p_lossy)
|
||||||
@@ -670,7 +653,8 @@ static int exfat_load_upcase_table(struct super_block *sb,
|
|||||||
{
|
{
|
||||||
struct exfat_sb_info *sbi = EXFAT_SB(sb);
|
struct exfat_sb_info *sbi = EXFAT_SB(sb);
|
||||||
unsigned int sect_size = sb->s_blocksize;
|
unsigned int sect_size = sb->s_blocksize;
|
||||||
unsigned int i, index = 0, checksum = 0;
|
unsigned int i, index = 0;
|
||||||
|
u32 chksum = 0;
|
||||||
int ret;
|
int ret;
|
||||||
unsigned char skip = false;
|
unsigned char skip = false;
|
||||||
unsigned short *upcase_table;
|
unsigned short *upcase_table;
|
||||||
@@ -687,8 +671,7 @@ static int exfat_load_upcase_table(struct super_block *sb,
|
|||||||
|
|
||||||
bh = sb_bread(sb, sector);
|
bh = sb_bread(sb, sector);
|
||||||
if (!bh) {
|
if (!bh) {
|
||||||
exfat_msg(sb, KERN_ERR,
|
exfat_err(sb, "failed to read sector(0x%llx)\n",
|
||||||
"failed to read sector(0x%llx)\n",
|
|
||||||
(unsigned long long)sector);
|
(unsigned long long)sector);
|
||||||
ret = -EIO;
|
ret = -EIO;
|
||||||
goto free_table;
|
goto free_table;
|
||||||
@@ -697,13 +680,6 @@ static int exfat_load_upcase_table(struct super_block *sb,
|
|||||||
for (i = 0; i < sect_size && index <= 0xFFFF; i += 2) {
|
for (i = 0; i < sect_size && index <= 0xFFFF; i += 2) {
|
||||||
unsigned short uni = get_unaligned_le16(bh->b_data + i);
|
unsigned short uni = get_unaligned_le16(bh->b_data + i);
|
||||||
|
|
||||||
checksum = ((checksum & 1) ? 0x80000000 : 0) +
|
|
||||||
(checksum >> 1) +
|
|
||||||
*(((unsigned char *)bh->b_data) + i);
|
|
||||||
checksum = ((checksum & 1) ? 0x80000000 : 0) +
|
|
||||||
(checksum >> 1) +
|
|
||||||
*(((unsigned char *)bh->b_data) + (i + 1));
|
|
||||||
|
|
||||||
if (skip) {
|
if (skip) {
|
||||||
index += uni;
|
index += uni;
|
||||||
skip = false;
|
skip = false;
|
||||||
@@ -716,15 +692,15 @@ static int exfat_load_upcase_table(struct super_block *sb,
|
|||||||
index++;
|
index++;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
chksum = exfat_calc_chksum32(bh->b_data, i, chksum, CS_DEFAULT);
|
||||||
brelse(bh);
|
brelse(bh);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (index >= 0xFFFF && utbl_checksum == checksum)
|
if (index >= 0xFFFF && utbl_checksum == chksum)
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
exfat_msg(sb, KERN_ERR,
|
exfat_err(sb, "failed to load upcase table (idx : 0x%08x, chksum : 0x%08x, utbl_chksum : 0x%08x)",
|
||||||
"failed to load upcase table (idx : 0x%08x, chksum : 0x%08x, utbl_chksum : 0x%08x)\n",
|
index, chksum, utbl_checksum);
|
||||||
index, checksum, utbl_checksum);
|
|
||||||
ret = -EINVAL;
|
ret = -EINVAL;
|
||||||
free_table:
|
free_table:
|
||||||
exfat_free_upcase_table(sbi);
|
exfat_free_upcase_table(sbi);
|
||||||
|
258
fs/exfat/super.c
258
fs/exfat/super.c
@@ -49,7 +49,7 @@ static void exfat_put_super(struct super_block *sb)
|
|||||||
sync_blockdev(sb->s_bdev);
|
sync_blockdev(sb->s_bdev);
|
||||||
exfat_set_vol_flags(sb, VOL_CLEAN);
|
exfat_set_vol_flags(sb, VOL_CLEAN);
|
||||||
exfat_free_bitmap(sbi);
|
exfat_free_bitmap(sbi);
|
||||||
brelse(sbi->pbr_bh);
|
brelse(sbi->boot_bh);
|
||||||
mutex_unlock(&sbi->s_lock);
|
mutex_unlock(&sbi->s_lock);
|
||||||
|
|
||||||
call_rcu(&sbi->rcu, exfat_delayed_free);
|
call_rcu(&sbi->rcu, exfat_delayed_free);
|
||||||
@@ -101,8 +101,8 @@ static int exfat_statfs(struct dentry *dentry, struct kstatfs *buf)
|
|||||||
int exfat_set_vol_flags(struct super_block *sb, unsigned short new_flag)
|
int exfat_set_vol_flags(struct super_block *sb, unsigned short new_flag)
|
||||||
{
|
{
|
||||||
struct exfat_sb_info *sbi = EXFAT_SB(sb);
|
struct exfat_sb_info *sbi = EXFAT_SB(sb);
|
||||||
struct pbr64 *bpb = (struct pbr64 *)sbi->pbr_bh->b_data;
|
struct boot_sector *p_boot = (struct boot_sector *)sbi->boot_bh->b_data;
|
||||||
bool sync = 0;
|
bool sync;
|
||||||
|
|
||||||
/* flags are not changed */
|
/* flags are not changed */
|
||||||
if (sbi->vol_flag == new_flag)
|
if (sbi->vol_flag == new_flag)
|
||||||
@@ -116,18 +116,18 @@ int exfat_set_vol_flags(struct super_block *sb, unsigned short new_flag)
|
|||||||
if (sb_rdonly(sb))
|
if (sb_rdonly(sb))
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
bpb->bsx.vol_flags = cpu_to_le16(new_flag);
|
p_boot->vol_flags = cpu_to_le16(new_flag);
|
||||||
|
|
||||||
if (new_flag == VOL_DIRTY && !buffer_dirty(sbi->pbr_bh))
|
if (new_flag == VOL_DIRTY && !buffer_dirty(sbi->boot_bh))
|
||||||
sync = true;
|
sync = true;
|
||||||
else
|
else
|
||||||
sync = false;
|
sync = false;
|
||||||
|
|
||||||
set_buffer_uptodate(sbi->pbr_bh);
|
set_buffer_uptodate(sbi->boot_bh);
|
||||||
mark_buffer_dirty(sbi->pbr_bh);
|
mark_buffer_dirty(sbi->boot_bh);
|
||||||
|
|
||||||
if (sync)
|
if (sync)
|
||||||
sync_dirty_buffer(sbi->pbr_bh);
|
sync_dirty_buffer(sbi->boot_bh);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -273,9 +273,8 @@ static int exfat_parse_param(struct fs_context *fc, struct fs_parameter *param)
|
|||||||
break;
|
break;
|
||||||
case Opt_charset:
|
case Opt_charset:
|
||||||
exfat_free_iocharset(sbi);
|
exfat_free_iocharset(sbi);
|
||||||
opts->iocharset = kstrdup(param->string, GFP_KERNEL);
|
opts->iocharset = param->string;
|
||||||
if (!opts->iocharset)
|
param->string = NULL;
|
||||||
return -ENOMEM;
|
|
||||||
break;
|
break;
|
||||||
case Opt_errors:
|
case Opt_errors:
|
||||||
opts->errors = result.uint_32;
|
opts->errors = result.uint_32;
|
||||||
@@ -366,151 +365,208 @@ static int exfat_read_root(struct inode *inode)
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
static struct pbr *exfat_read_pbr_with_logical_sector(struct super_block *sb)
|
static int exfat_calibrate_blocksize(struct super_block *sb, int logical_sect)
|
||||||
{
|
{
|
||||||
struct exfat_sb_info *sbi = EXFAT_SB(sb);
|
struct exfat_sb_info *sbi = EXFAT_SB(sb);
|
||||||
struct pbr *p_pbr = (struct pbr *) (sbi->pbr_bh)->b_data;
|
|
||||||
unsigned short logical_sect = 0;
|
|
||||||
|
|
||||||
logical_sect = 1 << p_pbr->bsx.f64.sect_size_bits;
|
|
||||||
|
|
||||||
if (!is_power_of_2(logical_sect) ||
|
if (!is_power_of_2(logical_sect) ||
|
||||||
logical_sect < 512 || logical_sect > 4096) {
|
logical_sect < 512 || logical_sect > 4096) {
|
||||||
exfat_msg(sb, KERN_ERR, "bogus logical sector size %u",
|
exfat_err(sb, "bogus logical sector size %u", logical_sect);
|
||||||
logical_sect);
|
return -EIO;
|
||||||
return NULL;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (logical_sect < sb->s_blocksize) {
|
if (logical_sect < sb->s_blocksize) {
|
||||||
exfat_msg(sb, KERN_ERR,
|
exfat_err(sb, "logical sector size too small for device (logical sector size = %u)",
|
||||||
"logical sector size too small for device (logical sector size = %u)",
|
|
||||||
logical_sect);
|
logical_sect);
|
||||||
return NULL;
|
return -EIO;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (logical_sect > sb->s_blocksize) {
|
if (logical_sect > sb->s_blocksize) {
|
||||||
brelse(sbi->pbr_bh);
|
brelse(sbi->boot_bh);
|
||||||
sbi->pbr_bh = NULL;
|
sbi->boot_bh = NULL;
|
||||||
|
|
||||||
if (!sb_set_blocksize(sb, logical_sect)) {
|
if (!sb_set_blocksize(sb, logical_sect)) {
|
||||||
exfat_msg(sb, KERN_ERR,
|
exfat_err(sb, "unable to set blocksize %u",
|
||||||
"unable to set blocksize %u", logical_sect);
|
logical_sect);
|
||||||
return NULL;
|
return -EIO;
|
||||||
}
|
}
|
||||||
sbi->pbr_bh = sb_bread(sb, 0);
|
sbi->boot_bh = sb_bread(sb, 0);
|
||||||
if (!sbi->pbr_bh) {
|
if (!sbi->boot_bh) {
|
||||||
exfat_msg(sb, KERN_ERR,
|
exfat_err(sb, "unable to read boot sector (logical sector size = %lu)",
|
||||||
"unable to read boot sector (logical sector size = %lu)",
|
|
||||||
sb->s_blocksize);
|
sb->s_blocksize);
|
||||||
return NULL;
|
return -EIO;
|
||||||
}
|
}
|
||||||
|
|
||||||
p_pbr = (struct pbr *)sbi->pbr_bh->b_data;
|
|
||||||
}
|
}
|
||||||
return p_pbr;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* mount the file system volume */
|
static int exfat_read_boot_sector(struct super_block *sb)
|
||||||
static int __exfat_fill_super(struct super_block *sb)
|
|
||||||
{
|
{
|
||||||
int ret;
|
struct boot_sector *p_boot;
|
||||||
struct pbr *p_pbr;
|
|
||||||
struct pbr64 *p_bpb;
|
|
||||||
struct exfat_sb_info *sbi = EXFAT_SB(sb);
|
struct exfat_sb_info *sbi = EXFAT_SB(sb);
|
||||||
|
|
||||||
/* set block size to read super block */
|
/* set block size to read super block */
|
||||||
sb_min_blocksize(sb, 512);
|
sb_min_blocksize(sb, 512);
|
||||||
|
|
||||||
/* read boot sector */
|
/* read boot sector */
|
||||||
sbi->pbr_bh = sb_bread(sb, 0);
|
sbi->boot_bh = sb_bread(sb, 0);
|
||||||
if (!sbi->pbr_bh) {
|
if (!sbi->boot_bh) {
|
||||||
exfat_msg(sb, KERN_ERR, "unable to read boot sector");
|
exfat_err(sb, "unable to read boot sector");
|
||||||
return -EIO;
|
return -EIO;
|
||||||
}
|
}
|
||||||
|
p_boot = (struct boot_sector *)sbi->boot_bh->b_data;
|
||||||
|
|
||||||
/* PRB is read */
|
/* check the validity of BOOT */
|
||||||
p_pbr = (struct pbr *)sbi->pbr_bh->b_data;
|
if (le16_to_cpu((p_boot->signature)) != BOOT_SIGNATURE) {
|
||||||
|
exfat_err(sb, "invalid boot record signature");
|
||||||
/* check the validity of PBR */
|
return -EINVAL;
|
||||||
if (le16_to_cpu((p_pbr->signature)) != PBR_SIGNATURE) {
|
|
||||||
exfat_msg(sb, KERN_ERR, "invalid boot record signature");
|
|
||||||
ret = -EINVAL;
|
|
||||||
goto free_bh;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (memcmp(p_boot->fs_name, STR_EXFAT, BOOTSEC_FS_NAME_LEN)) {
|
||||||
/* check logical sector size */
|
exfat_err(sb, "invalid fs_name"); /* fs_name may unprintable */
|
||||||
p_pbr = exfat_read_pbr_with_logical_sector(sb);
|
return -EINVAL;
|
||||||
if (!p_pbr) {
|
|
||||||
ret = -EIO;
|
|
||||||
goto free_bh;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* res_zero field must be filled with zero to prevent mounting
|
* must_be_zero field must be filled with zero to prevent mounting
|
||||||
* from FAT volume.
|
* from FAT volume.
|
||||||
*/
|
*/
|
||||||
if (memchr_inv(p_pbr->bpb.f64.res_zero, 0,
|
if (memchr_inv(p_boot->must_be_zero, 0, sizeof(p_boot->must_be_zero)))
|
||||||
sizeof(p_pbr->bpb.f64.res_zero))) {
|
return -EINVAL;
|
||||||
ret = -EINVAL;
|
|
||||||
goto free_bh;
|
if (p_boot->num_fats != 1 && p_boot->num_fats != 2) {
|
||||||
|
exfat_err(sb, "bogus number of FAT structure");
|
||||||
|
return -EINVAL;
|
||||||
}
|
}
|
||||||
|
|
||||||
p_bpb = (struct pbr64 *)p_pbr;
|
sbi->sect_per_clus = 1 << p_boot->sect_per_clus_bits;
|
||||||
if (!p_bpb->bsx.num_fats) {
|
sbi->sect_per_clus_bits = p_boot->sect_per_clus_bits;
|
||||||
exfat_msg(sb, KERN_ERR, "bogus number of FAT structure");
|
sbi->cluster_size_bits = p_boot->sect_per_clus_bits +
|
||||||
ret = -EINVAL;
|
p_boot->sect_size_bits;
|
||||||
goto free_bh;
|
|
||||||
}
|
|
||||||
|
|
||||||
sbi->sect_per_clus = 1 << p_bpb->bsx.sect_per_clus_bits;
|
|
||||||
sbi->sect_per_clus_bits = p_bpb->bsx.sect_per_clus_bits;
|
|
||||||
sbi->cluster_size_bits = sbi->sect_per_clus_bits + sb->s_blocksize_bits;
|
|
||||||
sbi->cluster_size = 1 << sbi->cluster_size_bits;
|
sbi->cluster_size = 1 << sbi->cluster_size_bits;
|
||||||
sbi->num_FAT_sectors = le32_to_cpu(p_bpb->bsx.fat_length);
|
sbi->num_FAT_sectors = le32_to_cpu(p_boot->fat_length);
|
||||||
sbi->FAT1_start_sector = le32_to_cpu(p_bpb->bsx.fat_offset);
|
sbi->FAT1_start_sector = le32_to_cpu(p_boot->fat_offset);
|
||||||
sbi->FAT2_start_sector = p_bpb->bsx.num_fats == 1 ?
|
sbi->FAT2_start_sector = le32_to_cpu(p_boot->fat_offset);
|
||||||
sbi->FAT1_start_sector :
|
if (p_boot->num_fats == 2)
|
||||||
sbi->FAT1_start_sector + sbi->num_FAT_sectors;
|
sbi->FAT2_start_sector += sbi->num_FAT_sectors;
|
||||||
sbi->data_start_sector = le32_to_cpu(p_bpb->bsx.clu_offset);
|
sbi->data_start_sector = le32_to_cpu(p_boot->clu_offset);
|
||||||
sbi->num_sectors = le64_to_cpu(p_bpb->bsx.vol_length);
|
sbi->num_sectors = le64_to_cpu(p_boot->vol_length);
|
||||||
/* because the cluster index starts with 2 */
|
/* because the cluster index starts with 2 */
|
||||||
sbi->num_clusters = le32_to_cpu(p_bpb->bsx.clu_count) +
|
sbi->num_clusters = le32_to_cpu(p_boot->clu_count) +
|
||||||
EXFAT_RESERVED_CLUSTERS;
|
EXFAT_RESERVED_CLUSTERS;
|
||||||
|
|
||||||
sbi->root_dir = le32_to_cpu(p_bpb->bsx.root_cluster);
|
sbi->root_dir = le32_to_cpu(p_boot->root_cluster);
|
||||||
sbi->dentries_per_clu = 1 <<
|
sbi->dentries_per_clu = 1 <<
|
||||||
(sbi->cluster_size_bits - DENTRY_SIZE_BITS);
|
(sbi->cluster_size_bits - DENTRY_SIZE_BITS);
|
||||||
|
|
||||||
sbi->vol_flag = le16_to_cpu(p_bpb->bsx.vol_flags);
|
sbi->vol_flag = le16_to_cpu(p_boot->vol_flags);
|
||||||
sbi->clu_srch_ptr = EXFAT_FIRST_CLUSTER;
|
sbi->clu_srch_ptr = EXFAT_FIRST_CLUSTER;
|
||||||
sbi->used_clusters = EXFAT_CLUSTERS_UNTRACKED;
|
sbi->used_clusters = EXFAT_CLUSTERS_UNTRACKED;
|
||||||
|
|
||||||
if (le16_to_cpu(p_bpb->bsx.vol_flags) & VOL_DIRTY) {
|
/* check consistencies */
|
||||||
sbi->vol_flag |= VOL_DIRTY;
|
if (sbi->num_FAT_sectors << p_boot->sect_size_bits <
|
||||||
exfat_msg(sb, KERN_WARNING,
|
sbi->num_clusters * 4) {
|
||||||
"Volume was not properly unmounted. Some data may be corrupt. Please run fsck.");
|
exfat_err(sb, "bogus fat length");
|
||||||
|
return -EINVAL;
|
||||||
}
|
}
|
||||||
|
if (sbi->data_start_sector <
|
||||||
|
sbi->FAT1_start_sector + sbi->num_FAT_sectors * p_boot->num_fats) {
|
||||||
|
exfat_err(sb, "bogus data start sector");
|
||||||
|
return -EINVAL;
|
||||||
|
}
|
||||||
|
if (sbi->vol_flag & VOL_DIRTY)
|
||||||
|
exfat_warn(sb, "Volume was not properly unmounted. Some data may be corrupt. Please run fsck.");
|
||||||
|
if (sbi->vol_flag & ERR_MEDIUM)
|
||||||
|
exfat_warn(sb, "Medium has reported failures. Some data may be lost.");
|
||||||
|
|
||||||
/* exFAT file size is limited by a disk volume size */
|
/* exFAT file size is limited by a disk volume size */
|
||||||
sb->s_maxbytes = (u64)(sbi->num_clusters - EXFAT_RESERVED_CLUSTERS) <<
|
sb->s_maxbytes = (u64)(sbi->num_clusters - EXFAT_RESERVED_CLUSTERS) <<
|
||||||
sbi->cluster_size_bits;
|
sbi->cluster_size_bits;
|
||||||
|
|
||||||
|
/* check logical sector size */
|
||||||
|
if (exfat_calibrate_blocksize(sb, 1 << p_boot->sect_size_bits))
|
||||||
|
return -EIO;
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int exfat_verify_boot_region(struct super_block *sb)
|
||||||
|
{
|
||||||
|
struct buffer_head *bh = NULL;
|
||||||
|
u32 chksum = 0;
|
||||||
|
__le32 *p_sig, *p_chksum;
|
||||||
|
int sn, i;
|
||||||
|
|
||||||
|
/* read boot sector sub-regions */
|
||||||
|
for (sn = 0; sn < 11; sn++) {
|
||||||
|
bh = sb_bread(sb, sn);
|
||||||
|
if (!bh)
|
||||||
|
return -EIO;
|
||||||
|
|
||||||
|
if (sn != 0 && sn <= 8) {
|
||||||
|
/* extended boot sector sub-regions */
|
||||||
|
p_sig = (__le32 *)&bh->b_data[sb->s_blocksize - 4];
|
||||||
|
if (le32_to_cpu(*p_sig) != EXBOOT_SIGNATURE)
|
||||||
|
exfat_warn(sb, "Invalid exboot-signature(sector = %d): 0x%08x",
|
||||||
|
sn, le32_to_cpu(*p_sig));
|
||||||
|
}
|
||||||
|
|
||||||
|
chksum = exfat_calc_chksum32(bh->b_data, sb->s_blocksize,
|
||||||
|
chksum, sn ? CS_DEFAULT : CS_BOOT_SECTOR);
|
||||||
|
brelse(bh);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* boot checksum sub-regions */
|
||||||
|
bh = sb_bread(sb, sn);
|
||||||
|
if (!bh)
|
||||||
|
return -EIO;
|
||||||
|
|
||||||
|
for (i = 0; i < sb->s_blocksize; i += sizeof(u32)) {
|
||||||
|
p_chksum = (__le32 *)&bh->b_data[i];
|
||||||
|
if (le32_to_cpu(*p_chksum) != chksum) {
|
||||||
|
exfat_err(sb, "Invalid boot checksum (boot checksum : 0x%08x, checksum : 0x%08x)",
|
||||||
|
le32_to_cpu(*p_chksum), chksum);
|
||||||
|
brelse(bh);
|
||||||
|
return -EINVAL;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
brelse(bh);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* mount the file system volume */
|
||||||
|
static int __exfat_fill_super(struct super_block *sb)
|
||||||
|
{
|
||||||
|
int ret;
|
||||||
|
struct exfat_sb_info *sbi = EXFAT_SB(sb);
|
||||||
|
|
||||||
|
ret = exfat_read_boot_sector(sb);
|
||||||
|
if (ret) {
|
||||||
|
exfat_err(sb, "failed to read boot sector");
|
||||||
|
goto free_bh;
|
||||||
|
}
|
||||||
|
|
||||||
|
ret = exfat_verify_boot_region(sb);
|
||||||
|
if (ret) {
|
||||||
|
exfat_err(sb, "invalid boot region");
|
||||||
|
goto free_bh;
|
||||||
|
}
|
||||||
|
|
||||||
ret = exfat_create_upcase_table(sb);
|
ret = exfat_create_upcase_table(sb);
|
||||||
if (ret) {
|
if (ret) {
|
||||||
exfat_msg(sb, KERN_ERR, "failed to load upcase table");
|
exfat_err(sb, "failed to load upcase table");
|
||||||
goto free_bh;
|
goto free_bh;
|
||||||
}
|
}
|
||||||
|
|
||||||
ret = exfat_load_bitmap(sb);
|
ret = exfat_load_bitmap(sb);
|
||||||
if (ret) {
|
if (ret) {
|
||||||
exfat_msg(sb, KERN_ERR, "failed to load alloc-bitmap");
|
exfat_err(sb, "failed to load alloc-bitmap");
|
||||||
goto free_upcase_table;
|
goto free_upcase_table;
|
||||||
}
|
}
|
||||||
|
|
||||||
ret = exfat_count_used_clusters(sb, &sbi->used_clusters);
|
ret = exfat_count_used_clusters(sb, &sbi->used_clusters);
|
||||||
if (ret) {
|
if (ret) {
|
||||||
exfat_msg(sb, KERN_ERR, "failed to scan clusters");
|
exfat_err(sb, "failed to scan clusters");
|
||||||
goto free_alloc_bitmap;
|
goto free_alloc_bitmap;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -521,7 +577,7 @@ free_alloc_bitmap:
|
|||||||
free_upcase_table:
|
free_upcase_table:
|
||||||
exfat_free_upcase_table(sbi);
|
exfat_free_upcase_table(sbi);
|
||||||
free_bh:
|
free_bh:
|
||||||
brelse(sbi->pbr_bh);
|
brelse(sbi->boot_bh);
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -539,8 +595,7 @@ static int exfat_fill_super(struct super_block *sb, struct fs_context *fc)
|
|||||||
struct request_queue *q = bdev_get_queue(sb->s_bdev);
|
struct request_queue *q = bdev_get_queue(sb->s_bdev);
|
||||||
|
|
||||||
if (!blk_queue_discard(q)) {
|
if (!blk_queue_discard(q)) {
|
||||||
exfat_msg(sb, KERN_WARNING,
|
exfat_warn(sb, "mounting with \"discard\" option, but the device does not support discard");
|
||||||
"mounting with \"discard\" option, but the device does not support discard");
|
|
||||||
opts->discard = 0;
|
opts->discard = 0;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -555,7 +610,7 @@ static int exfat_fill_super(struct super_block *sb, struct fs_context *fc)
|
|||||||
|
|
||||||
err = __exfat_fill_super(sb);
|
err = __exfat_fill_super(sb);
|
||||||
if (err) {
|
if (err) {
|
||||||
exfat_msg(sb, KERN_ERR, "failed to recognize exfat type");
|
exfat_err(sb, "failed to recognize exfat type");
|
||||||
goto check_nls_io;
|
goto check_nls_io;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -567,7 +622,7 @@ static int exfat_fill_super(struct super_block *sb, struct fs_context *fc)
|
|||||||
else {
|
else {
|
||||||
sbi->nls_io = load_nls(sbi->options.iocharset);
|
sbi->nls_io = load_nls(sbi->options.iocharset);
|
||||||
if (!sbi->nls_io) {
|
if (!sbi->nls_io) {
|
||||||
exfat_msg(sb, KERN_ERR, "IO charset %s not found",
|
exfat_err(sb, "IO charset %s not found",
|
||||||
sbi->options.iocharset);
|
sbi->options.iocharset);
|
||||||
err = -EINVAL;
|
err = -EINVAL;
|
||||||
goto free_table;
|
goto free_table;
|
||||||
@@ -581,7 +636,7 @@ static int exfat_fill_super(struct super_block *sb, struct fs_context *fc)
|
|||||||
|
|
||||||
root_inode = new_inode(sb);
|
root_inode = new_inode(sb);
|
||||||
if (!root_inode) {
|
if (!root_inode) {
|
||||||
exfat_msg(sb, KERN_ERR, "failed to allocate root inode.");
|
exfat_err(sb, "failed to allocate root inode");
|
||||||
err = -ENOMEM;
|
err = -ENOMEM;
|
||||||
goto free_table;
|
goto free_table;
|
||||||
}
|
}
|
||||||
@@ -590,7 +645,7 @@ static int exfat_fill_super(struct super_block *sb, struct fs_context *fc)
|
|||||||
inode_set_iversion(root_inode, 1);
|
inode_set_iversion(root_inode, 1);
|
||||||
err = exfat_read_root(root_inode);
|
err = exfat_read_root(root_inode);
|
||||||
if (err) {
|
if (err) {
|
||||||
exfat_msg(sb, KERN_ERR, "failed to initialize root inode.");
|
exfat_err(sb, "failed to initialize root inode");
|
||||||
goto put_inode;
|
goto put_inode;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -599,7 +654,7 @@ static int exfat_fill_super(struct super_block *sb, struct fs_context *fc)
|
|||||||
|
|
||||||
sb->s_root = d_make_root(root_inode);
|
sb->s_root = d_make_root(root_inode);
|
||||||
if (!sb->s_root) {
|
if (!sb->s_root) {
|
||||||
exfat_msg(sb, KERN_ERR, "failed to get the root dentry");
|
exfat_err(sb, "failed to get the root dentry");
|
||||||
err = -ENOMEM;
|
err = -ENOMEM;
|
||||||
goto put_inode;
|
goto put_inode;
|
||||||
}
|
}
|
||||||
@@ -613,7 +668,7 @@ put_inode:
|
|||||||
free_table:
|
free_table:
|
||||||
exfat_free_upcase_table(sbi);
|
exfat_free_upcase_table(sbi);
|
||||||
exfat_free_bitmap(sbi);
|
exfat_free_bitmap(sbi);
|
||||||
brelse(sbi->pbr_bh);
|
brelse(sbi->boot_bh);
|
||||||
|
|
||||||
check_nls_io:
|
check_nls_io:
|
||||||
unload_nls(sbi->nls_io);
|
unload_nls(sbi->nls_io);
|
||||||
@@ -630,7 +685,12 @@ static int exfat_get_tree(struct fs_context *fc)
|
|||||||
|
|
||||||
static void exfat_free(struct fs_context *fc)
|
static void exfat_free(struct fs_context *fc)
|
||||||
{
|
{
|
||||||
kfree(fc->s_fs_info);
|
struct exfat_sb_info *sbi = fc->s_fs_info;
|
||||||
|
|
||||||
|
if (sbi) {
|
||||||
|
exfat_free_iocharset(sbi);
|
||||||
|
kfree(sbi);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static const struct fs_context_operations exfat_context_ops = {
|
static const struct fs_context_operations exfat_context_ops = {
|
||||||
|
Reference in New Issue
Block a user