Commit 5b138907 authored by Vojtech Pavlik's avatar Vojtech Pavlik

input: More locking improvements (and a fix) for serio. This

       merges both my and Dmitry's changes.
Signed-off-by: default avatarVojtech Pavlik <vojtech@suse.cz>
parent cdc6f197
...@@ -68,8 +68,8 @@ struct serio_event { ...@@ -68,8 +68,8 @@ struct serio_event {
}; };
spinlock_t serio_event_lock = SPIN_LOCK_UNLOCKED; /* protects serio_event_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 DECLARE_MUTEX(serio_sem); /* protects serio_list and serio_dev_list */
static LIST_HEAD(serio_list); static LIST_HEAD(serio_list);
static LIST_HEAD(serio_dev_list); static LIST_HEAD(serio_dev_list);
static LIST_HEAD(serio_event_list); static LIST_HEAD(serio_event_list);
...@@ -99,16 +99,21 @@ static void serio_find_dev(struct serio *serio) ...@@ -99,16 +99,21 @@ static void serio_find_dev(struct serio *serio)
static DECLARE_WAIT_QUEUE_HEAD(serio_wait); static DECLARE_WAIT_QUEUE_HEAD(serio_wait);
static DECLARE_COMPLETION(serio_exited); 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; struct serio_event *event;
unsigned long flags; unsigned long flags;
spin_lock_irqsave(&serio_event_lock, flags); spin_lock_irqsave(&serio_event_lock, flags);
list_for_each_entry(event, &serio_event_list, node) list_for_each_safe(node, next, &serio_event_list) {
if (event->serio == serio) event = container_of(node, struct serio_event, node);
event->serio = NULL; if (event->serio == serio) {
list_del_init(node);
kfree(event);
}
}
spin_unlock_irqrestore(&serio_event_lock, flags); spin_unlock_irqrestore(&serio_event_lock, flags);
} }
...@@ -137,9 +142,6 @@ void serio_handle_events(void) ...@@ -137,9 +142,6 @@ void serio_handle_events(void)
down(&serio_sem); down(&serio_sem);
if (event->serio == NULL) /*!!!*/
goto event_done;
switch (event->type) { switch (event->type) {
case SERIO_REGISTER_PORT : case SERIO_REGISTER_PORT :
__serio_register_port(event->serio); __serio_register_port(event->serio);
...@@ -163,7 +165,7 @@ void serio_handle_events(void) ...@@ -163,7 +165,7 @@ void serio_handle_events(void)
default: default:
break; break;
} }
event_done:
up(&serio_sem); up(&serio_sem);
kfree(event); kfree(event);
} }
...@@ -224,7 +226,7 @@ irqreturn_t serio_interrupt(struct serio *serio, ...@@ -224,7 +226,7 @@ irqreturn_t serio_interrupt(struct serio *serio,
spin_lock_irqsave(&serio->lock, flags); spin_lock_irqsave(&serio->lock, flags);
if (serio->dev && serio->dev->interrupt) { if (likely(serio->dev)) {
ret = serio->dev->interrupt(serio, data, dfl, regs); ret = serio->dev->interrupt(serio, data, dfl, regs);
} else { } else {
if (!dfl) { if (!dfl) {
...@@ -294,7 +296,7 @@ void serio_unregister_port_delayed(struct serio *serio) ...@@ -294,7 +296,7 @@ void serio_unregister_port_delayed(struct serio *serio)
*/ */
void __serio_unregister_port(struct serio *serio) void __serio_unregister_port(struct serio *serio)
{ {
serio_invalidate_pending_events(serio); serio_remove_pending_events(serio);
list_del_init(&serio->node); list_del_init(&serio->node);
if (serio->dev && serio->dev->disconnect) if (serio->dev && serio->dev->disconnect)
serio->dev->disconnect(serio); serio->dev->disconnect(serio);
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment