From 4b1ac34b51a3783ab528b1af307156fab057b543 Mon Sep 17 00:00:00 2001 From: Neil Brown Date: Tue, 3 May 2005 23:44:18 +0000 Subject: Separate sueprblock handling into separate file Signed-off-by: Neil Brown --- Assemble.c | 241 ++++++++++++++++++++++--------------------------------------- 1 file changed, 88 insertions(+), 153 deletions(-) (limited to 'Assemble.c') diff --git a/Assemble.c b/Assemble.c index ace5717..e762c69 100644 --- a/Assemble.c +++ b/Assemble.c @@ -28,8 +28,6 @@ */ #include "mdadm.h" -#include "md_u.h" -#include "md_p.h" int Assemble(char *mddev, int mdfd, mddev_ident_t ident, char *conffile, @@ -52,11 +50,11 @@ int Assemble(char *mddev, int mdfd, * Much of the work of Assemble is in finding and/or * checking the disks to make sure they look right. * - * If mddev is not set, then scan must be and we + * If mddev is not set, then scan must be set and we * read through the config file for dev+uuid mapping * We recurse, setting mddev, for each device that * - isn't running - * - has a valid uuid (or any uuid if !uuidset + * - has a valid uuid (or any uuid if !uuidset) * * If mddev is set, we try to determine state of md. * check version - must be at least 0.90.0 @@ -74,9 +72,9 @@ int Assemble(char *mddev, int mdfd, * Check superblock - discard if bad * Check uuid (set if we don't have one) - discard if no match * Check superblock similarity if we have a superblock - discard if different - * Record events, devicenum, utime + * Record events, devicenum * This should give us a list of devices for the array - * We should collect the most recent event and utime numbers + * We should collect the most recent event number * * Count disks with recent enough event count * While force && !enough disks @@ -94,14 +92,12 @@ int Assemble(char *mddev, int mdfd, */ int old_linux = 0; int vers; - mdu_array_info_t array; - mdp_super_t first_super, super; + void *first_super = NULL, *super = NULL; struct { char *devname; unsigned int major, minor; unsigned int oldmajor, oldminor; long long events; - time_t utime; int uptodate; int state; int raid_disk; @@ -119,6 +115,7 @@ int Assemble(char *mddev, int mdfd, int start_partial_ok = force || devlist==NULL; unsigned int num_devs; mddev_dev_t tmpdev; + struct mdinfo info; vers = md_get_version(mdfd); if (vers <= 0) { @@ -133,7 +130,7 @@ int Assemble(char *mddev, int mdfd, if (get_linux_version() < 2004000) old_linux = 1; - if (ioctl(mdfd, GET_ARRAY_INFO, &array)>=0) { + if (ioctl(mdfd, GET_ARRAY_INFO, &info.array)>=0) { fprintf(stderr, Name ": device %s already active - cannot assemble it\n", mddev); return 1; @@ -165,12 +162,8 @@ int Assemble(char *mddev, int mdfd, num_devs++; tmpdev = tmpdev->next; } - best = malloc(num_devs * sizeof(*best)); devices = malloc(num_devs * sizeof(*devices)); - first_super.md_magic = 0; - for (i=0; idevname; devlist = devlist->next; @@ -192,6 +183,11 @@ int Assemble(char *mddev, int mdfd, fprintf(stderr, Name ": %s is not one of %s\n", devname, ident->devices); continue; } + + if (super) { + free(super); + super = NULL; + } dfd = open(devname, O_RDONLY|O_EXCL, 0); if (dfd < 0) { @@ -207,40 +203,39 @@ int Assemble(char *mddev, int mdfd, fprintf(stderr, Name ": %s is not a block device.\n", devname); close(dfd); - } else if (load_super(dfd, &super)) { + } else if (load_super0(dfd, &super, NULL)) { if (inargv || verbose) fprintf( stderr, Name ": no RAID superblock on %s\n", devname); close(dfd); } else { - havesuper =1; - uuid_from_super(this_uuid, &super); + getinfo_super0(&info, super); close(dfd); } if (ident->uuid_set && - (!havesuper || same_uuid(this_uuid, ident->uuid)==0)) { + (!super || same_uuid(info.uuid, ident->uuid)==0)) { if (inargv || verbose) fprintf(stderr, Name ": %s has wrong uuid.\n", devname); continue; } if (ident->super_minor != UnSet && - (!havesuper || ident->super_minor != super.md_minor)) { + (!super || ident->super_minor != info.array.md_minor)) { if (inargv || verbose) fprintf(stderr, Name ": %s has wrong super-minor.\n", devname); continue; } if (ident->level != UnSet && - (!havesuper|| ident->level != (int)super.level)) { + (!super|| ident->level != info.array.level)) { if (inargv || verbose) fprintf(stderr, Name ": %s has wrong raid level.\n", devname); continue; } if (ident->raid_disks != UnSet && - (!havesuper || ident->raid_disks!= super.raid_disks)) { + (!super || ident->raid_disks!= info.array.raid_disks)) { if (inargv || verbose) fprintf(stderr, Name ": %s requires wrong number of drives.\n", devname); @@ -252,85 +247,34 @@ int Assemble(char *mddev, int mdfd, * then we cannot continue */ - if (!havesuper) { + if (!super) { fprintf(stderr, Name ": %s has no superblock - assembly aborted\n", devname); + free(first_super); return 1; } - if (compare_super(&first_super, &super)) { + if (compare_super0(&first_super, super)) { fprintf(stderr, Name ": superblock on %s doesn't match others - assembly aborted\n", devname); + free(super); + free(first_super); return 1; } - - /* this is needed until we get a more relaxed super block format */ - if (devcnt >= MD_SB_DISKS) { - fprintf(stderr, Name ": ouch - too many devices appear to be in this array. Ignoring %s\n", - devname); - continue; - } - /* looks like a good enough match to update the super block if needed */ if (update) { - if (strcmp(update, "sparc2.2")==0 ) { - /* 2.2 sparc put the events in the wrong place - * So we copy the tail of the superblock - * up 4 bytes before continuing - */ - __u32 *sb32 = (__u32*)&super; - memcpy(sb32+MD_SB_GENERIC_CONSTANT_WORDS+7, - sb32+MD_SB_GENERIC_CONSTANT_WORDS+7+1, - (MD_SB_WORDS - (MD_SB_GENERIC_CONSTANT_WORDS+7+1))*4); - fprintf (stderr, Name ": adjusting superblock of %s for 2.2/sparc compatability.\n", - devname); - } - if (strcmp(update, "super-minor") ==0) { - struct stat stb2; - fstat(mdfd, &stb2); - super.md_minor = minor(stb2.st_rdev); - if (verbose) - fprintf(stderr, Name ": updating superblock of %s with minor number %d\n", - devname, super.md_minor); - } - if (strcmp(update, "summaries") == 0) { - /* set nr_disks, active_disks, working_disks, - * failed_disks, spare_disks based on disks[] - * array in superblock. - * Also make sure extra slots aren't 'failed' - */ - super.nr_disks = super.active_disks = - super.working_disks = super.failed_disks = - super.spare_disks = 0; - for (i=0; i < MD_SB_DISKS ; i++) - if (super.disks[i].major || - super.disks[i].minor) { - int state = super.disks[i].state; - if (state & (1<= super.raid_disks && super.disks[i].number == 0) - super.disks[i].state = 0; - } - if (strcmp(update, "resync") == 0) { - /* make sure resync happens */ - super.state &= ~(1<= 0) @@ -339,23 +283,22 @@ int Assemble(char *mddev, int mdfd, if (verbose) fprintf(stderr, Name ": %s is identified as a member of %s, slot %d.\n", - devname, mddev, super.this_disk.raid_disk); + devname, mddev, info.disk.raid_disk); devices[devcnt].devname = devname; devices[devcnt].major = major(stb.st_rdev); devices[devcnt].minor = minor(stb.st_rdev); - devices[devcnt].oldmajor = super.this_disk.major; - devices[devcnt].oldminor = super.this_disk.minor; - devices[devcnt].events = md_event(&super); - devices[devcnt].utime = super.utime; - devices[devcnt].raid_disk = super.this_disk.raid_disk; + devices[devcnt].oldmajor = info.disk.major; + devices[devcnt].oldminor = info.disk.minor; + devices[devcnt].events = info.events; + devices[devcnt].raid_disk = info.disk.raid_disk; devices[devcnt].uptodate = 0; - devices[devcnt].state = super.this_disk.state; + devices[devcnt].state = info.disk.state; if (most_recent < devcnt) { if (devices[devcnt].events > devices[most_recent].events) most_recent = devcnt; } - if ((int)super.level == -4) + if (info.array.level == -4) /* with multipath, the raid_disk from the superblock is meaningless */ i = devcnt; else @@ -381,11 +324,19 @@ int Assemble(char *mddev, int mdfd, devcnt++; } + if (super) + free(super); + super = NULL; + if (devcnt == 0) { fprintf(stderr, Name ": no devices found for %s\n", mddev); + free(first_super); return 1; } + + getinfo_super0(&info, first_super); + /* now we have some devices that might be suitable. * I wonder how many */ @@ -398,7 +349,7 @@ int Assemble(char *mddev, int mdfd, /* note: we ignore error flags in multipath arrays * as they don't make sense */ - if ((int)first_super.level != -4) + if (info.array.level != -4) if (!(devices[j].state & (1<= devices[most_recent].events) { devices[j].uptodate = 1; - if (i < first_super.raid_disks) + if (i < info.array.raid_disks) okcnt++; else sparecnt++; } } - while (force && !enough(first_super.level, first_super.raid_disks, okcnt)) { + while (force && !enough(info.array.level, info.array.raid_disks, okcnt)) { /* Choose the newest best drive which is * not up-to-date, update the superblock * and add it. */ int fd; chosen_drive = -1; - for (i=0; i=0 && !devices[j].uptodate && @@ -442,31 +393,29 @@ int Assemble(char *mddev, int mdfd, devices[chosen_drive].events = 0; continue; } - if (load_super(fd, &super)) { + if (load_super0(fd, &super, NULL)) { close(fd); fprintf(stderr, Name ": RAID superblock disappeared from %s - not updating.\n", devices[chosen_drive].devname); devices[chosen_drive].events = 0; continue; } - super.events_hi = (devices[most_recent].events>>32)&0xFFFFFFFF; - super.events_lo = (devices[most_recent].events)&0xFFFFFFFF; - if (super.level == 5 || super.level == 4) { - /* need to force clean */ - super.state = (1<= req_cnt || start_partial_ok) ))) { if (ioctl(mdfd, RUN_ARRAY, NULL)==0) { fprintf(stderr, Name ": %s has been started with %d drive%s", mddev, okcnt, okcnt==1?"":"s"); - if (okcnt < first_super.raid_disks) - fprintf(stderr, " (out of %d)", first_super.raid_disks); + if (okcnt < info.array.raid_disks) + fprintf(stderr, " (out of %d)", info.array.raid_disks); if (sparecnt) fprintf(stderr, " and %d spare%s", sparecnt, sparecnt==1?"":"s"); fprintf(stderr, ".\n"); @@ -649,13 +584,13 @@ This doesnt work yet fprintf(stderr, Name ": %s assembled from %d drive%s", mddev, okcnt, okcnt==1?"":"s"); if (sparecnt) fprintf(stderr, " and %d spare%s", sparecnt, sparecnt==1?"":"s"); - if (!enough(first_super.level, first_super.raid_disks, okcnt)) + if (!enough(info.array.level, info.array.raid_disks, okcnt)) fprintf(stderr, " - not enough to start the array.\n"); else { - if (req_cnt == first_super.raid_disks) + if (req_cnt == info.array.raid_disks) fprintf(stderr, " - need all %d to start it", req_cnt); else - fprintf(stderr, " - need %d of %d to start", req_cnt, first_super.raid_disks); + fprintf(stderr, " - need %d of %d to start", req_cnt, info.array.raid_disks); fprintf(stderr, " (use --run to insist).\n"); } return 1; -- cgit