int dosyslog = 0;
int rebuild_map = 0;
int auto_update_home = 0;
+ char *subarray = NULL;
int print_help = 0;
FILE *outf;
continue;
case 'b':
- if (mode == ASSEMBLE || mode == BUILD || mode == CREATE || mode == GROW)
+ if (mode == ASSEMBLE || mode == BUILD || mode == CREATE || mode == GROW ||
+ mode == INCREMENTAL || mode == MANAGE)
break; /* b means bitmap */
brief = 1;
- if (optarg) {
- fprintf(stderr, Name ": -b cannot have any extra immediately after it, sorry.\n");
- exit(2);
- }
continue;
case 'Y': export++;
case 'W':
case Waitclean:
case DetailPlatform:
+ case KillSubarray:
+ if (opt == KillSubarray)
+ subarray = optarg;
case 'K': if (!mode) newmode = MISC; break;
}
if (mode && newmode == mode) {
continue;
}
/* No mode yet, and this is the second device ... */
- fprintf(stderr, Name ": An option must be given to set the mode before a second device is listed\n");
+ fprintf(stderr, Name ": An option must be given to set the mode before a second device\n"
+ " (%s) is listed\n", optarg);
exit(2);
}
if (option_index >= 0)
case O(MISC,'W'):
case O(MISC, Waitclean):
case O(MISC, DetailPlatform):
+ case O(MISC, KillSubarray):
if (devmode && devmode != opt &&
(devmode == 'E' || (opt == 'E' && devmode != 'Q'))) {
fprintf(stderr, Name ": --examine/-E cannot be given with -%c\n",
continue;
}
/* probable typo */
- fprintf(stderr, Name ": bitmap file must contain a '/', or be 'internal', or 'none'\n");
+ fprintf(stderr, Name ": bitmap file must contain a '/', or be 'internal', or 'none'\n"
+ " not '%s'\n", optarg);
exit(2);
case O(GROW,BitmapChunk):
}
}
+ if ((mode != MISC || devmode != 'E') &&
+ geteuid() != 0) {
+ fprintf(stderr, Name ": must be super-user to perform this action\n");
+ exit(1);
+ }
+
ident.autof = autof;
rv = 0;
verbose-quiet, force);
}
} else {
- mddev_ident_t array_list = conf_get_ident(NULL);
+ mddev_ident_t a, array_list = conf_get_ident(NULL);
mddev_dev_t devlist = conf_get_devs();
int cnt = 0;
+ int failures, successes;
if (devlist == NULL) {
fprintf(stderr, Name ": No devices listed in conf file were found.\n");
exit(1);
fprintf(stderr, Name ": --backup_file not meaningful with a --scan assembly.\n");
exit(1);
}
- for (; array_list; array_list = array_list->next) {
- if (array_list->devname &&
- strcasecmp(array_list->devname, "<ignore>") == 0)
- continue;
- if (array_list->autof == 0)
- array_list->autof = autof;
+ for (a = array_list; a ; a = a->next) {
+ a->assembled = 0;
+ if (a->autof == 0)
+ a->autof = autof;
+ }
+ do {
+ failures = 0;
+ successes = 0;
+ rv = 0;
+ for (a = array_list; a ; a = a->next) {
+ int r;
+ if (a->assembled)
+ continue;
+ if (a->devname &&
+ strcasecmp(a->devname, "<ignore>") == 0)
+ continue;
- rv |= Assemble(ss, array_list->devname,
- array_list,
- NULL, NULL,
- readonly, runstop, NULL,
- homehost, require_homehost,
- verbose-quiet, force);
- cnt++;
- }
+ r = Assemble(ss, a->devname,
+ a,
+ NULL, NULL,
+ readonly, runstop, NULL,
+ homehost, require_homehost,
+ verbose-quiet, force);
+ if (r == 0) {
+ a->assembled = 1;
+ successes++;
+ } else
+ failures++;
+ rv |= r;
+ cnt++;
+ }
+ } while (failures && successes);
if (homehost && cnt == 0) {
/* Maybe we can auto-assemble something.
* Repeatedly call Assemble in auto-assemble mode
export, test, homehost);
continue;
case 'K': /* Zero superblock */
- rv |= Kill(dv->devname, force, quiet,0);
+ if (ss)
+ rv |= Kill(dv->devname, ss, force, quiet,0);
+ else {
+ int q = quiet;
+ do {
+ rv |= Kill(dv->devname, NULL, force, q, 0);
+ q = 1;
+ } while (rv == 0);
+ rv &= ~2;
+ }
continue;
case 'Q':
rv |= Query(dv->devname); continue;
rv |= Wait(dv->devname); continue;
case Waitclean:
rv |= WaitClean(dv->devname, -1, verbose-quiet); continue;
+ case KillSubarray:
+ rv |= Kill_subarray(dv->devname, subarray, quiet);
+ continue;
}
mdfd = open_mddev(dv->devname, 1);
if (mdfd>=0) {
if (array_size >= 0) {
/* alway impose array size first, independent of
* anything else
+ * Do not allow level or raid_disks changes at the
+ * same time as that can be irreversibly destructive.
*/
struct mdinfo sra;
int err;
+ if (raiddisks || level != UnSet) {
+ fprintf(stderr, Name ": cannot change array size in same operation "
+ "as changing raiddisks or level.\n"
+ " Change size first, then check that data is still intact.\n");
+ rv = 1;
+ break;
+ }
sysfs_init(&sra, mdfd, 0);
if (array_size == 0)
err = sysfs_set_str(&sra, NULL, "array_size", "default");