INIT_WORK(&bbio->end_io_work, btrfs_end_bio_work);
                queue_work(btrfs_end_io_wq(fs_info, bio), &bbio->end_io_work);
        } else {
+               if (bio_op(bio) == REQ_OP_ZONE_APPEND)
+                       btrfs_record_physical_zoned(bbio);
                bbio->end_io(bbio);
        }
 }
        }
 
        if (btrfs_op(bio) == BTRFS_MAP_WRITE) {
+               if (bio_op(bio) == REQ_OP_ZONE_APPEND) {
+                       ret = btrfs_extract_ordered_extent(btrfs_bio(bio));
+                       if (ret)
+                               goto fail;
+               }
+
                /*
                 * Csum items for reloc roots have already been cloned at this
                 * point, so they are handled as part of the no-checksum case.
 
                        int mirror_num, enum btrfs_compression_type compress_type);
 int btrfs_check_sector_csum(struct btrfs_fs_info *fs_info, struct page *page,
                            u32 pgoff, u8 *csum, const u8 * const csum_expected);
+blk_status_t btrfs_extract_ordered_extent(struct btrfs_bio *bbio);
 bool btrfs_data_csum_ok(struct btrfs_bio *bbio, struct btrfs_device *dev,
                        u32 bio_offset, struct bio_vec *bv);
 noinline int can_nocow_extent(struct inode *inode, u64 offset, u64 *len,
 
        if (refcount_dec_and_test(&cb->pending_ios)) {
                struct btrfs_fs_info *fs_info = btrfs_sb(cb->inode->i_sb);
 
-               btrfs_record_physical_zoned(cb->inode, cb->start, &bbio->bio);
                queue_work(fs_info->compressed_write_workers, &cb->write_end_work);
        }
        bio_put(&bbio->bio);
 
        u64 start;
        u64 end;
        struct bvec_iter_all iter_all;
-       bool first_bvec = true;
 
        ASSERT(!bio_flagged(bio, BIO_CLONED));
        bio_for_each_segment_all(bvec, bio, iter_all) {
                start = page_offset(page) + bvec->bv_offset;
                end = start + bvec->bv_len - 1;
 
-               if (first_bvec) {
-                       btrfs_record_physical_zoned(inode, start, bio);
-                       first_bvec = false;
-               }
-
                end_extent_writepage(page, error, start, end);
 
                btrfs_page_clear_writeback(fs_info, page, start, bvec->bv_len);
 
        return ret;
 }
 
-static blk_status_t extract_ordered_extent(struct btrfs_inode *inode,
-                                          struct bio *bio, loff_t file_offset)
+blk_status_t btrfs_extract_ordered_extent(struct btrfs_bio *bbio)
 {
+       u64 start = (u64)bbio->bio.bi_iter.bi_sector << SECTOR_SHIFT;
+       u64 len = bbio->bio.bi_iter.bi_size;
+       struct btrfs_inode *inode = bbio->inode;
        struct btrfs_ordered_extent *ordered;
-       u64 start = (u64)bio->bi_iter.bi_sector << SECTOR_SHIFT;
        u64 file_len;
-       u64 len = bio->bi_iter.bi_size;
        u64 end = start + len;
        u64 ordered_end;
        u64 pre, post;
        int ret = 0;
 
-       ordered = btrfs_lookup_ordered_extent(inode, file_offset);
+       ordered = btrfs_lookup_ordered_extent(inode, bbio->file_offset);
        if (WARN_ON_ONCE(!ordered))
                return BLK_STS_IOERR;
 
        ret = btrfs_split_ordered_extent(ordered, pre, post);
        if (ret)
                goto out;
-       ret = split_zoned_em(inode, file_offset, file_len, pre, post);
+       ret = split_zoned_em(inode, bbio->file_offset, file_len, pre, post);
 
 out:
        btrfs_put_ordered_extent(ordered);
 
 void btrfs_submit_data_write_bio(struct btrfs_inode *inode, struct bio *bio, int mirror_num)
 {
-       struct btrfs_fs_info *fs_info = inode->root->fs_info;
-       blk_status_t ret;
-
-       if (bio_op(bio) == REQ_OP_ZONE_APPEND) {
-               ret = extract_ordered_extent(inode, bio,
-                               page_offset(bio_first_bvec_all(bio)->bv_page));
-               if (ret) {
-                       btrfs_bio_end_io(btrfs_bio(bio), ret);
-                       return;
-               }
-       }
-
-       btrfs_submit_bio(fs_info, bio, mirror_num);
+       btrfs_submit_bio(inode->root->fs_info, bio, mirror_num);
 }
 
 void btrfs_submit_data_read_bio(struct btrfs_inode *inode, struct bio *bio,
                dip->bio.bi_status = err;
        }
 
-       btrfs_record_physical_zoned(&dip->inode->vfs_inode, bbio->file_offset, bio);
-
        bio_put(bio);
        btrfs_dio_private_put(dip);
 }
                                              dip);
                btrfs_bio(bio)->file_offset = file_offset;
 
-               if (bio_op(bio) == REQ_OP_ZONE_APPEND) {
-                       status = extract_ordered_extent(BTRFS_I(inode), bio,
-                                                       file_offset);
-                       if (status) {
-                               bio_put(bio);
-                               goto out_err;
-                       }
-               }
-
                ASSERT(submit_len >= clone_len);
                submit_len -= clone_len;
 
 
 #include "space-info.h"
 #include "fs.h"
 #include "accessors.h"
+#include "bio.h"
 
 /* Maximum number of zones to report per blkdev_report_zones() call */
 #define BTRFS_REPORT_NR_ZONES   4096
        return ret;
 }
 
-void btrfs_record_physical_zoned(struct inode *inode, u64 file_offset,
-                                struct bio *bio)
+void btrfs_record_physical_zoned(struct btrfs_bio *bbio)
 {
+       const u64 physical = bbio->bio.bi_iter.bi_sector << SECTOR_SHIFT;
        struct btrfs_ordered_extent *ordered;
-       const u64 physical = bio->bi_iter.bi_sector << SECTOR_SHIFT;
 
-       if (bio_op(bio) != REQ_OP_ZONE_APPEND)
-               return;
-
-       ordered = btrfs_lookup_ordered_extent(BTRFS_I(inode), file_offset);
+       ordered = btrfs_lookup_ordered_extent(bbio->inode, bbio->file_offset);
        if (WARN_ON(!ordered))
                return;
 
        ordered->physical = physical;
-       ordered->bdev = bio->bi_bdev;
+       ordered->bdev = bbio->bio.bi_bdev;
 
        btrfs_put_ordered_extent(ordered);
 }
 
                            struct extent_buffer *eb);
 void btrfs_free_redirty_list(struct btrfs_transaction *trans);
 bool btrfs_use_zone_append(struct btrfs_inode *inode, u64 start);
-void btrfs_record_physical_zoned(struct inode *inode, u64 file_offset,
-                                struct bio *bio);
+void btrfs_record_physical_zoned(struct btrfs_bio *bbio);
 void btrfs_rewrite_logical_zoned(struct btrfs_ordered_extent *ordered);
 bool btrfs_check_meta_write_pointer(struct btrfs_fs_info *fs_info,
                                    struct extent_buffer *eb,
        return false;
 }
 
-static inline void btrfs_record_physical_zoned(struct inode *inode,
-                                              u64 file_offset, struct bio *bio)
+static inline void btrfs_record_physical_zoned(struct btrfs_bio *bbio)
 {
 }