struct btrfs_fs_info *fs_info = fc->s_fs_info;
        struct btrfs_fs_context *ctx = fc->fs_private;
        struct btrfs_fs_devices *fs_devices = NULL;
-       struct block_device *bdev;
        struct btrfs_device *device;
        struct super_block *sb;
        blk_mode_t mode = btrfs_open_mode(fc);
                mutex_unlock(&uuid_mutex);
                return PTR_ERR(device);
        }
-
        fs_devices = device->fs_devices;
+       /*
+        * We cannot hold uuid_mutex calling sget_fc(), it will lead to a
+        * locking order reversal with s_umount.
+        *
+        * So here we increase the holding number of fs_devices, this will ensure
+        * the fs_devices itself won't be freed.
+        */
+       btrfs_fs_devices_inc_holding(fs_devices);
        fs_info->fs_devices = fs_devices;
-
-       ret = btrfs_open_devices(fs_devices, mode, &btrfs_fs_type);
        mutex_unlock(&uuid_mutex);
-       if (ret)
-               return ret;
 
-       if (!(fc->sb_flags & SB_RDONLY) && fs_devices->rw_devices == 0)
-               return -EACCES;
-
-       bdev = fs_devices->latest_dev->bdev;
 
        sb = sget_fc(fc, btrfs_fc_test_super, set_anon_super_fc);
-       if (IS_ERR(sb))
+       if (IS_ERR(sb)) {
+               mutex_lock(&uuid_mutex);
+               btrfs_fs_devices_dec_holding(fs_devices);
+               /*
+                * Since the fs_devices is not opened, it can be freed at any
+                * time after unlocking uuid_mutex.  We need to avoid double
+                * free through put_fs_context()->btrfs_free_fs_info().
+                * So here we reset fs_info->fs_devices to NULL, and let the
+                * regular fs_devices reclaim path to handle it.
+                *
+                * This applies to all later branches where no fs_devices is
+                * opened.
+                */
+               fs_info->fs_devices = NULL;
+               mutex_unlock(&uuid_mutex);
                return PTR_ERR(sb);
+       }
 
        set_device_specific_options(fs_info);
 
                 *
                 * fc->s_fs_info is not touched and will be later freed by
                 * put_fs_context() through btrfs_free_fs_context().
-                *
-                * The fs_info->fs_devices will also be closed by btrfs_free_fs_context().
                 */
                ASSERT(fc->s_fs_info == fs_info);
 
+               mutex_lock(&uuid_mutex);
+               btrfs_fs_devices_dec_holding(fs_devices);
+               fs_info->fs_devices = NULL;
+               mutex_unlock(&uuid_mutex);
                /*
                 * At this stage we may have RO flag mismatch between
                 * fc->sb_flags and sb->s_flags.  Caller should detect such
                 * needed.
                 */
        } else {
+               struct block_device *bdev;
+
                /*
                 * The first mount of the fs thus a new superblock, fc->s_fs_info
                 * must be NULL, and the ownership of our fs_info and fs_devices is
                 */
                ASSERT(fc->s_fs_info == NULL);
 
+               mutex_lock(&uuid_mutex);
+               btrfs_fs_devices_dec_holding(fs_devices);
+               ret = btrfs_open_devices(fs_devices, mode, &btrfs_fs_type);
+               if (ret < 0)
+                       fs_info->fs_devices = NULL;
+               mutex_unlock(&uuid_mutex);
+               if (ret < 0) {
+                       deactivate_locked_super(sb);
+                       return ret;
+               }
+               if (!(fc->sb_flags & SB_RDONLY) && fs_devices->rw_devices == 0) {
+                       deactivate_locked_super(sb);
+                       return -EACCES;
+               }
+               bdev = fs_devices->latest_dev->bdev;
                snprintf(sb->s_id, sizeof(sb->s_id), "%pg", bdev);
                shrinker_debugfs_rename(sb->s_shrink, "sb-btrfs:%s", sb->s_id);
                btrfs_sb(sb)->bdev_holder = &btrfs_fs_type;
 
        struct btrfs_device *device;
 
        WARN_ON(fs_devices->opened);
+       WARN_ON(fs_devices->holding);
        while (!list_empty(&fs_devices->devices)) {
                device = list_first_entry(&fs_devices->devices,
                                          struct btrfs_device, dev_list);
                                continue;
                        if (devt && devt != device->devt)
                                continue;
-                       if (fs_devices->opened) {
+                       if (fs_devices->opened || fs_devices->holding) {
                                if (devt)
                                        ret = -EBUSY;
                                break;
 
        mutex_lock(&uuid_mutex);
        close_fs_devices(fs_devices);
-       if (!fs_devices->opened) {
+       if (!fs_devices->opened && !fs_devices->holding) {
                list_splice_init(&fs_devices->seed_list, &list);
 
                /*
 
        /* Count fs-devices opened. */
        int opened;
 
+       /*
+        * Counter of the processes that are holding this fs_devices but not
+        * yet opened.
+        * This is for mounting handling, as we can only open the fs_devices
+        * after a super block is created.  But we cannot take uuid_mutex
+        * during sget_fc(), thus we have to hold the fs_devices (meaning it
+        * cannot be released) until a super block is returned.
+        */
+       int holding;
+
        /* Set when we find or add a device that doesn't have the nonrot flag set. */
        bool rotating;
        /* Devices support TRIM/discard commands. */
        WARN_ONCE(1, "unknown allocation policy %d, fallback to regular", pol);
 }
 
+static inline void btrfs_fs_devices_inc_holding(struct btrfs_fs_devices *fs_devices)
+{
+       lockdep_assert_held(&uuid_mutex);
+       ASSERT(fs_devices->holding >= 0);
+       fs_devices->holding++;
+}
+
+static inline void btrfs_fs_devices_dec_holding(struct btrfs_fs_devices *fs_devices)
+{
+       lockdep_assert_held(&uuid_mutex);
+       ASSERT(fs_devices->holding > 0);
+       fs_devices->holding--;
+}
+
 void btrfs_commit_device_sizes(struct btrfs_transaction *trans);
 
 struct list_head * __attribute_const__ btrfs_get_fs_uuids(void);