Commit 5d2c74f3 authored by Hannes Reinecke's avatar Hannes Reinecke Committed by Mike Snitzer

dm zoned: allocate temporary superblock for tertiary devices

Checking the tertiary superblock just consists of validating UUIDs,
crcs, and the generation number; it doesn't have contents which would
be required during the actual operation.

So allocate a temporary superblock when checking tertiary devices to
avoid having to store it together with the 'real' superblocks.
Signed-off-by: default avatarHannes Reinecke <hare@suse.de>
Reviewed-by: default avatarDamien Le Moal <damien.lemoal@wdc.com>
Signed-off-by: default avatarMike Snitzer <snitzer@redhat.com>
parent a92fbc44
...@@ -174,7 +174,7 @@ struct dmz_metadata { ...@@ -174,7 +174,7 @@ struct dmz_metadata {
/* Zone information array */ /* Zone information array */
struct xarray zones; struct xarray zones;
struct dmz_sb sb[3]; struct dmz_sb sb[2];
unsigned int mblk_primary; unsigned int mblk_primary;
unsigned int sb_version; unsigned int sb_version;
u64 sb_gen; u64 sb_gen;
...@@ -1016,10 +1016,11 @@ int dmz_flush_metadata(struct dmz_metadata *zmd) ...@@ -1016,10 +1016,11 @@ int dmz_flush_metadata(struct dmz_metadata *zmd)
/* /*
* Check super block. * Check super block.
*/ */
static int dmz_check_sb(struct dmz_metadata *zmd, unsigned int set) static int dmz_check_sb(struct dmz_metadata *zmd, struct dmz_sb *dsb,
bool tertiary)
{ {
struct dmz_super *sb = zmd->sb[set].sb; struct dmz_super *sb = dsb->sb;
struct dmz_dev *dev = zmd->sb[set].dev; struct dmz_dev *dev = dsb->dev;
unsigned int nr_meta_zones, nr_data_zones; unsigned int nr_meta_zones, nr_data_zones;
u32 crc, stored_crc; u32 crc, stored_crc;
u64 gen; u64 gen;
...@@ -1036,7 +1037,7 @@ static int dmz_check_sb(struct dmz_metadata *zmd, unsigned int set) ...@@ -1036,7 +1037,7 @@ static int dmz_check_sb(struct dmz_metadata *zmd, unsigned int set)
DMZ_META_VER, zmd->sb_version); DMZ_META_VER, zmd->sb_version);
return -EINVAL; return -EINVAL;
} }
if ((zmd->sb_version < 1) && (set == 2)) { if (zmd->sb_version < 2 && tertiary) {
dmz_dev_err(dev, "Tertiary superblocks are not supported"); dmz_dev_err(dev, "Tertiary superblocks are not supported");
return -EINVAL; return -EINVAL;
} }
...@@ -1080,7 +1081,7 @@ static int dmz_check_sb(struct dmz_metadata *zmd, unsigned int set) ...@@ -1080,7 +1081,7 @@ static int dmz_check_sb(struct dmz_metadata *zmd, unsigned int set)
return -ENXIO; return -ENXIO;
} }
if (set == 2) { if (tertiary) {
/* /*
* Generation number should be 0, but it doesn't * Generation number should be 0, but it doesn't
* really matter if it isn't. * really matter if it isn't.
...@@ -1129,14 +1130,13 @@ static int dmz_check_sb(struct dmz_metadata *zmd, unsigned int set) ...@@ -1129,14 +1130,13 @@ static int dmz_check_sb(struct dmz_metadata *zmd, unsigned int set)
/* /*
* Read the first or second super block from disk. * Read the first or second super block from disk.
*/ */
static int dmz_read_sb(struct dmz_metadata *zmd, unsigned int set) static int dmz_read_sb(struct dmz_metadata *zmd, struct dmz_sb *sb, int set)
{ {
dmz_zmd_debug(zmd, "read superblock set %d dev %s block %llu", dmz_zmd_debug(zmd, "read superblock set %d dev %s block %llu",
set, zmd->sb[set].dev->name, set, sb->dev->name, sb->block);
zmd->sb[set].block);
return dmz_rdwr_block(zmd->sb[set].dev, REQ_OP_READ, return dmz_rdwr_block(sb->dev, REQ_OP_READ,
zmd->sb[set].block, zmd->sb[set].mblk->page); sb->block, sb->mblk->page);
} }
/* /*
...@@ -1164,7 +1164,7 @@ static int dmz_lookup_secondary_sb(struct dmz_metadata *zmd) ...@@ -1164,7 +1164,7 @@ static int dmz_lookup_secondary_sb(struct dmz_metadata *zmd)
zmd->sb[1].zone = dmz_get(zmd, zone_id + 1); zmd->sb[1].zone = dmz_get(zmd, zone_id + 1);
zmd->sb[1].dev = zmd->sb[0].dev; zmd->sb[1].dev = zmd->sb[0].dev;
for (i = 1; i < zmd->nr_rnd_zones; i++) { for (i = 1; i < zmd->nr_rnd_zones; i++) {
if (dmz_read_sb(zmd, 1) != 0) if (dmz_read_sb(zmd, &zmd->sb[1], 1) != 0)
break; break;
if (le32_to_cpu(zmd->sb[1].sb->magic) == DMZ_MAGIC) if (le32_to_cpu(zmd->sb[1].sb->magic) == DMZ_MAGIC)
return 0; return 0;
...@@ -1181,9 +1181,9 @@ static int dmz_lookup_secondary_sb(struct dmz_metadata *zmd) ...@@ -1181,9 +1181,9 @@ static int dmz_lookup_secondary_sb(struct dmz_metadata *zmd)
} }
/* /*
* Read the first or second super block from disk. * Read a super block from disk.
*/ */
static int dmz_get_sb(struct dmz_metadata *zmd, unsigned int set) static int dmz_get_sb(struct dmz_metadata *zmd, struct dmz_sb *sb, int set)
{ {
struct dmz_mblock *mblk; struct dmz_mblock *mblk;
int ret; int ret;
...@@ -1193,14 +1193,14 @@ static int dmz_get_sb(struct dmz_metadata *zmd, unsigned int set) ...@@ -1193,14 +1193,14 @@ static int dmz_get_sb(struct dmz_metadata *zmd, unsigned int set)
if (!mblk) if (!mblk)
return -ENOMEM; return -ENOMEM;
zmd->sb[set].mblk = mblk; sb->mblk = mblk;
zmd->sb[set].sb = mblk->data; sb->sb = mblk->data;
/* Read super block */ /* Read super block */
ret = dmz_read_sb(zmd, set); ret = dmz_read_sb(zmd, sb, set);
if (ret) { if (ret) {
dmz_free_mblock(zmd, mblk); dmz_free_mblock(zmd, mblk);
zmd->sb[set].mblk = NULL; sb->mblk = NULL;
return ret; return ret;
} }
...@@ -1274,13 +1274,13 @@ static int dmz_load_sb(struct dmz_metadata *zmd) ...@@ -1274,13 +1274,13 @@ static int dmz_load_sb(struct dmz_metadata *zmd)
/* Read and check the primary super block */ /* Read and check the primary super block */
zmd->sb[0].block = dmz_start_block(zmd, zmd->sb[0].zone); zmd->sb[0].block = dmz_start_block(zmd, zmd->sb[0].zone);
zmd->sb[0].dev = dmz_zone_to_dev(zmd, zmd->sb[0].zone); zmd->sb[0].dev = dmz_zone_to_dev(zmd, zmd->sb[0].zone);
ret = dmz_get_sb(zmd, 0); ret = dmz_get_sb(zmd, &zmd->sb[0], 0);
if (ret) { if (ret) {
dmz_dev_err(zmd->sb[0].dev, "Read primary super block failed"); dmz_dev_err(zmd->sb[0].dev, "Read primary super block failed");
return ret; return ret;
} }
ret = dmz_check_sb(zmd, 0); ret = dmz_check_sb(zmd, &zmd->sb[0], false);
/* Read and check secondary super block */ /* Read and check secondary super block */
if (ret == 0) { if (ret == 0) {
...@@ -1293,7 +1293,7 @@ static int dmz_load_sb(struct dmz_metadata *zmd) ...@@ -1293,7 +1293,7 @@ static int dmz_load_sb(struct dmz_metadata *zmd)
} }
zmd->sb[1].block = dmz_start_block(zmd, zmd->sb[1].zone); zmd->sb[1].block = dmz_start_block(zmd, zmd->sb[1].zone);
zmd->sb[1].dev = zmd->sb[0].dev; zmd->sb[1].dev = zmd->sb[0].dev;
ret = dmz_get_sb(zmd, 1); ret = dmz_get_sb(zmd, &zmd->sb[1], 1);
} else } else
ret = dmz_lookup_secondary_sb(zmd); ret = dmz_lookup_secondary_sb(zmd);
...@@ -1302,7 +1302,7 @@ static int dmz_load_sb(struct dmz_metadata *zmd) ...@@ -1302,7 +1302,7 @@ static int dmz_load_sb(struct dmz_metadata *zmd)
return ret; return ret;
} }
ret = dmz_check_sb(zmd, 1); ret = dmz_check_sb(zmd, &zmd->sb[1], false);
if (ret == 0) if (ret == 0)
sb_good[1] = true; sb_good[1] = true;
...@@ -1347,20 +1347,40 @@ static int dmz_load_sb(struct dmz_metadata *zmd) ...@@ -1347,20 +1347,40 @@ static int dmz_load_sb(struct dmz_metadata *zmd)
"Using super block %u (gen %llu)", "Using super block %u (gen %llu)",
zmd->mblk_primary, zmd->sb_gen); zmd->mblk_primary, zmd->sb_gen);
if ((zmd->sb_version > 1) && zmd->sb[2].zone) { if (zmd->sb_version > 1) {
zmd->sb[2].block = dmz_start_block(zmd, zmd->sb[2].zone); int i;
zmd->sb[2].dev = dmz_zone_to_dev(zmd, zmd->sb[2].zone); struct dmz_sb *sb;
ret = dmz_get_sb(zmd, 2);
if (ret) { sb = kzalloc(sizeof(struct dmz_sb), GFP_KERNEL);
dmz_dev_err(zmd->sb[2].dev, if (!sb)
"Read tertiary super block failed"); return -ENOMEM;
return ret; for (i = 1; i < zmd->nr_devs; i++) {
sb->block = 0;
sb->zone = dmz_get(zmd, zmd->dev[i].zone_offset);
sb->dev = &zmd->dev[i];
if (!dmz_is_meta(sb->zone)) {
dmz_dev_err(sb->dev,
"Tertiary super block zone %u not marked as metadata zone",
sb->zone->id);
ret = -EINVAL;
goto out_kfree;
}
ret = dmz_get_sb(zmd, sb, i + 1);
if (ret) {
dmz_dev_err(sb->dev,
"Read tertiary super block failed");
dmz_free_mblock(zmd, sb->mblk);
goto out_kfree;
}
ret = dmz_check_sb(zmd, sb, true);
dmz_free_mblock(zmd, sb->mblk);
if (ret == -EINVAL)
goto out_kfree;
} }
ret = dmz_check_sb(zmd, 2); out_kfree:
if (ret == -EINVAL) kfree(sb);
return ret;
} }
return 0; return ret;
} }
/* /*
...@@ -1417,12 +1437,15 @@ static int dmz_init_zone(struct blk_zone *blkz, unsigned int num, void *data) ...@@ -1417,12 +1437,15 @@ static int dmz_init_zone(struct blk_zone *blkz, unsigned int num, void *data)
zmd->sb[0].zone = zone; zmd->sb[0].zone = zone;
} }
} }
if (zmd->nr_devs > 1 && !zmd->sb[2].zone) { if (zmd->nr_devs > 1 && num == 0) {
/* Tertiary superblock zone */ /*
zmd->sb[2].zone = zone; * Tertiary superblock zones are always at the
* start of the zoned devices, so mark them
* as metadata zone.
*/
set_bit(DMZ_META, &zone->flags);
} }
} }
return 0; return 0;
} }
...@@ -2860,16 +2883,6 @@ int dmz_ctr_metadata(struct dmz_dev *dev, int num_dev, ...@@ -2860,16 +2883,6 @@ int dmz_ctr_metadata(struct dmz_dev *dev, int num_dev,
} }
set_bit(DMZ_META, &zone->flags); set_bit(DMZ_META, &zone->flags);
} }
if (zmd->sb[2].zone) {
zone = dmz_get(zmd, zmd->sb[2].zone->id);
if (!zone) {
dmz_zmd_err(zmd,
"Tertiary metadata zone not present");
ret = -ENXIO;
goto err;
}
set_bit(DMZ_META, &zone->flags);
}
/* Load mapping table */ /* Load mapping table */
ret = dmz_load_mapping(zmd); ret = dmz_load_mapping(zmd);
if (ret) if (ret)
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment