aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/tty
diff options
context:
space:
mode:
authorPeter Hurley <peter@hurleysoftware.com>2013-01-30 12:43:50 -0500
committerGreg Kroah-Hartman <gregkh@linuxfoundation.org>2013-02-04 18:40:28 -0500
commit699390354da6c258b65bf8fa79cfd5feaede50b6 (patch)
treea2a1cde7fdc06a6ca684ac1c9bf57bef5d6f75ca /drivers/tty
parent7acf6cd80b201f77371a5374a786144153629be8 (diff)
pty: Ignore slave pty close() if never successfully opened
If the master and slave ptys are opened in parallel, the slave open fails because the pty is still locked. This is as designed. However, pty_close() is still called for the slave pty which sets TTY_OTHER_CLOSED in the master pty. This can cause the master open to fail as well. Use a common pattern in other tty drivers by setting TTY_IO_ERROR until the open is successful and only closing the pty if not set. Note: the master pty always closes regardless of whether the open was successful, so that proper cleanup can occur. Signed-off-by: Peter Hurley <peter@hurleysoftware.com> Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
Diffstat (limited to 'drivers/tty')
-rw-r--r--drivers/tty/pty.c6
1 files changed, 6 insertions, 0 deletions
diff --git a/drivers/tty/pty.c b/drivers/tty/pty.c
index 96dc6dd31425..d38455fab4b7 100644
--- a/drivers/tty/pty.c
+++ b/drivers/tty/pty.c
@@ -38,9 +38,12 @@ static void pty_close(struct tty_struct *tty, struct file *filp)
38 if (tty->driver->subtype == PTY_TYPE_MASTER) 38 if (tty->driver->subtype == PTY_TYPE_MASTER)
39 WARN_ON(tty->count > 1); 39 WARN_ON(tty->count > 1);
40 else { 40 else {
41 if (test_bit(TTY_IO_ERROR, &tty->flags))
42 return;
41 if (tty->count > 2) 43 if (tty->count > 2)
42 return; 44 return;
43 } 45 }
46 set_bit(TTY_IO_ERROR, &tty->flags);
44 wake_up_interruptible(&tty->read_wait); 47 wake_up_interruptible(&tty->read_wait);
45 wake_up_interruptible(&tty->write_wait); 48 wake_up_interruptible(&tty->write_wait);
46 tty->packet = 0; 49 tty->packet = 0;
@@ -246,6 +249,8 @@ static int pty_open(struct tty_struct *tty, struct file *filp)
246 if (!tty || !tty->link) 249 if (!tty || !tty->link)
247 goto out; 250 goto out;
248 251
252 set_bit(TTY_IO_ERROR, &tty->flags);
253
249 retval = -EIO; 254 retval = -EIO;
250 if (test_bit(TTY_OTHER_CLOSED, &tty->flags)) 255 if (test_bit(TTY_OTHER_CLOSED, &tty->flags))
251 goto out; 256 goto out;
@@ -254,6 +259,7 @@ static int pty_open(struct tty_struct *tty, struct file *filp)
254 if (tty->link->count != 1) 259 if (tty->link->count != 1)
255 goto out; 260 goto out;
256 261
262 clear_bit(TTY_IO_ERROR, &tty->flags);
257 clear_bit(TTY_OTHER_CLOSED, &tty->link->flags); 263 clear_bit(TTY_OTHER_CLOSED, &tty->link->flags);
258 set_bit(TTY_THROTTLED, &tty->flags); 264 set_bit(TTY_THROTTLED, &tty->flags);
259 retval = 0; 265 retval = 0;