Commit f4ab7818 authored by Johannes Berg's avatar Johannes Berg Committed by Richard Weinberger

um: line: Don't free winch (with IRQ) under spinlock

Lockdep correctly complains that one shouldn't call um_free_irq()
with free_irq() inside under a spinlock since that will attempt
to acquire a mutex.

Rearrange the code to keep the list manipulations under the lock
while moving the actual freeing outside of it, to avoid this.

In particular, this removes the lockdep complaint at shutdown that
I was seeing with lockdep enabled.
Signed-off-by: default avatarJohannes Berg <johannes.berg@intel.com>
Acked-By: anton.ivanov@cambridgegreys.com
Signed-off-by: default avatarRichard Weinberger <richard@nod.at>
parent fc6b6a87
...@@ -608,7 +608,6 @@ static void free_winch(struct winch *winch) ...@@ -608,7 +608,6 @@ static void free_winch(struct winch *winch)
winch->fd = -1; winch->fd = -1;
if (fd != -1) if (fd != -1)
os_close_file(fd); os_close_file(fd);
list_del(&winch->list);
__free_winch(&winch->work); __free_winch(&winch->work);
} }
...@@ -709,6 +708,8 @@ static void unregister_winch(struct tty_struct *tty) ...@@ -709,6 +708,8 @@ static void unregister_winch(struct tty_struct *tty)
winch = list_entry(ele, struct winch, list); winch = list_entry(ele, struct winch, list);
wtty = tty_port_tty_get(winch->port); wtty = tty_port_tty_get(winch->port);
if (wtty == tty) { if (wtty == tty) {
list_del(&winch->list);
spin_unlock(&winch_handler_lock);
free_winch(winch); free_winch(winch);
break; break;
} }
...@@ -719,14 +720,17 @@ static void unregister_winch(struct tty_struct *tty) ...@@ -719,14 +720,17 @@ static void unregister_winch(struct tty_struct *tty)
static void winch_cleanup(void) static void winch_cleanup(void)
{ {
struct list_head *ele, *next;
struct winch *winch; struct winch *winch;
spin_lock(&winch_handler_lock); spin_lock(&winch_handler_lock);
while ((winch = list_first_entry_or_null(&winch_handlers,
struct winch, list))) {
list_del(&winch->list);
spin_unlock(&winch_handler_lock);
list_for_each_safe(ele, next, &winch_handlers) {
winch = list_entry(ele, struct winch, list);
free_winch(winch); free_winch(winch);
spin_lock(&winch_handler_lock);
} }
spin_unlock(&winch_handler_lock); spin_unlock(&winch_handler_lock);
......
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