Commit fae02687 authored by Linus Torvalds's avatar Linus Torvalds

Merge tag 'vfs-6.9-rc3.fixes' of git://git.kernel.org/pub/scm/linux/kernel/git/vfs/vfs

Pull vfs fixes from Christian Brauner:
 "This contains a few small fixes. This comes with some delay because I
  wanted to wait on people running their reproducers and the Easter
  Holidays meant that those replies came in a little later than usual:

   - Fix handling of preventing writes to mounted block devices.

     Since last kernel we allow to prevent writing to mounted block
     devices provided CONFIG_BLK_DEV_WRITE_MOUNTED isn't set and the
     block device is opened with restricted writes. When we switched to
     opening block devices as files we altered the mechanism by which we
     recognize when a block device has been opened with write
     restrictions.

     The detection logic assumed that only read-write mounted
     filesystems would apply write restrictions to their block devices
     from other openers. That of course is not true since it also makes
     sense to apply write restrictions for filesystems that are
     read-only.

     Fix the detection logic using an FMODE_* bit. We still have a few
     left since we freed up a couple a while ago. I also picked up a
     patch to free up four additional FMODE_* bits scheduled for the
     next merge window.

   - Fix counting the number of writers to a block device. This just
     changes the logic to be consistent.

   - Fix a bug in aio causing a NULL pointer derefernce after we
     implemented batched processing in aio.

   - Finally, add the changes we discussed that allows to yield block
     devices early even though file closing itself is deferred.

     This also allows us to remove two holder operations to get and
     release the holder to align lifetime of file and holder of the
     block device"

* tag 'vfs-6.9-rc3.fixes' of git://git.kernel.org/pub/scm/linux/kernel/git/vfs/vfs:
  aio: Fix null ptr deref in aio_complete() wakeup
  fs,block: yield devices early
  block: count BLK_OPEN_RESTRICT_WRITES openers
  block: handle BLK_OPEN_RESTRICT_WRITES correctly
parents 8cb4a9a8 caeb4b0a
...@@ -583,9 +583,6 @@ static void bd_finish_claiming(struct block_device *bdev, void *holder, ...@@ -583,9 +583,6 @@ static void bd_finish_claiming(struct block_device *bdev, void *holder,
mutex_unlock(&bdev->bd_holder_lock); mutex_unlock(&bdev->bd_holder_lock);
bd_clear_claiming(whole, holder); bd_clear_claiming(whole, holder);
mutex_unlock(&bdev_lock); mutex_unlock(&bdev_lock);
if (hops && hops->get_holder)
hops->get_holder(holder);
} }
/** /**
...@@ -608,7 +605,6 @@ EXPORT_SYMBOL(bd_abort_claiming); ...@@ -608,7 +605,6 @@ EXPORT_SYMBOL(bd_abort_claiming);
static void bd_end_claim(struct block_device *bdev, void *holder) static void bd_end_claim(struct block_device *bdev, void *holder)
{ {
struct block_device *whole = bdev_whole(bdev); struct block_device *whole = bdev_whole(bdev);
const struct blk_holder_ops *hops = bdev->bd_holder_ops;
bool unblock = false; bool unblock = false;
/* /*
...@@ -631,9 +627,6 @@ static void bd_end_claim(struct block_device *bdev, void *holder) ...@@ -631,9 +627,6 @@ static void bd_end_claim(struct block_device *bdev, void *holder)
whole->bd_holder = NULL; whole->bd_holder = NULL;
mutex_unlock(&bdev_lock); mutex_unlock(&bdev_lock);
if (hops && hops->put_holder)
hops->put_holder(holder);
/* /*
* If this was the last claim, remove holder link and unblock evpoll if * If this was the last claim, remove holder link and unblock evpoll if
* it was a write holder. * it was a write holder.
...@@ -776,17 +769,17 @@ void blkdev_put_no_open(struct block_device *bdev) ...@@ -776,17 +769,17 @@ void blkdev_put_no_open(struct block_device *bdev)
static bool bdev_writes_blocked(struct block_device *bdev) static bool bdev_writes_blocked(struct block_device *bdev)
{ {
return bdev->bd_writers == -1; return bdev->bd_writers < 0;
} }
static void bdev_block_writes(struct block_device *bdev) static void bdev_block_writes(struct block_device *bdev)
{ {
bdev->bd_writers = -1; bdev->bd_writers--;
} }
static void bdev_unblock_writes(struct block_device *bdev) static void bdev_unblock_writes(struct block_device *bdev)
{ {
bdev->bd_writers = 0; bdev->bd_writers++;
} }
static bool bdev_may_open(struct block_device *bdev, blk_mode_t mode) static bool bdev_may_open(struct block_device *bdev, blk_mode_t mode)
...@@ -813,6 +806,11 @@ static void bdev_claim_write_access(struct block_device *bdev, blk_mode_t mode) ...@@ -813,6 +806,11 @@ static void bdev_claim_write_access(struct block_device *bdev, blk_mode_t mode)
bdev->bd_writers++; bdev->bd_writers++;
} }
static inline bool bdev_unclaimed(const struct file *bdev_file)
{
return bdev_file->private_data == BDEV_I(bdev_file->f_mapping->host);
}
static void bdev_yield_write_access(struct file *bdev_file) static void bdev_yield_write_access(struct file *bdev_file)
{ {
struct block_device *bdev; struct block_device *bdev;
...@@ -820,14 +818,15 @@ static void bdev_yield_write_access(struct file *bdev_file) ...@@ -820,14 +818,15 @@ static void bdev_yield_write_access(struct file *bdev_file)
if (bdev_allow_write_mounted) if (bdev_allow_write_mounted)
return; return;
if (bdev_unclaimed(bdev_file))
return;
bdev = file_bdev(bdev_file); bdev = file_bdev(bdev_file);
/* Yield exclusive or shared write access. */
if (bdev_file->f_mode & FMODE_WRITE) { if (bdev_file->f_mode & FMODE_WRITE_RESTRICTED)
if (bdev_writes_blocked(bdev)) bdev_unblock_writes(bdev);
bdev_unblock_writes(bdev); else if (bdev_file->f_mode & FMODE_WRITE)
else bdev->bd_writers--;
bdev->bd_writers--;
}
} }
/** /**
...@@ -907,6 +906,8 @@ int bdev_open(struct block_device *bdev, blk_mode_t mode, void *holder, ...@@ -907,6 +906,8 @@ int bdev_open(struct block_device *bdev, blk_mode_t mode, void *holder,
bdev_file->f_mode |= FMODE_BUF_RASYNC | FMODE_CAN_ODIRECT; bdev_file->f_mode |= FMODE_BUF_RASYNC | FMODE_CAN_ODIRECT;
if (bdev_nowait(bdev)) if (bdev_nowait(bdev))
bdev_file->f_mode |= FMODE_NOWAIT; bdev_file->f_mode |= FMODE_NOWAIT;
if (mode & BLK_OPEN_RESTRICT_WRITES)
bdev_file->f_mode |= FMODE_WRITE_RESTRICTED;
bdev_file->f_mapping = bdev->bd_inode->i_mapping; bdev_file->f_mapping = bdev->bd_inode->i_mapping;
bdev_file->f_wb_err = filemap_sample_wb_err(bdev_file->f_mapping); bdev_file->f_wb_err = filemap_sample_wb_err(bdev_file->f_mapping);
bdev_file->private_data = holder; bdev_file->private_data = holder;
...@@ -1012,6 +1013,20 @@ struct file *bdev_file_open_by_path(const char *path, blk_mode_t mode, ...@@ -1012,6 +1013,20 @@ struct file *bdev_file_open_by_path(const char *path, blk_mode_t mode,
} }
EXPORT_SYMBOL(bdev_file_open_by_path); EXPORT_SYMBOL(bdev_file_open_by_path);
static inline void bd_yield_claim(struct file *bdev_file)
{
struct block_device *bdev = file_bdev(bdev_file);
void *holder = bdev_file->private_data;
lockdep_assert_held(&bdev->bd_disk->open_mutex);
if (WARN_ON_ONCE(IS_ERR_OR_NULL(holder)))
return;
if (!bdev_unclaimed(bdev_file))
bd_end_claim(bdev, holder);
}
void bdev_release(struct file *bdev_file) void bdev_release(struct file *bdev_file)
{ {
struct block_device *bdev = file_bdev(bdev_file); struct block_device *bdev = file_bdev(bdev_file);
...@@ -1036,7 +1051,7 @@ void bdev_release(struct file *bdev_file) ...@@ -1036,7 +1051,7 @@ void bdev_release(struct file *bdev_file)
bdev_yield_write_access(bdev_file); bdev_yield_write_access(bdev_file);
if (holder) if (holder)
bd_end_claim(bdev, holder); bd_yield_claim(bdev_file);
/* /*
* Trigger event checking and tell drivers to flush MEDIA_CHANGE * Trigger event checking and tell drivers to flush MEDIA_CHANGE
...@@ -1056,6 +1071,39 @@ void bdev_release(struct file *bdev_file) ...@@ -1056,6 +1071,39 @@ void bdev_release(struct file *bdev_file)
blkdev_put_no_open(bdev); blkdev_put_no_open(bdev);
} }
/**
* bdev_fput - yield claim to the block device and put the file
* @bdev_file: open block device
*
* Yield claim on the block device and put the file. Ensure that the
* block device can be reclaimed before the file is closed which is a
* deferred operation.
*/
void bdev_fput(struct file *bdev_file)
{
if (WARN_ON_ONCE(bdev_file->f_op != &def_blk_fops))
return;
if (bdev_file->private_data) {
struct block_device *bdev = file_bdev(bdev_file);
struct gendisk *disk = bdev->bd_disk;
mutex_lock(&disk->open_mutex);
bdev_yield_write_access(bdev_file);
bd_yield_claim(bdev_file);
/*
* Tell release we already gave up our hold on the
* device and if write restrictions are available that
* we already gave up write access to the device.
*/
bdev_file->private_data = BDEV_I(bdev_file->f_mapping->host);
mutex_unlock(&disk->open_mutex);
}
fput(bdev_file);
}
EXPORT_SYMBOL(bdev_fput);
/** /**
* lookup_bdev() - Look up a struct block_device by name. * lookup_bdev() - Look up a struct block_device by name.
* @pathname: Name of the block device in the filesystem. * @pathname: Name of the block device in the filesystem.
......
...@@ -209,7 +209,7 @@ static void block2mtd_free_device(struct block2mtd_dev *dev) ...@@ -209,7 +209,7 @@ static void block2mtd_free_device(struct block2mtd_dev *dev)
if (dev->bdev_file) { if (dev->bdev_file) {
invalidate_mapping_pages(dev->bdev_file->f_mapping, 0, -1); invalidate_mapping_pages(dev->bdev_file->f_mapping, 0, -1);
fput(dev->bdev_file); bdev_fput(dev->bdev_file);
} }
kfree(dev); kfree(dev);
......
...@@ -1202,8 +1202,8 @@ static void aio_complete(struct aio_kiocb *iocb) ...@@ -1202,8 +1202,8 @@ static void aio_complete(struct aio_kiocb *iocb)
spin_lock_irqsave(&ctx->wait.lock, flags); spin_lock_irqsave(&ctx->wait.lock, flags);
list_for_each_entry_safe(curr, next, &ctx->wait.head, w.entry) list_for_each_entry_safe(curr, next, &ctx->wait.head, w.entry)
if (avail >= curr->min_nr) { if (avail >= curr->min_nr) {
list_del_init_careful(&curr->w.entry);
wake_up_process(curr->w.private); wake_up_process(curr->w.private);
list_del_init_careful(&curr->w.entry);
} }
spin_unlock_irqrestore(&ctx->wait.lock, flags); spin_unlock_irqrestore(&ctx->wait.lock, flags);
} }
......
...@@ -143,7 +143,7 @@ void bch2_free_super(struct bch_sb_handle *sb) ...@@ -143,7 +143,7 @@ void bch2_free_super(struct bch_sb_handle *sb)
{ {
kfree(sb->bio); kfree(sb->bio);
if (!IS_ERR_OR_NULL(sb->s_bdev_file)) if (!IS_ERR_OR_NULL(sb->s_bdev_file))
fput(sb->s_bdev_file); bdev_fput(sb->s_bdev_file);
kfree(sb->holder); kfree(sb->holder);
kfree(sb->sb_name); kfree(sb->sb_name);
......
...@@ -495,7 +495,7 @@ static void cramfs_kill_sb(struct super_block *sb) ...@@ -495,7 +495,7 @@ static void cramfs_kill_sb(struct super_block *sb)
sb->s_mtd = NULL; sb->s_mtd = NULL;
} else if (IS_ENABLED(CONFIG_CRAMFS_BLOCKDEV) && sb->s_bdev) { } else if (IS_ENABLED(CONFIG_CRAMFS_BLOCKDEV) && sb->s_bdev) {
sync_blockdev(sb->s_bdev); sync_blockdev(sb->s_bdev);
fput(sb->s_bdev_file); bdev_fput(sb->s_bdev_file);
} }
kfree(sbi); kfree(sbi);
} }
......
...@@ -5668,7 +5668,7 @@ failed_mount9: __maybe_unused ...@@ -5668,7 +5668,7 @@ failed_mount9: __maybe_unused
brelse(sbi->s_sbh); brelse(sbi->s_sbh);
if (sbi->s_journal_bdev_file) { if (sbi->s_journal_bdev_file) {
invalidate_bdev(file_bdev(sbi->s_journal_bdev_file)); invalidate_bdev(file_bdev(sbi->s_journal_bdev_file));
fput(sbi->s_journal_bdev_file); bdev_fput(sbi->s_journal_bdev_file);
} }
out_fail: out_fail:
invalidate_bdev(sb->s_bdev); invalidate_bdev(sb->s_bdev);
...@@ -5913,7 +5913,7 @@ static struct file *ext4_get_journal_blkdev(struct super_block *sb, ...@@ -5913,7 +5913,7 @@ static struct file *ext4_get_journal_blkdev(struct super_block *sb,
out_bh: out_bh:
brelse(bh); brelse(bh);
out_bdev: out_bdev:
fput(bdev_file); bdev_fput(bdev_file);
return ERR_PTR(errno); return ERR_PTR(errno);
} }
...@@ -5952,7 +5952,7 @@ static journal_t *ext4_open_dev_journal(struct super_block *sb, ...@@ -5952,7 +5952,7 @@ static journal_t *ext4_open_dev_journal(struct super_block *sb,
out_journal: out_journal:
jbd2_journal_destroy(journal); jbd2_journal_destroy(journal);
out_bdev: out_bdev:
fput(bdev_file); bdev_fput(bdev_file);
return ERR_PTR(errno); return ERR_PTR(errno);
} }
...@@ -7327,7 +7327,7 @@ static void ext4_kill_sb(struct super_block *sb) ...@@ -7327,7 +7327,7 @@ static void ext4_kill_sb(struct super_block *sb)
kill_block_super(sb); kill_block_super(sb);
if (bdev_file) if (bdev_file)
fput(bdev_file); bdev_fput(bdev_file);
} }
static struct file_system_type ext4_fs_type = { static struct file_system_type ext4_fs_type = {
......
...@@ -1558,7 +1558,7 @@ static void destroy_device_list(struct f2fs_sb_info *sbi) ...@@ -1558,7 +1558,7 @@ static void destroy_device_list(struct f2fs_sb_info *sbi)
for (i = 0; i < sbi->s_ndevs; i++) { for (i = 0; i < sbi->s_ndevs; i++) {
if (i > 0) if (i > 0)
fput(FDEV(i).bdev_file); bdev_fput(FDEV(i).bdev_file);
#ifdef CONFIG_BLK_DEV_ZONED #ifdef CONFIG_BLK_DEV_ZONED
kvfree(FDEV(i).blkz_seq); kvfree(FDEV(i).blkz_seq);
#endif #endif
......
...@@ -1141,7 +1141,7 @@ int lmLogOpen(struct super_block *sb) ...@@ -1141,7 +1141,7 @@ int lmLogOpen(struct super_block *sb)
lbmLogShutdown(log); lbmLogShutdown(log);
close: /* close external log device */ close: /* close external log device */
fput(bdev_file); bdev_fput(bdev_file);
free: /* free log descriptor */ free: /* free log descriptor */
mutex_unlock(&jfs_log_mutex); mutex_unlock(&jfs_log_mutex);
...@@ -1485,7 +1485,7 @@ int lmLogClose(struct super_block *sb) ...@@ -1485,7 +1485,7 @@ int lmLogClose(struct super_block *sb)
bdev_file = log->bdev_file; bdev_file = log->bdev_file;
rc = lmLogShutdown(log); rc = lmLogShutdown(log);
fput(bdev_file); bdev_fput(bdev_file);
kfree(log); kfree(log);
......
...@@ -2589,7 +2589,7 @@ static void journal_list_init(struct super_block *sb) ...@@ -2589,7 +2589,7 @@ static void journal_list_init(struct super_block *sb)
static void release_journal_dev(struct reiserfs_journal *journal) static void release_journal_dev(struct reiserfs_journal *journal)
{ {
if (journal->j_bdev_file) { if (journal->j_bdev_file) {
fput(journal->j_bdev_file); bdev_fput(journal->j_bdev_file);
journal->j_bdev_file = NULL; journal->j_bdev_file = NULL;
} }
} }
......
...@@ -594,7 +594,7 @@ static void romfs_kill_sb(struct super_block *sb) ...@@ -594,7 +594,7 @@ static void romfs_kill_sb(struct super_block *sb)
#ifdef CONFIG_ROMFS_ON_BLOCK #ifdef CONFIG_ROMFS_ON_BLOCK
if (sb->s_bdev) { if (sb->s_bdev) {
sync_blockdev(sb->s_bdev); sync_blockdev(sb->s_bdev);
fput(sb->s_bdev_file); bdev_fput(sb->s_bdev_file);
} }
#endif #endif
} }
......
...@@ -1515,29 +1515,11 @@ static int fs_bdev_thaw(struct block_device *bdev) ...@@ -1515,29 +1515,11 @@ static int fs_bdev_thaw(struct block_device *bdev)
return error; return error;
} }
static void fs_bdev_super_get(void *data)
{
struct super_block *sb = data;
spin_lock(&sb_lock);
sb->s_count++;
spin_unlock(&sb_lock);
}
static void fs_bdev_super_put(void *data)
{
struct super_block *sb = data;
put_super(sb);
}
const struct blk_holder_ops fs_holder_ops = { const struct blk_holder_ops fs_holder_ops = {
.mark_dead = fs_bdev_mark_dead, .mark_dead = fs_bdev_mark_dead,
.sync = fs_bdev_sync, .sync = fs_bdev_sync,
.freeze = fs_bdev_freeze, .freeze = fs_bdev_freeze,
.thaw = fs_bdev_thaw, .thaw = fs_bdev_thaw,
.get_holder = fs_bdev_super_get,
.put_holder = fs_bdev_super_put,
}; };
EXPORT_SYMBOL_GPL(fs_holder_ops); EXPORT_SYMBOL_GPL(fs_holder_ops);
...@@ -1562,7 +1544,7 @@ int setup_bdev_super(struct super_block *sb, int sb_flags, ...@@ -1562,7 +1544,7 @@ int setup_bdev_super(struct super_block *sb, int sb_flags,
* writable from userspace even for a read-only block device. * writable from userspace even for a read-only block device.
*/ */
if ((mode & BLK_OPEN_WRITE) && bdev_read_only(bdev)) { if ((mode & BLK_OPEN_WRITE) && bdev_read_only(bdev)) {
fput(bdev_file); bdev_fput(bdev_file);
return -EACCES; return -EACCES;
} }
...@@ -1573,7 +1555,7 @@ int setup_bdev_super(struct super_block *sb, int sb_flags, ...@@ -1573,7 +1555,7 @@ int setup_bdev_super(struct super_block *sb, int sb_flags,
if (atomic_read(&bdev->bd_fsfreeze_count) > 0) { if (atomic_read(&bdev->bd_fsfreeze_count) > 0) {
if (fc) if (fc)
warnf(fc, "%pg: Can't mount, blockdev is frozen", bdev); warnf(fc, "%pg: Can't mount, blockdev is frozen", bdev);
fput(bdev_file); bdev_fput(bdev_file);
return -EBUSY; return -EBUSY;
} }
spin_lock(&sb_lock); spin_lock(&sb_lock);
...@@ -1693,7 +1675,7 @@ void kill_block_super(struct super_block *sb) ...@@ -1693,7 +1675,7 @@ void kill_block_super(struct super_block *sb)
generic_shutdown_super(sb); generic_shutdown_super(sb);
if (bdev) { if (bdev) {
sync_blockdev(bdev); sync_blockdev(bdev);
fput(sb->s_bdev_file); bdev_fput(sb->s_bdev_file);
} }
} }
......
...@@ -2030,7 +2030,7 @@ xfs_free_buftarg( ...@@ -2030,7 +2030,7 @@ xfs_free_buftarg(
fs_put_dax(btp->bt_daxdev, btp->bt_mount); fs_put_dax(btp->bt_daxdev, btp->bt_mount);
/* the main block device is closed by kill_block_super */ /* the main block device is closed by kill_block_super */
if (btp->bt_bdev != btp->bt_mount->m_super->s_bdev) if (btp->bt_bdev != btp->bt_mount->m_super->s_bdev)
fput(btp->bt_bdev_file); bdev_fput(btp->bt_bdev_file);
kfree(btp); kfree(btp);
} }
......
...@@ -485,7 +485,7 @@ xfs_open_devices( ...@@ -485,7 +485,7 @@ xfs_open_devices(
mp->m_logdev_targp = mp->m_ddev_targp; mp->m_logdev_targp = mp->m_ddev_targp;
/* Handle won't be used, drop it */ /* Handle won't be used, drop it */
if (logdev_file) if (logdev_file)
fput(logdev_file); bdev_fput(logdev_file);
} }
return 0; return 0;
...@@ -497,10 +497,10 @@ xfs_open_devices( ...@@ -497,10 +497,10 @@ xfs_open_devices(
xfs_free_buftarg(mp->m_ddev_targp); xfs_free_buftarg(mp->m_ddev_targp);
out_close_rtdev: out_close_rtdev:
if (rtdev_file) if (rtdev_file)
fput(rtdev_file); bdev_fput(rtdev_file);
out_close_logdev: out_close_logdev:
if (logdev_file) if (logdev_file)
fput(logdev_file); bdev_fput(logdev_file);
return error; return error;
} }
......
...@@ -1505,16 +1505,6 @@ struct blk_holder_ops { ...@@ -1505,16 +1505,6 @@ struct blk_holder_ops {
* Thaw the file system mounted on the block device. * Thaw the file system mounted on the block device.
*/ */
int (*thaw)(struct block_device *bdev); int (*thaw)(struct block_device *bdev);
/*
* If needed, get a reference to the holder.
*/
void (*get_holder)(void *holder);
/*
* Release the holder.
*/
void (*put_holder)(void *holder);
}; };
/* /*
...@@ -1585,6 +1575,7 @@ static inline int early_lookup_bdev(const char *pathname, dev_t *dev) ...@@ -1585,6 +1575,7 @@ static inline int early_lookup_bdev(const char *pathname, dev_t *dev)
int bdev_freeze(struct block_device *bdev); int bdev_freeze(struct block_device *bdev);
int bdev_thaw(struct block_device *bdev); int bdev_thaw(struct block_device *bdev);
void bdev_fput(struct file *bdev_file);
struct io_comp_batch { struct io_comp_batch {
struct request *req_list; struct request *req_list;
......
...@@ -121,6 +121,8 @@ typedef int (dio_iodone_t)(struct kiocb *iocb, loff_t offset, ...@@ -121,6 +121,8 @@ typedef int (dio_iodone_t)(struct kiocb *iocb, loff_t offset,
#define FMODE_PWRITE ((__force fmode_t)0x10) #define FMODE_PWRITE ((__force fmode_t)0x10)
/* File is opened for execution with sys_execve / sys_uselib */ /* File is opened for execution with sys_execve / sys_uselib */
#define FMODE_EXEC ((__force fmode_t)0x20) #define FMODE_EXEC ((__force fmode_t)0x20)
/* File writes are restricted (block device specific) */
#define FMODE_WRITE_RESTRICTED ((__force fmode_t)0x40)
/* 32bit hashes as llseek() offset (for directories) */ /* 32bit hashes as llseek() offset (for directories) */
#define FMODE_32BITHASH ((__force fmode_t)0x200) #define FMODE_32BITHASH ((__force fmode_t)0x200)
/* 64bit hashes as llseek() offset (for directories) */ /* 64bit hashes as llseek() offset (for directories) */
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment