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?

-- 
Dmitry


===================================================================


[EMAIL PROTECTED], 2005-02-05 01:48:45-05:00, [EMAIL PROTECTED]
  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 <[EMAIL PROTECTED]>


 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);
 }
-
To unsubscribe from this list: send the line "unsubscribe linux-kernel" in
the body of a message to [EMAIL PROTECTED]
More majordomo info at  http://vger.kernel.org/majordomo-info.html
Please read the FAQ at  http://www.tux.org/lkml/

Reply via email to