X-Git-Url: http://git.ipfire.org/?p=thirdparty%2Fmdadm.git;a=blobdiff_plain;f=Examine.c;h=7013480d6dd80846cc2f464d3ba23c9e8e21284d;hp=6263866ec14e20b8c4f3bafdb664851571021fc0;hb=b06815989179e0f153e44e4336290e655edce9a1;hpb=aa88f531b468349982905ecacf11da2cb6678ce6 diff --git a/Examine.c b/Examine.c index 6263866e..7013480d 100644 --- a/Examine.c +++ b/Examine.c @@ -1,7 +1,7 @@ /* * mdadm - manage Linux "md" devices aka RAID arrays. * - * Copyright (C) 2001-2002 Neil Brown + * Copyright (C) 2001-2013 Neil Brown * * * This program is free software; you can redistribute it and/or modify @@ -19,12 +19,7 @@ * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA * * Author: Neil Brown - * Email: - * Paper: Neil Brown - * School of Computer Science and Engineering - * The University of New South Wales - * Sydney, 2052 - * Australia + * Email: */ #include "mdadm.h" @@ -35,7 +30,9 @@ #endif #include "md_u.h" #include "md_p.h" -int Examine(mddev_dev_t devlist, int brief, int scan, int SparcAdjust) +int Examine(struct mddev_dev *devlist, + struct context *c, + struct supertype *forcest) { /* Read the raid superblock from a device and @@ -50,205 +47,182 @@ int Examine(mddev_dev_t devlist, int brief, int scan, int SparcAdjust) * * utime, state etc * - * If (brief) gather devices for same array and just print a mdadm.conf line including devices= - * if devlist==NULL, use conf_get_devs( + * If (brief) gather devices for same array and just print a mdadm.conf + * line including devices= + * if devlist==NULL, use conf_get_devs() */ - int fd; - time_t atime; - mdp_super_t super; - int d; - char *c; + int fd; int rv = 0; - int err; struct array { - mdp_super_t super; + struct supertype *st; + struct mdinfo info; void *devs; struct array *next; + int spares; } *arrays = NULL; - for (; devlist ; devlist=devlist->next) { - fd = open(devlist->devname, O_RDONLY); + for (; devlist ; devlist = devlist->next) { + struct supertype *st; + int have_container = 0; + int err = 0; + int container = 0; + + fd = dev_open(devlist->devname, O_RDONLY); if (fd < 0) { - if (!scan) - fprintf(stderr,Name ": cannot open %s: %s\n", - devlist->devname, strerror(errno)); - err = 1; + if (!c->scan) { + pr_err("cannot open %s: %s\n", + devlist->devname, strerror(errno)); + rv = 1; + } + continue; } - else { - err = load_super(fd, &super); - close(fd); + + if (forcest) + st = dup_super(forcest); + else if (must_be_container(fd)) { + /* might be a container */ + st = super_by_fd(fd, NULL); + container = 1; + } else + st = guess_super(fd); + if (st) { + err = 1; + st->ignore_hw_compat = 1; + if (!container) + err = st->ss->load_super(st, fd, + (c->brief||c->scan) ? NULL + :devlist->devname); + if (err && st->ss->load_container) { + err = st->ss->load_container(st, fd, + (c->brief||c->scan) ? NULL + :devlist->devname); + if (!err) + have_container = 1; + } + st->ignore_hw_compat = 0; + } else { + if (!c->brief) { + pr_err("No md superblock detected on %s.\n", devlist->devname); + rv = 1; + } + err = 1; } - if (err && (brief||scan)) - continue; - if (err) rv =1; - switch(err) { - case 1: - fprintf(stderr, Name ": cannot find device size for %s: %s\n", - devlist->devname, strerror(errno)); - continue; - case 2: -/* fprintf(stderr, Name ": %s is too small for md: size is %ld sectors\n", - devlist->devname, size); -*/ - fprintf(stderr, Name ": %s is too small for md\n", - devlist->devname); - continue; - case 3: - fprintf(stderr, Name ": Cannot seek to superblock on %s: %s\n", - devlist->devname, strerror(errno)); - continue; - case 4: - fprintf(stderr, Name ": Cannot read superblock on %s\n", - devlist->devname); - continue; - case 5: - fprintf(stderr, Name ": No super block found on %s (Expected magic %08x, got %08x)\n", - devlist->devname, MD_SB_MAGIC, super.md_magic); - continue; - case 6: - fprintf(stderr, Name ": Cannot interpret superblock on %s - version is %d\n", - devlist->devname, super.major_version); + close(fd); + + if (err) { + if (st) + st->ss->free_super(st); continue; } - + + if (c->SparcAdjust) + st->ss->update_super(st, NULL, "sparc2.2", + devlist->devname, 0, 0, NULL); /* Ok, its good enough to try, though the checksum could be wrong */ - if (brief) { + + if (c->brief && st->ss->brief_examine_super == NULL) { + if (!c->scan) + pr_err("No brief listing for %s on %s\n", + st->ss->name, devlist->devname); + } else if (c->brief) { struct array *ap; char *d; - for (ap=arrays; ap; ap=ap->next) { - if (compare_super(&ap->super, &super)==0) + for (ap = arrays; ap; ap = ap->next) { + if (st->ss == ap->st->ss && + st->ss->compare_super(ap->st, st) == 0) break; } if (!ap) { - ap = malloc(sizeof(*ap)); - ap->super = super; + ap = xmalloc(sizeof(*ap)); ap->devs = dl_head(); ap->next = arrays; + ap->spares = 0; + ap->st = st; arrays = ap; - } + st->ss->getinfo_super(st, &ap->info, NULL); + } else + st->ss->getinfo_super(st, &ap->info, NULL); + if (!have_container && + !(ap->info.disk.state & (1<spares++; d = dl_strdup(devlist->devname); dl_add(ap->devs, d); + } else if (c->export) { + if (st->ss->export_examine_super) + st->ss->export_examine_super(st); + st->ss->free_super(st); } else { printf("%s:\n",devlist->devname); - printf(" Magic : %08x\n", super.md_magic); - printf(" Version : %02d.%02d.%02d\n", super.major_version, super.minor_version, - super.patch_version); - if (super.minor_version >= 90) - printf(" UUID : %08x:%08x:%08x:%08x\n", super.set_uuid0, super.set_uuid1, - super.set_uuid2, super.set_uuid3); - else - printf(" UUID : %08x\n", super.set_uuid0); - - atime = super.ctime; - printf(" Creation Time : %.24s\n", ctime(&atime)); - c=map_num(pers, super.level); - printf(" Raid Level : %s\n", c?c:"-unknown-"); - printf(" Device Size : %d%s\n", super.size, human_size((long long)super.size<<10)); - printf(" Raid Devices : %d\n", super.raid_disks); - printf(" Total Devices : %d\n", super.nr_disks); - printf("Preferred Minor : %d\n", super.md_minor); - printf("\n"); - atime = super.utime; - printf(" Update Time : %.24s\n", ctime(&atime)); - printf(" State : %s, %serrors\n", - (super.state&(1<=0) dp = &super.disks[d]; - else dp = &super.this_disk; - sprintf(nb, "%4d", d); - printf("%4s %5d %5d %5d %5d ", d < 0 ? "this" : nb, - dp->number, dp->major, dp->minor, dp->raid_disk); - if (dp->state & (1<state & (1<state & (1<state & (1<state == 0) printf(" spare"); - if ((dv=map_dev(dp->major, dp->minor))) - printf(" %s", dv); - printf("\n"); - } - } - if (SparcAdjust == 2) { - printf(" ----- updating superblock on device ----\n"); - fd = open(devlist->devname, O_RDWR); - if (fd < 0) { - fprintf(stderr, Name ": cannot open %s to update superblock: %s\n", - devlist->devname, strerror(errno)); - err = 1; - } else { - super.sb_csum = calc_sb_csum(&super); - if (store_super(fd, &super)) { - fprintf(stderr, Name ": Count not re-write superblock on %s\n", - devlist->devname); - err = 1; - } - close(fd); - } + st->ss->examine_super(st, c->homehost); + st->ss->free_super(st); } } - if (brief) { + if (c->brief) { struct array *ap; - for (ap=arrays; ap; ap=ap->next) { + for (ap = arrays; ap; ap = ap->next) { char sep='='; - char *c=map_num(pers, ap->super.level); char *d; - printf("ARRAY /dev/md%d level=%s num-devices=%d UUID=", - ap->super.md_minor, c?c:"-unknown-", ap->super.raid_disks); - if (ap->super.minor_version >= 90) - printf("%08x:%08x:%08x:%08x", ap->super.set_uuid0, ap->super.set_uuid1, - ap->super.set_uuid2, ap->super.set_uuid3); - else - printf("%08x", ap->super.set_uuid0); - printf("\n devices"); - for (d=dl_next(ap->devs); d!= ap->devs; d=dl_next(d)) { - printf("%c%s", sep, d); - sep=','; + int newline = 0; + + ap->st->ss->brief_examine_super(ap->st, c->verbose > 0); + if (ap->spares) + newline += printf(" spares=%d", ap->spares); + if (c->verbose > 0) { + newline += printf(" devices"); + for (d = dl_next(ap->devs); + d != ap->devs; + d=dl_next(d)) { + printf("%c%s", sep, d); + sep=','; + } + } + if (ap->st->ss->brief_examine_subarrays) { + if (newline) + printf("\n"); + ap->st->ss->brief_examine_subarrays(ap->st, c->verbose); } - printf("\n"); + ap->st->ss->free_super(ap->st); + /* FIXME free ap */ + if (ap->spares || c->verbose > 0) + printf("\n"); } } return rv; } + +int ExamineBadblocks(char *devname, int brief, struct supertype *forcest) +{ + int fd = dev_open(devname, O_RDONLY); + struct supertype *st = forcest; + int err = 1; + + if (fd < 0) { + pr_err("cannot open %s: %s\n", devname, strerror(errno)); + return 1; + } + if (!st) + st = guess_super(fd); + if (!st) { + if (!brief) + pr_err("No md superblock detected on %s\n", devname); + goto out; + } + if (!st->ss->examine_badblocks) { + pr_err("%s metadata does not support badblocks\n", st->ss->name); + goto out; + } + err = st->ss->load_super(st, fd, brief ? NULL : devname); + if (err) + goto out; + err = st->ss->examine_badblocks(st, fd, devname); + +out: + if (fd >= 0) + close(fd); + if (st) { + st->ss->free_super(st); + free(st); + } + return err; +}