On Tuesday, January 29, 2013 04:27:28 PM Toshi Kani wrote: > On Wed, 2013-01-30 at 00:19 +0100, Rafael J. Wysocki wrote: > > On Tuesday, January 29, 2013 03:57:10 PM Toshi Kani wrote: > > > On Tue, 2013-01-29 at 22:32 +0100, Rafael J. Wysocki wrote: > > > > On Tuesday, January 29, 2013 07:50:43 AM Toshi Kani wrote: > > > > > On Tue, 2013-01-29 at 12:28 +0100, Rafael J. Wysocki wrote: > > > > > > On Monday, January 28, 2013 07:35:39 PM Toshi Kani wrote: > > > > > > > On Mon, 2013-01-28 at 13:59 +0100, Rafael J. Wysocki wrote: > > > > > > > > From: Rafael J. Wysocki <rafael.j.wysocki@xxxxxxxxx> > > > > > > > > > > > > > > > > Introduce struct acpi_scan_handler for representing objects that > > > > > > > > will do configuration tasks depending on ACPI device nodes' > > > > > > > > hardware IDs (HIDs). > > > > > > > > > > > > > > > > Currently, those tasks are done either directly by the ACPI namespace > > > > > > > > scanning code or by ACPI device drivers designed specifically for > > > > > > > > this purpose. None of the above is desirable, however, because > > > > > > > > doing that directly in the namespace scanning code makes that code > > > > > > > > overly complicated and difficult to follow and doing that in > > > > > > > > "special" device drivers leads to a great deal of confusion about > > > > > > > > their role and to confusing interactions with the driver core (for > > > > > > > > example, sysfs directories are created for those drivers, but they > > > > > > > > are completely unnecessary and only increase the kernel's memory > > > > > > > > footprint in vain). > > > > > > > > > > > > > > > > Signed-off-by: Rafael J. Wysocki <rafael.j.wysocki@xxxxxxxxx> > > > > > > > > --- > > > > > > > > Documentation/acpi/scan_handlers.txt | 77 +++++++++++++++++++++++++++++++++++ > > > > > > > > drivers/acpi/scan.c | 60 ++++++++++++++++++++++++--- > > > > > > > > include/acpi/acpi_bus.h | 14 ++++++ > > > > > > > > 3 files changed, 144 insertions(+), 7 deletions(-) > > > > > > > > > > > > > > > > Index: test/include/acpi/acpi_bus.h > > > > > > > > =================================================================== > > > > > > > > --- test.orig/include/acpi/acpi_bus.h > > > > > > > > +++ test/include/acpi/acpi_bus.h > > > > > > > > @@ -84,6 +84,18 @@ struct acpi_driver; > > > > > > > > struct acpi_device; > > > > > > > > > > > > > > > > /* > > > > > > > > + * ACPI Scan Handler > > > > > > > > + * ----------------- > > > > > > > > + */ > > > > > > > > + > > > > > > > > +struct acpi_scan_handler { > > > > > > > > + const struct acpi_device_id *ids; > > > > > > > > + struct list_head list_node; > > > > > > > > + int (*attach)(struct acpi_device *dev, const struct acpi_device_id *id); > > > > > > > > + void (*detach)(struct acpi_device *dev); > > > > > > > > +}; > > > > > > > > + > > > > > > > > +/* > > > > > > > > * ACPI Driver > > > > > > > > * ----------- > > > > > > > > */ > > > > > > > > @@ -269,6 +281,7 @@ struct acpi_device { > > > > > > > > struct acpi_device_wakeup wakeup; > > > > > > > > struct acpi_device_perf performance; > > > > > > > > struct acpi_device_dir dir; > > > > > > > > + struct acpi_scan_handler *handler; > > > > > > > > struct acpi_driver *driver; > > > > > > > > void *driver_data; > > > > > > > > struct device dev; > > > > > > > > @@ -382,6 +395,7 @@ int acpi_bus_receive_event(struct acpi_b > > > > > > > > static inline int acpi_bus_generate_proc_event(struct acpi_device *device, u8 type, int data) > > > > > > > > { return 0; } > > > > > > > > #endif > > > > > > > > +int acpi_scan_add_handler(struct acpi_scan_handler *handler); > > > > > > > > int acpi_bus_register_driver(struct acpi_driver *driver); > > > > > > > > void acpi_bus_unregister_driver(struct acpi_driver *driver); > > > > > > > > int acpi_bus_scan(acpi_handle handle); > > > > > > > > Index: test/drivers/acpi/scan.c > > > > > > > > =================================================================== > > > > > > > > --- test.orig/drivers/acpi/scan.c > > > > > > > > +++ test/drivers/acpi/scan.c > > > > > > > > @@ -53,6 +53,7 @@ static const struct acpi_device_id acpi_ > > > > > > > > static LIST_HEAD(acpi_device_list); > > > > > > > > static LIST_HEAD(acpi_bus_id_list); > > > > > > > > static DEFINE_MUTEX(acpi_scan_lock); > > > > > > > > +static LIST_HEAD(acpi_scan_handlers_list); > > > > > > > > DEFINE_MUTEX(acpi_device_lock); > > > > > > > > LIST_HEAD(acpi_wakeup_device_list); > > > > > > > > > > > > > > > > @@ -62,6 +63,15 @@ struct acpi_device_bus_id{ > > > > > > > > struct list_head node; > > > > > > > > }; > > > > > > > > > > > > > > > > +int acpi_scan_add_handler(struct acpi_scan_handler *handler) > > > > > > > > +{ > > > > > > > > + if (!handler || !handler->attach) > > > > > > > > + return -EINVAL; > > > > > > > > + > > > > > > > > + list_add_tail(&handler->list_node, &acpi_scan_handlers_list); > > > > > > > > + return 0; > > > > > > > > +} > > > > > > > > + > > > > > > > > /* > > > > > > > > * Creates hid/cid(s) string needed for modalias and uevent > > > > > > > > * e.g. on a device with hid:IBM0001 and cid:ACPI0001 you get: > > > > > > > > @@ -1570,20 +1580,42 @@ static acpi_status acpi_bus_check_add(ac > > > > > > > > return AE_OK; > > > > > > > > } > > > > > > > > > > > > > > > > +static int acpi_scan_attach_handler(struct acpi_device *device) > > > > > > > > +{ > > > > > > > > + struct acpi_scan_handler *handler; > > > > > > > > + int ret = 0; > > > > > > > > + > > > > > > > > + list_for_each_entry(handler, &acpi_scan_handlers_list, list_node) { > > > > > > > > + const struct acpi_device_id *id; > > > > > > > > + > > > > > > > > + id = __acpi_match_device(device, handler->ids); > > > > > > > > + if (!id) > > > > > > > > + continue; > > > > > > > > + > > > > > > > > + ret = handler->attach(device, id); > > > > > > > > + if (ret > 0) { > > > > > > > > + device->handler = handler; > > > > > > > > + break; > > > > > > > > + } else if (ret < 0) { > > > > > > > > + break; > > > > > > > > + } > > > > > > > > + } > > > > > > > > + return ret; > > > > > > > > +} > > > > > > > > > > > > > > Now that we have full control over the attach logic, it would be great > > > > > > > if we can update it to match with the ACPI spec -- HID has priority over > > > > > > > CIDs, and CIDs are also listed in their priority. For example, Device-X > > > > > > > has HID and CID. In this case, this Device-X should be attached to > > > > > > > Handler-A since it supports HID. The current logic simply chooses a > > > > > > > handler whichever registered before. > > > > > > > > > > > > > > Device-X: HID PNPID-A, CID PNPID-B > > > > > > > Handler-A: PNPID-A > > > > > > > Handler-B: PNPID-B > > > > > > > > > > > > > > So, the attach logic should be something like: > > > > > > > > > > > > > > list_for_each_entry(hwid, acpi_device->pnp.ids,) { > > > > > > > list_for_each_entry(,&acpi_scan_handlers_list,) > > > > > > > check if this handler supports a given hwid > > > > > > > } > > > > > > > > > > > > OK, I see the problem, but I think it's better to address it in a separate > > > > > > patch on top of the current series. > > > > > > > > > > I agree. > > > > > > > > > > > I'm not sure what approach is best, though. Do you think there should be two > > > > > > passes the first of which will check HIDs only and the second one will check > > > > > > CIDs? Or do you have something different in mind? > > > > > > > > > > HID and CIDs are already listed in their priority order in > > > > > acpi_device->pnp.ids. So, the single pass like I described above should > > > > > work. > > > > > > > > Well, I'm not sure if I understand you correctly. > > > > > > > > The device is given and we need to find a handler for it. So, it looks like > > > > we first should check if any handler matches the HID. This has to be a pass > > > > through all handlers. If there's no match, we need to check if any handler > > > > matches any of the device IDs. That will be the second pass, won't it? > > > > > > acpi_device->pnp.ids has a list of HID->CID[0]->CID[1]..CID[n] since > > > acpi_device_set_id() checks HID before CIDs. So, the first pass is to > > > check with the first entry of pnp.ids. If no handler is found, then > > > check the second entry, and so on. > > > > I'd prefer to check HID and then all CIDs at once in accordance with the > > following rule: > > 1) Use the first handler that matches HID exactly. If there's none: > > 2) Use the first handler compatible with the given device. > > What do you mean by all CIDs at once? CIDs also have priority when > there are multiple CIDs. That is, CID[0] takes priority over CID[1]. > So, going down the pnp.ids list handles the ordering correctly. OK, I see what you mean now. > > > > The difficulty is that the first item in pnp.ids need not be *the* HID. > > > > It only will be the HID if ACPI_VALID_HID is set in the device info in > > > > acpi_device_set_id(). So perhaps we need to add a hid_valid bit in > > > > device->flags and only do the "HID pass" if that is set? > > > > > > I do not think such bit is necessary for this. _HID is required (unless > > > it has _ADR), but _CID is optional. So, the valid cases are that a > > > device object has HID only, has both HID and CID(s), or has none of them > > > (i.e. _ADR device). If there is a device with CID only, this is a FW > > > bug and we just have to check with CID then. > > > > In addition to that there's a number of objects that we assign artificial HIDs. > > I think we should treat them as CIDs rather than as real HIDs in this regard. > > When an object does not support the _HID/_CID combination, it means that > this type of objects can only support a single characteristic, which is > the same as a device object with HID only. Therefore, an artificial ID > should be considered as an HID. I'm not sure about that, but it shouldn't really matter a lot anyway. :-) Thanks, Rafael -- I speak only for myself. Rafael J. Wysocki, Intel Open Source Technology Center. -- 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