diff options
Diffstat (limited to 'fs')
-rw-r--r-- | fs/btrfs/volumes.c | 22 |
1 files changed, 21 insertions, 1 deletions
diff --git a/fs/btrfs/volumes.c b/fs/btrfs/volumes.c index 7c538f65214b..5700ab03e84b 100644 --- a/fs/btrfs/volumes.c +++ b/fs/btrfs/volumes.c | |||
@@ -532,8 +532,19 @@ static noinline int device_list_add(const char *path, | |||
532 | * As of now don't allow update to btrfs_fs_device through | 532 | * As of now don't allow update to btrfs_fs_device through |
533 | * the btrfs dev scan cli, after FS has been mounted. | 533 | * the btrfs dev scan cli, after FS has been mounted. |
534 | */ | 534 | */ |
535 | if (fs_devices->opened) | 535 | if (fs_devices->opened) { |
536 | return -EBUSY; | 536 | return -EBUSY; |
537 | } else { | ||
538 | /* | ||
539 | * That is if the FS is _not_ mounted and if you | ||
540 | * are here, that means there is more than one | ||
541 | * disk with same uuid and devid.We keep the one | ||
542 | * with larger generation number or the last-in if | ||
543 | * generation are equal. | ||
544 | */ | ||
545 | if (found_transid < device->generation) | ||
546 | return -EEXIST; | ||
547 | } | ||
537 | 548 | ||
538 | name = rcu_string_strdup(path, GFP_NOFS); | 549 | name = rcu_string_strdup(path, GFP_NOFS); |
539 | if (!name) | 550 | if (!name) |
@@ -546,6 +557,15 @@ static noinline int device_list_add(const char *path, | |||
546 | } | 557 | } |
547 | } | 558 | } |
548 | 559 | ||
560 | /* | ||
561 | * Unmount does not free the btrfs_device struct but would zero | ||
562 | * generation along with most of the other members. So just update | ||
563 | * it back. We need it to pick the disk with largest generation | ||
564 | * (as above). | ||
565 | */ | ||
566 | if (!fs_devices->opened) | ||
567 | device->generation = found_transid; | ||
568 | |||
549 | if (found_transid > fs_devices->latest_trans) { | 569 | if (found_transid > fs_devices->latest_trans) { |
550 | fs_devices->latest_devid = devid; | 570 | fs_devices->latest_devid = devid; |
551 | fs_devices->latest_trans = found_transid; | 571 | fs_devices->latest_trans = found_transid; |