From 1ec038fd23618828a34ff8341022c50f6f765373 Mon Sep 17 00:00:00 2001 From: Sergey Bolshakov Date: Wed, 15 Dec 2004 20:43:02 +0000 Subject: [PATCH] updated pci probing scheme: stop compile in pci ids form now, separate map file /modules/modules.map used instead and parsed at runtime. see related changes in mar module also. --- Makefile | 14 +---- probing.c | 142 +++++++++++++++++++++++++++++------------- probing.h | 9 +++ usb-resource/Makefile | 4 +- 4 files changed, 111 insertions(+), 58 deletions(-) diff --git a/Makefile b/Makefile index eb0ac5a..0e969d9 100644 --- a/Makefile +++ b/Makefile @@ -56,7 +56,6 @@ include Makefile.common STAGE1BINS = $(METHODS:%=stage1-%) BINS = init $(STAGE1BINS) -DIRS += pci-resource usb-resource DEFS = -DENABLE_USB ifneq ($(WITH_SHELL),) DEFS += -DSPAWN_SHELL @@ -137,12 +136,7 @@ endif endif -all: version.h dirs $(BINS) - -dirs: - @for n in . $(DIRS); do \ - [ "$$n" = "." ] || make -C $$n ;\ - done +all: version.h $(BINS) init: $(INITOBJS) $(INIT_LIBC) $(CC) -o $@ $^ $(LDFLAGS_INIT) @@ -180,11 +174,7 @@ version.h: /etc/altlinux-release echo \\#define VERSION \\"\2\\";echo \\#define DISTRIB_NAME \\"ALT Linux\2\1 \\(\3\\)\\"|' < $^ |sh > $@ clean: - @for n in $(DIRS); do \ - (cd $$n; make clean) \ - done - rm -f *.o .depend *.rdz *.img $(BINS) - rm -f version.h + rm -f *.o .depend $(BINS) version.h .depend: $(CPP) $(CFLAGS) -M $(ALLSRC) > .depend diff --git a/probing.c b/probing.c index 66d0403..87b1601 100644 --- a/probing.c +++ b/probing.c @@ -45,20 +45,38 @@ #include "log.h" #include "frontend.h" #include "modules.h" -#include "pci-resource/pci-ids.h" -#ifdef ENABLE_USB -#include "usb-resource/usb-ids.h" -#endif - #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; +}; + +#ifndef DISABLE_USB + +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 *)) +#endif static void warning_insmod_failed(enum insmod_return r) { @@ -120,6 +138,42 @@ char * get_net_intf_description(char * intf_name) } #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; +} + static void probe_that_type(enum driver_type type) { if (IS_EXPERT) { @@ -130,11 +184,13 @@ static void probe_that_type(enum driver_type type) /* ---- PCI probe ---------------------------------------------- */ { FILE * f; - int n, len = 0; + int n; char buf[200]; char devname[22]; u_int8_t devdata[48]; - struct pci_module_map * pcidb = NULL; + struct pci_module_map * pci_ids = NULL; + int that_class; + #ifdef ENABLE_USB static int need_usb_hcd[HCD_NUM]; #endif @@ -142,37 +198,38 @@ static void probe_that_type(enum driver_type type) switch (type) { #ifndef DISABLE_MEDIAS case SCSI_ADAPTERS: - pcidb = medias_pci_ids; - len = medias_num_ids; + that_class = PCI_CLASS_STORAGE_SCSI << 8; break; #endif #ifndef DISABLE_NETWORK #ifndef DISABLE_PCINET case NETWORK_DEVICES: - pcidb = network_pci_ids; - len = network_num_ids; + that_class = PCI_CLASS_NETWORK_ETHERNET << 8; break; #endif #endif #ifdef ENABLE_USB case USB_CONTROLLERS: - pcidb = usb_pci_ids; - len = usb_num_ids; + that_class = PCI_CLASS_SERIAL_USB << 8; break; #endif default: goto end_pci_probe; } - + + if (NULL == (pci_ids = get_pci_ids())) { + log_message("PCI: could not get pci ids"); + goto end_pci_probe; + } + if (!(f = fopen("/proc/bus/pci/devices", "rb"))) { log_message("PCI: could not open proc file"); goto end_pci_probe; } - log_message("PCI: XXX probing %d XXX", type); - while (1) { - int i, j, dfn, vendor, device, class, subv, subid; + int i, dfn, vendor, device, class, subv, subid; + struct pci_module_map * pcidb; if (!fgets(buf, sizeof(buf), f)) break; @@ -197,19 +254,20 @@ static void probe_that_type(enum driver_type type) log_message("found pci device: %04x %04x %06x %04x %04x", vendor, device, class, subv, subid); - for (i = 0; i < len; i++) { - if (pcidb[i].vendor == vendor && pcidb[i].device == device) { - log_message("module is \"%s\" (%s)", pcidb[i].name, pcidb[i].module); + for (pcidb = pci_ids; pcidb; pcidb = pcidb->next) { + if (that_class == (class & 0xffff00) && + pcidb->vendor == vendor && pcidb->device == device) { + log_message("module is \"%s\"", pcidb->module); #ifndef DISABLE_MEDIAS if (type == SCSI_ADAPTERS) { int wait_msg = 0; enum insmod_return failed; if (IS_AUTOMATIC) { - wait_message("Loading driver for SCSI adapter:\n \n%s", pcidb[i].name); + wait_message("Loading driver for SCSI adapter:\n \n%s", pcidb->module); wait_msg = 1; } else - stg1_info_message("About to load driver for SCSI adapter:\n \n%s", pcidb[i].name); - failed = my_insmod(pcidb[i].module, SCSI_ADAPTERS, NULL); + stg1_info_message("About to load driver for SCSI adapter:\n \n%s", pcidb->module); + failed = my_insmod(pcidb->module, SCSI_ADAPTERS, NULL); if (wait_msg) remove_wait_message(); warning_insmod_failed(failed); @@ -217,9 +275,9 @@ static void probe_that_type(enum driver_type type) #endif /* DISABLE_MEDIAS */ #ifndef DISABLE_NETWORK if (type == NETWORK_DEVICES) { - stg1_info_message("About to load driver for network device:\n \n%s", pcidb[i].name); - prepare_intf_descr(pcidb[i].name); - warning_insmod_failed(my_insmod(pcidb[i].module, NETWORK_DEVICES, NULL)); + stg1_info_message("About to load driver for network device:\n \n%s", pcidb->module); + prepare_intf_descr(pcidb->module); + warning_insmod_failed(my_insmod(pcidb->module, NETWORK_DEVICES, NULL)); if (intf_descr_for_discover) /* for modules providing more than one net intf */ net_discovered_interface(NULL); } @@ -227,10 +285,10 @@ static void probe_that_type(enum driver_type type) #ifdef ENABLE_USB if (type == USB_CONTROLLERS) { /* found explicitly declared module */ - for (j=0; j < HCD_NUM; j++) { - if(ptr_begins_static_str(pcidb[i].module, usb_hcd[j])) { - log_message("found explicit %s", usb_hcd[j]); - need_usb_hcd[j] = 1; + for (i=0; i < HCD_NUM; i++) { + if(ptr_begins_static_str(pcidb->module, usb_hcd[i])) { + log_message("found explicit %s", usb_hcd[i]); + need_usb_hcd[i] = 1; break; } } @@ -277,23 +335,20 @@ static void probe_that_type(enum driver_type type) static int already_mounted_usbdev = 0; FILE * f; - int len = 0; char buf[200]; - struct usb_module_map * usbdb = NULL; + static struct usb_module_map * usb_ids = NULL; switch (type) { #ifdef ENABLE_USBNET case NETWORK_DEVICES: /* usbdb = usbnet_usb_ids; - len = usbnet_usb_num_ids; */ break; #endif case SCSI_ADAPTERS: /* usbdb = usbstorage_usb_ids; - len = usbstorage_usb_num_ids; */ break; default: @@ -328,7 +383,8 @@ static void probe_that_type(enum driver_type type) } while (1) { - int i, vendor, id; + int vendor, device; + struct usb_module_map * usbdb; if (!fgets(buf, sizeof(buf), f)) break; @@ -337,17 +393,17 @@ static void probe_that_type(enum driver_type type) my_insmod("keybdev", ANY_DRIVER_TYPE, NULL); } - if (sscanf(buf, "P: Vendor=%x ProdID=%x", &vendor, &id) != 2) + if (sscanf(buf, "P: Vendor=%x ProdID=%x", &vendor, &device) != 2) continue; - for (i = 0; i < len; i++) { - if (usbdb[i].vendor == vendor && usbdb[i].id == id) { - log_message("USB: device %04x %04x is \"%s\" (%s)", vendor, id, usbdb[i].name, usbdb[i].module); + for (usbdb = usb_ids; usbdb; usbdb = usbdb->next) { + if (usbdb->vendor == vendor && usbdb->device == device) { + log_message("USB: device %04x %04x is \"%s\" (%s)", vendor, device, usbdb->module); #ifdef ENABLE_USBNET if (type == NETWORK_DEVICES) { - stg1_info_message("About to load driver for usb network device:\n \n%s", usbdb[i].name); - prepare_intf_descr(usbdb[i].name); - warning_insmod_failed(my_insmod(usbdb[i].module, NETWORK_DEVICES, NULL)); + stg1_info_message("About to load driver for usb network device:\n \n%s", usbdb->module); + prepare_intf_descr(usbdb->module); + warning_insmod_failed(my_insmod(usbdb->module, NETWORK_DEVICES, NULL)); if (intf_descr_for_discover) /* for modules providing more than one net intf */ net_discovered_interface(NULL); } diff --git a/probing.h b/probing.h index cd48062..c0adec0 100644 --- a/probing.h +++ b/probing.h @@ -40,4 +40,13 @@ void prepare_intf_descr(const char * intf_descr); #define PCI_FUNC(devfn) ((devfn) & 0x07) #endif +/* + extract from linux/pci_ids.h +*/ + +#define PCI_CLASS_STORAGE_SCSI 0x0100 +#define PCI_CLASS_NETWORK_ETHERNET 0x0200 +#define PCI_CLASS_SERIAL_FIREWIRE 0x0c00 +#define PCI_CLASS_SERIAL_USB 0x0c03 + #endif diff --git a/usb-resource/Makefile b/usb-resource/Makefile index 601bd40..404fdc7 100644 --- a/usb-resource/Makefile +++ b/usb-resource/Makefile @@ -15,11 +15,9 @@ # #***************************************************************************** -VPATH = $(MODDIR) - all: usb-ids.h -usb-ids.h: medias/modules.mar network/modules.mar update-usb-ids.sh +usb-ids.h: update-usb-ids.sh ./update-usb-ids.sh $^ > $@ || rm -f $@ clean: