[PATCH 12/47] More locking improvements (and a fix) for serio.

From: Vojtech Pavlik
Date: Thu Jul 29 2004 - 13:59:52 EST


You can pull this changeset from:
bk://kernel.bkbits.net/vojtech/input

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

ChangeSet@xxxxxxxxxxx, 2004-06-02 16:09:25+02:00, vojtech@xxxxxxx
input: More locking improvements (and a fix) for serio. This
merges both my and Dmitry's changes.

Signed-off-by: Vojtech Pavlik <vojtech@xxxxxxx>


serio.c | 26 ++++++++++++++------------
1 files changed, 14 insertions(+), 12 deletions(-)

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

diff -Nru a/drivers/input/serio/serio.c b/drivers/input/serio/serio.c
--- a/drivers/input/serio/serio.c Thu Jul 29 14:41:31 2004
+++ b/drivers/input/serio/serio.c Thu Jul 29 14:41:31 2004
@@ -68,8 +68,8 @@
};


-spinlock_t serio_event_lock = SPIN_LOCK_UNLOCKED; /* protects serio_event_list */
-static DECLARE_MUTEX(serio_sem); /* protects serio_list and serio_dev_list */
+static spinlock_t serio_event_lock = SPIN_LOCK_UNLOCKED; /* protects serio_event_list */
+static DECLARE_MUTEX(serio_sem); /* protects serio_list and serio_dev_list */
static LIST_HEAD(serio_list);
static LIST_HEAD(serio_dev_list);
static LIST_HEAD(serio_event_list);
@@ -99,16 +99,21 @@
static DECLARE_WAIT_QUEUE_HEAD(serio_wait);
static DECLARE_COMPLETION(serio_exited);

-static void serio_invalidate_pending_events(struct serio *serio)
+static void serio_remove_pending_events(struct serio *serio)
{
+ struct list_head *node, *next;
struct serio_event *event;
unsigned long flags;

spin_lock_irqsave(&serio_event_lock, flags);

- list_for_each_entry(event, &serio_event_list, node)
- if (event->serio == serio)
- event->serio = NULL;
+ list_for_each_safe(node, next, &serio_event_list) {
+ event = container_of(node, struct serio_event, node);
+ if (event->serio == serio) {
+ list_del_init(node);
+ kfree(event);
+ }
+ }

spin_unlock_irqrestore(&serio_event_lock, flags);
}
@@ -137,9 +142,6 @@

down(&serio_sem);

- if (event->serio == NULL) /*!!!*/
- goto event_done;
-
switch (event->type) {
case SERIO_REGISTER_PORT :
__serio_register_port(event->serio);
@@ -163,7 +165,7 @@
default:
break;
}
-event_done:
+
up(&serio_sem);
kfree(event);
}
@@ -224,7 +226,7 @@

spin_lock_irqsave(&serio->lock, flags);

- if (serio->dev && serio->dev->interrupt) {
+ if (likely(serio->dev)) {
ret = serio->dev->interrupt(serio, data, dfl, regs);
} else {
if (!dfl) {
@@ -294,7 +296,7 @@
*/
void __serio_unregister_port(struct serio *serio)
{
- serio_invalidate_pending_events(serio);
+ serio_remove_pending_events(serio);
list_del_init(&serio->node);
if (serio->dev && serio->dev->disconnect)
serio->dev->disconnect(serio);

-
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/