Date: Thu, 12 Jul 2007 19:41:45 GMT From: Maxim Zhuravlev <thioretic@FreeBSD.org> To: Perforce Change Reviews <perforce@FreeBSD.org> Subject: PERFORCE change 123400 for review Message-ID: <200707121941.l6CJfjl6097058@repoman.freebsd.org>
next in thread | raw e-mail | index | archive | help
http://perforce.freebsd.org/chv.cgi?CH=123400 Change 123400 by thioretic@thioretic on 2007/07/12 19:41:00 TODO: bus_generic_* and resource_* stuff , cleanups, including proper naming :) (probably they should go first), locking and IO requests processing stuff. DONE: device_set_driver semantics. ALSO: my thoughts about drv_compat_ctrl_driver should now be more obvious. Affected files ... .. //depot/projects/soc2007/thioretic_gidl/TODO#12 edit .. //depot/projects/soc2007/thioretic_gidl/kern/subr_bus.c#13 edit .. //depot/projects/soc2007/thioretic_gidl/sys/bus.h#6 edit Differences ... ==== //depot/projects/soc2007/thioretic_gidl/TODO#12 (text+ko) ==== @@ -107,10 +107,6 @@ # a driver not just process, but also produce and destroy an io request. # Inportant note: from above-said one can see, that since now newbus # won't be a tree, it will be a graph! - # New logic: - # probe/attach: the dev->driver is set to drv_compat(driver) to be - # probed and attached. - # detach: will detach DRV_LOWEST driver and all filters consequently. # As for functional devices (let's call like this all drivers with # DR_REQDEV flag set), the following scenario looks reasonable: # @@ -132,4 +128,11 @@ # device_set_driver_softc(). # All these return softc of a specified driver for a device. Old functions # (those without '_driver_') return softc of DRV_LOWEST driver (as expected). - + a.1.5 device_set_driver semantics + SOLUTION: add DS_RAW state value && add per-pdrv_compat state + FILE(S) AFFECTED: kern/subr_bus.c, sys/bus.h + # As now device hosts many drivers, dev->driver should point to currently + # probed/attached/.. driver. If driver to be set to is DR_LOWEST not equal + # to that already in stack or is NULL, the whole driver configuration is + # destroyed. Device won't process any i/o requests till its state is set + # to DS_ALIVE (i.e. there is no DS_RAW or other drivers in stack). ==== //depot/projects/soc2007/thioretic_gidl/kern/subr_bus.c#13 (text+ko) ==== @@ -87,6 +87,7 @@ void *softc; #define DF_EXTERNALSOFTC 1 /* softc not allocated by us */ int flags; + device_state_t state; TAILQ_ENTRY(pdrv_compat) link; }; @@ -953,133 +954,181 @@ return((level>DR_LOWEST & level<DR_TOPMOST) ? TRUE : FALSE); } + /** * device control multiplexing entries */ +#define PREFIX drv_compat_ctrl + +#define FUNC(retval, prefix, name, ...) \ + static retval \ + prefix##_##name (__VA_ARGS__) + +#define FOR_RAW_DRIVER(func, ...) \ + return(func(__VA_ARGS__)); + +#define FOR_ALL_BUT_RAW_DRIVER (func, memb, ...) \ + int level; \ + pdrv_compat *pdc; \ + for (level = DRV_LOWEST; level <= DRV_TOPMOST; level++){ \ + if(TAILQ_EMPTY(&(dev->drivers[level]))) \ + continue; \ + TAILQ_FOREACH (pdc, &(dev->drivers[level]), link){ \ + func(pdc->pdriver->memb, __VA_ARGS__); \ + } \ + } + + +FUNC(int, PREFIX, probe, device_t dev){ + FOR_RAW_DRIVER(DEVICE_PROBE, dev->driver->pdriver->topology_ops); +} + +FUNC(int, PREFIX, identify, driver_t *driver, device_t parent){ + FOR_RAW_DRIVER(DEVICE_IDENTIFY, dev->driver->pdriver->driver, parent); +} + +FUNC(int, PREFIX, attach, device_t dev){ + FOR_RAW_DRIVER(DEVICE_ATTACH, dev->driver->pdriver->topology_ops); +} + +FUNC(int, PREFIX, detach, device_t dev){ + FOR_RAW_DRIVER(DEVICE_DETACH, dev->driver->pdriver->topology_ops); +} + +FUNC(int, PREFIX, shutdown, device_t parent){ + FOR_RAW_DRIVER(DEVICE_SHUTDOWN, dev->driver->pdriver->topology_ops); +} + +FUNC(int, PREFIX, suspend, device_t dev){ + FOR_RAW_DRIVER(DEVICE_SUSPEND, dev->driver->pdriver->topology_ops); +} + +FUNC(int, PREFIX, resume, device_t dev){ + FOR_RAW_DRIVER(DEVICE_RESUME, dev->driver->pdriver->topology_ops); +} + +FUNC(int, PREFIX, quiesce, device_t dev){ + FOR_RAW_DRIVER(DEVICE_QUIESCE, dev->driver->pdriver->topology_ops); +} -static int drv_compat_ctrl_probe (device_t dev); -static int drv_compat_ctrl_identify (driver_t *driver, device_t parent); -static int drv_compat_ctrl_attach (device_t dev); -static int drv_compat_ctrl_detach (device_t dev); -static int drv_compat_ctrl_shutdown (device_t dev); -static int drv_compat_ctrl_suspend (device_t dev); -static int drv_compat_ctrl_resume (device_t dev); -static int drv_compat_ctrl_quiesce (device_t dev); -static void* drv_compat_ctrl_info (device_t dev); +FUNC(void *, PREFIX, info, driver_t *driver, device_t parent){ + FOR_ALL_BUT_RAW_DRIVER(DEVICE_INFO, topology_ops); +} + /** * bus control multiplexing entries */ static int drv_compat_ctrl_print_child (device_t dev, - device_t child); + device_t child); static void drv_compat_ctrl_probe_nomatch (device_t dev, - device_t child); + device_t child); static int drv_compat_ctrl_read_ivar (device_t dev, - device_t child, - int index, - uintptr_t *result); + device_t child, + int index, + uintptr_t *result); static int drv_compat_ctrl_write_ivar (device_t dev, - device_t child, - int index, - uintptr_t value); + device_t child, + int index, + uintptr_t value); static void drv_compat_ctrl_child_detached (device_t dev, - device_t child); + device_t child); static void drv_compat_ctrl_driver_added (device_t dev, - driver_t *driver); + driver_t *driver); static device_t drv_compat_ctrl_add_child (device_t dev, - int order, - const char* name, - int unit); + int order, + const char* name, + int unit); static struct_resource * drv_compat_ctrl_alloc_resource (device_t dev, - device_t child, - int type, - int *rid, - u_long start, - u_long end, - u_long count, - u_int flags); + device_t child, + int type, + int *rid, + u_long start, + u_long end, + u_long count, + u_int flags); static int drv_compat_ctrl_activate_resource (device_t dev, - device_t child, - int type, - int rid, - struct resource *r); + device_t child, + int type, + int rid, + struct resource *r); static int drv_compat_ctrl_deactivate_resource (device_t dev, - device_t child, - int type, - int rid, - struct resource *r); + device_t child, + int type, + int rid, + struct resource *r); static int drv_compat_ctrl_release_resource (device_t dev, - device_t child, - int type, - int rid, - struct resource *res); + device_t child, + int type, + int rid, + struct resource *res); static int drv_compat_ctrl_setup_intr (device_t dev, - device_t child, - struct resource *irq, - int flags, - driver_intr_t *intr, - void *arg, - void **cookiep); + device_t child, + struct resource *irq, + int flags, + driver_intr_t *intr, + void *arg, + void **cookiep); static int drv_compat_ctrl_teardown_intr (device_t dev, - device_t child, - struct resource *irq, - void *cookie); + device_t child, + struct resource *irq, + void *cookie); static int drv_compat_ctrl_set_resource (device_t dev, - device_t child, - int type, - int rid, - u_long start, - u_long count); + device_t child, + int type, + int rid, + u_long start, + u_long count); static int drv_compat_ctrl_get_resource (device_t dev, - device_t child, - int type, - int rid, - u_long *startp, - u_long *countp); + device_t child, + int type, + int rid, + u_long *startp, + u_long *countp); static void drv_compat_ctrl_delete_resource (device_t dev, - device_t child, - int type, - int rid); + device_t child, + int type, + int rid); static struct resource_list * drv_compat_ctrl_get_resource_list (device_t dev, - device_t child); + device_t child); static int drv_compat_ctrl_child_present (device_t dev, - device_t child); + device_t child); static int drv_compat_ctrl_child_pnpinfo_str (device_t dev, - device_t child, - char *buf, - size_t buflen); + device_t child, + char *buf, + size_t buflen); static int -drv_compat_ctrl_child_location_str (device_t dev, - device_t child, - char *buf, - size_t buflen); +drv_compat_ctrl_child_location_str (device_t dev, + device_t child, + char *buf, + size_t buflen); static int -drv_compat_ctrl_config_intr (device_t dev, - int irq, - enum intr_trigger trig, - enum intr_polarity pol); +drv_compat_ctrl_config_intr (device_t dev, + int irq, + enum intr_trigger trig, + enum intr_polarity pol); static void -drv_compat_ctrl_hinted_child (device_t dev, - const char *dname, - int dunit); +drv_compat_ctrl_hinted_child (device_t dev, + const char *dname, + int dunit); static device_method_t drv_compat_ctrl_methods[] = { DEVMETHOD(device_probe, drv_compat_ctrl_probe), @@ -1889,7 +1938,7 @@ drv_compat_t drvc; int level; - device_set_driver (dev, dl->driver); + device_set_driver (dev, drvc->driver); if (DEVICE_PROBE(dev)<=0){ drvc = dev->driver; level = drv_compat_flags2idx(drvc->flags); @@ -2053,7 +2102,7 @@ } static int -device_add_to_dev_list (device_t addwhat, pdevice_list_t *addwhere){ +device_add_to_dev_list (device_t addwhat, pdevice_list_t *addwhere, int incr_refs){ pdevice *pd; TAILQ_FOREACH (pd, addwhere, link){ if (pd->device_ptr == addwhat) @@ -2066,7 +2115,8 @@ pd->device_ptr = addwhat; TAILQ_INSERT_TAIL(addwhere, pd, link); - pd->device_ptr->refs++; + if (incr_refs) + pd->device_ptr->refs++; bus_data_generation_update(); return (1); @@ -2079,7 +2129,8 @@ */ int device_add_existing_child (device_t dev, device_t child){ - device_add_to_dev_list (child, &(dev->children)); + device_add_to_dev_list (child, &(dev->children), TRUE); + device_add_to_dev_list (dev, &(child->parents), FALSE); } /** * @brief Add a new parent, which is a device, that already exists @@ -2088,7 +2139,8 @@ */ int device_add_existing_parent (device_t dev, device_t parent){ - device_add_to_dev_list (parent, &(dev->parents)); + device_add_to_dev_list (parent, &(dev->parents), FALSE); + device_add_to_dev_list (dev, &(parent->children), TRUE); } @@ -2149,6 +2201,8 @@ } child->device_ptr->refs++; + device_add_existing_parent (child->device_ptr, dev); + bus_data_generation_update(); return (child); } @@ -2162,27 +2216,32 @@ pdevice *grand; pdevice *pd; pdevice_list_t *list; + device_t child, parent; PDEBUG(("%s from %s", DEVICENAME(child), DEVICENAME(dev))); + if (direction==PARENTS) + return (destroy_recurse (devtodel, dev, CHILDREN)); + /* remove children first */ - list = (direction == CHILDREN) ? &devtodel->children : &devtodel->parents; - if (--devtodel->refs) goto deleteself; - while ( (grand = TAILQ_FIRST(list)) ) { + if (devtodel->refs>1) goto deletefromparents; + while ( (grand = TAILQ_FIRST(&devtodel->children)) ) { error = destroy_recurse(devtodel, grand->device_ptr, direction); if (error) return (error); + TAILQ_REMOVE (&devtodel->children, grand, link) free (grand); } - if ((error = device_detach(devtodel)) != 0) +/*!!!*/ if ((error = device_detach(devtodel)) != 0) return (error); if (devtodel->devclass) devclass_delete_device(devtodel->devclass, devtodel); -deletefromparent: - TAILQ_FOREACH_SAFE(pd, list, link){ - if (pd->device_ptr == devtodel){ - TAILQ_REMOVE(list, devtodel, link); +deletefromparents: + TAILQ_FOREACH_SAFE(pd, &devtodel->parents, link){ + if (pd->device_ptr == dev){ + TAILQ_REMOVE(&devtodel->parents, pd, link); + devtodel->refs--; free (pd); break; } @@ -2205,7 +2264,7 @@ } void -device_add_existing_parent (device_t dev, device_t parent){ +device_delete_existing_parent (device_t dev, device_t parent){ destroy_recurse (dev, parent, PARENTS); } @@ -2983,6 +3042,25 @@ return (error); } +static int +device_destroy_configuration (device_t dev){ + int level, error; + pdrv_compat *pdrvc; + + for (level = DRV_TOPMOST; level>= DRV_LOWEST; level--){ + if (TAILQ_EMPTY(&(dev->drivers[level]))) + continue; + while(pdrvc = TAILQ_FIRST(&(dev->drivers[level]))){ + pdrvc->state = DS_RAW; + if(error = device_detach(dev)) + return(error); + TAILQ_REMOVE(&(dev->drivers[level]), pdrvc, link); + free(pdrvc); + + } + } +} + /** * @brief Set the driver of a device * @@ -2994,30 +3072,42 @@ device_set_driver(device_t dev, driver_t *driver) /*TODO*/ { drv_compat_t drvc; - pdrv_compat *drvctmp; + pdrv_compat *pdrvc; uint32_t flags; int level; if (dev->state >= DS_ATTACHED) return (EBUSY); - drvc = drv_compat_find_driver (driver); + if (driver != NULL){ + drvc = drv_compat_find_driver (driver); + + drv_compat_get_flags(driver, &flags); + level = drv_compat_flags2idx(flags); - TAILQ_FOREACH(drvctmp, &(dev->drivers[level]), link){ - if (drvctmp->pdriver == drvc) - break; + TAILQ_FOREACH(pdrvc, &(dev->drivers[level]), link){ + if (pdrvc->pdriver == drvc) + break; + } + if (pdrvc){ + pdrvc->state = DS_RAW; + return (0); + } } - if (!drvctmp){ - + if(driver == NULL || level == DRV_LOWEST){ + if(!device_destroy_configuration(dev)) + return (EBUSY); } - dev->driver = drvctmp; - dev->driver_level = level; + pdrvc = malloc (sizeof(struct pdrv_compat), M_BUS, M_NOWAIT|M_ZERO); + pdrvc->pdriver = drvc; + pdrvc->state = DS_RAW; + + TAILQ_INSERT_TAIL (&(dev->drivers[level]), pdrvc); - /* - if (dev->driver == driver) - return (0); + dev->driver = pdrvc; +// dev->driver_level = level; if (dev->softc && !(dev->flags & DF_EXTERNALSOFTC)) { free(dev->softc, M_BUS_SC); @@ -3030,7 +3120,9 @@ if (!dev->softc) { //kobj_delete((kobj_t) dev, 0); //kobj_init((kobj_t) dev, &null_class); - dev->driver = NULL; + //dev->driver = NULL; + TAILQ_REMOVE(&(dev->drivers[level]), pdrvc, link); + free (pdrvc); return (ENOMEM); } } @@ -3141,9 +3233,10 @@ } else { drvc = dev->driver; dev->devclass = devclass_find (DRIVERNAME(drvc->driver)); - level = drv_compat_flags2idx(drvc->flags); - TAILQ_INSERT_TAIL (&((dev->drivers)[level]), &drvc, link); - dev->driver = NULL; + //level = drv_compat_flags2idx(drvc->flags); + + //TAILQ_INSERT_TAIL (&((dev->drivers)[level]), &drvc, link); + //dev->driver = NULL; } dev->state = DS_ATTACHED; devadded(dev); ==== //depot/projects/soc2007/thioretic_gidl/sys/bus.h#6 (text+ko) ==== @@ -47,6 +47,7 @@ * @brief State of the device. */ typedef enum device_state { + DS_RAW, DS_NOTPRESENT, /**< @brief not probed or probe failed */ DS_ALIVE, /**< @brief probe succeeded */ DS_ATTACHED, /**< @brief attach method called */
Want to link to this message? Use this URL: <https://mail-archive.FreeBSD.org/cgi/mid.cgi?200707121941.l6CJfjl6097058>