}
 
 
-void bch_cached_dev_run(struct cached_dev *dc)
+int bch_cached_dev_run(struct cached_dev *dc)
 {
        struct bcache_device *d = &dc->disk;
        char *buf = kmemdup_nul(dc->sb.label, SB_LABEL_SIZE, GFP_KERNEL);
                NULL,
        };
 
+       if (dc->io_disable)
+               return -EIO;
+
        if (atomic_xchg(&dc->running, 1)) {
                kfree(env[1]);
                kfree(env[2]);
                kfree(buf);
-               return;
+               return -EBUSY;
        }
 
        if (!d->c &&
        kfree(buf);
 
        if (sysfs_create_link(&d->kobj, &disk_to_dev(d->disk)->kobj, "dev") ||
-           sysfs_create_link(&disk_to_dev(d->disk)->kobj, &d->kobj, "bcache"))
+           sysfs_create_link(&disk_to_dev(d->disk)->kobj,
+                             &d->kobj, "bcache")) {
                pr_debug("error creating sysfs link");
+               return -ENOMEM;
+       }
 
        dc->status_update_thread = kthread_run(cached_dev_status_update,
                                               dc, "bcache_status_update");
                        "continue to run without monitoring backing "
                        "device status");
        }
+
+       return 0;
 }
 
 /*
        uint32_t rtime = cpu_to_le32((u32)ktime_get_real_seconds());
        struct uuid_entry *u;
        struct cached_dev *exist_dc, *t;
+       int ret = 0;
 
        if ((set_uuid && memcmp(set_uuid, c->sb.set_uuid, 16)) ||
            (!set_uuid && memcmp(dc->sb.set_uuid, c->sb.set_uuid, 16)))
 
        bch_sectors_dirty_init(&dc->disk);
 
-       bch_cached_dev_run(dc);
+       ret = bch_cached_dev_run(dc);
+       if (ret && (ret != -EBUSY)) {
+               up_write(&dc->writeback_lock);
+               return ret;
+       }
+
        bcache_device_link(&dc->disk, c, "bdev");
        atomic_inc(&c->attached_dev_nr);
 
 {
        const char *err = "cannot allocate memory";
        struct cache_set *c;
+       int ret = -ENOMEM;
 
        bdevname(bdev, dc->backing_dev_name);
        memcpy(&dc->sb, sb, sizeof(struct cache_sb));
                bch_cached_dev_attach(dc, c, NULL);
 
        if (BDEV_STATE(&dc->sb) == BDEV_STATE_NONE ||
-           BDEV_STATE(&dc->sb) == BDEV_STATE_STALE)
-               bch_cached_dev_run(dc);
+           BDEV_STATE(&dc->sb) == BDEV_STATE_STALE) {
+               err = "failed to run cached device";
+               ret = bch_cached_dev_run(dc);
+               if (ret)
+                       goto err;
+       }
 
        return 0;
 err:
        pr_notice("error %s: %s", dc->backing_dev_name, err);
        bcache_device_stop(&dc->disk);
-       return -EIO;
+       return ret;
 }
 
 /* Flash only volumes */