• R
    ext4: Start with shared i_rwsem in case of DIO instead of exclusive · 535b1683
    Ritesh Harjani 提交于
    commit aa9714d0e39788d0688474c9d5f6a9a36159599f upstream.
    
    Earlier there was no shared lock in DIO read path. But this patch
    (16c54688: ext4: Allow parallel DIO reads)
    simplified some of the locking mechanism while still allowing for parallel DIO
    reads by adding shared lock in inode DIO read path.
    
    But this created problem with mixed read/write workload. It is due to the fact
    that in DIO path, we first start with exclusive lock and only when we determine
    that it is a ovewrite IO, we downgrade the lock. This causes the problem, since
    we still have shared locking in DIO reads.
    
    So, this patch tries to fix this issue by starting with shared lock and then
    switching to exclusive lock only when required based on ext4_dio_write_checks().
    
    Other than that, it also simplifies below cases:-
    
    1. Simplified ext4_unaligned_aio API to ext4_unaligned_io. Previous API was
    abused in the sense that it was not really checking for AIO anywhere also it
    used to check for extending writes. So this API was renamed and simplified to
    ext4_unaligned_io() which actully only checks if the IO is really unaligned.
    
    Now, in case of unaligned direct IO, iomap_dio_rw needs to do zeroing of partial
    block and that will require serialization against other direct IOs in the same
    block. So we take a exclusive inode lock for any unaligned DIO. In case of AIO
    we also need to wait for any outstanding IOs to complete so that conversion from
    unwritten to written is completed before anyone try to map the overlapping block.
    Hence we take exclusive inode lock and also wait for inode_dio_wait() for
    unaligned DIO case. Please note since we are anyway taking an exclusive lock in
    unaligned IO, inode_dio_wait() becomes a no-op in case of non-AIO DIO.
    
    2. Added ext4_extending_io(). This checks if the IO is extending the file.
    
    3. Added ext4_dio_write_checks(). In this we start with shared inode lock and
    only switch to exclusive lock if required. So in most cases with aligned,
    non-extending, dioread_nolock & overwrites, it tries to write with a shared
    lock. If not, then we restart the operation in ext4_dio_write_checks(), after
    acquiring exclusive lock.
    Reviewed-by: NJan Kara <jack@suse.cz>
    Tested-by: NJoseph Qi <joseph.qi@linux.alibaba.com>
    Signed-off-by: NRitesh Harjani <riteshh@linux.ibm.com>
    Link: https://lore.kernel.org/r/20191212055557.11151-3-riteshh@linux.ibm.comSigned-off-by: NTheodore Ts'o <tytso@mit.edu>
    Signed-off-by: NJoseph Qi <joseph.qi@linux.alibaba.com>
    Reviewed-by: NXiaoguang Wang <xiaoguang.wang@linux.alibaba.com>
    535b1683
file.c 22.5 KB