summaryrefslogtreecommitdiffstats
path: root/mapfile.c
diff options
context:
space:
mode:
authorNeilBrown <neilb@suse.de>2008-12-18 16:23:46 +1100
committerNeilBrown <neilb@suse.de>2008-12-18 16:23:46 +1100
commit3a56f223e94106102eed3fd2b08d3ecad353361b (patch)
tree8604aae606146e65447d58a653a69a7a0da72721 /mapfile.c
parentacee8e896406b686d4b4f12bb748352d20ef83a5 (diff)
downloadmdadm-3a56f223e94106102eed3fd2b08d3ecad353361b.tar.gz
mdadm-3a56f223e94106102eed3fd2b08d3ecad353361b.tar.xz
mdadm-3a56f223e94106102eed3fd2b08d3ecad353361b.zip
map: rebuild map if it doesn't exist.
It is possible for some arrays to be created e.g. by initrd, and so not get mentioned in /var/run/mdadm/map. As "-I" depends on things being listed in 'map', we create it by scanning all devices if it doesn't exist. Signed-off-by: NeilBrown <neilb@suse.de>
Diffstat (limited to 'mapfile.c')
-rw-r--r--mapfile.c53
1 files changed, 53 insertions, 0 deletions
diff --git a/mapfile.c b/mapfile.c
index f616f15..0d0aa39 100644
--- a/mapfile.c
+++ b/mapfile.c
@@ -113,6 +113,12 @@ void map_read(struct map_ent **melp)
f = fopen("/var/run/mdadm/map", "r");
if (!f)
f = fopen("/var/run/mdadm.map", "r");
+ if (!f) {
+ RebuildMap();
+ f = fopen("/var/run/mdadm/map", "r");
+ }
+ if (!f)
+ f = fopen("/var/run/mdadm.map", "r");
if (!f)
return;
@@ -195,3 +201,50 @@ struct map_ent *map_by_uuid(struct map_ent **map, int uuid[4])
return NULL;
}
+
+void RebuildMap(void)
+{
+ struct mdstat_ent *mdstat = mdstat_read(0, 0);
+ struct mdstat_ent *md;
+ struct map_ent *map = NULL;
+ int mdp = get_mdp_major();
+
+ for (md = mdstat ; md ; md = md->next) {
+ struct mdinfo *sra = sysfs_read(-1, md->devnum, GET_DEVS);
+ struct mdinfo *sd;
+
+ for (sd = sra->devs ; sd ; sd = sd->next) {
+ char dn[30];
+ int dfd;
+ int ok;
+ struct supertype *st;
+ char *path;
+ struct mdinfo info;
+
+ sprintf(dn, "%d:%d", sd->disk.major, sd->disk.minor);
+ dfd = dev_open(dn, O_RDONLY);
+ if (dfd < 0)
+ continue;
+ st = guess_super(dfd);
+ if ( st == NULL)
+ ok = -1;
+ else
+ ok = st->ss->load_super(st, dfd, NULL);
+ close(dfd);
+ if (ok != 0)
+ continue;
+ st->ss->getinfo_super(st, &info);
+ if (md->devnum > 0)
+ path = map_dev(MD_MAJOR, md->devnum, 0);
+ else
+ path = map_dev(mdp, (-1-md->devnum)<< 6, 0);
+ map_add(&map, md->devnum, st->ss->major,
+ st->minor_version,
+ info.uuid, path ? : "/unknown");
+ st->ss->free_super(st);
+ break;
+ }
+ }
+ map_write(map);
+ map_free(map);
+}