2 * The new sysinstall program.
4 * This is probably the last program in the `sysinstall' line - the next
5 * generation being essentially a complete rewrite.
10 * Jordan Hubbard. All rights reserved.
12 * Redistribution and use in source and binary forms, with or without
13 * modification, are permitted provided that the following conditions
15 * 1. Redistributions of source code must retain the above copyright
16 * notice, this list of conditions and the following disclaimer,
17 * verbatim and that no modifications are made prior to this
19 * 2. Redistributions in binary form must reproduce the above copyright
20 * notice, this list of conditions and the following disclaimer in the
21 * documentation and/or other materials provided with the distribution.
23 * THIS SOFTWARE IS PROVIDED BY JORDAN HUBBARD ``AS IS'' AND
24 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
25 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
26 * ARE DISCLAIMED. IN NO EVENT SHALL JORDAN HUBBARD OR HIS PETS BE LIABLE
27 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
28 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
29 * OR SERVICES; LOSS OF USE, DATA, LIFE OR PROFITS; OR BUSINESS INTERRUPTION)
30 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
31 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
32 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
37 #include "sysinstall.h"
38 #include <sys/fcntl.h>
39 #include <sys/param.h>
40 #include <sys/socket.h>
41 #include <sys/ioctl.h>
42 #include <sys/errno.h>
45 #include <net/if_var.h>
46 #include <net/if_dl.h>
47 #include <netinet/in.h>
48 #include <netinet/in_var.h>
49 #include <arpa/inet.h>
53 /* how much to bias minor number for a given /dev/<ct#><un#>s<s#> slice */
54 #define SLICE_DELTA (0x10000)
56 static Device *Devices[DEV_MAX];
59 static struct _devname {
63 int major, minor, delta, max;
65 { DEVICE_TYPE_CDROM, "cd%d", "SCSI CDROM drive", 15, 2, 8, 4 },
66 { DEVICE_TYPE_CDROM, "mcd%d", "Mitsumi (old model) CDROM drive", 29, 0, 8, 4 },
67 { DEVICE_TYPE_CDROM, "scd%d", "Sony CDROM drive - CDU31/33A type", 45, 0, 8, 4 },
69 { DEVICE_TYPE_CDROM, "matcd%d", "Matsushita CDROM ('sound blaster' type)", 46, 0, 8, 4 },
71 { DEVICE_TYPE_CDROM, "acd%d", "ATAPI/IDE CDROM", 117, 0, 8, 4 },
72 { DEVICE_TYPE_TAPE, "sa%d", "SCSI tape drive", 14, 0, 16, 4 },
73 { DEVICE_TYPE_TAPE, "rwt%d", "Wangtek tape drive", 10, 0, 1, 4 },
74 { DEVICE_TYPE_DISK, "da%d", "SCSI disk device", 13, 65538, 8, 16 },
75 { DEVICE_TYPE_DISK, "ad%d", "ATA/IDE disk device", 116, 65538, 8, 16 },
76 { DEVICE_TYPE_DISK, "ar%d", "ATA/IDE RAID device", 157, 65538, 8, 16 },
77 { DEVICE_TYPE_DISK, "afd%d", "ATAPI/IDE floppy device", 118, 65538, 8, 4 },
78 { DEVICE_TYPE_DISK, "mlxd%d", "Mylex RAID disk", 131, 65538, 8, 4 },
79 { DEVICE_TYPE_DISK, "amrd%d", "AMI MegaRAID drive", 133, 65538, 8, 4 },
80 { DEVICE_TYPE_DISK, "idad%d", "Compaq RAID array", 109, 65538, 8, 4 },
81 { DEVICE_TYPE_DISK, "twed%d", "3ware ATA RAID array", 147, 65538, 8, 4 },
82 { DEVICE_TYPE_DISK, "aacd%d", "Adaptec FSA RAID array", 151, 65538, 8, 4 },
83 { DEVICE_TYPE_DISK, "ipsd%d", "IBM ServeRAID RAID array", 176, 65538, 8, 4 },
84 { DEVICE_TYPE_FLOPPY, "fd%d", "floppy drive unit A", 9, 0, 64, 4 },
85 { DEVICE_TYPE_NETWORK, "an", "Aironet 4500/4800 802.11 wireless adapter" },
86 { DEVICE_TYPE_NETWORK, "aue", "ADMtek USB ethernet adapter" },
87 { DEVICE_TYPE_NETWORK, "axe", "ASIX Electronics USB ethernet adapter" },
88 { DEVICE_TYPE_NETWORK, "bfe", "Broadcom BCM440x PCI ethernet card" },
89 { DEVICE_TYPE_NETWORK, "bge", "Broadcom BCM570x PCI gigabit ethernet card" },
90 { DEVICE_TYPE_NETWORK, "cue", "CATC USB ethernet adapter" },
91 { DEVICE_TYPE_NETWORK, "fpa", "DEC DEFPA PCI FDDI card" },
92 { DEVICE_TYPE_NETWORK, "sr", "SDL T1/E1 sync serial PCI card" },
93 { DEVICE_TYPE_NETWORK, "cc3i", "SDL HSSI sync serial PCI card" },
94 { DEVICE_TYPE_NETWORK, "en", "Efficient Networks ATM PCI card" },
95 { DEVICE_TYPE_NETWORK, "dc", "DEC/Intel 21143 (and clones) PCI fast ethernet card" },
96 { DEVICE_TYPE_NETWORK, "de", "DEC DE435 PCI NIC or other DC21040-AA based card" },
97 { DEVICE_TYPE_NETWORK, "fxp", "Intel EtherExpress Pro/100B PCI Fast Ethernet card" },
98 { DEVICE_TYPE_NETWORK, "ed", "Novell NE1000/2000; 3C503; NE2000-compatible PCMCIA" },
99 { DEVICE_TYPE_NETWORK, "ep", "3Com 3C509 ethernet card/3C589 PCMCIA" },
100 { DEVICE_TYPE_NETWORK, "el", "3Com 3C501 ethernet card" },
101 { DEVICE_TYPE_NETWORK, "em", "Intel(R) PRO/1000 ethernet card" },
102 { DEVICE_TYPE_NETWORK, "ex", "Intel EtherExpress Pro/10 ethernet card" },
103 { DEVICE_TYPE_NETWORK, "fe", "Fujitsu MB86960A/MB86965A ethernet card" },
104 { DEVICE_TYPE_NETWORK, "gem", "Apple/Sun GMAC ethernet adapter" },
105 { DEVICE_TYPE_NETWORK, "ie", "AT&T StarLAN 10 and EN100; 3Com 3C507; NI5210" },
106 { DEVICE_TYPE_NETWORK, "ix", "Intel Etherexpress ethernet card" },
107 { DEVICE_TYPE_NETWORK, "kue", "Kawasaki LSI USB ethernet adapter" },
108 { DEVICE_TYPE_NETWORK, "le", "DEC EtherWorks 2 or 3 ethernet card" },
109 { DEVICE_TYPE_NETWORK, "lnc", "Lance/PCnet (Isolan/Novell NE2100/NE32-VL) ethernet" },
110 { DEVICE_TYPE_NETWORK, "lge", "Level 1 LXT1001 gigabit ethernet card" },
111 { DEVICE_TYPE_NETWORK, "nge", "NatSemi PCI gigabit ethernet card" },
112 { DEVICE_TYPE_NETWORK, "pcn", "AMD Am79c79x PCI ethernet card" },
113 { DEVICE_TYPE_NETWORK, "ray", "Raytheon Raylink 802.11 wireless adaptor" },
114 { DEVICE_TYPE_NETWORK, "re", "RealTek 8139C+/8169/8169S/8110S PCI ethernet card" },
115 { DEVICE_TYPE_NETWORK, "rl", "RealTek 8129/8139 PCI ethernet card" },
116 { DEVICE_TYPE_NETWORK, "rue", "RealTek USB ethernet card" },
117 { DEVICE_TYPE_NETWORK, "sf", "Adaptec AIC-6915 PCI ethernet card" },
118 { DEVICE_TYPE_NETWORK, "sis", "SiS 900/SiS 7016 PCI ethernet card" },
120 { DEVICE_TYPE_NETWORK, "snc", "SONIC ethernet card" },
122 { DEVICE_TYPE_NETWORK, "sn", "SMC/Megahertz ethernet card" },
123 { DEVICE_TYPE_NETWORK, "ste", "Sundance ST201 PCI ethernet card" },
124 { DEVICE_TYPE_NETWORK, "sk", "SysKonnect PCI gigabit ethernet card" },
125 { DEVICE_TYPE_NETWORK, "tx", "SMC 9432TX ethernet card" },
126 { DEVICE_TYPE_NETWORK, "txp", "3Com 3cR990 ethernet card" },
127 { DEVICE_TYPE_NETWORK, "ti", "Alteon Networks PCI gigabit ethernet card" },
128 { DEVICE_TYPE_NETWORK, "tl", "Texas Instruments ThunderLAN PCI ethernet card" },
129 { DEVICE_TYPE_NETWORK, "vge", "VIA VT612x PCI gigabit ethernet card" },
130 { DEVICE_TYPE_NETWORK, "vr", "VIA VT3043/VT86C100A Rhine PCI ethernet card" },
131 { DEVICE_TYPE_NETWORK, "vlan", "IEEE 802.1Q VLAN network interface" },
132 { DEVICE_TYPE_NETWORK, "vx", "3COM 3c590 / 3c595 ethernet card" },
133 { DEVICE_TYPE_NETWORK, "wb", "Winbond W89C840F PCI ethernet card" },
134 { DEVICE_TYPE_NETWORK, "wi", "Lucent WaveLAN/IEEE 802.11 wireless adapter" },
135 { DEVICE_TYPE_NETWORK, "wx", "Intel Gigabit Ethernet (82452) card" },
136 { DEVICE_TYPE_NETWORK, "xe", "Xircom/Intel EtherExpress Pro100/16 ethernet card" },
137 { DEVICE_TYPE_NETWORK, "xl", "3COM 3c90x / 3c90xB PCI ethernet card" },
138 { DEVICE_TYPE_NETWORK, "cuad%d", "%s on device %s (COM%d)", 28, 128, 1, 16 },
139 { DEVICE_TYPE_NETWORK, "fwe", "FireWire Ethernet emulation" },
140 { DEVICE_TYPE_NETWORK, "lp", "Parallel Port IP (PLIP) peer connection" },
141 { DEVICE_TYPE_NETWORK, "lo", "Loop-back (local) network interface" },
142 { DEVICE_TYPE_NETWORK, "disc", "Software discard network interface" },
147 new_device(char *name)
151 dev = safe_malloc(sizeof(Device));
152 bzero(dev, sizeof(Device));
154 SAFE_STRCPY(dev->name, name);
158 /* Stubs for unimplemented strategy routines */
160 dummyInit(Device *dev)
166 dummyGet(Device *dev, char *dist, Boolean probe)
172 dummyShutdown(Device *dev)
178 deviceTry(struct _devname dev, char *try, int i)
186 snprintf(unit, sizeof unit, dev.name, i);
187 snprintf(try, FILENAME_MAX, "/dev/%s", unit);
189 msgDebug("deviceTry: attempting to open %s\n", try);
190 fd = open(try, O_RDONLY);
193 msgDebug("deviceTry: open of %s succeeded on first try.\n", try);
197 d = makedev(dev.major, dev.minor + (i * dev.delta));
199 msgDebug("deviceTry: Making %s device for %s [%d, %d]\n", m & S_IFCHR ? "raw" : "block", try, dev.major, dev.minor + (i * dev.delta));
200 fail = mknod(try, m, d);
201 fd = open(try, O_RDONLY);
204 msgDebug("deviceTry: open of %s succeeded on second try.\n", try);
209 /* Don't try a "make-under" here since we're using a fixit floppy in this case */
210 snprintf(try, FILENAME_MAX, "/mnt/dev/%s", unit);
211 fd = open(try, O_RDONLY);
213 msgDebug("deviceTry: final attempt for %s returns %d\n", try, fd);
217 /* Register a new device in the devices array */
219 deviceRegister(char *name, char *desc, char *devname, DeviceType type, Boolean enabled,
220 Boolean (*init)(Device *), FILE * (*get)(Device *, char *, Boolean),
221 void (*shutdown)(Device *), void *private)
223 Device *newdev = NULL;
225 if (numDevs == DEV_MAX)
226 msgFatal("Too many devices found!");
228 newdev = new_device(name);
229 newdev->description = desc;
230 newdev->devname = devname;
232 newdev->enabled = enabled;
233 newdev->init = init ? init : dummyInit;
234 newdev->get = get ? get : dummyGet;
235 newdev->shutdown = shutdown ? shutdown : dummyShutdown;
236 newdev->private = private;
237 Devices[numDevs] = newdev;
238 Devices[++numDevs] = NULL;
243 /* Reset the registered device chain */
249 for (i = 0; i < numDevs; i++) {
250 DEVICE_SHUTDOWN(Devices[i]);
252 /* XXX this potentially leaks Devices[i]->private if it's being
253 * used to point to something dynamic, but you're not supposed
254 * to call this routine at such times that some open instance
255 * has its private ptr pointing somewhere anyway. XXX
259 Devices[numDevs = 0] = NULL;
262 /* Get all device information for devices we have attached */
268 struct ifreq *ifptr, *end;
270 char buffer[INTERFACE_MAX * sizeof(struct ifreq)];
273 msgNotify("Probing devices, please wait (this can take a while)...");
274 /* First go for the network interfaces. Stolen shamelessly from ifconfig! */
275 ifc.ifc_len = sizeof(buffer);
276 ifc.ifc_buf = buffer;
278 s = socket(AF_INET, SOCK_DGRAM, 0);
280 goto skipif; /* Jump over network iface probing */
282 if (ioctl(s, SIOCGIFCONF, (char *) &ifc) < 0)
283 goto skipif; /* Jump over network iface probing */
286 ifflags = ifc.ifc_req->ifr_flags;
287 end = (struct ifreq *) (ifc.ifc_buf + ifc.ifc_len);
288 for (ifptr = ifc.ifc_req; ifptr < end; ifptr++) {
291 /* If it's not a link entry, forget it */
292 if (ifptr->ifr_ifru.ifru_addr.sa_family != AF_LINK)
295 /* Eliminate network devices that don't make sense */
296 if (!strncmp(ifptr->ifr_name, "lo", 2))
299 /* If we have a slip device, don't register it */
300 if (!strncmp(ifptr->ifr_name, "sl", 2)) {
303 /* And the same for ppp */
304 if (!strncmp(ifptr->ifr_name, "tun", 3) || !strncmp(ifptr->ifr_name, "ppp", 3)) {
307 /* Try and find its description */
308 for (i = 0, descr = NULL; device_names[i].name; i++) {
309 int len = strlen(device_names[i].name);
311 if (!ifptr->ifr_name || !ifptr->ifr_name[0])
313 else if (!strncmp(ifptr->ifr_name, device_names[i].name, len)) {
314 descr = device_names[i].description;
319 descr = "<unknown network interface type>";
321 deviceRegister(ifptr->ifr_name, descr, strdup(ifptr->ifr_name), DEVICE_TYPE_NETWORK, TRUE,
322 mediaInitNetwork, NULL, mediaShutdownNetwork, NULL);
324 msgDebug("Found a network device named %s\n", ifptr->ifr_name);
326 if ((s = socket(AF_INET, SOCK_DGRAM, 0)) < 0)
330 if (ifptr->ifr_addr.sa_len) /* I'm not sure why this is here - it's inherited */
331 ifptr = (struct ifreq *)((caddr_t)ifptr + ifptr->ifr_addr.sa_len - sizeof(struct sockaddr));
336 /* Next, try to find all the types of devices one might need
337 * during the second stage of the installation.
339 for (i = 0; device_names[i].name; i++) {
340 for (j = 0; j < device_names[i].max; j++) {
341 char try[FILENAME_MAX];
343 switch(device_names[i].type) {
344 case DEVICE_TYPE_CDROM:
345 fd = deviceTry(device_names[i], try, j);
346 if (fd >= 0 || errno == EBUSY) { /* EBUSY if already mounted */
349 if (fd >= 0) close(fd);
350 snprintf(n, sizeof n, device_names[i].name, j);
351 deviceRegister(strdup(n), device_names[i].description, strdup(try),
352 DEVICE_TYPE_CDROM, TRUE, mediaInitCDROM, mediaGetCDROM,
353 mediaShutdownCDROM, NULL);
355 msgDebug("Found a CDROM device for %s\n", try);
359 case DEVICE_TYPE_TAPE:
360 fd = deviceTry(device_names[i], try, j);
365 snprintf(n, sizeof n, device_names[i].name, j);
366 deviceRegister(strdup(n), device_names[i].description, strdup(try),
367 DEVICE_TYPE_TAPE, TRUE, mediaInitTape, mediaGetTape, mediaShutdownTape, NULL);
369 msgDebug("Found a TAPE device for %s\n", try);
373 case DEVICE_TYPE_DISK:
374 fd = deviceTry(device_names[i], try, j);
375 if (fd >= 0 && RunningAsInit) {
379 char unit[80], slice[80];
382 /* Make associated slice entries */
383 for (s = 1; s < 8; s++) {
384 snprintf(unit, sizeof unit, device_names[i].name, j);
385 snprintf(slice, sizeof slice, "/dev/%ss%d", unit, s);
386 d = makedev(device_names[i].major, device_names[i].minor +
387 (j * device_names[i].delta) + (s * SLICE_DELTA));
389 fail = mknod(slice, m, d);
390 fd = open(slice, O_RDONLY);
399 case DEVICE_TYPE_FLOPPY:
400 fd = deviceTry(device_names[i], try, j);
405 snprintf(n, sizeof n, device_names[i].name, j);
406 deviceRegister(strdup(n), device_names[i].description, strdup(try),
407 DEVICE_TYPE_FLOPPY, TRUE, mediaInitFloppy, mediaGetFloppy,
408 mediaShutdownFloppy, NULL);
410 msgDebug("Found a floppy device for %s\n", try);
414 case DEVICE_TYPE_NETWORK:
415 fd = deviceTry(device_names[i], try, j);
416 /* The only network devices that you can open this way are serial ones */
421 cp = device_names[i].description;
422 /* Serial devices get a slip and ppp device each, if supported */
423 newdesc = safe_malloc(strlen(cp) + 40);
424 sprintf(newdesc, cp, "SLIP interface", try, j + 1);
425 deviceRegister("sl0", newdesc, strdup(try), DEVICE_TYPE_NETWORK, TRUE, mediaInitNetwork,
426 NULL, mediaShutdownNetwork, NULL);
427 msgDebug("Add mapping for %s to sl0\n", try);
428 newdesc = safe_malloc(strlen(cp) + 50);
429 sprintf(newdesc, cp, "PPP interface", try, j + 1);
430 deviceRegister("ppp0", newdesc, strdup(try), DEVICE_TYPE_NETWORK, TRUE, mediaInitNetwork,
431 NULL, mediaShutdownNetwork, NULL);
433 msgDebug("Add mapping for %s to ppp0\n", try);
443 /* Finally, go get the disks and look for DOS partitions to register */
444 if ((names = Disk_Names()) != NULL) {
447 for (i = 0; names[i]; i++) {
451 /* Ignore memory disks */
452 if (!strncmp(names[i], "md", 2))
457 * Due to unknown reasons, Disk_Names() returns SCSI CDROM as a
458 * valid disk. This is main reason why sysinstall presents SCSI
459 * CDROM to available disks in Fdisk/Label menu. In addition,
460 * adding a blank SCSI CDROM to the menu generates floating point
461 * exception in sparc64. Disk_Names() just extracts sysctl
462 * "kern.disks". Why GEOM treats SCSI CDROM as a disk is beyond
463 * me and that should be investigated.
464 * For temporary workaround, ignore SCSI CDROM device.
466 if (!strncmp(names[i], "cd", 2))
469 d = Open_Disk(names[i]);
471 msgDebug("Unable to open disk %s\n", names[i]);
475 deviceRegister(names[i], names[i], d->name, DEVICE_TYPE_DISK, FALSE,
476 dummyInit, dummyGet, dummyShutdown, d);
478 msgDebug("Found a disk device named %s\n", names[i]);
480 /* Look for existing DOS partitions to register as "DOS media devices" */
481 for (c1 = d->chunks->part; c1; c1 = c1->next) {
482 if (c1->type == fat || c1->type == efi || c1->type == extended) {
487 snprintf(devname, sizeof devname, "/dev/%s", c1->name);
488 dev = deviceRegister(c1->name, c1->name, strdup(devname), DEVICE_TYPE_DOS, TRUE,
489 mediaInitDOS, mediaGetDOS, mediaShutdownDOS, NULL);
492 msgDebug("Found a DOS partition %s on drive %s\n", c1->name, d->name);
498 dialog_clear_norefresh();
501 /* Rescan all devices, after closing previous set - convenience function */
510 * Find all devices that match the criteria, allowing "wildcarding" as well
511 * by allowing NULL or ANY values to match all. The array returned is static
512 * and may be used until the next invocation of deviceFind().
515 deviceFind(char *name, DeviceType class)
517 static Device *found[DEV_MAX];
521 for (i = 0; i < numDevs; i++) {
522 if ((!name || !strcmp(Devices[i]->name, name))
523 && (class == DEVICE_TYPE_ANY || class == Devices[i]->type))
524 found[j++] = Devices[i];
527 return j ? found : NULL;
531 deviceFindDescr(char *name, char *desc, DeviceType class)
533 static Device *found[DEV_MAX];
537 for (i = 0; i < numDevs; i++) {
538 if ((!name || !strcmp(Devices[i]->name, name)) &&
539 (!desc || !strcmp(Devices[i]->description, desc)) &&
540 (class == DEVICE_TYPE_ANY || class == Devices[i]->type))
541 found[j++] = Devices[i];
544 return j ? found : NULL;
548 deviceCount(Device **devs)
554 for (i = 0; devs[i]; i++);
559 * Create a menu listing all the devices of a certain type in the system.
560 * The passed-in menu is expected to be a "prototype" from which the new
564 deviceCreateMenu(DMenu *menu, DeviceType type, int (*hook)(dialogMenuItem *d), int (*check)(dialogMenuItem *d))
571 devs = deviceFind(NULL, type);
572 numdevs = deviceCount(devs);
575 tmp = (DMenu *)safe_malloc(sizeof(DMenu) + (sizeof(dialogMenuItem) * (numdevs + 1)));
576 bcopy(menu, tmp, sizeof(DMenu));
577 for (i = 0; devs[i]; i++) {
578 tmp->items[i].prompt = devs[i]->name;
579 for (j = 0; j < numDevs; j++) {
580 if (devs[i] == Devices[j]) {
581 tmp->items[i].title = Devices[j]->description;
586 tmp->items[i].title = "<unknown device type>";
587 tmp->items[i].fire = hook;
588 tmp->items[i].checked = check;
590 tmp->items[i].title = NULL;