usb: Bind devices to ACPI devices when possible
authorMatthew Garrett <mjg@redhat.com>
Fri, 11 May 2012 08:08:27 +0000 (16:08 +0800)
committerGreg Kroah-Hartman <gregkh@linuxfoundation.org>
Sat, 12 May 2012 00:06:13 +0000 (17:06 -0700)
Built-in USB devices will typically have a representation in the system
ACPI tables. Add support for binding the two together so the USB code can
make use of the associated methods.

Signed-off-by: Matthew Garrett <mjg@redhat.com>
Signed-off-by: Lan Tianyu <tianyu.lan@intel.com>
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
drivers/usb/core/Makefile
drivers/usb/core/usb-acpi.c [new file with mode: 0644]
drivers/usb/core/usb.c
drivers/usb/core/usb.h

index c4ea846d3c0213b61069126975441a9077dfde41..26059b93dbf4bac36e72cb580adea681436ce88f 100644 (file)
@@ -9,5 +9,6 @@ usbcore-y += config.o file.o buffer.o sysfs.o endpoint.o
 usbcore-y += devio.o notify.o generic.o quirks.o devices.o
 
 usbcore-$(CONFIG_PCI)          += hcd-pci.o
+usbcore-$(CONFIG_ACPI)         += usb-acpi.o
 
 obj-$(CONFIG_USB)              += usbcore.o
diff --git a/drivers/usb/core/usb-acpi.c b/drivers/usb/core/usb-acpi.c
new file mode 100644 (file)
index 0000000..cab5cb7
--- /dev/null
@@ -0,0 +1,60 @@
+/*
+ * USB-ACPI glue code
+ *
+ * Copyright 2012 Red Hat <mjg@redhat.com>
+ *
+ * This program is free software; you can redistribute it and/or modify it
+ * under the terms of the GNU General Public License as published by the Free
+ * Software Foundation, version 2.
+ *
+ */
+#include <linux/module.h>
+#include <linux/usb.h>
+#include <linux/device.h>
+#include <linux/errno.h>
+#include <linux/kernel.h>
+#include <linux/acpi.h>
+#include <linux/pci.h>
+#include <acpi/acpi_bus.h>
+
+#include "usb.h"
+
+static int usb_acpi_find_device(struct device *dev, acpi_handle *handle)
+{
+       struct usb_device *udev;
+       struct device *parent;
+       acpi_handle *parent_handle;
+
+       if (!is_usb_device(dev))
+               return -ENODEV;
+
+       udev = to_usb_device(dev);
+       parent = dev->parent;
+       parent_handle = DEVICE_ACPI_HANDLE(parent);
+
+       if (!parent_handle)
+               return -ENODEV;
+
+       *handle = acpi_get_child(parent_handle, udev->portnum);
+
+       if (!*handle)
+               return -ENODEV;
+
+       return 0;
+}
+
+static struct acpi_bus_type usb_acpi_bus = {
+       .bus = &usb_bus_type,
+       .find_bridge = NULL,
+       .find_device = usb_acpi_find_device,
+};
+
+int usb_acpi_register(void)
+{
+       return register_acpi_bus_type(&usb_acpi_bus);
+}
+
+void usb_acpi_unregister(void)
+{
+       unregister_acpi_bus_type(&usb_acpi_bus);
+}
index 0ce862bfdd77af804551486fd9621315d40196f2..7998a67503c1f627ea6f79392bb63617fe3bbe55 100644 (file)
@@ -1015,6 +1015,9 @@ static int __init usb_init(void)
        if (retval)
                goto out;
 
+       retval = usb_acpi_register();
+       if (retval)
+               goto acpi_register_failed;
        retval = bus_register(&usb_bus_type);
        if (retval)
                goto bus_register_failed;
@@ -1049,6 +1052,8 @@ major_init_failed:
 bus_notifier_failed:
        bus_unregister(&usb_bus_type);
 bus_register_failed:
+       usb_acpi_unregister();
+acpi_register_failed:
        usb_debugfs_cleanup();
 out:
        return retval;
@@ -1070,6 +1075,7 @@ static void __exit usb_exit(void)
        usb_hub_cleanup();
        bus_unregister_notifier(&usb_bus_type, &usb_bus_nb);
        bus_unregister(&usb_bus_type);
+       usb_acpi_unregister();
        usb_debugfs_cleanup();
 }
 
index 71648dcbe4386a98cf5e571516b3186d3b3ba749..5c5c538ea73dbac77adbb462a027ac4d49d7396c 100644 (file)
@@ -156,3 +156,10 @@ extern void usb_notify_remove_device(struct usb_device *udev);
 extern void usb_notify_add_bus(struct usb_bus *ubus);
 extern void usb_notify_remove_bus(struct usb_bus *ubus);
 
+#ifdef CONFIG_ACPI
+extern int usb_acpi_register(void);
+extern void usb_acpi_unregister(void);
+#else
+static inline int usb_acpi_register(void) { return 0; };
+static inline void usb_acpi_unregister(void) { };
+#endif