Re: [PATCH 2.6.11-rc3] IBM Trackpoint support

From: Vojtech Pavlik
Date: Sat Feb 05 2005 - 07:33:47 EST


On Sat, Feb 05, 2005 at 01:56:07AM -0500, Dmitry Torokhov wrote:
> On Friday 04 February 2005 09:45, Vojtech Pavlik wrote:
> > On Fri, Feb 04, 2005 at 09:17:33AM -0500, Dmitry Torokhov wrote:
> >
> > > It is still a problem if driver is registered after the port has been
> > > detected wich quite often is the case as many people have psmouse as a
> > > module.
> > >
> > > I wonder if we should make driver registration asynchronous too.
> >
> > Probably yes.
> >
> > > I
> > > don't forsee any issues providing that I bump up module's reference
> > > count while driver structure is "in flight", do you?
> >
> > No, looks OK to me, too.
>
> Ok, what about the following patch then?

Applied. ;)

>
> --
> Dmitry
>
>
> ===================================================================
>
>
> ChangeSet@xxxxxx, 2005-02-05 01:48:45-05:00, dtor_core@xxxxxxxxxxxxx
> Input: make serio drivers register asynchronously. This should
> speed up boot process as some drivers take a long time
> probing for supported devices.
>
> Also change __inline__ to inline in serio.h
>
> Signed-off-by: Dmitry Torokhov <dtor@xxxxxxx>
>
>
> drivers/input/serio/serio.c | 65 ++++++++++++++++++++++++--------------------
> include/linux/serio.h | 25 ++++++++++------
> 2 files changed, 51 insertions(+), 39 deletions(-)
>
>
> ===================================================================
>
>
>
> diff -Nru a/drivers/input/serio/serio.c b/drivers/input/serio/serio.c
> --- a/drivers/input/serio/serio.c 2005-02-05 01:53:56 -05:00
> +++ b/drivers/input/serio/serio.c 2005-02-05 01:53:56 -05:00
> @@ -44,7 +44,7 @@
> EXPORT_SYMBOL(__serio_register_port);
> EXPORT_SYMBOL(serio_unregister_port);
> EXPORT_SYMBOL(__serio_unregister_port_delayed);
> -EXPORT_SYMBOL(serio_register_driver);
> +EXPORT_SYMBOL(__serio_register_driver);
> EXPORT_SYMBOL(serio_unregister_driver);
> EXPORT_SYMBOL(serio_open);
> EXPORT_SYMBOL(serio_close);
> @@ -120,18 +120,19 @@
> * Serio event processing.
> */
>
> -struct serio_event {
> - int type;
> - struct serio *serio;
> - struct module *owner;
> - struct list_head node;
> -};
> -
> enum serio_event_type {
> SERIO_RESCAN,
> SERIO_RECONNECT,
> SERIO_REGISTER_PORT,
> SERIO_UNREGISTER_PORT,
> + SERIO_REGISTER_DRIVER,
> +};
> +
> +struct serio_event {
> + enum serio_event_type type;
> + void *object;
> + struct module *owner;
> + struct list_head node;
> };
>
> static DEFINE_SPINLOCK(serio_event_lock); /* protects serio_event_list */
> @@ -140,7 +141,7 @@
> static DECLARE_COMPLETION(serio_exited);
> static int serio_pid;
>
> -static void serio_queue_event(struct serio *serio, struct module *owner,
> +static void serio_queue_event(void *object, struct module *owner,
> enum serio_event_type event_type)
> {
> unsigned long flags;
> @@ -156,7 +157,7 @@
> * we need to preseve sequence of distinct events.
> */
> list_for_each_entry_reverse(event, &serio_event_list, node) {
> - if (event->serio == serio) {
> + if (event->object == object) {
> if (event->type == event_type)
> goto out;
> break;
> @@ -170,7 +171,7 @@
> }
>
> event->type = event_type;
> - event->serio = serio;
> + event->object = object;
> event->owner = owner;
>
> list_add_tail(&event->node, &serio_event_list);
> @@ -198,7 +199,7 @@
>
> list_for_each_safe(node, next, &serio_event_list) {
> e = list_entry(node, struct serio_event, node);
> - if (event->serio == e->serio) {
> + if (event->object == e->object) {
> /*
> * If this event is of different type we should not
> * look further - we only suppress duplicate events
> @@ -241,6 +242,7 @@
> static void serio_handle_events(void)
> {
> struct serio_event *event;
> + struct serio_driver *serio_drv;
>
> down(&serio_sem);
>
> @@ -248,21 +250,26 @@
>
> switch (event->type) {
> case SERIO_REGISTER_PORT:
> - serio_add_port(event->serio);
> + serio_add_port(event->object);
> break;
>
> case SERIO_UNREGISTER_PORT:
> - serio_disconnect_port(event->serio);
> - serio_destroy_port(event->serio);
> + serio_disconnect_port(event->object);
> + serio_destroy_port(event->object);
> break;
>
> case SERIO_RECONNECT:
> - serio_reconnect_port(event->serio);
> + serio_reconnect_port(event->object);
> break;
>
> case SERIO_RESCAN:
> - serio_disconnect_port(event->serio);
> - serio_find_driver(event->serio);
> + serio_disconnect_port(event->object);
> + serio_find_driver(event->object);
> + break;
> +
> + case SERIO_REGISTER_DRIVER:
> + serio_drv = event->object;
> + driver_register(&serio_drv->driver);
> break;
>
> default:
> @@ -289,7 +296,7 @@
>
> list_for_each_safe(node, next, &serio_event_list) {
> event = list_entry(node, struct serio_event, node);
> - if (event->serio == serio) {
> + if (event->object == serio) {
> list_del_init(node);
> serio_free_event(event);
> }
> @@ -309,20 +316,23 @@
> static struct serio *serio_get_pending_child(struct serio *parent)
> {
> struct serio_event *event;
> - struct serio *serio = NULL;
> + struct serio *serio, *child = NULL;
> unsigned long flags;
>
> spin_lock_irqsave(&serio_event_lock, flags);
>
> list_for_each_entry(event, &serio_event_list, node) {
> - if (event->type == SERIO_REGISTER_PORT && event->serio->parent == parent) {
> - serio = event->serio;
> - break;
> + if (event->type == SERIO_REGISTER_PORT) {
> + serio = event->object;
> + if (serio->parent == parent) {
> + child = serio;
> + break;
> + }
> }
> }
>
> spin_unlock_irqrestore(&serio_event_lock, flags);
> - return serio;
> + return child;
> }
>
> static int serio_thread(void *nothing)
> @@ -672,16 +682,13 @@
> return 0;
> }
>
> -void serio_register_driver(struct serio_driver *drv)
> +void __serio_register_driver(struct serio_driver *drv, struct module *owner)
> {
> - down(&serio_sem);
> -
> drv->driver.bus = &serio_bus;
> drv->driver.probe = serio_driver_probe;
> drv->driver.remove = serio_driver_remove;
> - driver_register(&drv->driver);
>
> - up(&serio_sem);
> + serio_queue_event(drv, owner, SERIO_REGISTER_DRIVER);
> }
>
> void serio_unregister_driver(struct serio_driver *drv)
> diff -Nru a/include/linux/serio.h b/include/linux/serio.h
> --- a/include/linux/serio.h 2005-02-05 01:53:56 -05:00
> +++ b/include/linux/serio.h 2005-02-05 01:53:56 -05:00
> @@ -95,10 +95,15 @@
> __serio_unregister_port_delayed(serio, THIS_MODULE);
> }
>
> -void serio_register_driver(struct serio_driver *drv);
> +void __serio_register_driver(struct serio_driver *drv, struct module *owner);
> +static inline void serio_register_driver(struct serio_driver *drv)
> +{
> + __serio_register_driver(drv, THIS_MODULE);
> +}
> +
> void serio_unregister_driver(struct serio_driver *drv);
>
> -static __inline__ int serio_write(struct serio *serio, unsigned char data)
> +static inline int serio_write(struct serio *serio, unsigned char data)
> {
> if (serio->write)
> return serio->write(serio, data);
> @@ -106,13 +111,13 @@
> return -1;
> }
>
> -static __inline__ void serio_drv_write_wakeup(struct serio *serio)
> +static inline void serio_drv_write_wakeup(struct serio *serio)
> {
> if (serio->drv && serio->drv->write_wakeup)
> serio->drv->write_wakeup(serio);
> }
>
> -static __inline__ void serio_cleanup(struct serio *serio)
> +static inline void serio_cleanup(struct serio *serio)
> {
> if (serio->drv && serio->drv->cleanup)
> serio->drv->cleanup(serio);
> @@ -122,12 +127,12 @@
> * Use the following fucntions to manipulate serio's per-port
> * driver-specific data.
> */
> -static __inline__ void *serio_get_drvdata(struct serio *serio)
> +static inline void *serio_get_drvdata(struct serio *serio)
> {
> return dev_get_drvdata(&serio->dev);
> }
>
> -static __inline__ void serio_set_drvdata(struct serio *serio, void *data)
> +static inline void serio_set_drvdata(struct serio *serio, void *data)
> {
> dev_set_drvdata(&serio->dev, data);
> }
> @@ -136,12 +141,12 @@
> * Use the following fucntions to protect critical sections in
> * driver code from port's interrupt handler
> */
> -static __inline__ void serio_pause_rx(struct serio *serio)
> +static inline void serio_pause_rx(struct serio *serio)
> {
> spin_lock_irq(&serio->lock);
> }
>
> -static __inline__ void serio_continue_rx(struct serio *serio)
> +static inline void serio_continue_rx(struct serio *serio)
> {
> spin_unlock_irq(&serio->lock);
> }
> @@ -149,12 +154,12 @@
> /*
> * Use the following fucntions to pin serio's driver in process context
> */
> -static __inline__ int serio_pin_driver(struct serio *serio)
> +static inline int serio_pin_driver(struct serio *serio)
> {
> return down_interruptible(&serio->drv_sem);
> }
>
> -static __inline__ void serio_unpin_driver(struct serio *serio)
> +static inline void serio_unpin_driver(struct serio *serio)
> {
> up(&serio->drv_sem);
> }
>

--
Vojtech Pavlik
SuSE Labs, SuSE CR
-
To unsubscribe from this list: send the line "unsubscribe linux-kernel" in
the body of a message to majordomo@xxxxxxxxxxxxxxx
More majordomo info at http://vger.kernel.org/majordomo-info.html
Please read the FAQ at http://www.tux.org/lkml/