2020-03-02 15:21:34 +09:00
|
|
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
|
|
|
/*
|
|
|
|
* Copyright (C) 2012-2013 Samsung Electronics Co., Ltd.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <linux/init.h>
|
|
|
|
#include <linux/buffer_head.h>
|
|
|
|
#include <linux/mpage.h>
|
|
|
|
#include <linux/bio.h>
|
|
|
|
#include <linux/blkdev.h>
|
|
|
|
#include <linux/time.h>
|
|
|
|
#include <linux/writeback.h>
|
|
|
|
#include <linux/uio.h>
|
|
|
|
#include <linux/random.h>
|
|
|
|
#include <linux/iversion.h>
|
|
|
|
|
|
|
|
#include "exfat_raw.h"
|
|
|
|
#include "exfat_fs.h"
|
|
|
|
|
2022-06-29 09:51:40 +08:00
|
|
|
int __exfat_write_inode(struct inode *inode, int sync)
|
2020-03-02 15:21:34 +09:00
|
|
|
{
|
|
|
|
unsigned long long on_disk_size;
|
|
|
|
struct exfat_dentry *ep, *ep2;
|
2022-11-17 11:37:13 +08:00
|
|
|
struct exfat_entry_set_cache es;
|
2020-03-02 15:21:34 +09:00
|
|
|
struct super_block *sb = inode->i_sb;
|
|
|
|
struct exfat_sb_info *sbi = EXFAT_SB(sb);
|
|
|
|
struct exfat_inode_info *ei = EXFAT_I(inode);
|
|
|
|
bool is_dir = (ei->type == TYPE_DIR) ? true : false;
|
2023-10-04 14:52:18 -04:00
|
|
|
struct timespec64 ts;
|
2020-03-02 15:21:34 +09:00
|
|
|
|
|
|
|
if (inode->i_ino == EXFAT_ROOT_INO)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
/*
|
2021-11-02 22:23:56 +01:00
|
|
|
* If the inode is already unlinked, there is no need for updating it.
|
2020-03-02 15:21:34 +09:00
|
|
|
*/
|
|
|
|
if (ei->dir.dir == DIR_DELETED)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
if (is_dir && ei->dir.dir == sbi->root_dir && ei->entry == -1)
|
|
|
|
return 0;
|
|
|
|
|
2020-07-31 14:58:26 +09:00
|
|
|
exfat_set_volume_dirty(sb);
|
2020-03-02 15:21:34 +09:00
|
|
|
|
|
|
|
/* get the directory entry of given file or directory */
|
2022-11-17 11:37:13 +08:00
|
|
|
if (exfat_get_dentry_set(&es, sb, &(ei->dir), ei->entry, ES_ALL_ENTRIES))
|
2020-03-02 15:21:34 +09:00
|
|
|
return -EIO;
|
2022-03-17 19:39:20 +08:00
|
|
|
ep = exfat_get_dentry_cached(&es, ES_IDX_FILE);
|
|
|
|
ep2 = exfat_get_dentry_cached(&es, ES_IDX_STREAM);
|
2020-03-02 15:21:34 +09:00
|
|
|
|
|
|
|
ep->dentry.file.attr = cpu_to_le16(exfat_make_attr(inode));
|
|
|
|
|
|
|
|
/* set FILE_INFO structure using the acquired struct exfat_dentry */
|
|
|
|
exfat_set_entry_time(sbi, &ei->i_crtime,
|
|
|
|
&ep->dentry.file.create_tz,
|
|
|
|
&ep->dentry.file.create_time,
|
|
|
|
&ep->dentry.file.create_date,
|
2020-04-22 08:30:56 +09:00
|
|
|
&ep->dentry.file.create_time_cs);
|
2023-10-31 17:52:13 +08:00
|
|
|
ts = inode_get_mtime(inode);
|
2023-10-04 14:52:18 -04:00
|
|
|
exfat_set_entry_time(sbi, &ts,
|
|
|
|
&ep->dentry.file.modify_tz,
|
|
|
|
&ep->dentry.file.modify_time,
|
|
|
|
&ep->dentry.file.modify_date,
|
|
|
|
&ep->dentry.file.modify_time_cs);
|
2023-10-31 17:52:13 +08:00
|
|
|
ts = inode_get_atime(inode);
|
2023-10-04 14:52:18 -04:00
|
|
|
exfat_set_entry_time(sbi, &ts,
|
|
|
|
&ep->dentry.file.access_tz,
|
|
|
|
&ep->dentry.file.access_time,
|
|
|
|
&ep->dentry.file.access_date,
|
|
|
|
NULL);
|
2020-03-02 15:21:34 +09:00
|
|
|
|
|
|
|
/* File size should be zero if there is no cluster allocated */
|
|
|
|
on_disk_size = i_size_read(inode);
|
|
|
|
|
|
|
|
if (ei->start_clu == EXFAT_EOF_CLUSTER)
|
|
|
|
on_disk_size = 0;
|
|
|
|
|
2023-03-13 12:38:53 +08:00
|
|
|
ep2->dentry.stream.size = cpu_to_le64(on_disk_size);
|
2023-05-30 17:35:00 +08:00
|
|
|
/*
|
|
|
|
* mmap write does not use exfat_write_end(), valid_size may be
|
|
|
|
* extended to the sector-aligned length in exfat_get_block().
|
|
|
|
* So we need to fixup valid_size to the writren length.
|
|
|
|
*/
|
|
|
|
if (on_disk_size < ei->valid_size)
|
|
|
|
ep2->dentry.stream.valid_size = ep2->dentry.stream.size;
|
|
|
|
else
|
|
|
|
ep2->dentry.stream.valid_size = cpu_to_le64(ei->valid_size);
|
|
|
|
|
2022-06-29 09:51:40 +08:00
|
|
|
if (on_disk_size) {
|
|
|
|
ep2->dentry.stream.flags = ei->flags;
|
|
|
|
ep2->dentry.stream.start_clu = cpu_to_le32(ei->start_clu);
|
|
|
|
} else {
|
|
|
|
ep2->dentry.stream.flags = ALLOC_FAT_CHAIN;
|
|
|
|
ep2->dentry.stream.start_clu = EXFAT_FREE_CLUSTER;
|
|
|
|
}
|
2020-03-02 15:21:34 +09:00
|
|
|
|
2022-08-05 16:57:04 +08:00
|
|
|
exfat_update_dir_chksum(&es);
|
2022-03-17 18:12:40 +08:00
|
|
|
return exfat_put_dentry_set(&es, sync);
|
2020-03-02 15:21:34 +09:00
|
|
|
}
|
|
|
|
|
|
|
|
int exfat_write_inode(struct inode *inode, struct writeback_control *wbc)
|
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
|
exfat: Implement sops->shutdown and ioctl
We found that when writing a large file through buffer write, if the
disk is inaccessible, exFAT does not return an error normally, which
leads to the writing process not stopping properly.
To easily reproduce this issue, you can follow the steps below:
1. format a device to exFAT and then mount (with a full disk erase)
2. dd if=/dev/zero of=/exfat_mount/test.img bs=1M count=8192
3. eject the device
You may find that the dd process does not stop immediately and may
continue for a long time.
The root cause of this issue is that during buffer write process,
exFAT does not need to access the disk to look up directory entries
or the FAT table (whereas FAT would do) every time data is written.
Instead, exFAT simply marks the buffer as dirty and returns,
delegating the writeback operation to the writeback process.
If the disk cannot be accessed at this time, the error will only be
returned to the writeback process, and the original process will not
receive the error, so it cannot be returned to the user side.
When the disk cannot be accessed normally, an error should be returned
to stop the writing process.
Implement sops->shutdown and ioctl to shut down the file system
when underlying block device is marked dead.
Signed-off-by: Dongliang Cui <dongliang.cui@unisoc.com>
Signed-off-by: Zhiguo Niu <zhiguo.niu@unisoc.com>
Signed-off-by: Namjae Jeon <linkinjeon@kernel.org>
2024-09-18 07:40:05 +09:00
|
|
|
if (unlikely(exfat_forced_shutdown(inode->i_sb)))
|
|
|
|
return -EIO;
|
|
|
|
|
2020-03-02 15:21:34 +09:00
|
|
|
mutex_lock(&EXFAT_SB(inode->i_sb)->s_lock);
|
|
|
|
ret = __exfat_write_inode(inode, wbc->sync_mode == WB_SYNC_ALL);
|
|
|
|
mutex_unlock(&EXFAT_SB(inode->i_sb)->s_lock);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
void exfat_sync_inode(struct inode *inode)
|
|
|
|
{
|
|
|
|
lockdep_assert_held(&EXFAT_SB(inode->i_sb)->s_lock);
|
|
|
|
__exfat_write_inode(inode, 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Input: inode, (logical) clu_offset, target allocation area
|
|
|
|
* Output: errcode, cluster number
|
|
|
|
* *clu = (~0), if it's unable to allocate a new cluster
|
|
|
|
*/
|
|
|
|
static int exfat_map_cluster(struct inode *inode, unsigned int clu_offset,
|
|
|
|
unsigned int *clu, int create)
|
|
|
|
{
|
2022-06-29 11:05:51 +08:00
|
|
|
int ret;
|
2020-03-02 15:21:34 +09:00
|
|
|
unsigned int last_clu;
|
|
|
|
struct exfat_chain new_clu;
|
|
|
|
struct super_block *sb = inode->i_sb;
|
|
|
|
struct exfat_sb_info *sbi = EXFAT_SB(sb);
|
|
|
|
struct exfat_inode_info *ei = EXFAT_I(inode);
|
|
|
|
unsigned int local_clu_offset = clu_offset;
|
2024-01-12 10:32:58 +08:00
|
|
|
unsigned int num_to_be_allocated = 0, num_clusters;
|
2020-03-02 15:21:34 +09:00
|
|
|
|
2024-01-12 10:32:58 +08:00
|
|
|
num_clusters = EXFAT_B_TO_CLU(exfat_ondisk_size(inode), sbi);
|
2020-03-02 15:21:34 +09:00
|
|
|
|
|
|
|
if (clu_offset >= num_clusters)
|
|
|
|
num_to_be_allocated = clu_offset - num_clusters + 1;
|
|
|
|
|
|
|
|
if (!create && (num_to_be_allocated > 0)) {
|
|
|
|
*clu = EXFAT_EOF_CLUSTER;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
*clu = last_clu = ei->start_clu;
|
|
|
|
|
|
|
|
if (ei->flags == ALLOC_NO_FAT_CHAIN) {
|
|
|
|
if (clu_offset > 0 && *clu != EXFAT_EOF_CLUSTER) {
|
|
|
|
last_clu += clu_offset - 1;
|
|
|
|
|
|
|
|
if (clu_offset == num_clusters)
|
|
|
|
*clu = EXFAT_EOF_CLUSTER;
|
|
|
|
else
|
|
|
|
*clu += clu_offset;
|
|
|
|
}
|
|
|
|
} else if (ei->type == TYPE_FILE) {
|
|
|
|
unsigned int fclus = 0;
|
|
|
|
int err = exfat_get_cluster(inode, clu_offset,
|
|
|
|
&fclus, clu, &last_clu, 1);
|
|
|
|
if (err)
|
|
|
|
return -EIO;
|
|
|
|
|
|
|
|
clu_offset -= fclus;
|
|
|
|
} else {
|
|
|
|
/* hint information */
|
|
|
|
if (clu_offset > 0 && ei->hint_bmap.off != EXFAT_EOF_CLUSTER &&
|
|
|
|
ei->hint_bmap.off > 0 && clu_offset >= ei->hint_bmap.off) {
|
|
|
|
clu_offset -= ei->hint_bmap.off;
|
|
|
|
/* hint_bmap.clu should be valid */
|
|
|
|
WARN_ON(ei->hint_bmap.clu < 2);
|
|
|
|
*clu = ei->hint_bmap.clu;
|
|
|
|
}
|
|
|
|
|
|
|
|
while (clu_offset > 0 && *clu != EXFAT_EOF_CLUSTER) {
|
|
|
|
last_clu = *clu;
|
|
|
|
if (exfat_get_next_cluster(sb, clu))
|
|
|
|
return -EIO;
|
|
|
|
clu_offset--;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (*clu == EXFAT_EOF_CLUSTER) {
|
2020-07-31 14:58:26 +09:00
|
|
|
exfat_set_volume_dirty(sb);
|
2020-03-02 15:21:34 +09:00
|
|
|
|
|
|
|
new_clu.dir = (last_clu == EXFAT_EOF_CLUSTER) ?
|
|
|
|
EXFAT_EOF_CLUSTER : last_clu + 1;
|
|
|
|
new_clu.size = 0;
|
|
|
|
new_clu.flags = ei->flags;
|
|
|
|
|
|
|
|
/* allocate a cluster */
|
|
|
|
if (num_to_be_allocated < 1) {
|
|
|
|
/* Broken FAT (i_sze > allocated FAT) */
|
|
|
|
exfat_fs_error(sb, "broken FAT chain.");
|
|
|
|
return -EIO;
|
|
|
|
}
|
|
|
|
|
2021-03-15 13:12:55 +09:00
|
|
|
ret = exfat_alloc_cluster(inode, num_to_be_allocated, &new_clu,
|
|
|
|
inode_needs_sync(inode));
|
2020-03-02 15:21:34 +09:00
|
|
|
if (ret)
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
if (new_clu.dir == EXFAT_EOF_CLUSTER ||
|
|
|
|
new_clu.dir == EXFAT_FREE_CLUSTER) {
|
|
|
|
exfat_fs_error(sb,
|
|
|
|
"bogus cluster new allocated (last_clu : %u, new_clu : %u)",
|
|
|
|
last_clu, new_clu.dir);
|
|
|
|
return -EIO;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* append to the FAT chain */
|
|
|
|
if (last_clu == EXFAT_EOF_CLUSTER) {
|
|
|
|
if (new_clu.flags == ALLOC_FAT_CHAIN)
|
|
|
|
ei->flags = ALLOC_FAT_CHAIN;
|
|
|
|
ei->start_clu = new_clu.dir;
|
|
|
|
} else {
|
|
|
|
if (new_clu.flags != ei->flags) {
|
|
|
|
/* no-fat-chain bit is disabled,
|
|
|
|
* so fat-chain should be synced with
|
|
|
|
* alloc-bitmap
|
|
|
|
*/
|
|
|
|
exfat_chain_cont_cluster(sb, ei->start_clu,
|
|
|
|
num_clusters);
|
|
|
|
ei->flags = ALLOC_FAT_CHAIN;
|
|
|
|
}
|
|
|
|
if (new_clu.flags == ALLOC_FAT_CHAIN)
|
|
|
|
if (exfat_ent_set(sb, last_clu, new_clu.dir))
|
|
|
|
return -EIO;
|
|
|
|
}
|
|
|
|
|
|
|
|
num_clusters += num_to_be_allocated;
|
|
|
|
*clu = new_clu.dir;
|
|
|
|
|
2023-01-04 14:37:47 +08:00
|
|
|
inode->i_blocks += EXFAT_CLU_TO_B(num_to_be_allocated, sbi) >> 9;
|
2020-03-02 15:21:34 +09:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Move *clu pointer along FAT chains (hole care) because the
|
|
|
|
* caller of this function expect *clu to be the last cluster.
|
|
|
|
* This only works when num_to_be_allocated >= 2,
|
|
|
|
* *clu = (the first cluster of the allocated chain) =>
|
|
|
|
* (the last cluster of ...)
|
|
|
|
*/
|
|
|
|
if (ei->flags == ALLOC_NO_FAT_CHAIN) {
|
|
|
|
*clu += num_to_be_allocated - 1;
|
|
|
|
} else {
|
|
|
|
while (num_to_be_allocated > 1) {
|
|
|
|
if (exfat_get_next_cluster(sb, clu))
|
|
|
|
return -EIO;
|
|
|
|
num_to_be_allocated--;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
/* hint information */
|
|
|
|
ei->hint_bmap.off = local_clu_offset;
|
|
|
|
ei->hint_bmap.clu = *clu;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int exfat_get_block(struct inode *inode, sector_t iblock,
|
|
|
|
struct buffer_head *bh_result, int create)
|
|
|
|
{
|
|
|
|
struct exfat_inode_info *ei = EXFAT_I(inode);
|
|
|
|
struct super_block *sb = inode->i_sb;
|
|
|
|
struct exfat_sb_info *sbi = EXFAT_SB(sb);
|
|
|
|
unsigned long max_blocks = bh_result->b_size >> inode->i_blkbits;
|
|
|
|
int err = 0;
|
|
|
|
unsigned long mapped_blocks = 0;
|
|
|
|
unsigned int cluster, sec_offset;
|
|
|
|
sector_t last_block;
|
|
|
|
sector_t phys = 0;
|
2023-03-13 12:38:53 +08:00
|
|
|
sector_t valid_blks;
|
2020-03-02 15:21:34 +09:00
|
|
|
|
|
|
|
mutex_lock(&sbi->s_lock);
|
|
|
|
last_block = EXFAT_B_TO_BLK_ROUND_UP(i_size_read(inode), sb);
|
|
|
|
if (iblock >= last_block && !create)
|
|
|
|
goto done;
|
|
|
|
|
|
|
|
/* Is this block already allocated? */
|
|
|
|
err = exfat_map_cluster(inode, iblock >> sbi->sect_per_clus_bits,
|
|
|
|
&cluster, create);
|
|
|
|
if (err) {
|
|
|
|
if (err != -ENOSPC)
|
|
|
|
exfat_fs_error_ratelimit(sb,
|
|
|
|
"failed to bmap (inode : %p iblock : %llu, err : %d)",
|
|
|
|
inode, (unsigned long long)iblock, err);
|
|
|
|
goto unlock_ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (cluster == EXFAT_EOF_CLUSTER)
|
|
|
|
goto done;
|
|
|
|
|
|
|
|
/* sector offset in cluster */
|
|
|
|
sec_offset = iblock & (sbi->sect_per_clus - 1);
|
|
|
|
|
|
|
|
phys = exfat_cluster_to_sector(sbi, cluster) + sec_offset;
|
|
|
|
mapped_blocks = sbi->sect_per_clus - sec_offset;
|
|
|
|
max_blocks = min(mapped_blocks, max_blocks);
|
|
|
|
|
2023-03-13 12:38:53 +08:00
|
|
|
map_bh(bh_result, sb, phys);
|
|
|
|
if (buffer_delay(bh_result))
|
|
|
|
clear_buffer_delay(bh_result);
|
|
|
|
|
2020-03-02 15:21:34 +09:00
|
|
|
if (create) {
|
2023-03-13 12:38:53 +08:00
|
|
|
valid_blks = EXFAT_B_TO_BLK_ROUND_UP(ei->valid_size, sb);
|
|
|
|
|
|
|
|
if (iblock + max_blocks < valid_blks) {
|
|
|
|
/* The range has been written, map it */
|
|
|
|
goto done;
|
|
|
|
} else if (iblock < valid_blks) {
|
|
|
|
/*
|
|
|
|
* The range has been partially written,
|
|
|
|
* map the written part.
|
|
|
|
*/
|
|
|
|
max_blocks = valid_blks - iblock;
|
|
|
|
goto done;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* The area has not been written, map and mark as new. */
|
2024-09-17 14:58:56 +09:00
|
|
|
set_buffer_new(bh_result);
|
2023-05-30 17:35:00 +08:00
|
|
|
|
|
|
|
ei->valid_size = EXFAT_BLK_TO_B(iblock + max_blocks, sb);
|
|
|
|
mark_inode_dirty(inode);
|
2023-03-13 12:38:53 +08:00
|
|
|
} else {
|
|
|
|
valid_blks = EXFAT_B_TO_BLK(ei->valid_size, sb);
|
|
|
|
|
|
|
|
if (iblock + max_blocks < valid_blks) {
|
|
|
|
/* The range has been written, map it */
|
|
|
|
goto done;
|
|
|
|
} else if (iblock < valid_blks) {
|
|
|
|
/*
|
|
|
|
* The area has been partially written,
|
|
|
|
* map the written part.
|
|
|
|
*/
|
|
|
|
max_blocks = valid_blks - iblock;
|
|
|
|
goto done;
|
|
|
|
} else if (iblock == valid_blks &&
|
|
|
|
(ei->valid_size & (sb->s_blocksize - 1))) {
|
|
|
|
/*
|
|
|
|
* The block has been partially written,
|
|
|
|
* zero the unwritten part and map the block.
|
|
|
|
*/
|
2024-09-17 14:58:56 +09:00
|
|
|
loff_t size, off, pos;
|
2023-03-13 12:38:53 +08:00
|
|
|
|
|
|
|
max_blocks = 1;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* For direct read, the unwritten part will be zeroed in
|
|
|
|
* exfat_direct_IO()
|
|
|
|
*/
|
|
|
|
if (!bh_result->b_folio)
|
|
|
|
goto done;
|
|
|
|
|
2024-09-17 14:58:56 +09:00
|
|
|
pos = EXFAT_BLK_TO_B(iblock, sb);
|
2023-03-13 12:38:53 +08:00
|
|
|
size = ei->valid_size - pos;
|
|
|
|
off = pos & (PAGE_SIZE - 1);
|
|
|
|
|
|
|
|
folio_set_bh(bh_result, bh_result->b_folio, off);
|
|
|
|
err = bh_read(bh_result, 0);
|
|
|
|
if (err < 0)
|
|
|
|
goto unlock_ret;
|
|
|
|
|
|
|
|
folio_zero_segment(bh_result->b_folio, off + size,
|
|
|
|
off + sb->s_blocksize);
|
|
|
|
} else {
|
|
|
|
/*
|
|
|
|
* The range has not been written, clear the mapped flag
|
|
|
|
* to only zero the cache and do not read from disk.
|
|
|
|
*/
|
|
|
|
clear_buffer_mapped(bh_result);
|
|
|
|
}
|
2020-03-02 15:21:34 +09:00
|
|
|
}
|
|
|
|
done:
|
|
|
|
bh_result->b_size = EXFAT_BLK_TO_B(max_blocks, sb);
|
|
|
|
unlock_ret:
|
|
|
|
mutex_unlock(&sbi->s_lock);
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
2022-04-29 11:47:39 -04:00
|
|
|
static int exfat_read_folio(struct file *file, struct folio *folio)
|
2020-03-02 15:21:34 +09:00
|
|
|
{
|
2022-04-29 11:47:39 -04:00
|
|
|
return mpage_read_folio(folio, exfat_get_block);
|
2020-03-02 15:21:34 +09:00
|
|
|
}
|
|
|
|
|
fs: convert mpage_readpages to mpage_readahead
Implement the new readahead aop and convert all callers (block_dev,
exfat, ext2, fat, gfs2, hpfs, isofs, jfs, nilfs2, ocfs2, omfs, qnx6,
reiserfs & udf).
The callers are all trivial except for GFS2 & OCFS2.
Signed-off-by: Matthew Wilcox (Oracle) <willy@infradead.org>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Reviewed-by: Junxiao Bi <junxiao.bi@oracle.com> # ocfs2
Reviewed-by: Joseph Qi <joseph.qi@linux.alibaba.com> # ocfs2
Reviewed-by: Dave Chinner <dchinner@redhat.com>
Reviewed-by: John Hubbard <jhubbard@nvidia.com>
Reviewed-by: Christoph Hellwig <hch@lst.de>
Reviewed-by: William Kucharski <william.kucharski@oracle.com>
Cc: Chao Yu <yuchao0@huawei.com>
Cc: Cong Wang <xiyou.wangcong@gmail.com>
Cc: Darrick J. Wong <darrick.wong@oracle.com>
Cc: Eric Biggers <ebiggers@google.com>
Cc: Gao Xiang <gaoxiang25@huawei.com>
Cc: Jaegeuk Kim <jaegeuk@kernel.org>
Cc: Michal Hocko <mhocko@suse.com>
Cc: Zi Yan <ziy@nvidia.com>
Cc: Johannes Thumshirn <johannes.thumshirn@wdc.com>
Cc: Miklos Szeredi <mszeredi@redhat.com>
Link: http://lkml.kernel.org/r/20200414150233.24495-17-willy@infradead.org
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
2020-06-01 21:47:02 -07:00
|
|
|
static void exfat_readahead(struct readahead_control *rac)
|
2020-03-02 15:21:34 +09:00
|
|
|
{
|
2023-03-13 12:38:53 +08:00
|
|
|
struct address_space *mapping = rac->mapping;
|
|
|
|
struct inode *inode = mapping->host;
|
|
|
|
struct exfat_inode_info *ei = EXFAT_I(inode);
|
|
|
|
loff_t pos = readahead_pos(rac);
|
|
|
|
|
|
|
|
/* Range cross valid_size, read it page by page. */
|
|
|
|
if (ei->valid_size < i_size_read(inode) &&
|
|
|
|
pos <= ei->valid_size &&
|
|
|
|
ei->valid_size < pos + readahead_length(rac))
|
|
|
|
return;
|
|
|
|
|
fs: convert mpage_readpages to mpage_readahead
Implement the new readahead aop and convert all callers (block_dev,
exfat, ext2, fat, gfs2, hpfs, isofs, jfs, nilfs2, ocfs2, omfs, qnx6,
reiserfs & udf).
The callers are all trivial except for GFS2 & OCFS2.
Signed-off-by: Matthew Wilcox (Oracle) <willy@infradead.org>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Reviewed-by: Junxiao Bi <junxiao.bi@oracle.com> # ocfs2
Reviewed-by: Joseph Qi <joseph.qi@linux.alibaba.com> # ocfs2
Reviewed-by: Dave Chinner <dchinner@redhat.com>
Reviewed-by: John Hubbard <jhubbard@nvidia.com>
Reviewed-by: Christoph Hellwig <hch@lst.de>
Reviewed-by: William Kucharski <william.kucharski@oracle.com>
Cc: Chao Yu <yuchao0@huawei.com>
Cc: Cong Wang <xiyou.wangcong@gmail.com>
Cc: Darrick J. Wong <darrick.wong@oracle.com>
Cc: Eric Biggers <ebiggers@google.com>
Cc: Gao Xiang <gaoxiang25@huawei.com>
Cc: Jaegeuk Kim <jaegeuk@kernel.org>
Cc: Michal Hocko <mhocko@suse.com>
Cc: Zi Yan <ziy@nvidia.com>
Cc: Johannes Thumshirn <johannes.thumshirn@wdc.com>
Cc: Miklos Szeredi <mszeredi@redhat.com>
Link: http://lkml.kernel.org/r/20200414150233.24495-17-willy@infradead.org
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
2020-06-01 21:47:02 -07:00
|
|
|
mpage_readahead(rac, exfat_get_block);
|
2020-03-02 15:21:34 +09:00
|
|
|
}
|
|
|
|
|
|
|
|
static int exfat_writepages(struct address_space *mapping,
|
|
|
|
struct writeback_control *wbc)
|
|
|
|
{
|
exfat: Implement sops->shutdown and ioctl
We found that when writing a large file through buffer write, if the
disk is inaccessible, exFAT does not return an error normally, which
leads to the writing process not stopping properly.
To easily reproduce this issue, you can follow the steps below:
1. format a device to exFAT and then mount (with a full disk erase)
2. dd if=/dev/zero of=/exfat_mount/test.img bs=1M count=8192
3. eject the device
You may find that the dd process does not stop immediately and may
continue for a long time.
The root cause of this issue is that during buffer write process,
exFAT does not need to access the disk to look up directory entries
or the FAT table (whereas FAT would do) every time data is written.
Instead, exFAT simply marks the buffer as dirty and returns,
delegating the writeback operation to the writeback process.
If the disk cannot be accessed at this time, the error will only be
returned to the writeback process, and the original process will not
receive the error, so it cannot be returned to the user side.
When the disk cannot be accessed normally, an error should be returned
to stop the writing process.
Implement sops->shutdown and ioctl to shut down the file system
when underlying block device is marked dead.
Signed-off-by: Dongliang Cui <dongliang.cui@unisoc.com>
Signed-off-by: Zhiguo Niu <zhiguo.niu@unisoc.com>
Signed-off-by: Namjae Jeon <linkinjeon@kernel.org>
2024-09-18 07:40:05 +09:00
|
|
|
if (unlikely(exfat_forced_shutdown(mapping->host->i_sb)))
|
|
|
|
return -EIO;
|
|
|
|
|
2020-03-02 15:21:34 +09:00
|
|
|
return mpage_writepages(mapping, wbc, exfat_get_block);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void exfat_write_failed(struct address_space *mapping, loff_t to)
|
|
|
|
{
|
|
|
|
struct inode *inode = mapping->host;
|
|
|
|
|
|
|
|
if (to > i_size_read(inode)) {
|
|
|
|
truncate_pagecache(inode, i_size_read(inode));
|
2023-10-04 14:52:18 -04:00
|
|
|
inode_set_mtime_to_ts(inode, inode_set_ctime_current(inode));
|
2022-11-17 10:36:21 +08:00
|
|
|
exfat_truncate(inode);
|
2020-03-02 15:21:34 +09:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static int exfat_write_begin(struct file *file, struct address_space *mapping,
|
2022-02-22 14:31:43 -05:00
|
|
|
loff_t pos, unsigned int len,
|
2024-07-15 14:24:01 -04:00
|
|
|
struct folio **foliop, void **fsdata)
|
2020-03-02 15:21:34 +09:00
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
|
exfat: Implement sops->shutdown and ioctl
We found that when writing a large file through buffer write, if the
disk is inaccessible, exFAT does not return an error normally, which
leads to the writing process not stopping properly.
To easily reproduce this issue, you can follow the steps below:
1. format a device to exFAT and then mount (with a full disk erase)
2. dd if=/dev/zero of=/exfat_mount/test.img bs=1M count=8192
3. eject the device
You may find that the dd process does not stop immediately and may
continue for a long time.
The root cause of this issue is that during buffer write process,
exFAT does not need to access the disk to look up directory entries
or the FAT table (whereas FAT would do) every time data is written.
Instead, exFAT simply marks the buffer as dirty and returns,
delegating the writeback operation to the writeback process.
If the disk cannot be accessed at this time, the error will only be
returned to the writeback process, and the original process will not
receive the error, so it cannot be returned to the user side.
When the disk cannot be accessed normally, an error should be returned
to stop the writing process.
Implement sops->shutdown and ioctl to shut down the file system
when underlying block device is marked dead.
Signed-off-by: Dongliang Cui <dongliang.cui@unisoc.com>
Signed-off-by: Zhiguo Niu <zhiguo.niu@unisoc.com>
Signed-off-by: Namjae Jeon <linkinjeon@kernel.org>
2024-09-18 07:40:05 +09:00
|
|
|
if (unlikely(exfat_forced_shutdown(mapping->host->i_sb)))
|
|
|
|
return -EIO;
|
|
|
|
|
2024-07-15 14:24:01 -04:00
|
|
|
ret = block_write_begin(mapping, pos, len, foliop, exfat_get_block);
|
2020-03-02 15:21:34 +09:00
|
|
|
|
|
|
|
if (ret < 0)
|
|
|
|
exfat_write_failed(mapping, pos+len);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int exfat_write_end(struct file *file, struct address_space *mapping,
|
|
|
|
loff_t pos, unsigned int len, unsigned int copied,
|
2024-07-10 15:45:32 -04:00
|
|
|
struct folio *folio, void *fsdata)
|
2020-03-02 15:21:34 +09:00
|
|
|
{
|
|
|
|
struct inode *inode = mapping->host;
|
|
|
|
struct exfat_inode_info *ei = EXFAT_I(inode);
|
|
|
|
int err;
|
|
|
|
|
2024-07-10 15:45:32 -04:00
|
|
|
err = generic_write_end(file, mapping, pos, len, copied, folio, fsdata);
|
2020-03-02 15:21:34 +09:00
|
|
|
if (err < len)
|
|
|
|
exfat_write_failed(mapping, pos+len);
|
|
|
|
|
2023-03-13 12:38:53 +08:00
|
|
|
if (!(err < 0) && pos + err > ei->valid_size) {
|
|
|
|
ei->valid_size = pos + err;
|
|
|
|
mark_inode_dirty(inode);
|
|
|
|
}
|
|
|
|
|
2023-10-30 20:53:18 +09:00
|
|
|
if (!(err < 0) && !(ei->attr & EXFAT_ATTR_ARCHIVE)) {
|
2023-10-04 14:52:18 -04:00
|
|
|
inode_set_mtime_to_ts(inode, inode_set_ctime_current(inode));
|
2023-10-30 20:53:18 +09:00
|
|
|
ei->attr |= EXFAT_ATTR_ARCHIVE;
|
2020-03-02 15:21:34 +09:00
|
|
|
mark_inode_dirty(inode);
|
|
|
|
}
|
|
|
|
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
static ssize_t exfat_direct_IO(struct kiocb *iocb, struct iov_iter *iter)
|
|
|
|
{
|
|
|
|
struct address_space *mapping = iocb->ki_filp->f_mapping;
|
|
|
|
struct inode *inode = mapping->host;
|
2023-03-13 12:38:53 +08:00
|
|
|
struct exfat_inode_info *ei = EXFAT_I(inode);
|
|
|
|
loff_t pos = iocb->ki_pos;
|
2024-01-18 09:52:52 +08:00
|
|
|
loff_t size = pos + iov_iter_count(iter);
|
2020-03-02 15:21:34 +09:00
|
|
|
int rw = iov_iter_rw(iter);
|
|
|
|
ssize_t ret;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Need to use the DIO_LOCKING for avoiding the race
|
|
|
|
* condition of exfat_get_block() and ->truncate().
|
|
|
|
*/
|
|
|
|
ret = blockdev_direct_IO(iocb, inode, iter, exfat_get_block);
|
2023-03-13 12:38:53 +08:00
|
|
|
if (ret < 0) {
|
2024-01-18 09:52:52 +08:00
|
|
|
if (rw == WRITE && ret != -EIOCBQUEUED)
|
2023-03-13 12:38:53 +08:00
|
|
|
exfat_write_failed(mapping, size);
|
|
|
|
|
2024-01-18 09:52:52 +08:00
|
|
|
return ret;
|
2023-03-13 12:38:53 +08:00
|
|
|
} else
|
|
|
|
size = pos + ret;
|
|
|
|
|
2024-09-17 14:58:56 +09:00
|
|
|
if (rw == WRITE) {
|
|
|
|
/*
|
|
|
|
* If the block had been partially written before this write,
|
|
|
|
* ->valid_size will not be updated in exfat_get_block(),
|
|
|
|
* update it here.
|
|
|
|
*/
|
|
|
|
if (ei->valid_size < size) {
|
|
|
|
ei->valid_size = size;
|
|
|
|
mark_inode_dirty(inode);
|
|
|
|
}
|
|
|
|
} else if (pos < ei->valid_size && ei->valid_size < size) {
|
|
|
|
/* zero the unwritten part in the partially written block */
|
2023-03-13 12:38:53 +08:00
|
|
|
iov_iter_revert(iter, size - ei->valid_size);
|
|
|
|
iov_iter_zero(size - ei->valid_size, iter);
|
|
|
|
}
|
|
|
|
|
2020-03-02 15:21:34 +09:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static sector_t exfat_aop_bmap(struct address_space *mapping, sector_t block)
|
|
|
|
{
|
|
|
|
sector_t blocknr;
|
|
|
|
|
|
|
|
/* exfat_get_cluster() assumes the requested blocknr isn't truncated. */
|
|
|
|
down_read(&EXFAT_I(mapping->host)->truncate_lock);
|
|
|
|
blocknr = generic_block_bmap(mapping, block, exfat_get_block);
|
|
|
|
up_read(&EXFAT_I(mapping->host)->truncate_lock);
|
|
|
|
return blocknr;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* exfat_block_truncate_page() zeroes out a mapping from file offset `from'
|
|
|
|
* up to the end of the block which corresponds to `from'.
|
|
|
|
* This is required during truncate to physically zeroout the tail end
|
|
|
|
* of that block so it doesn't yield old data if the file is later grown.
|
|
|
|
* Also, avoid causing failure from fsx for cases of "data past EOF"
|
|
|
|
*/
|
|
|
|
int exfat_block_truncate_page(struct inode *inode, loff_t from)
|
|
|
|
{
|
|
|
|
return block_truncate_page(inode->i_mapping, from, exfat_get_block);
|
|
|
|
}
|
|
|
|
|
|
|
|
static const struct address_space_operations exfat_aops = {
|
2022-02-09 20:22:12 +00:00
|
|
|
.dirty_folio = block_dirty_folio,
|
2022-02-09 20:21:34 +00:00
|
|
|
.invalidate_folio = block_invalidate_folio,
|
2022-04-29 11:47:39 -04:00
|
|
|
.read_folio = exfat_read_folio,
|
fs: convert mpage_readpages to mpage_readahead
Implement the new readahead aop and convert all callers (block_dev,
exfat, ext2, fat, gfs2, hpfs, isofs, jfs, nilfs2, ocfs2, omfs, qnx6,
reiserfs & udf).
The callers are all trivial except for GFS2 & OCFS2.
Signed-off-by: Matthew Wilcox (Oracle) <willy@infradead.org>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Reviewed-by: Junxiao Bi <junxiao.bi@oracle.com> # ocfs2
Reviewed-by: Joseph Qi <joseph.qi@linux.alibaba.com> # ocfs2
Reviewed-by: Dave Chinner <dchinner@redhat.com>
Reviewed-by: John Hubbard <jhubbard@nvidia.com>
Reviewed-by: Christoph Hellwig <hch@lst.de>
Reviewed-by: William Kucharski <william.kucharski@oracle.com>
Cc: Chao Yu <yuchao0@huawei.com>
Cc: Cong Wang <xiyou.wangcong@gmail.com>
Cc: Darrick J. Wong <darrick.wong@oracle.com>
Cc: Eric Biggers <ebiggers@google.com>
Cc: Gao Xiang <gaoxiang25@huawei.com>
Cc: Jaegeuk Kim <jaegeuk@kernel.org>
Cc: Michal Hocko <mhocko@suse.com>
Cc: Zi Yan <ziy@nvidia.com>
Cc: Johannes Thumshirn <johannes.thumshirn@wdc.com>
Cc: Miklos Szeredi <mszeredi@redhat.com>
Link: http://lkml.kernel.org/r/20200414150233.24495-17-willy@infradead.org
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
2020-06-01 21:47:02 -07:00
|
|
|
.readahead = exfat_readahead,
|
2020-03-02 15:21:34 +09:00
|
|
|
.writepages = exfat_writepages,
|
|
|
|
.write_begin = exfat_write_begin,
|
|
|
|
.write_end = exfat_write_end,
|
|
|
|
.direct_IO = exfat_direct_IO,
|
2022-12-02 11:26:38 +01:00
|
|
|
.bmap = exfat_aop_bmap,
|
|
|
|
.migrate_folio = buffer_migrate_folio,
|
2020-03-02 15:21:34 +09:00
|
|
|
};
|
|
|
|
|
|
|
|
static inline unsigned long exfat_hash(loff_t i_pos)
|
|
|
|
{
|
|
|
|
return hash_32(i_pos, EXFAT_HASH_BITS);
|
|
|
|
}
|
|
|
|
|
|
|
|
void exfat_hash_inode(struct inode *inode, loff_t i_pos)
|
|
|
|
{
|
|
|
|
struct exfat_sb_info *sbi = EXFAT_SB(inode->i_sb);
|
|
|
|
struct hlist_head *head = sbi->inode_hashtable + exfat_hash(i_pos);
|
|
|
|
|
|
|
|
spin_lock(&sbi->inode_hash_lock);
|
|
|
|
EXFAT_I(inode)->i_pos = i_pos;
|
|
|
|
hlist_add_head(&EXFAT_I(inode)->i_hash_fat, head);
|
|
|
|
spin_unlock(&sbi->inode_hash_lock);
|
|
|
|
}
|
|
|
|
|
|
|
|
void exfat_unhash_inode(struct inode *inode)
|
|
|
|
{
|
|
|
|
struct exfat_sb_info *sbi = EXFAT_SB(inode->i_sb);
|
|
|
|
|
|
|
|
spin_lock(&sbi->inode_hash_lock);
|
|
|
|
hlist_del_init(&EXFAT_I(inode)->i_hash_fat);
|
|
|
|
EXFAT_I(inode)->i_pos = 0;
|
|
|
|
spin_unlock(&sbi->inode_hash_lock);
|
|
|
|
}
|
|
|
|
|
|
|
|
struct inode *exfat_iget(struct super_block *sb, loff_t i_pos)
|
|
|
|
{
|
|
|
|
struct exfat_sb_info *sbi = EXFAT_SB(sb);
|
|
|
|
struct exfat_inode_info *info;
|
|
|
|
struct hlist_head *head = sbi->inode_hashtable + exfat_hash(i_pos);
|
|
|
|
struct inode *inode = NULL;
|
|
|
|
|
|
|
|
spin_lock(&sbi->inode_hash_lock);
|
|
|
|
hlist_for_each_entry(info, head, i_hash_fat) {
|
|
|
|
WARN_ON(info->vfs_inode.i_sb != sb);
|
|
|
|
|
|
|
|
if (i_pos != info->i_pos)
|
|
|
|
continue;
|
|
|
|
inode = igrab(&info->vfs_inode);
|
|
|
|
if (inode)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
spin_unlock(&sbi->inode_hash_lock);
|
|
|
|
return inode;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* doesn't deal with root inode */
|
|
|
|
static int exfat_fill_inode(struct inode *inode, struct exfat_dir_entry *info)
|
|
|
|
{
|
|
|
|
struct exfat_sb_info *sbi = EXFAT_SB(inode->i_sb);
|
|
|
|
struct exfat_inode_info *ei = EXFAT_I(inode);
|
|
|
|
loff_t size = info->size;
|
|
|
|
|
2020-09-11 13:45:19 +09:00
|
|
|
ei->dir = info->dir;
|
2020-03-02 15:21:34 +09:00
|
|
|
ei->entry = info->entry;
|
|
|
|
ei->attr = info->attr;
|
|
|
|
ei->start_clu = info->start_clu;
|
|
|
|
ei->flags = info->flags;
|
|
|
|
ei->type = info->type;
|
2023-03-13 12:38:53 +08:00
|
|
|
ei->valid_size = info->valid_size;
|
2020-03-02 15:21:34 +09:00
|
|
|
|
|
|
|
ei->version = 0;
|
|
|
|
ei->hint_stat.eidx = 0;
|
|
|
|
ei->hint_stat.clu = info->start_clu;
|
|
|
|
ei->hint_femp.eidx = EXFAT_HINT_NONE;
|
|
|
|
ei->hint_bmap.off = EXFAT_EOF_CLUSTER;
|
|
|
|
ei->i_pos = 0;
|
|
|
|
|
|
|
|
inode->i_uid = sbi->options.fs_uid;
|
|
|
|
inode->i_gid = sbi->options.fs_gid;
|
|
|
|
inode_inc_iversion(inode);
|
2022-10-05 17:43:22 +02:00
|
|
|
inode->i_generation = get_random_u32();
|
2020-03-02 15:21:34 +09:00
|
|
|
|
2023-10-30 20:53:18 +09:00
|
|
|
if (info->attr & EXFAT_ATTR_SUBDIR) { /* directory */
|
2020-03-02 15:21:34 +09:00
|
|
|
inode->i_generation &= ~1;
|
|
|
|
inode->i_mode = exfat_make_mode(sbi, info->attr, 0777);
|
|
|
|
inode->i_op = &exfat_dir_inode_operations;
|
|
|
|
inode->i_fop = &exfat_dir_operations;
|
|
|
|
set_nlink(inode, info->num_subdirs);
|
|
|
|
} else { /* regular file */
|
|
|
|
inode->i_generation |= 1;
|
|
|
|
inode->i_mode = exfat_make_mode(sbi, info->attr, 0777);
|
|
|
|
inode->i_op = &exfat_file_inode_operations;
|
|
|
|
inode->i_fop = &exfat_file_operations;
|
|
|
|
inode->i_mapping->a_ops = &exfat_aops;
|
|
|
|
inode->i_mapping->nrpages = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
i_size_write(inode, size);
|
|
|
|
|
|
|
|
exfat_save_attr(inode, info->attr);
|
|
|
|
|
2023-01-04 14:37:47 +08:00
|
|
|
inode->i_blocks = round_up(i_size_read(inode), sbi->cluster_size) >> 9;
|
2023-10-04 14:52:18 -04:00
|
|
|
inode_set_mtime_to_ts(inode, info->mtime);
|
2023-07-05 15:01:05 -04:00
|
|
|
inode_set_ctime_to_ts(inode, info->mtime);
|
2020-03-02 15:21:34 +09:00
|
|
|
ei->i_crtime = info->crtime;
|
2023-10-04 14:52:18 -04:00
|
|
|
inode_set_atime_to_ts(inode, info->atime);
|
2020-03-02 15:21:34 +09:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
struct inode *exfat_build_inode(struct super_block *sb,
|
|
|
|
struct exfat_dir_entry *info, loff_t i_pos)
|
|
|
|
{
|
|
|
|
struct inode *inode;
|
|
|
|
int err;
|
|
|
|
|
|
|
|
inode = exfat_iget(sb, i_pos);
|
|
|
|
if (inode)
|
|
|
|
goto out;
|
|
|
|
inode = new_inode(sb);
|
|
|
|
if (!inode) {
|
|
|
|
inode = ERR_PTR(-ENOMEM);
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
inode->i_ino = iunique(sb, EXFAT_ROOT_INO);
|
|
|
|
inode_set_iversion(inode, 1);
|
|
|
|
err = exfat_fill_inode(inode, info);
|
|
|
|
if (err) {
|
|
|
|
iput(inode);
|
|
|
|
inode = ERR_PTR(err);
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
exfat_hash_inode(inode, i_pos);
|
|
|
|
insert_inode_hash(inode);
|
|
|
|
out:
|
|
|
|
return inode;
|
|
|
|
}
|
|
|
|
|
|
|
|
void exfat_evict_inode(struct inode *inode)
|
|
|
|
{
|
|
|
|
truncate_inode_pages(&inode->i_data, 0);
|
|
|
|
|
|
|
|
if (!inode->i_nlink) {
|
|
|
|
i_size_write(inode, 0);
|
|
|
|
mutex_lock(&EXFAT_SB(inode->i_sb)->s_lock);
|
2022-03-28 16:37:58 +08:00
|
|
|
__exfat_truncate(inode);
|
2020-03-02 15:21:34 +09:00
|
|
|
mutex_unlock(&EXFAT_SB(inode->i_sb)->s_lock);
|
|
|
|
}
|
|
|
|
|
|
|
|
invalidate_inode_buffers(inode);
|
|
|
|
clear_inode(inode);
|
|
|
|
exfat_cache_inval_inode(inode);
|
|
|
|
exfat_unhash_inode(inode);
|
|
|
|
}
|