ACPIPNP devices may have two levels of ID - the HID (a single string defining the hardware) and the CIDs (zero or more strings defining interfaces compatible with the hardware). If a driver matching a CID is bound first, it will be impossible to bind a driver that matches the HID despite it being more specific. This has been seen in the wild with platforms that define an IPMI device as:
Device (NIPM) { Name (_HID, EisaId ("IPI0001")) Name (_CID, EisaId ("PNP0C01")) resulting in the device being claimed by the generic motherboard resource driver binding to the PNP0C01 CID. The IPMI driver attempts to bind later and fails, preventing the IPMI device from being made available to the ACPI layer. This can be avoided at driver probe time by detaching the existing driver if the new driver matches the device HID. Since each device can only have a single HID this will only permit more specific drivers to dislodge more generic drivers. Signed-off-by: Matthew Garrett <m...@redhat.com> --- drivers/pnp/driver.c | 42 +++++++++++++++++++++++++++++++++++++++--- 1 file changed, 39 insertions(+), 3 deletions(-) diff --git a/drivers/pnp/driver.c b/drivers/pnp/driver.c index 00e9403..8d22836 100644 --- a/drivers/pnp/driver.c +++ b/drivers/pnp/driver.c @@ -25,6 +25,14 @@ static int compare_func(const char *ida, const char *idb) return 1; } +static int compare_single_pnp_id(const char *ida, const char *idb) +{ + if (memcmp(ida, idb, 3) == 0) + if (compare_func(ida, idb) == 1) + return 1; + return 0; +} + int compare_pnp_id(struct pnp_id *pos, const char *id) { if (!pos || !id || (strlen(id) != 7)) @@ -32,9 +40,8 @@ int compare_pnp_id(struct pnp_id *pos, const char *id) if (memcmp(id, "ANYDEVS", 7) == 0) return 1; while (pos) { - if (memcmp(pos->id, id, 3) == 0) - if (compare_func(pos->id, id) == 1) - return 1; + if (compare_single_pnp_id(pos->id, id) == 1) + return 1; pos = pos->next; } return 0; @@ -56,6 +63,22 @@ static const struct pnp_device_id *match_device(struct pnp_driver *drv, return NULL; } +static int match_hid(struct pnp_driver *drv, struct pnp_dev *dev) +{ + const struct pnp_device_id *drv_id = drv->id_table; + + if (!drv_id) + return 0; + + while (&drv_id->id) { + /* The first ID is _HID */ + if (compare_single_pnp_id(dev->id->id, drv_id->id)) + return 1; + drv_id++; + } + return 0; +} + int pnp_device_attach(struct pnp_dev *pnp_dev) { spin_lock(&pnp_lock); @@ -151,6 +174,19 @@ static int pnp_bus_match(struct device *dev, struct device_driver *drv) if (match_device(pnp_drv, pnp_dev) == NULL) return 0; + + /* + * ACPIPNP offers two levels of device ID - HID and CID. HID defines + * the specific device ID while CID represents the device + * compatibility IDs. If a device is matched by a compatibility ID + * first, it will be impossible for a hardware-specific driver to + * bind since there will already be a driver. We can handle this case + * by unbinding the original driver if the device has one bound and + * if the new driver matches the HID rather than a compatibility ID. + */ + if (dev->driver && match_hid(pnp_drv, pnp_dev)) + device_release_driver(dev); + return 1; } -- 1.7.11.4 -- To unsubscribe from this list: send the line "unsubscribe linux-kernel" in the body of a message to majord...@vger.kernel.org More majordomo info at http://vger.kernel.org/majordomo-info.html Please read the FAQ at http://www.tux.org/lkml/