On Thu, Feb 28, 2013 at 1:53 PM, Rafael J. Wysocki <rjw@xxxxxxx> wrote: > From: Rafael J. Wysocki <rafael.j.wysocki@xxxxxxxxx> > > USB uses the .find_bridge() callback from struct acpi_bus_type > incorrectly, because as a result of the way it is used by USB every > device in the system that doesn't have a bus type or parent is > passed to usb_acpi_find_device() for inspection. > > What USB actually needs, though, is to call usb_acpi_find_device() > for USB ports that don't have a bus type defined, but have > usb_port_device_type as their device type, as well as for USB > devices. > > To fix that replace the struct bus_type pointer in struct > acpi_bus_type used for matching devices to specific subsystems > with a .match() callback to be used for this purpose and update > the users of struct acpi_bus_type, including USB, accordingly. > Define the .match() callback routine for USB, usb_acpi_bus_match(), > in such a way that it will cover both USB devices and USB ports > and remove the now redundant .find_bridge() callback pointer from > usb_acpi_bus. > > Signed-off-by: Rafael J. Wysocki <rafael.j.wysocki@xxxxxxxxx> > --- > drivers/acpi/glue.c | 39 +++++++++++++-------------------------- > drivers/ata/libata-acpi.c | 1 + > drivers/pci/pci-acpi.c | 8 +++++++- > drivers/pnp/pnpacpi/core.c | 8 +++++++- > drivers/scsi/scsi_lib.c | 7 ++++++- > drivers/usb/core/usb-acpi.c | 9 +++++++-- > include/acpi/acpi_bus.h | 3 ++- > 7 files changed, 43 insertions(+), 32 deletions(-) > > Index: linux-pm/include/acpi/acpi_bus.h > =================================================================== > --- linux-pm.orig/include/acpi/acpi_bus.h > +++ linux-pm/include/acpi/acpi_bus.h > @@ -437,7 +437,8 @@ void acpi_remove_dir(struct acpi_device > */ > struct acpi_bus_type { > struct list_head list; > - struct bus_type *bus; > + const char *name; > + bool (*match)(struct device *dev); > /* For general devices under the bus */ > int (*find_device) (struct device *, acpi_handle *); > /* For bridges, such as PCI root bridge, IDE controller */ > Index: linux-pm/drivers/acpi/glue.c > =================================================================== > --- linux-pm.orig/drivers/acpi/glue.c > +++ linux-pm/drivers/acpi/glue.c > @@ -36,12 +36,11 @@ int register_acpi_bus_type(struct acpi_b > { > if (acpi_disabled) > return -ENODEV; > - if (type && type->bus && type->find_device) { > + if (type && type->match && type->find_device) { > down_write(&bus_type_sem); > list_add_tail(&type->list, &bus_type_list); > up_write(&bus_type_sem); > - printk(KERN_INFO PREFIX "bus type %s registered\n", > - type->bus->name); > + printk(KERN_INFO PREFIX "bus type %s registered\n", type->name); > return 0; > } > return -ENODEV; > @@ -56,24 +55,21 @@ int unregister_acpi_bus_type(struct acpi > down_write(&bus_type_sem); > list_del_init(&type->list); > up_write(&bus_type_sem); > - printk(KERN_INFO PREFIX "ACPI bus type %s unregistered\n", > - type->bus->name); > + printk(KERN_INFO PREFIX "bus type %s unregistered\n", > + type->name); > return 0; > } > return -ENODEV; > } > EXPORT_SYMBOL_GPL(unregister_acpi_bus_type); > > -static struct acpi_bus_type *acpi_get_bus_type(struct bus_type *type) > +static struct acpi_bus_type *acpi_get_bus_type(struct device *dev) > { > struct acpi_bus_type *tmp, *ret = NULL; > > - if (!type) > - return NULL; > - > down_read(&bus_type_sem); > list_for_each_entry(tmp, &bus_type_list, list) { > - if (tmp->bus == type) { > + if (tmp->match(dev)) { > ret = tmp; > break; > } > @@ -261,26 +257,17 @@ err: > > static int acpi_platform_notify(struct device *dev) > { > - struct acpi_bus_type *type; > + struct acpi_bus_type *type = acpi_get_bus_type(dev); > acpi_handle handle; > int ret; > > ret = acpi_bind_one(dev, NULL); > - if (ret && (!dev->bus || !dev->parent)) { > - /* bridge devices genernally haven't bus or parent */ > - ret = acpi_find_bridge_device(dev, &handle); > - if (!ret) { > - ret = acpi_bind_one(dev, handle); > - if (ret) > - goto out; > - } > - } > - > - type = acpi_get_bus_type(dev->bus); > if (ret) { > - if (!type || !type->find_device) { > - DBG("No ACPI bus support for %s\n", dev_name(dev)); > - ret = -EINVAL; > + if (!type) { > + ret = acpi_find_bridge_device(dev, &handle); > + if (!ret) > + ret = acpi_bind_one(dev, handle); > + > goto out; > } > > @@ -316,7 +303,7 @@ static int acpi_platform_notify_remove(s > { > struct acpi_bus_type *type; > > - type = acpi_get_bus_type(dev->bus); > + type = acpi_get_bus_type(dev); > if (type && type->cleanup) > type->cleanup(dev); > > Index: linux-pm/drivers/scsi/scsi_lib.c > =================================================================== > --- linux-pm.orig/drivers/scsi/scsi_lib.c > +++ linux-pm/drivers/scsi/scsi_lib.c > @@ -71,9 +71,14 @@ struct kmem_cache *scsi_sdb_cache; > #ifdef CONFIG_ACPI > #include <acpi/acpi_bus.h> > > +static bool acpi_scsi_bus_match(struct device *dev) > +{ > + return dev->bus == &scsi_bus_type; > +} > + > int scsi_register_acpi_bus_type(struct acpi_bus_type *bus) > { > - bus->bus = &scsi_bus_type; > + bus->match = acpi_scsi_bus_match; > return register_acpi_bus_type(bus); > } > EXPORT_SYMBOL_GPL(scsi_register_acpi_bus_type); > Index: linux-pm/drivers/pci/pci-acpi.c > =================================================================== > --- linux-pm.orig/drivers/pci/pci-acpi.c > +++ linux-pm/drivers/pci/pci-acpi.c > @@ -331,8 +331,14 @@ static void pci_acpi_cleanup(struct devi > } > } > > +static bool pci_acpi_bus_match(struct device *dev) > +{ > + return dev->bus == &pci_bus_type; > +} > + > static struct acpi_bus_type acpi_pci_bus = { > - .bus = &pci_bus_type, > + .name = "PCI", > + .match = pci_acpi_bus_match, > .find_device = acpi_pci_find_device, > .setup = pci_acpi_setup, > .cleanup = pci_acpi_cleanup, > Index: linux-pm/drivers/pnp/pnpacpi/core.c > =================================================================== > --- linux-pm.orig/drivers/pnp/pnpacpi/core.c > +++ linux-pm/drivers/pnp/pnpacpi/core.c > @@ -353,8 +353,14 @@ static int __init acpi_pnp_find_device(s > /* complete initialization of a PNPACPI device includes having > * pnpdev->dev.archdata.acpi_handle point to its ACPI sibling. > */ > +static bool acpi_pnp_bus_match(struct device *dev) > +{ > + return dev->bus == &pnp_bus_type; > +} > + > static struct acpi_bus_type __initdata acpi_pnp_bus = { > - .bus = &pnp_bus_type, > + .name = "PNP", > + .match = acpi_pnp_bus_match, > .find_device = acpi_pnp_find_device, > }; > > Index: linux-pm/drivers/usb/core/usb-acpi.c > =================================================================== > --- linux-pm.orig/drivers/usb/core/usb-acpi.c > +++ linux-pm/drivers/usb/core/usb-acpi.c > @@ -210,9 +210,14 @@ static int usb_acpi_find_device(struct d > return 0; > } > > +static bool usb_acpi_bus_match(struct device *dev) > +{ > + return is_usb_device(dev) || is_usb_port(dev); > +} > + > static struct acpi_bus_type usb_acpi_bus = { > - .bus = &usb_bus_type, > - .find_bridge = usb_acpi_find_device, > + .name = "USB", > + .match = usb_acpi_bus_match, > .find_device = usb_acpi_find_device, > }; yes, much clean. Don't need to add extra bus type for usb port device. For 1-2. Acked-by: Yinghai Lu <yinghai@xxxxxxxxxx> Thanks Yinghai > > Index: linux-pm/drivers/ata/libata-acpi.c > =================================================================== > --- linux-pm.orig/drivers/ata/libata-acpi.c > +++ linux-pm/drivers/ata/libata-acpi.c > @@ -1150,6 +1150,7 @@ static int ata_acpi_find_dummy(struct de > } > > static struct acpi_bus_type ata_acpi_bus = { > + .name = "ATA", > .find_bridge = ata_acpi_find_dummy, > .find_device = ata_acpi_find_device, > }; -- To unsubscribe from this list: send the line "unsubscribe linux-acpi" in the body of a message to majordomo@xxxxxxxxxxxxxxx More majordomo info at http://vger.kernel.org/majordomo-info.html