#include "md_u.h"
#include "md_p.h"
-int Create(char *mddev, int mdfd,
+int Create(struct supertype *st, char *mddev, int mdfd,
int chunk, int level, int layout, unsigned long size, int raiddisks, int sparedisks,
int subdevs, mddev_dev_t devlist,
int runstop, int verbose, int force)
* abort.
*
* SET_ARRAY_INFO and ADD_NEW_DISK, and
- * if runstop==run, or raiddisks diskswere used,
+ * if runstop==run, or raiddisks disks were used,
* RUN_ARRAY
*/
unsigned long long minsize=0, maxsize=0;
mddev_dev_t dv;
int fail=0, warn=0;
struct stat stb;
- int first_missing = MD_SB_DISKS*2;
+ int first_missing = subdevs * 2;
int missing_disks = 0;
- int insert_point = MD_SB_DISKS*2; /* where to insert a missing drive */
- mddev_dev_t moved_disk = NULL; /* the disk that was moved out of the insert point */
+ int insert_point = subdevs * 2; /* where to insert a missing drive */
+ void *super;
+ int pass;
+ int vers;
+ int rv;
mdu_array_info_t array;
-
- if (md_get_version(mdfd) < 9000) {
+ vers = md_get_version(mdfd);
+ if (vers < 9000) {
fprintf(stderr, Name ": Create requires md driver version 0.90.0 or later\n");
return 1;
}
Name ": at least 2 raid-devices needed for level 4 or 5\n");
return 1;
}
- if (raiddisks+sparedisks > MD_SB_DISKS) {
- fprintf(stderr,
- Name ": too many devices requested: %d+%d > %d\n",
- raiddisks, sparedisks, MD_SB_DISKS);
- return 1;
- }
if (subdevs > raiddisks+sparedisks) {
fprintf(stderr, Name ": You have listed more devices (%d) than are in the array(%d)!\n", subdevs, raiddisks+sparedisks);
return 1;
}
else {
ldsize = dsize;
- dsize <<= 9;
+ ldsize <<= 9;
}
- if (ldsize < MD_RESERVED_SECTORS*2LL*512LL) {
+ freesize = st->ss->avail_size(ldsize >> 9);
+ if (freesize == 0) {
fprintf(stderr, Name ": %s is too small: %luK\n",
dname, (unsigned long)(ldsize>>10));
fail = 1;
close(fd);
continue;
}
- freesize = MD_NEW_SIZE_SECTORS((ldsize>>9));
- freesize /= 2;
+
+ freesize /= 2; /* convert to K */
if (size && freesize < size) {
fprintf(stderr, Name ": %s is smaller that given size."
array.nr_disks = array.working_disks + array.failed_disks;
array.layout = layout;
array.chunk_size = chunk*1024;
+ printf("VERS = %d\n", vers);
+
+ if (!st->ss->init_super(&super, &array))
+ return 1;
- if (ioctl(mdfd, SET_ARRAY_INFO, &array)) {
+ if ((vers % 100) >= 1) { /* can use different versions */
+ mdu_array_info_t inf;
+ memset(&inf, 0, sizeof(inf));
+ inf.major_version = st->ss->major;
+ inf.minor_version = st->minor_version;
+ rv = ioctl(mdfd, SET_ARRAY_INFO, &inf);
+ } else
+ rv = ioctl(mdfd, SET_ARRAY_INFO, NULL);
+ if (rv) {
fprintf(stderr, Name ": SET_ARRAY_INFO failed for %s: %s\n",
mddev, strerror(errno));
return 1;
}
-
- for (dnum=0, dv = devlist ; dv ; dv=(dv->next)?(dv->next):moved_disk, dnum++) {
- int fd;
- struct stat stb;
- mdu_disk_info_t disk;
- disk.number = dnum;
- if (dnum == insert_point) {
- moved_disk = dv;
- }
- disk.raid_disk = disk.number;
- if (disk.raid_disk < raiddisks)
- disk.state = 6; /* active and in sync */
- else
- disk.state = 0;
- if (dnum == insert_point ||
- strcasecmp(dv->devname, "missing")==0) {
- disk.major = 0;
- disk.minor = 0;
- disk.state = 1; /* faulty */
- } else {
- fd = open(dv->devname, O_RDONLY|O_EXCL, 0);
- if (fd < 0) {
- fprintf(stderr, Name ": failed to open %s after earlier success - aborting\n",
- dv->devname);
- return 1;
+
+
+ for (pass=1; pass <=2 ; pass++) {
+ mddev_dev_t moved_disk = NULL; /* the disk that was moved out of the insert point */
+
+ for (dnum=0, dv = devlist ; dv ;
+ dv=(dv->next)?(dv->next):moved_disk, dnum++) {
+ int fd;
+ struct stat stb;
+ mdu_disk_info_t disk;
+
+ disk.number = dnum;
+ if (dnum == insert_point) {
+ moved_disk = dv;
}
- fstat(fd, &stb);
- disk.major = major(stb.st_rdev);
- disk.minor = minor(stb.st_rdev);
- close(fd);
- }
- if (ioctl(mdfd, ADD_NEW_DISK, &disk)) {
- fprintf(stderr, Name ": ADD_NEW_DISK for %s failed: %s\n",
- dv->devname, strerror(errno));
- return 1;
+ disk.raid_disk = disk.number;
+ if (disk.raid_disk < raiddisks)
+ disk.state = 6; /* active and in sync */
+ else
+ disk.state = 0;
+ if (dnum == insert_point ||
+ strcasecmp(dv->devname, "missing")==0) {
+ disk.major = 0;
+ disk.minor = 0;
+ disk.state = 1; /* faulty */
+ } else {
+ fd = open(dv->devname, O_RDONLY|O_EXCL, 0);
+ if (fd < 0) {
+ fprintf(stderr, Name ": failed to open %s after earlier success - aborting\n",
+ dv->devname);
+ return 1;
+ }
+ fstat(fd, &stb);
+ disk.major = major(stb.st_rdev);
+ disk.minor = minor(stb.st_rdev);
+ close(fd);
+ }
+ if (disk.state != 1)
+ switch(pass){
+ case 1:
+ st->ss->add_to_super(super, &disk);
+ break;
+ case 2:
+ st->ss->write_init_super(st, super, &disk, dv->devname);
+
+ if (ioctl(mdfd, ADD_NEW_DISK, &disk)) {
+ fprintf(stderr, Name ": ADD_NEW_DISK for %s failed: %s\n",
+ dv->devname, strerror(errno));
+ free(super);
+ return 1;
+ }
+
+ break;
+ }
+ if (dv == moved_disk && dnum != insert_point) break;
}
- if (dv == moved_disk && dnum != insert_point) break;
}
+ free(super);
/* param is not actually used */
if (runstop == 1 || subdevs >= raiddisks) {