On Thursday, November 08, 2012 01:23:44 PM Toshi Kani wrote: > Added a new .sys_notify interface, which allows ACPI drivers to > register their system-level (ex. hotplug) notify handlers through > their acpi_driver table. This removes redundant ACPI namespace > walks from ACPI drivers for faster booting. > > The global notify handler acpi_bus_notify() is called for all > system-level ACPI notifications, which then calls an appropriate > driver's handler if any. ACPI drivers no longer need to register > or unregister driver's handler to each ACPI device object. It also > supports dynamic ACPI namespace with LoadTable & Unload opcode > without any modification in ACPI drivers. > > Added a common system notify handler acpi_bus_sys_notify(), which > allows ACPI drivers to set it to .sys_notify when this function is > fully implemented. I don't really understand this. > It removes functional conflict between driver's > notify handler and the global notify handler acpi_bus_notify(). > > Note that the changes maintain backward compatibility for ACPI > drivers. Any drivers registered their hotplug handler through the > existing interfaces, such as acpi_install_notify_handler() and > register_acpi_bus_notifier(), will continue to work as before. I really wouldn't like to add new callbacks to struct acpi_device_ops, because I'd like that whole thing to go away entirely eventually, along with struct acpi_driver. Moreover, in this particular case, it really is not useful to have to define a struct acpi_driver so that one can register for receiving system notifications from ACPI. It would be really nice if non-ACPI drivers, such as PCI or platform, could do that too. Besides, acpi_os_execute_deferred() is always run on CPU0, because of some SMI-related peculiarity, which is not very efficient as far as the events handling is concerned, but we can improve the situation a bit by queing the execution of the registered handlers in a different workqueue. Maybe it's worth considering if we're going to change this code anyway? > Signed-off-by: Toshi Kani <toshi.kani@xxxxxx> > --- > drivers/acpi/bus.c | 64 ++++++++++++++++++++++++++++---------- > drivers/acpi/scan.c | 83 +++++++++++++++++++++++++++++++++++++++++++++++++ > include/acpi/acpi_bus.h | 6 ++++ > 3 files changed, 137 insertions(+), 16 deletions(-) > > diff --git a/drivers/acpi/bus.c b/drivers/acpi/bus.c > index 07a20ee..b256bcf2 100644 > --- a/drivers/acpi/bus.c > +++ b/drivers/acpi/bus.c > @@ -779,21 +779,16 @@ void unregister_acpi_bus_notifier(struct notifier_block *nb) > EXPORT_SYMBOL_GPL(unregister_acpi_bus_notifier); > > /** > - * acpi_bus_notify > - * --------------- > - * Callback for all 'system-level' device notifications (values 0x00-0x7F). > + * acpi_bus_sys_notify: Common system notify handler > + * > + * ACPI drivers may specify this common handler to its sys_notify entry. > + * TBD: This handler is not implemented yet. > */ > -static void acpi_bus_notify(acpi_handle handle, u32 type, void *data) > +void acpi_bus_sys_notify(acpi_handle handle, u32 type, void *data) This isn't used anywhere. Are drivers supposed to use it? If so, what about the BUS_CHECK and DEVICE_CHECK notifications? > { > - struct acpi_device *device = NULL; > - struct acpi_driver *driver; > - > ACPI_DEBUG_PRINT((ACPI_DB_INFO, "Notification %#02x to handle %p\n", > type, handle)); > > - blocking_notifier_call_chain(&acpi_bus_notify_list, > - type, (void *)handle); > - > switch (type) { > > case ACPI_NOTIFY_BUS_CHECK: > @@ -842,14 +837,51 @@ static void acpi_bus_notify(acpi_handle handle, u32 type, void *data) > type)); > break; > } > +} > + > +/** > + * acpi_bus_drv_notify: Call driver's system-level notify handler > + */ > +void acpi_bus_drv_notify(struct acpi_driver *driver, > + struct acpi_device *device, acpi_handle handle, > + u32 type, void *data) > +{ > + BUG_ON(!driver); Rule: Don't crash the kernel if you don't have to. Try to recover instead. It seems that if (WARN_ON(!driver)) return; would be sufficient in this particulare case, wouldn't it? > + > + if (driver->ops.sys_notify) > + driver->ops.sys_notify(handle, type, data); > + else if (device && driver->ops.notify && Why "else if"? The existing code does this unconditionally. Is that incorrect? > + (driver->flags & ACPI_DRIVER_ALL_NOTIFY_EVENTS)) > + driver->ops.notify(device, type); > + > + return; > +} > + > +/** > + * acpi_bus_notify: The system-level global notify handler > + * > + * The global notify handler for all 'system-level' device notifications > + * (values 0x00-0x7F). This handler calls a driver's notify handler for > + * the notified ACPI device. > + */ > +static void acpi_bus_notify(acpi_handle handle, u32 type, void *data) > +{ > + struct acpi_device *device = NULL; > + > + /* call registered handlers in the bus notify list */ > + blocking_notifier_call_chain(&acpi_bus_notify_list, > + type, (void *)handle); > > + /* obtain an associated driver if already bound */ > acpi_bus_get_device(handle, &device); > - if (device) { > - driver = device->driver; > - if (driver && driver->ops.notify && > - (driver->flags & ACPI_DRIVER_ALL_NOTIFY_EVENTS)) > - driver->ops.notify(device, type); > - } > + > + /* call the driver's system-level notify handler */ > + if (device && device->driver) > + acpi_bus_drv_notify(device->driver, device, handle, type, data); > + else > + acpi_lookup_drv_notify(handle, type, data); > + > + return; > } > > /* -------------------------------------------------------------------------- > diff --git a/drivers/acpi/scan.c b/drivers/acpi/scan.c > index 95ff1e8..333e57f 100644 > --- a/drivers/acpi/scan.c > +++ b/drivers/acpi/scan.c > @@ -1758,3 +1758,86 @@ int __init acpi_scan_init(void) > > return result; > } > + > +/* callback args for acpi_match_drv_notify() */ > +struct acpi_notify_args { > + struct acpi_device *device; > + acpi_handle handle; > + u32 event; > + void *data; > +}; > + > +static int acpi_match_drv_notify(struct device_driver *drv, void *data) > +{ > + struct acpi_driver *driver = to_acpi_driver(drv); > + struct acpi_notify_args *args = (struct acpi_notify_args *) data; > + > + /* check if this driver matches with the device */ > + if (acpi_match_device_ids(args->device, driver->ids)) > + return 0; > + > + /* call the driver's notify handler */ > + acpi_bus_drv_notify(driver, NULL, args->handle, > + args->event, args->data); > + > + return 1; > +} > + > +/** > + * acpi_lookup_drv_notify: Look up and call driver's notify handler > + * @handle: ACPI handle of the notified device object > + * @event: Notify event > + * @data: Context > + * > + * Look up and call the associated driver's notify handler for the ACPI > + * device object by walking through the list of ACPI drivers. What exactly is the purpose of this? > + */ > +void acpi_lookup_drv_notify(acpi_handle handle, u32 event, void *data) > +{ > + struct acpi_notify_args args; > + struct acpi_device *device; > + unsigned long long sta; > + int type; > + int ret; > + > + /* allocate a temporary device object */ > + device = kzalloc(sizeof(struct acpi_device), GFP_KERNEL); > + if (!device) { > + pr_err(PREFIX "No memory to allocate a tmp device\n"); > + return; > + } > + INIT_LIST_HEAD(&device->pnp.ids); > + > + /* obtain device type */ > + ret = acpi_bus_type_and_status(handle, &type, &sta); > + if (ret) { > + pr_err(PREFIX "Failed to get device type\n"); > + goto out; > + } > + > + /* setup this temporary device object */ > + device->device_type = type; > + device->handle = handle; > + device->parent = acpi_bus_get_parent(handle); > + device->dev.bus = &acpi_bus_type; > + device->driver = NULL; > + STRUCT_TO_INT(device->status) = sta; > + device->status.present = 1; > + > + /* set HID to this device object */ > + acpi_device_set_id(device); > + > + /* set args */ > + args.device = device; > + args.handle = handle; > + args.event = event; > + args.data = data; > + > + /* call a matched driver's notify handler */ > + (void) bus_for_each_drv(device->dev.bus, NULL, > + &args, acpi_match_drv_notify); Excuse me? What makes you think I would accept anything like this? > + > +out: > + acpi_device_release(&device->dev); > + return; > +} > diff --git a/include/acpi/acpi_bus.h b/include/acpi/acpi_bus.h > index 2242c10..4052406 100644 > --- a/include/acpi/acpi_bus.h > +++ b/include/acpi/acpi_bus.h > @@ -94,6 +94,7 @@ typedef int (*acpi_op_start) (struct acpi_device * device); > typedef int (*acpi_op_bind) (struct acpi_device * device); > typedef int (*acpi_op_unbind) (struct acpi_device * device); > typedef void (*acpi_op_notify) (struct acpi_device * device, u32 event); > +typedef void (*acpi_op_sys_notify) (acpi_handle handle, u32 event, void *data); > > struct acpi_bus_ops { > u32 acpi_op_add:1; > @@ -107,6 +108,7 @@ struct acpi_device_ops { > acpi_op_bind bind; > acpi_op_unbind unbind; > acpi_op_notify notify; > + acpi_op_sys_notify sys_notify; > }; > > #define ACPI_DRIVER_ALL_NOTIFY_EVENTS 0x1 /* system AND device events */ > @@ -328,6 +330,10 @@ extern int unregister_acpi_notifier(struct notifier_block *); > > extern int register_acpi_bus_notifier(struct notifier_block *nb); > extern void unregister_acpi_bus_notifier(struct notifier_block *nb); > +extern void acpi_lookup_drv_notify(acpi_handle handle, u32 event, void *data); > +extern void acpi_bus_drv_notify(struct acpi_driver *driver, > + struct acpi_device *device, acpi_handle handle, u32 type, void *data); > + > /* > * External Functions > */ > 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