Date: Thu, 16 Nov 2006 03:20:39 GMT From: Marcel Moolenaar <marcel@FreeBSD.org> To: Perforce Change Reviews <perforce@freebsd.org> Subject: PERFORCE change 110092 for review Message-ID: <200611160320.kAG3Kdlg058147@repoman.freebsd.org>
next in thread | raw e-mail | index | archive | help
http://perforce.freebsd.org/chv.cgi?CH=110092 Change 110092 by marcel@marcel_nfs on 2006/11/16 03:19:58 Declare the classes as weak. This allows device driver classes to be compiled-out without causing unresolved symbols. Implement the DT tag fully. Affected files ... .. //depot/projects/uart/dev/uart/uart.h#12 edit .. //depot/projects/uart/dev/uart/uart_core.c#56 edit .. //depot/projects/uart/dev/uart/uart_cpu.h#21 edit .. //depot/projects/uart/dev/uart/uart_cpu_amd64.c#12 edit .. //depot/projects/uart/dev/uart/uart_cpu_i386.c#13 edit .. //depot/projects/uart/dev/uart/uart_cpu_ia64.c#14 edit .. //depot/projects/uart/dev/uart/uart_cpu_pc98.c#14 edit .. //depot/projects/uart/dev/uart/uart_cpu_powerpc.c#3 edit .. //depot/projects/uart/dev/uart/uart_subr.c#8 edit Differences ... ==== //depot/projects/uart/dev/uart/uart.h#12 (text+ko) ==== @@ -64,10 +64,9 @@ */ struct uart_class; -extern struct uart_class uart_ns8250_class; -extern struct uart_class uart_quicc_class; -extern struct uart_class uart_sab82532_class; -extern struct uart_class uart_z8530_class; +extern struct uart_class uart_ns8250_class __attribute__((weak)); +extern struct uart_class uart_sab82532_class __attribute__((weak)); +extern struct uart_class uart_z8530_class __attribute__((weak)); /* * Device flags. ==== //depot/projects/uart/dev/uart/uart_core.c#56 (text+ko) ==== @@ -70,16 +70,22 @@ SLIST_INSERT_HEAD(&uart_sysdevs, di, next); } +const char * +uart_getname(struct uart_class *uc) +{ + return ((uc != NULL) ? uc->name : NULL); +} + struct uart_ops * uart_getops(struct uart_class *uc) { - return (uc->uc_ops); + return ((uc != NULL) ? uc->uc_ops : NULL); } int uart_getrange(struct uart_class *uc) { - return (uc->uc_range); + return ((uc != NULL) ? uc->uc_range : 0); } /* @@ -303,18 +309,26 @@ struct uart_devinfo *sysdev; int error; + sc = device_get_softc(dev); + /* + * All uart_class references are weak. Check that the needed + * class has been compiled-in. Fail if not. + */ + if (sc->sc_class == NULL) + return (ENXIO); + + /* * Initialize the instance. Note that the instance (=softc) does * not necessarily match the hardware specific softc. We can't do * anything about it now, because we may not attach to the device. * Hardware drivers cannot use any of the class specific fields * while probing. */ - sc = device_get_softc(dev); kobj_init((kobj_t)sc, (kobj_class_t)sc->sc_class); sc->sc_dev = dev; if (device_get_desc(dev) == NULL) - device_set_desc(dev, sc->sc_class->name); + device_set_desc(dev, uart_getname(sc->sc_class)); /* * Allocate the register resource. We assume that all UARTs have @@ -326,12 +340,13 @@ sc->sc_rrid = rid; sc->sc_rtype = SYS_RES_IOPORT; sc->sc_rres = bus_alloc_resource(dev, sc->sc_rtype, &sc->sc_rrid, - 0, ~0, sc->sc_class->uc_range, RF_ACTIVE); + 0, ~0, uart_getrange(sc->sc_class), RF_ACTIVE); if (sc->sc_rres == NULL) { sc->sc_rrid = rid; sc->sc_rtype = SYS_RES_MEMORY; sc->sc_rres = bus_alloc_resource(dev, sc->sc_rtype, - &sc->sc_rrid, 0, ~0, sc->sc_class->uc_range, RF_ACTIVE); + &sc->sc_rrid, 0, ~0, uart_getrange(sc->sc_class), + RF_ACTIVE); if (sc->sc_rres == NULL) return (ENXIO); } @@ -400,7 +415,7 @@ * collected by uart_bus_probe() intact. */ sc->sc_rres = bus_alloc_resource(dev, sc->sc_rtype, &sc->sc_rrid, - 0, ~0, sc->sc_class->uc_range, RF_ACTIVE); + 0, ~0, uart_getrange(sc->sc_class), RF_ACTIVE); if (sc->sc_rres == NULL) { mtx_destroy(&sc->sc_hwmtx_s); return (ENXIO); ==== //depot/projects/uart/dev/uart/uart_cpu.h#21 (text+ko) ==== @@ -74,6 +74,7 @@ int uart_cpu_getdev(int, struct uart_devinfo *); int uart_getenv(int, struct uart_devinfo *, struct uart_class *); +const char *uart_getname(struct uart_class *); struct uart_ops *uart_getops(struct uart_class *); int uart_getrange(struct uart_class *); ==== //depot/projects/uart/dev/uart/uart_cpu_amd64.c#12 (text+ko) ==== @@ -53,6 +53,8 @@ unsigned int i, ivar; class = &uart_ns8250_class; + if (class == NULL) + return (ENXIO); /* Check the environment. */ if (uart_getenv(devtype, di, class) == 0) ==== //depot/projects/uart/dev/uart/uart_cpu_i386.c#13 (text+ko) ==== @@ -53,6 +53,8 @@ unsigned int i, ivar; class = &uart_ns8250_class; + if (class == NULL) + return (ENXIO); /* Check the environment. */ if (uart_getenv(devtype, di, class) == 0) ==== //depot/projects/uart/dev/uart/uart_cpu_ia64.c#14 (text+ko) ==== @@ -65,6 +65,8 @@ unsigned int i; class = &uart_ns8250_class; + if (class == NULL) + return (ENXIO); /* * Use the DIG64 HCDP table if present. ==== //depot/projects/uart/dev/uart/uart_cpu_pc98.c#14 (text+ko) ==== @@ -53,6 +53,8 @@ unsigned int i, ivar, flags; class = &uart_ns8250_class; + if (class == NULL) + return (ENXIO); /* Check the environment. */ if (uart_getenv(devtype, di, class) == 0) ==== //depot/projects/uart/dev/uart/uart_cpu_powerpc.c#3 (text) ==== @@ -52,9 +52,14 @@ uart_cpu_getdev(int devtype, struct uart_devinfo *di) { char buf[64]; + struct uart_class *class; phandle_t input, opts; int error; + class = &uart_z8530_class; + if (class == NULL) + return (ENXIO); + if ((opts = OF_finddevice("/options")) == -1) return (ENXIO); switch (devtype) { @@ -93,7 +98,7 @@ if (error) return (error); - di->ops = uart_z8530_ops; + di->ops = uart_getops(class); di->bas.rclk = 230400; di->bas.chan = 1; ==== //depot/projects/uart/dev/uart/uart_subr.c#8 (text+ko) ==== @@ -48,6 +48,13 @@ #define UART_TAG_SB 8 #define UART_TAG_XO 9 +static struct uart_class *uart_classes[] = { + &uart_ns8250_class, + &uart_sab82532_class, + &uart_z8530_class, +}; +static size_t uart_nclasses = sizeof(uart_classes) / sizeof(uart_classes[0]); + static bus_addr_t uart_parse_addr(__const char **p) { @@ -57,6 +64,22 @@ static struct uart_class * uart_parse_class(struct uart_class *class, __const char **p) { + struct uart_class *uc; + const char *nm; + size_t len; + u_int i; + + for (i = 0; i < uart_nclasses; i++) { + uc = uart_classes[i]; + nm = uart_getname(uc); + if (nm == NULL || *nm == '\0') + continue; + len = strlen(nm); + if (strncmp(nm, *p, len) == 0) { + *p += len; + return (uc); + } + } return (class); } @@ -173,9 +196,12 @@ bus_addr_t addr = ~0U; int error; - /* We must get a valid default device class. */ + /* + * All uart_class references are weak. Make sure the default + * device class has been compiled-in. + */ if (class == NULL) - return (EDOOFUS); + return (ENXIO); /* * Check the environment variables "hw.uart.console" and
Want to link to this message? Use this URL: <https://mail-archive.FreeBSD.org/cgi/mid.cgi?200611160320.kAG3Kdlg058147>