* Each bit represents a sector. Thus caller should ensure @csum_buf passed
  * in is large enough to contain all csums.
  */
-int btrfs_lookup_csums_bitmap(struct btrfs_root *root, u64 start, u64 end,
-                             u8 *csum_buf, unsigned long *csum_bitmap,
-                             bool search_commit)
+int btrfs_lookup_csums_bitmap(struct btrfs_root *root, struct btrfs_path *path,
+                             u64 start, u64 end, u8 *csum_buf,
+                             unsigned long *csum_bitmap)
 {
        struct btrfs_fs_info *fs_info = root->fs_info;
        struct btrfs_key key;
-       struct btrfs_path *path;
        struct extent_buffer *leaf;
        struct btrfs_csum_item *item;
        const u64 orig_start = start;
+       bool free_path = false;
        int ret;
 
        ASSERT(IS_ALIGNED(start, fs_info->sectorsize) &&
               IS_ALIGNED(end + 1, fs_info->sectorsize));
 
-       path = btrfs_alloc_path();
-       if (!path)
-               return -ENOMEM;
+       if (!path) {
+               path = btrfs_alloc_path();
+               if (!path)
+                       return -ENOMEM;
+               free_path = true;
+       }
 
-       if (search_commit) {
-               path->skip_locking = 1;
-               path->reada = READA_FORWARD;
-               path->search_commit_root = 1;
+       /* Check if we can reuse the previous path. */
+       if (path->nodes[0]) {
+               btrfs_item_key_to_cpu(path->nodes[0], &key, path->slots[0]);
+
+               if (key.objectid == BTRFS_EXTENT_CSUM_OBJECTID &&
+                   key.type == BTRFS_EXTENT_CSUM_KEY &&
+                   key.offset <= start)
+                       goto search_forward;
+               btrfs_release_path(path);
        }
 
        key.objectid = BTRFS_EXTENT_CSUM_OBJECTID;
                }
        }
 
+search_forward:
        while (start <= end) {
                u64 csum_end;
 
        }
        ret = 0;
 fail:
-       btrfs_free_path(path);
+       if (free_path)
+               btrfs_free_path(path);
        return ret;
 }
 
 
 int btrfs_lookup_csums_list(struct btrfs_root *root, u64 start, u64 end,
                            struct list_head *list, int search_commit,
                            bool nowait);
-int btrfs_lookup_csums_bitmap(struct btrfs_root *root, u64 start, u64 end,
-                             u8 *csum_buf, unsigned long *csum_bitmap,
-                             bool search_commit);
+int btrfs_lookup_csums_bitmap(struct btrfs_root *root, struct btrfs_path *path,
+                             u64 start, u64 end, u8 *csum_buf,
+                             unsigned long *csum_bitmap);
 void btrfs_extent_item_to_extent_map(struct btrfs_inode *inode,
                                     const struct btrfs_path *path,
                                     struct btrfs_file_extent_item *fi,
 
        struct scrub_stripe     *raid56_data_stripes;
        struct btrfs_fs_info    *fs_info;
        struct btrfs_path       extent_path;
+       struct btrfs_path       csum_path;
        int                     first_free;
        int                     cur_stripe;
        atomic_t                cancel_req;
        sctx->fs_info = fs_info;
        sctx->extent_path.search_commit_root = 1;
        sctx->extent_path.skip_locking = 1;
+       sctx->csum_path.search_commit_root = 1;
+       sctx->csum_path.skip_locking = 1;
        for (i = 0; i < SCRUB_STRIPES_PER_SCTX; i++) {
                int ret;
 
  */
 static int scrub_find_fill_first_stripe(struct btrfs_block_group *bg,
                                        struct btrfs_path *extent_path,
+                                       struct btrfs_path *csum_path,
                                        struct btrfs_device *dev, u64 physical,
                                        int mirror_num, u64 logical_start,
                                        u32 logical_len,
                 */
                ASSERT(BITS_PER_LONG >= BTRFS_STRIPE_LEN >> fs_info->sectorsize_bits);
 
-               ret = btrfs_lookup_csums_bitmap(csum_root, stripe->logical,
-                                               stripe_end, stripe->csums,
-                                               &csum_bitmap, true);
+               ret = btrfs_lookup_csums_bitmap(csum_root, csum_path,
+                                               stripe->logical, stripe_end,
+                                               stripe->csums, &csum_bitmap);
                if (ret < 0)
                        goto out;
                if (ret > 0)
 
        /* We can queue one stripe using the remaining slot. */
        scrub_reset_stripe(stripe);
-       ret = scrub_find_fill_first_stripe(bg, &sctx->extent_path, dev,
-                                          physical, mirror_num, logical,
-                                          length, stripe);
+       ret = scrub_find_fill_first_stripe(bg, &sctx->extent_path,
+                                          &sctx->csum_path, dev, physical,
+                                          mirror_num, logical, length, stripe);
        /* Either >0 as no more extents or <0 for error. */
        if (ret)
                return ret;
        struct btrfs_raid_bio *rbio;
        struct btrfs_io_context *bioc = NULL;
        struct btrfs_path extent_path = { 0 };
+       struct btrfs_path csum_path = { 0 };
        struct bio *bio;
        struct scrub_stripe *stripe;
        bool all_empty = true;
        ASSERT(sctx->raid56_data_stripes);
 
        /*
-        * For data stripe search, we cannot re-use the same extent path, as
-        * the data stripe bytenr may be smaller than previous extent.  Thus we
-        * have to use our own extent path.
+        * For data stripe search, we cannot re-use the same extent/csum paths,
+        * as the data stripe bytenr may be smaller than previous extent.  Thus
+        * we have to use our own extent/csum paths.
         */
        extent_path.search_commit_root = 1;
        extent_path.skip_locking = 1;
+       csum_path.search_commit_root = 1;
+       csum_path.skip_locking = 1;
 
        for (int i = 0; i < data_stripes; i++) {
                int stripe_index;
 
                scrub_reset_stripe(stripe);
                set_bit(SCRUB_STRIPE_FLAG_NO_REPORT, &stripe->state);
-               ret = scrub_find_fill_first_stripe(bg, &extent_path,
+               ret = scrub_find_fill_first_stripe(bg, &extent_path, &csum_path,
                                map->stripes[stripe_index].dev, physical, 1,
                                full_stripe_start + btrfs_stripe_nr_to_offset(i),
                                BTRFS_STRIPE_LEN, stripe);
        btrfs_bio_counter_dec(fs_info);
 
        btrfs_release_path(&extent_path);
+       btrfs_release_path(&csum_path);
 out:
        return ret;
 }
        if (!ret)
                ret = ret2;
        btrfs_release_path(&sctx->extent_path);
+       btrfs_release_path(&sctx->csum_path);
 
        if (sctx->raid56_data_stripes) {
                for (int i = 0; i < nr_data_stripes(map); i++)