/* * Guillaume Cottenceau (gc@mandrakesoft.com) * * Copyright 2000 MandrakeSoft * * This software may be freely redistributed under the terms of the GNU * public license. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. * */ /* * Portions from Erik Troan (ewt@redhat.com) * * Copyright 1996 Red Hat Software * */ /* * This contains stuff related to probing: * (1) any (actually only SCSI, NET, CPQ, USB Controllers) devices (autoprobe for PCI and USB) * (2) IDE media * (3) SCSI media * (4) ETH devices */ #include #include #include #include #include #include #include // #include #include #include #include #include #include "stage1.h" #include "log.h" #include "frontend.h" #include "modules.h" #include "probing.h" struct media_info { char * name; char * model; enum media_type type; }; struct pci_module_map { unsigned short vendor; /* PCI vendor id */ unsigned short device; /* PCI device id */ char * module; /* module to load */ struct pci_module_map * next; }; struct usb_module_map { unsigned short vendor; /* PCI vendor id */ unsigned short device; /* PCI device id */ char * module; /* module to load */ struct usb_module_map * next; }; char *usb_hcd[] = { "uhci-hcd", "ohci-hcd", "ehci-hcd", }; #define HCD_NUM (sizeof(usb_hcd) / sizeof(char *)) static void warning_insmod_failed(enum insmod_return r) { if (IS_AUTOMATIC && r == INSMOD_FAILED_FILE_NOT_FOUND) return; if (r != INSMOD_OK) { if (r == INSMOD_FAILED_FILE_NOT_FOUND) stg1_error_message("This floppy doesn't contain the driver."); else stg1_error_message("Warning, installation of driver failed. (please include msg from for bugreports)"); } } #ifndef DISABLE_NETWORK struct net_description_elem { char * intf_name; char * intf_description; }; static struct net_description_elem net_descriptions[50]; static int net_descr_number = 0; static char * intf_descr_for_discover = NULL; static char * net_intf_too_early_name[50]; /* for modules providing more than one net intf */ static int net_intf_too_early_number = 0; static int net_intf_too_early_ptr = 0; void prepare_intf_descr(const char * intf_descr) { intf_descr_for_discover = strdup(intf_descr); } void net_discovered_interface(char * intf_name) { if (!intf_descr_for_discover) { net_intf_too_early_name[net_intf_too_early_number++] = strdup(intf_name); return; } if (!intf_name) { if (net_intf_too_early_ptr >= net_intf_too_early_number) { log_message("NET: was expecting another network interface (broken net module?)"); return; } net_descriptions[net_descr_number].intf_name = net_intf_too_early_name[net_intf_too_early_ptr++]; } else net_descriptions[net_descr_number].intf_name = strdup(intf_name); net_descriptions[net_descr_number].intf_description = strdup(intf_descr_for_discover); intf_descr_for_discover = NULL; net_descr_number++; } char * get_net_intf_description(char * intf_name) { int i; for (i = 0; i < net_descr_number ; i++) if (!strcmp(net_descriptions[i].intf_name, intf_name)) return net_descriptions[i].intf_description; return strdup("unknown"); } #endif static struct pci_module_map * get_pci_ids() { static struct pci_module_map * pcidb = NULL; struct pci_module_map * last, *new; char buf[50]; int v, d; FILE *f; if (pcidb) return pcidb; log_message("loading pcimap file"); if (!(f = fopen("/modules/modules.map", "rb"))) { log_message("couldn't open pcimap file"); return NULL; } while (3 == (fscanf(f, "%x %x %48s", &v, &d, buf))) { new = (struct pci_module_map *)malloc(sizeof(*pcidb)); new->vendor = v; new->device = d; new->module = strdup(buf); new->next = NULL; if (!pcidb) { pcidb = last = new; } else { last->next = new; last = new; } } fclose(f); return pcidb; } /* forward */ static void probe_that_type(enum driver_type type); /* ---- PCI probe ---------------------------------------------- */ static void pci_probe(enum driver_type type) { FILE * f; int n; char buf[200]; char devname[22]; u_int8_t devdata[48]; static int need_usb_hcd[HCD_NUM]; struct pci_module_map * pci_ids = NULL; int that_class; switch (type) { case SCSI_ADAPTERS: that_class = PCI_CLASS_STORAGE_SCSI << 8; break; case IDE_ADAPTERS: that_class = PCI_CLASS_STORAGE_IDE << 8; break; case RAID_ADAPTERS: that_class = PCI_CLASS_STORAGE_RAID << 8; break; case NETWORK_DEVICES: that_class = PCI_CLASS_NETWORK_ETHERNET << 8; break; case BRIDGE_OTHER: that_class = PCI_CLASS_BRIDGE_OTHER << 8; break; case USB_CONTROLLERS: that_class = PCI_CLASS_SERIAL_USB << 8; break; default: return; } if (NULL == (pci_ids = get_pci_ids())) { log_message("PCI: could not get pci ids"); return; } if (!(f = fopen("/proc/bus/pci/devices", "r"))) { log_message("PCI: could not open proc file"); return; } while (NULL != fgets(buf, sizeof(buf), f)) { int i, matched, dfn, vendor, device, class, subv, subid; struct pci_module_map * pcidb; sscanf(buf, "%x %x", &dfn, &vendor); device = vendor & 0xFFFF; /* because scanf from dietlibc does not support %4f */ vendor = (vendor >> 16) & 0xFFFF; snprintf(devname, sizeof(devname), "/proc/bus/pci/%02x/%02x.%x", dfn >> 8, PCI_SLOT(dfn & 0xff), PCI_FUNC(dfn & 0xff)); log_message("gathering info for %s", devname); if ((i = open(devname, O_RDONLY)) != -1) { read(i, devdata, sizeof(devdata)); close(i); } else continue; class = devdata[9] | (devdata[10] << 8) | (devdata[11] << 16); subv = devdata[0x2c] | (devdata[0x2d] << 8); subid = devdata[0x2e] | (devdata[0x2f] << 8); if (that_class != (class & 0xffff00)) continue; log_message("found pci device: %04x %04x %06x %04x %04x", vendor, device, class, subv, subid); for (matched = 0, pcidb = pci_ids; pcidb; pcidb = pcidb->next) { if (pcidb->vendor == vendor && pcidb->device == device) { /* vendor & device matched */ log_message("(pcimap) module is \"%s\"", pcidb->module); #ifndef DISABLE_MEDIAS if (type == IDE_ADAPTERS || type == SCSI_ADAPTERS || type == RAID_ADAPTERS) { int wait_msg = 0; enum insmod_return failed; if (IS_AUTOMATIC) { wait_message("Loading driver for storage adapter: %s", pcidb->module); wait_msg = 1; } else stg1_info_message("About to load driver for storage adapter: %s", pcidb->module); failed = my_insmod(pcidb->module, type, NULL); if (wait_msg) remove_wait_message(); warning_insmod_failed(failed); } #endif /* DISABLE_MEDIAS */ #ifndef DISABLE_NETWORK if (type == NETWORK_DEVICES || type == BRIDGE_OTHER) { int wait_msg = 0; enum insmod_return failed; if (IS_AUTOMATIC) { wait_message("Loading driver for network device: %s", pcidb->module); wait_msg = 1; } else stg1_info_message("About to load driver for network device: %s", pcidb->module); prepare_intf_descr(pcidb->module); failed = my_insmod(pcidb->module, type, NULL); if (wait_msg) remove_wait_message(); warning_insmod_failed(failed); if (intf_descr_for_discover) /* for modules providing more than one net intf */ net_discovered_interface(NULL); } #endif /* DISABLE_NETWORK */ if (type == USB_CONTROLLERS) { /* found explicitly declared module */ for (i=0; i < HCD_NUM; i++) { if(ptr_begins_static_str(pcidb->module, usb_hcd[i])) { need_usb_hcd[i] = 1; break; } } } matched = 1; break; } } /* end of pcidb table */ #ifndef DISABLE_MEDIAS if (!matched && type == IDE_ADAPTERS) { /* probe ide-generic as last resort */ log_message("(guess) module is \"ide-generic\""); int wait_msg = 0; enum insmod_return failed; if (IS_AUTOMATIC) { wait_message("Loading driver for IDE adapter: ide-generic"); wait_msg = 1; } else stg1_info_message("About to load driver for IDE adapter: ide-generic"); failed = my_insmod("ide-generic", type, NULL); if (wait_msg) remove_wait_message(); warning_insmod_failed(failed); } #endif /* DISABLE_MEDIAS */ if (!matched && type == USB_CONTROLLERS && (class & 0xffff0f) == 0x0c0300) { /* no module found, trying to identify one by class: HCD: PCI Class: uhci-hcd 0x000c0300 ohci-hcd 0x000c0310 ehci-hcd 0x000c0320 */ log_message("(guess) module is \"%s\"", usb_hcd[(class & 0xf0)>>4]); need_usb_hcd[(class & 0xf0)>>4] = 1; } } /* end of this vendor & device */ fclose(f); /* load all usb controller modules now, starting from possible ehci-hcd */ /* to prevent case when old-timed module sitting on newer host controller */ if (type == USB_CONTROLLERS) { enum insmod_return failed; for (n=HCD_NUM-1; n >= 0; n--) { if (need_usb_hcd[n]) { /* do it AUTOMATIC -- for usb kbd case */ /* and ever silent wait_message("Loading driver for USB controller: %s", usb_hcd[n]); */ failed = my_insmod(usb_hcd[n], USB_CONTROLLERS, NULL); /* remove_wait_message(); */ warning_insmod_failed(failed); } } } } /* ---- USB probe ---------------------------------------------- */ static void usb_probe(enum driver_type type) { static int already_probed_usb_controllers = 0; static int already_mounted_usbdev = 0; static int already_probed_hid = 0; FILE * f; char buf[200]; static struct usb_module_map * usb_ids = NULL; switch (type) { #ifdef ENABLE_USBNET case NETWORK_DEVICES: #endif case MEDIA_ADAPTERS: case HID_DEVICES: break; default: return; } if (!already_probed_usb_controllers) { already_probed_usb_controllers = 1; probe_that_type(USB_CONTROLLERS); } if (!already_mounted_usbdev) { already_mounted_usbdev = 1; if (mount("/proc/bus/usb", "/proc/bus/usb", "usbfs", 0, NULL)) { log_message("USB: couldn't mount /proc/bus/usb"); return; } /* no need to show wait message -- we're doing this very first wait_message("Waiting for USB stuff to show up."); */ sleep(2); /* sucking background work */ /* remove_wait_message(); */ } /* dirty hacks */ if (type == MEDIA_ADAPTERS) { stg1_info_message("About to load driver for usb storage device: usb-storage"); my_insmod("usb-storage", ANY_DRIVER_TYPE, NULL); sleep(5); /* wait for dust settles down */ } #ifdef ENABLE_USBNET if (type == NETWORK_DEVICES) { stg1_info_message("About to load driver for usb network device: usbnet"); my_insmod("usbnet", ANY_DRIVER_TYPE, NULL); } #endif if (!(f = fopen("/proc/bus/usb/devices", "r"))) { log_message("USB: could not open proc file"); return; } if (type == HID_DEVICES && !already_probed_hid) { while (NULL != fgets(buf, sizeof(buf), f)) { if (strstr(buf, "Cls=03")) { my_insmod("usbhid", ANY_DRIVER_TYPE, NULL); already_probed_hid = 1; break; } } } fclose(f); } static void probe_that_type(enum driver_type type) { if (IS_EXPERT) { ask_insmod(type); return; } if (type == MEDIA_ADAPTERS) { update_splash(); pci_probe(IDE_ADAPTERS); update_splash(); pci_probe(SCSI_ADAPTERS); update_splash(); pci_probe(RAID_ADAPTERS); update_splash(); usb_probe(MEDIA_ADAPTERS); update_splash(); } else { update_splash(); pci_probe(type); update_splash(); usb_probe(type); update_splash(); } } #ifndef DISABLE_MEDIAS static struct media_info * medias = NULL; static void find_media(void) { FILE *f; DIR *dir = NULL; struct dirent *dirent = NULL; struct stat st; struct media_info tmp[50]; char path[SYSFS_PATH_MAX]; char buf[512]; char *s; int count = 0; if (!medias) probe_that_type(MEDIA_ADAPTERS); else free(medias); /* that does not free the strings, by the way */ if ((dir = opendir("/sys/block")) == NULL) { log_message("failed to open /sys/block directory"); return; } while ((dirent = readdir(dir)) != NULL) { if (!strcmp(dirent->d_name, ".") || !strcmp(dirent->d_name, "..")) continue; memset(path, 0, SYSFS_PATH_MAX); strcpy(path, "/sys/block/"); strcat(path, dirent->d_name); s = path + strlen(path); /* probe for scsi type */ strcat(path, "/device/type"); if (lstat(path, &st) == 0 && S_ISREG(st.st_mode)) { tmp[count].name = strdup(dirent->d_name); tmp[count].type = UNKNOWN_MEDIA; if ((f = fopen(path, "r")) != NULL) { int type; if (fscanf(f, "%d", &type)) { if (type == SCSI_TYPE_DISK) tmp[count].type = DISK; else if (type == SCSI_TYPE_ROM) tmp[count].type = CDROM; else if (type == SCSI_TYPE_TAPE) tmp[count].type = TAPE; } fclose(f); } *s = 0; strcat(path, "/device/model"); if ((f = fopen(path, "r")) != NULL) { if (fgets(buf, sizeof(buf), f)) { tmp[count].model = strdup(buf); } fclose(f); } else { tmp[count].model = strdup("(unknown)"); } log_message("SCSI/%d: %s is a %s", tmp[count].type, tmp[count].name, tmp[count].model); count++; continue; } /* assume ide */ *s = 0; strcat(s, "/device/media"); if (lstat(path, &st) == 0 && S_ISREG(st.st_mode)) { tmp[count].name = strdup(dirent->d_name); tmp[count].type = UNKNOWN_MEDIA; log_message("IDE: name: %s", dirent->d_name); if ((f = fopen(path, "r")) != NULL) { if (fgets(buf, sizeof(buf), f)) { log_message("IDE: type: %s", buf); if (!strncmp("disk", buf, 4)) tmp[count].type = DISK; else if (!strncmp("cdrom", buf, 5)) tmp[count].type = CDROM; else if (!strncmp("tape", buf, 4)) tmp[count].type = TAPE; else if (!strncmp("floppy", buf, 6)) tmp[count].type = FLOPPY; } fclose(f); } /* grab model */ strcpy(path, "/proc/ide/"); strcat(path, dirent->d_name); strcat(path, "/model"); if ((f = fopen(path, "r")) != NULL) { if (fgets(buf, sizeof(buf), f)) { tmp[count].model = strdup(buf); } fclose(f); } else { tmp[count].model = strdup("(none)"); } log_message("IDE/%d: %s is a %s", tmp[count].type, tmp[count].name, tmp[count].model); count++; } } closedir(dir); tmp[count].name = NULL; count++; medias = memdup(tmp, sizeof(struct media_info) * count); } /* Finds by media */ void get_medias(enum media_type media, char *** names, char *** models) { struct media_info * m; char * tmp_names[50]; char * tmp_models[50]; int count; find_media(); m = medias; count = 0; while (m && m->name) { if (m->type == media) { tmp_names[count] = strdup(m->name); tmp_models[count++] = strdup(m->model); } m++; } tmp_names[count] = NULL; tmp_models[count++] = NULL; *names = memdup(tmp_names, sizeof(char *) * count); *models = memdup(tmp_models, sizeof(char *) * count); } #endif /* DISABLE_MEDIAS */ void probe_hiddev() { usb_probe(HID_DEVICES); } #ifndef DISABLE_NETWORK int net_device_available(char * device) { struct ifreq req; int s; s = socket(AF_INET, SOCK_DGRAM, 0); if (s < 0) { log_perror(device); return 0; } strcpy(req.ifr_name, device); if (ioctl(s, SIOCGIFFLAGS, &req)) { /* if we can't get the flags, the networking device isn't available */ close(s); return 0; } close(s); return 1; } char ** get_net_devices(void) { char * devices[] = { "eth0", "eth1", "eth2", "eth3", "eth4", "eth5", "tr0", "plip0", "plip1", "plip2", "fddi0", #ifdef ENABLE_USBNET "usb0", "usb1", "usb2", "usb3", #endif NULL }; char ** ptr = devices; char * tmp[50]; int i = 0; static int already_probed = 0; if (!already_probed) { already_probed = 1; /* cut off loop brought by: probe_that_type => my_insmod => get_net_devices */ probe_that_type(NETWORK_DEVICES); /* for some chipsets having nic in it, i.e. nForcex */ probe_that_type(BRIDGE_OTHER); } while (ptr && *ptr) { if (net_device_available(*ptr)) tmp[i++] = strdup(*ptr); ptr++; } tmp[i++] = NULL; return memdup(tmp, sizeof(char *) * i); } #endif /* DISABLE_NETWORK */