diff options
Diffstat (limited to 'drivers/char/tty_io.c')
-rw-r--r-- | drivers/char/tty_io.c | 41 |
1 files changed, 20 insertions, 21 deletions
diff --git a/drivers/char/tty_io.c b/drivers/char/tty_io.c index bb22cf495435..dcf7d368639e 100644 --- a/drivers/char/tty_io.c +++ b/drivers/char/tty_io.c | |||
@@ -492,10 +492,8 @@ EXPORT_SYMBOL_GPL(tty_wakeup); | |||
492 | * tasklist_lock to walk task list for hangup event | 492 | * tasklist_lock to walk task list for hangup event |
493 | * ->siglock to protect ->signal/->sighand | 493 | * ->siglock to protect ->signal/->sighand |
494 | */ | 494 | */ |
495 | static void do_tty_hangup(struct work_struct *work) | 495 | void tty_vhangup_locked(struct tty_struct *tty) |
496 | { | 496 | { |
497 | struct tty_struct *tty = | ||
498 | container_of(work, struct tty_struct, hangup_work); | ||
499 | struct file *cons_filp = NULL; | 497 | struct file *cons_filp = NULL; |
500 | struct file *filp, *f = NULL; | 498 | struct file *filp, *f = NULL; |
501 | struct task_struct *p; | 499 | struct task_struct *p; |
@@ -517,8 +515,6 @@ static void do_tty_hangup(struct work_struct *work) | |||
517 | /* inuse_filps is protected by the single tty lock, | 515 | /* inuse_filps is protected by the single tty lock, |
518 | this really needs to change if we want to flush the | 516 | this really needs to change if we want to flush the |
519 | workqueue with the lock held */ | 517 | workqueue with the lock held */ |
520 | tty_lock_nested(); /* called with BTM held from pty_close and | ||
521 | others */ | ||
522 | check_tty_count(tty, "do_tty_hangup"); | 518 | check_tty_count(tty, "do_tty_hangup"); |
523 | 519 | ||
524 | file_list_lock(); | 520 | file_list_lock(); |
@@ -598,11 +594,20 @@ static void do_tty_hangup(struct work_struct *work) | |||
598 | */ | 594 | */ |
599 | set_bit(TTY_HUPPED, &tty->flags); | 595 | set_bit(TTY_HUPPED, &tty->flags); |
600 | tty_ldisc_enable(tty); | 596 | tty_ldisc_enable(tty); |
601 | tty_unlock(); | ||
602 | if (f) | 597 | if (f) |
603 | fput(f); | 598 | fput(f); |
604 | } | 599 | } |
605 | 600 | ||
601 | static void do_tty_hangup(struct work_struct *work) | ||
602 | { | ||
603 | struct tty_struct *tty = | ||
604 | container_of(work, struct tty_struct, hangup_work); | ||
605 | |||
606 | tty_lock(); | ||
607 | tty_vhangup_locked(tty); | ||
608 | tty_unlock(); | ||
609 | } | ||
610 | |||
606 | /** | 611 | /** |
607 | * tty_hangup - trigger a hangup event | 612 | * tty_hangup - trigger a hangup event |
608 | * @tty: tty to hangup | 613 | * @tty: tty to hangup |
@@ -638,7 +643,9 @@ void tty_vhangup(struct tty_struct *tty) | |||
638 | 643 | ||
639 | printk(KERN_DEBUG "%s vhangup...\n", tty_name(tty, buf)); | 644 | printk(KERN_DEBUG "%s vhangup...\n", tty_name(tty, buf)); |
640 | #endif | 645 | #endif |
641 | do_tty_hangup(&tty->hangup_work); | 646 | tty_lock(); |
647 | tty_vhangup_locked(tty); | ||
648 | tty_unlock(); | ||
642 | } | 649 | } |
643 | 650 | ||
644 | EXPORT_SYMBOL(tty_vhangup); | 651 | EXPORT_SYMBOL(tty_vhangup); |
@@ -719,10 +726,12 @@ void disassociate_ctty(int on_exit) | |||
719 | tty = get_current_tty(); | 726 | tty = get_current_tty(); |
720 | if (tty) { | 727 | if (tty) { |
721 | tty_pgrp = get_pid(tty->pgrp); | 728 | tty_pgrp = get_pid(tty->pgrp); |
722 | tty_lock_nested(); /* see above */ | 729 | if (on_exit) { |
723 | if (on_exit && tty->driver->type != TTY_DRIVER_TYPE_PTY) | 730 | tty_lock(); |
724 | tty_vhangup(tty); | 731 | if (tty->driver->type != TTY_DRIVER_TYPE_PTY) |
725 | tty_unlock(); | 732 | tty_vhangup_locked(tty); |
733 | tty_unlock(); | ||
734 | } | ||
726 | tty_kref_put(tty); | 735 | tty_kref_put(tty); |
727 | } else if (on_exit) { | 736 | } else if (on_exit) { |
728 | struct pid *old_pgrp; | 737 | struct pid *old_pgrp; |
@@ -1213,18 +1222,14 @@ static int tty_driver_install_tty(struct tty_driver *driver, | |||
1213 | int ret; | 1222 | int ret; |
1214 | 1223 | ||
1215 | if (driver->ops->install) { | 1224 | if (driver->ops->install) { |
1216 | tty_lock_nested(); /* already called with BTM held */ | ||
1217 | ret = driver->ops->install(driver, tty); | 1225 | ret = driver->ops->install(driver, tty); |
1218 | tty_unlock(); | ||
1219 | return ret; | 1226 | return ret; |
1220 | } | 1227 | } |
1221 | 1228 | ||
1222 | if (tty_init_termios(tty) == 0) { | 1229 | if (tty_init_termios(tty) == 0) { |
1223 | tty_lock_nested(); | ||
1224 | tty_driver_kref_get(driver); | 1230 | tty_driver_kref_get(driver); |
1225 | tty->count++; | 1231 | tty->count++; |
1226 | driver->ttys[idx] = tty; | 1232 | driver->ttys[idx] = tty; |
1227 | tty_unlock(); | ||
1228 | return 0; | 1233 | return 0; |
1229 | } | 1234 | } |
1230 | return -ENOMEM; | 1235 | return -ENOMEM; |
@@ -1317,15 +1322,11 @@ struct tty_struct *tty_init_dev(struct tty_driver *driver, int idx, | |||
1317 | struct tty_struct *tty; | 1322 | struct tty_struct *tty; |
1318 | int retval; | 1323 | int retval; |
1319 | 1324 | ||
1320 | tty_lock_nested(); /* always called with tty lock held already */ | ||
1321 | |||
1322 | /* Check if pty master is being opened multiple times */ | 1325 | /* Check if pty master is being opened multiple times */ |
1323 | if (driver->subtype == PTY_TYPE_MASTER && | 1326 | if (driver->subtype == PTY_TYPE_MASTER && |
1324 | (driver->flags & TTY_DRIVER_DEVPTS_MEM) && !first_ok) { | 1327 | (driver->flags & TTY_DRIVER_DEVPTS_MEM) && !first_ok) { |
1325 | tty_unlock(); | ||
1326 | return ERR_PTR(-EIO); | 1328 | return ERR_PTR(-EIO); |
1327 | } | 1329 | } |
1328 | tty_unlock(); | ||
1329 | 1330 | ||
1330 | /* | 1331 | /* |
1331 | * First time open is complex, especially for PTY devices. | 1332 | * First time open is complex, especially for PTY devices. |
@@ -1369,9 +1370,7 @@ release_mem_out: | |||
1369 | if (printk_ratelimit()) | 1370 | if (printk_ratelimit()) |
1370 | printk(KERN_INFO "tty_init_dev: ldisc open failed, " | 1371 | printk(KERN_INFO "tty_init_dev: ldisc open failed, " |
1371 | "clearing slot %d\n", idx); | 1372 | "clearing slot %d\n", idx); |
1372 | tty_lock_nested(); | ||
1373 | release_tty(tty, idx); | 1373 | release_tty(tty, idx); |
1374 | tty_unlock(); | ||
1375 | return ERR_PTR(retval); | 1374 | return ERR_PTR(retval); |
1376 | } | 1375 | } |
1377 | 1376 | ||