trace_ext4_ext_map_blocks_enter(inode, map->m_lblk, map->m_len, flags);
 
        /* find extent for this block */
-       path = ext4_find_extent(inode, map->m_lblk, NULL, 0);
+       path = ext4_find_extent(inode, map->m_lblk, NULL, flags);
        if (IS_ERR(path)) {
                err = PTR_ERR(path);
                goto out;
                goto out;
        ar.lright = map->m_lblk;
        err = ext4_ext_search_right(inode, path, &ar.lright, &ar.pright,
-                                   &ex2, 0);
+                                   &ex2, flags);
        if (err < 0)
                goto out;
 
                                break;
                        }
                }
+               /*
+                * Do not cache any unrelated extents, as it does not hold the
+                * i_rwsem or invalidate_lock, which could corrupt the extent
+                * status tree.
+                */
                ret = ext4_map_blocks(handle, inode, &map,
-                                     EXT4_GET_BLOCKS_IO_CONVERT_EXT);
+                                     EXT4_GET_BLOCKS_IO_CONVERT_EXT |
+                                     EXT4_EX_NOCACHE);
                if (ret <= 0)
                        ext4_warning(inode->i_sb,
                                     "inode #%lu: block %u: len %u: "
 
 #endif /* ES_AGGRESSIVE_TEST */
 
 static int ext4_map_query_blocks(handle_t *handle, struct inode *inode,
-                                struct ext4_map_blocks *map)
+                                struct ext4_map_blocks *map, int flags)
 {
        unsigned int status;
        int retval;
 
+       flags &= EXT4_EX_FILTER;
        if (ext4_test_inode_flag(inode, EXT4_INODE_EXTENTS))
-               retval = ext4_ext_map_blocks(handle, inode, map, 0);
+               retval = ext4_ext_map_blocks(handle, inode, map, flags);
        else
-               retval = ext4_ind_map_blocks(handle, inode, map, 0);
+               retval = ext4_ind_map_blocks(handle, inode, map, flags);
 
        if (retval <= 0)
                return retval;
        if (unlikely(map->m_lblk >= EXT_MAX_BLOCKS))
                return -EFSCORRUPTED;
 
+       /*
+        * Do not allow caching of unrelated ranges of extents during I/O
+        * submission.
+        */
+       if (flags & EXT4_GET_BLOCKS_IO_SUBMIT)
+               WARN_ON_ONCE(!(flags & EXT4_EX_NOCACHE));
+
        /* Lookup extent status tree firstly */
        if (!(EXT4_SB(inode->i_sb)->s_mount_state & EXT4_FC_REPLAY) &&
            ext4_es_lookup_extent(inode, map->m_lblk, NULL, &es)) {
         * file system block.
         */
        down_read(&EXT4_I(inode)->i_data_sem);
-       retval = ext4_map_query_blocks(handle, inode, map);
+       retval = ext4_map_query_blocks(handle, inode, map, flags);
        up_read((&EXT4_I(inode)->i_data_sem));
 
 found:
        if (ext4_has_inline_data(inode))
                retval = 0;
        else
-               retval = ext4_map_query_blocks(NULL, inode, map);
+               retval = ext4_map_query_blocks(NULL, inode, map, 0);
        up_read(&EXT4_I(inode)->i_data_sem);
        if (retval)
                return retval < 0 ? retval : 0;
                        goto found;
                }
        } else if (!ext4_has_inline_data(inode)) {
-               retval = ext4_map_query_blocks(NULL, inode, map);
+               retval = ext4_map_query_blocks(NULL, inode, map, 0);
                if (retval) {
                        up_write(&EXT4_I(inode)->i_data_sem);
                        return retval < 0 ? retval : 0;
         * previously reserved. However we must not fail because we're in
         * writeback and there is nothing we can do about it so it might result
         * in data loss.  So use reserved blocks to allocate metadata if
-        * possible.
+        * possible. In addition, do not cache any unrelated extents, as it
+        * only holds the folio lock but does not hold the i_rwsem or
+        * invalidate_lock, which could corrupt the extent status tree.
         */
        get_blocks_flags = EXT4_GET_BLOCKS_CREATE |
                           EXT4_GET_BLOCKS_METADATA_NOFAIL |
-                          EXT4_GET_BLOCKS_IO_SUBMIT;
+                          EXT4_GET_BLOCKS_IO_SUBMIT |
+                          EXT4_EX_NOCACHE;
+
        dioread_nolock = ext4_should_dioread_nolock(inode);
        if (dioread_nolock)
                get_blocks_flags |= EXT4_GET_BLOCKS_IO_CREATE_EXT;