struct scrub_bio {
        int                     index;
        struct scrub_ctx        *sctx;
+       struct btrfs_device     *dev;
        struct bio              *bio;
        int                     err;
        u64                     logical;
 
 struct scrub_ctx {
        struct scrub_bio        *bios[SCRUB_BIOS_PER_CTX];
-       struct btrfs_device     *dev;
+       struct btrfs_root       *dev_root;
        int                     first_free;
        int                     curr;
        atomic_t                in_flight;
 
 struct scrub_fixup_nodatasum {
        struct scrub_ctx        *sctx;
+       struct btrfs_device     *dev;
        u64                     logical;
        struct btrfs_root       *root;
        struct btrfs_work       work;
 static int scrub_add_page_to_bio(struct scrub_ctx *sctx,
                                 struct scrub_page *spage);
 static int scrub_pages(struct scrub_ctx *sctx, u64 logical, u64 len,
-                      u64 physical, u64 flags, u64 gen, int mirror_num,
-                      u8 *csum, int force);
+                      u64 physical, struct btrfs_device *dev, u64 flags,
+                      u64 gen, int mirror_num, u8 *csum, int force);
 static void scrub_bio_end_io(struct bio *bio, int err);
 static void scrub_bio_end_io_worker(struct btrfs_work *work);
 static void scrub_block_complete(struct scrub_block *sblock);
        sctx = kzalloc(sizeof(*sctx), GFP_NOFS);
        if (!sctx)
                goto nomem;
-       sctx->dev = dev;
        sctx->pages_per_bio = pages_per_bio;
        sctx->curr = -1;
+       sctx->dev_root = dev->dev_root;
        for (i = 0; i < SCRUB_BIOS_PER_CTX; ++i) {
                struct scrub_bio *sbio;
 
 
 static void scrub_print_warning(const char *errstr, struct scrub_block *sblock)
 {
-       struct btrfs_device *dev = sblock->sctx->dev;
-       struct btrfs_fs_info *fs_info = dev->dev_root->fs_info;
+       struct btrfs_device *dev;
+       struct btrfs_fs_info *fs_info;
        struct btrfs_path *path;
        struct btrfs_key found_key;
        struct extent_buffer *eb;
        const int bufsize = 4096;
        int ret;
 
+       WARN_ON(sblock->page_count < 1);
+       dev = sblock->pagev[0].dev;
+       fs_info = sblock->sctx->dev_root->fs_info;
+
        path = btrfs_alloc_path();
 
        swarn.scratch_buf = kmalloc(bufsize, GFP_NOFS);
        swarn.msg_buf = kmalloc(bufsize, GFP_NOFS);
-       BUG_ON(sblock->page_count < 1);
        swarn.sector = (sblock->pagev[0].physical) >> 9;
        swarn.logical = sblock->pagev[0].logical;
        swarn.errstr = errstr;
-       swarn.dev = dev;
+       swarn.dev = NULL;
        swarn.msg_bufsize = bufsize;
        swarn.scratch_bufsize = bufsize;
 
                } while (ret != 1);
        } else {
                swarn.path = path;
+               swarn.dev = dev;
                iterate_extent_inodes(fs_info, found_key.objectid,
                                        extent_item_pos, 1,
                                        scrub_print_warning_inode, &swarn);
                printk_ratelimited_in_rcu(KERN_ERR
                        "btrfs: unable to fixup (nodatasum) error at logical %llu on dev %s\n",
                        (unsigned long long)fixup->logical,
-                       rcu_str_deref(sctx->dev->name));
+                       rcu_str_deref(fixup->dev->name));
        }
 
        btrfs_free_path(path);
 static int scrub_handle_errored_block(struct scrub_block *sblock_to_check)
 {
        struct scrub_ctx *sctx = sblock_to_check->sctx;
+       struct btrfs_device *dev;
        struct btrfs_fs_info *fs_info;
        u64 length;
        u64 logical;
                                      DEFAULT_RATELIMIT_BURST);
 
        BUG_ON(sblock_to_check->page_count < 1);
-       fs_info = sctx->dev->dev_root->fs_info;
+       fs_info = sctx->dev_root->fs_info;
        length = sblock_to_check->page_count * PAGE_SIZE;
        logical = sblock_to_check->pagev[0].logical;
        generation = sblock_to_check->pagev[0].generation;
                        BTRFS_EXTENT_FLAG_DATA);
        have_csum = sblock_to_check->pagev[0].have_csum;
        csum = sblock_to_check->pagev[0].csum;
+       dev = sblock_to_check->pagev[0].dev;
 
        /*
         * read all mirrors one after the other. This includes to
                sctx->stat.read_errors++;
                sctx->stat.uncorrectable_errors++;
                spin_unlock(&sctx->stat_lock);
-               btrfs_dev_stat_inc_and_print(sctx->dev,
-                                            BTRFS_DEV_STAT_READ_ERRS);
+               btrfs_dev_stat_inc_and_print(dev, BTRFS_DEV_STAT_READ_ERRS);
                goto out;
        }
 
                sctx->stat.read_errors++;
                sctx->stat.uncorrectable_errors++;
                spin_unlock(&sctx->stat_lock);
-               btrfs_dev_stat_inc_and_print(sctx->dev,
-                                            BTRFS_DEV_STAT_READ_ERRS);
+               btrfs_dev_stat_inc_and_print(dev, BTRFS_DEV_STAT_READ_ERRS);
                goto out;
        }
        BUG_ON(failed_mirror_index >= BTRFS_MAX_MIRRORS);
                sctx->stat.read_errors++;
                sctx->stat.uncorrectable_errors++;
                spin_unlock(&sctx->stat_lock);
-               btrfs_dev_stat_inc_and_print(sctx->dev,
-                                            BTRFS_DEV_STAT_READ_ERRS);
+               btrfs_dev_stat_inc_and_print(dev, BTRFS_DEV_STAT_READ_ERRS);
                goto out;
        }
 
                spin_unlock(&sctx->stat_lock);
                if (__ratelimit(&_rs))
                        scrub_print_warning("i/o error", sblock_to_check);
-               btrfs_dev_stat_inc_and_print(sctx->dev,
-                                            BTRFS_DEV_STAT_READ_ERRS);
+               btrfs_dev_stat_inc_and_print(dev, BTRFS_DEV_STAT_READ_ERRS);
        } else if (sblock_bad->checksum_error) {
                spin_lock(&sctx->stat_lock);
                sctx->stat.csum_errors++;
                spin_unlock(&sctx->stat_lock);
                if (__ratelimit(&_rs))
                        scrub_print_warning("checksum error", sblock_to_check);
-               btrfs_dev_stat_inc_and_print(sctx->dev,
+               btrfs_dev_stat_inc_and_print(dev,
                                             BTRFS_DEV_STAT_CORRUPTION_ERRS);
        } else if (sblock_bad->header_error) {
                spin_lock(&sctx->stat_lock);
                        scrub_print_warning("checksum/header error",
                                            sblock_to_check);
                if (sblock_bad->generation_error)
-                       btrfs_dev_stat_inc_and_print(sctx->dev,
+                       btrfs_dev_stat_inc_and_print(dev,
                                BTRFS_DEV_STAT_GENERATION_ERRS);
                else
-                       btrfs_dev_stat_inc_and_print(sctx->dev,
+                       btrfs_dev_stat_inc_and_print(dev,
                                BTRFS_DEV_STAT_CORRUPTION_ERRS);
        }
 
                if (!fixup_nodatasum)
                        goto did_not_correct_error;
                fixup_nodatasum->sctx = sctx;
+               fixup_nodatasum->dev = dev;
                fixup_nodatasum->logical = logical;
                fixup_nodatasum->root = fs_info->extent_root;
                fixup_nodatasum->mirror_num = failed_mirror_index + 1;
                        printk_ratelimited_in_rcu(KERN_ERR
                                "btrfs: fixed up error at logical %llu on dev %s\n",
                                (unsigned long long)logical,
-                               rcu_str_deref(sctx->dev->name));
+                               rcu_str_deref(dev->name));
                }
        } else {
 did_not_correct_error:
                printk_ratelimited_in_rcu(KERN_ERR
                        "btrfs: unable to fixup (regular) error at logical %llu on dev %s\n",
                        (unsigned long long)logical,
-                       rcu_str_deref(sctx->dev->name));
+                       rcu_str_deref(dev->name));
        }
 
 out:
        void *buffer;
        u32 crc = ~(u32)0;
        int fail = 0;
-       struct btrfs_root *root = sctx->dev->dev_root;
+       struct btrfs_root *root = sctx->dev_root;
        u64 len;
        int index;
 
 {
        struct scrub_ctx *sctx = sblock->sctx;
        struct btrfs_header *h;
-       struct btrfs_root *root = sctx->dev->dev_root;
+       struct btrfs_root *root = sctx->dev_root;
        struct btrfs_fs_info *fs_info = root->fs_info;
        u8 calculated_csum[BTRFS_CSUM_SIZE];
        u8 on_disk_csum[BTRFS_CSUM_SIZE];
 {
        struct btrfs_super_block *s;
        struct scrub_ctx *sctx = sblock->sctx;
-       struct btrfs_root *root = sctx->dev->dev_root;
+       struct btrfs_root *root = sctx->dev_root;
        struct btrfs_fs_info *fs_info = root->fs_info;
        u8 calculated_csum[BTRFS_CSUM_SIZE];
        u8 on_disk_csum[BTRFS_CSUM_SIZE];
                ++sctx->stat.super_errors;
                spin_unlock(&sctx->stat_lock);
                if (fail_cor)
-                       btrfs_dev_stat_inc_and_print(sctx->dev,
+                       btrfs_dev_stat_inc_and_print(sblock->pagev[0].dev,
                                BTRFS_DEV_STAT_CORRUPTION_ERRS);
                else
-                       btrfs_dev_stat_inc_and_print(sctx->dev,
+                       btrfs_dev_stat_inc_and_print(sblock->pagev[0].dev,
                                BTRFS_DEV_STAT_GENERATION_ERRS);
        }
 
 
                sbio->physical = spage->physical;
                sbio->logical = spage->logical;
+               sbio->dev = spage->dev;
                bio = sbio->bio;
                if (!bio) {
                        bio = bio_alloc(GFP_NOFS, sctx->pages_per_bio);
 
                bio->bi_private = sbio;
                bio->bi_end_io = scrub_bio_end_io;
-               bio->bi_bdev = sctx->dev->bdev;
-               bio->bi_sector = spage->physical >> 9;
+               bio->bi_bdev = sbio->dev->bdev;
+               bio->bi_sector = sbio->physical >> 9;
                sbio->err = 0;
        } else if (sbio->physical + sbio->page_count * PAGE_SIZE !=
                   spage->physical ||
                   sbio->logical + sbio->page_count * PAGE_SIZE !=
-                  spage->logical) {
+                  spage->logical ||
+                  sbio->dev != spage->dev) {
                scrub_submit(sctx);
                goto again;
        }
 }
 
 static int scrub_pages(struct scrub_ctx *sctx, u64 logical, u64 len,
-                      u64 physical, u64 flags, u64 gen, int mirror_num,
-                      u8 *csum, int force)
+                      u64 physical, struct btrfs_device *dev, u64 flags,
+                      u64 gen, int mirror_num, u8 *csum, int force)
 {
        struct scrub_block *sblock;
        int index;
                        return -ENOMEM;
                }
                spage->sblock = sblock;
-               spage->dev = sctx->dev;
+               spage->dev = dev;
                spage->flags = flags;
                spage->generation = gen;
                spage->logical = logical;
 static void scrub_bio_end_io(struct bio *bio, int err)
 {
        struct scrub_bio *sbio = bio->bi_private;
-       struct scrub_ctx *sctx = sbio->sctx;
-       struct btrfs_fs_info *fs_info = sctx->dev->dev_root->fs_info;
+       struct btrfs_fs_info *fs_info = sbio->dev->dev_root->fs_info;
 
        sbio->err = err;
        sbio->bio = bio;
 
 /* scrub extent tries to collect up to 64 kB for each bio */
 static int scrub_extent(struct scrub_ctx *sctx, u64 logical, u64 len,
-                       u64 physical, u64 flags, u64 gen, int mirror_num)
+                       u64 physical, struct btrfs_device *dev, u64 flags,
+                       u64 gen, int mirror_num)
 {
        int ret;
        u8 csum[BTRFS_CSUM_SIZE];
                        if (have_csum == 0)
                                ++sctx->stat.no_csum;
                }
-               ret = scrub_pages(sctx, logical, l, physical, flags, gen,
+               ret = scrub_pages(sctx, logical, l, physical, dev, flags, gen,
                                  mirror_num, have_csum ? csum : NULL, 0);
                if (ret)
                        return ret;
 }
 
 static noinline_for_stack int scrub_stripe(struct scrub_ctx *sctx,
-       struct map_lookup *map, int num, u64 base, u64 length)
+                                          struct map_lookup *map,
+                                          struct btrfs_device *scrub_dev,
+                                          int num, u64 base, u64 length)
 {
        struct btrfs_path *path;
-       struct btrfs_fs_info *fs_info = sctx->dev->dev_root->fs_info;
+       struct btrfs_fs_info *fs_info = sctx->dev_root->fs_info;
        struct btrfs_root *root = fs_info->extent_root;
        struct btrfs_root *csum_root = fs_info->csum_root;
        struct btrfs_extent_item *extent;
        struct reada_control *reada2;
        struct btrfs_key key_start;
        struct btrfs_key key_end;
-
        u64 increment = map->stripe_len;
        u64 offset;
 
 
                        ret = scrub_extent(sctx, key.objectid, key.offset,
                                           key.objectid - logical + physical,
-                                          flags, generation, mirror_num);
+                                          scrub_dev, flags, generation,
+                                          mirror_num);
                        if (ret)
                                goto out;
 
 }
 
 static noinline_for_stack int scrub_chunk(struct scrub_ctx *sctx,
-       u64 chunk_tree, u64 chunk_objectid, u64 chunk_offset, u64 length,
-       u64 dev_offset)
+                                         struct btrfs_device *scrub_dev,
+                                         u64 chunk_tree, u64 chunk_objectid,
+                                         u64 chunk_offset, u64 length,
+                                         u64 dev_offset)
 {
        struct btrfs_mapping_tree *map_tree =
-               &sctx->dev->dev_root->fs_info->mapping_tree;
+               &sctx->dev_root->fs_info->mapping_tree;
        struct map_lookup *map;
        struct extent_map *em;
        int i;
                goto out;
 
        for (i = 0; i < map->num_stripes; ++i) {
-               if (map->stripes[i].dev == sctx->dev &&
+               if (map->stripes[i].dev->bdev == scrub_dev->bdev &&
                    map->stripes[i].physical == dev_offset) {
-                       ret = scrub_stripe(sctx, map, i, chunk_offset, length);
+                       ret = scrub_stripe(sctx, map, scrub_dev, i,
+                                          chunk_offset, length);
                        if (ret)
                                goto out;
                }
 }
 
 static noinline_for_stack
-int scrub_enumerate_chunks(struct scrub_ctx *sctx, u64 start, u64 end)
+int scrub_enumerate_chunks(struct scrub_ctx *sctx,
+                          struct btrfs_device *scrub_dev, u64 start, u64 end)
 {
        struct btrfs_dev_extent *dev_extent = NULL;
        struct btrfs_path *path;
-       struct btrfs_root *root = sctx->dev->dev_root;
+       struct btrfs_root *root = sctx->dev_root;
        struct btrfs_fs_info *fs_info = root->fs_info;
        u64 length;
        u64 chunk_tree;
        path->search_commit_root = 1;
        path->skip_locking = 1;
 
-       key.objectid = sctx->dev->devid;
+       key.objectid = scrub_dev->devid;
        key.offset = 0ull;
        key.type = BTRFS_DEV_EXTENT_KEY;
 
-
        while (1) {
                ret = btrfs_search_slot(NULL, root, &key, path, 0, 0);
                if (ret < 0)
 
                btrfs_item_key_to_cpu(l, &found_key, slot);
 
-               if (found_key.objectid != sctx->dev->devid)
+               if (found_key.objectid != scrub_dev->devid)
                        break;
 
                if (btrfs_key_type(&found_key) != BTRFS_DEV_EXTENT_KEY)
                        ret = -ENOENT;
                        break;
                }
-               ret = scrub_chunk(sctx, chunk_tree, chunk_objectid,
+               ret = scrub_chunk(sctx, scrub_dev, chunk_tree, chunk_objectid,
                                  chunk_offset, length, found_key.offset);
                btrfs_put_block_group(cache);
                if (ret)
        return ret < 0 ? ret : 0;
 }
 
-static noinline_for_stack int scrub_supers(struct scrub_ctx *sctx)
+static noinline_for_stack int scrub_supers(struct scrub_ctx *sctx,
+                                          struct btrfs_device *scrub_dev)
 {
        int     i;
        u64     bytenr;
        u64     gen;
        int     ret;
-       struct btrfs_device *device = sctx->dev;
-       struct btrfs_root *root = device->dev_root;
+       struct btrfs_root *root = sctx->dev_root;
 
        if (root->fs_info->fs_state & BTRFS_SUPER_FLAG_ERROR)
                return -EIO;
 
        for (i = 0; i < BTRFS_SUPER_MIRROR_MAX; i++) {
                bytenr = btrfs_sb_offset(i);
-               if (bytenr + BTRFS_SUPER_INFO_SIZE > device->total_bytes)
+               if (bytenr + BTRFS_SUPER_INFO_SIZE > scrub_dev->total_bytes)
                        break;
 
                ret = scrub_pages(sctx, bytenr, BTRFS_SUPER_INFO_SIZE, bytenr,
-                                    BTRFS_EXTENT_FLAG_SUPER, gen, i, NULL, 1);
+                                 scrub_dev, BTRFS_EXTENT_FLAG_SUPER, gen, i,
+                                 NULL, 1);
                if (ret)
                        return ret;
        }
        mutex_unlock(&root->fs_info->fs_devices->device_list_mutex);
 
        down_read(&fs_info->scrub_super_lock);
-       ret = scrub_supers(sctx);
+       ret = scrub_supers(sctx, dev);
        up_read(&fs_info->scrub_super_lock);
 
        if (!ret)
-               ret = scrub_enumerate_chunks(sctx, start, end);
+               ret = scrub_enumerate_chunks(sctx, dev, start, end);
 
        wait_event(sctx->list_wait, atomic_read(&sctx->in_flight) == 0);
        atomic_dec(&fs_info->scrubs_running);