Commit 87951a91 authored by Peter Hurley's avatar Peter Hurley Committed by Kleber Sacilotto de Souza

tty: Prepare for destroying line discipline on hangup

BugLink: http://bugs.launchpad.net/bugs/1721065

tty file_operations (read/write/ioctl) wait for the ldisc reference
indefinitely (until ldisc lifetime events, such as hangup or TIOCSETD,
finish). Since hangup now destroys the ldisc and does not instance
another copy, file_operations must now be prepared to receive a NULL
ldisc reference from tty_ldisc_ref_wait():

CPU 0                                   CPU 1
-----                                   -----
(*f_op->read)() => tty_read()
                                        __tty_hangup()
                                        ...
                                        f_op = &hung_up_tty_fops;
                                        ...
                                        tty_ldisc_hangup()
                                           tty_ldisc_lock()
                                           tty_ldisc_kill()
                                              tty->ldisc = NULL
                                           tty_ldisc_unlock()
ld = tty_ldisc_ref_wait()
/* ld == NULL */

Instead, the action taken now is to return the same value as if the
tty had been hungup a moment earlier:

CPU 0                                   CPU 1
-----                                   -----
                                        __tty_hangup()
                                        ...
                                        f_op = &hung_up_tty_fops;
(*f_op->read)() => hung_up_tty_read()
return 0;
                                        ...
                                        tty_ldisc_hangup()
                                           tty_ldisc_lock()
                                           tty_ldisc_kill()
                                              tty->ldisc = NULL
                                           tty_ldisc_unlock()
Signed-off-by: default avatarPeter Hurley <peter@hurleysoftware.com>
Signed-off-by: default avatarGreg Kroah-Hartman <gregkh@linuxfoundation.org>
(cherry picked from commit e55afd11)
Signed-off-by: default avatarJoseph Salisbury <joseph.salisbury@canonical.com>
Acked-by: default avatarStefan Bader <stefan.bader@canonical.com>
Acked-by: default avatarKleber Sacilotto de Souza <kleber.souza@canonical.com>
Signed-off-by: default avatarKleber Sacilotto de Souza <kleber.souza@canonical.com>
parent bf853ce7
...@@ -1066,6 +1066,8 @@ static ssize_t tty_read(struct file *file, char __user *buf, size_t count, ...@@ -1066,6 +1066,8 @@ static ssize_t tty_read(struct file *file, char __user *buf, size_t count,
/* We want to wait for the line discipline to sort out in this /* We want to wait for the line discipline to sort out in this
situation */ situation */
ld = tty_ldisc_ref_wait(tty); ld = tty_ldisc_ref_wait(tty);
if (!ld)
return hung_up_tty_read(file, buf, count, ppos);
if (ld->ops->read) if (ld->ops->read)
i = ld->ops->read(tty, file, buf, count); i = ld->ops->read(tty, file, buf, count);
else else
...@@ -1241,6 +1243,8 @@ static ssize_t tty_write(struct file *file, const char __user *buf, ...@@ -1241,6 +1243,8 @@ static ssize_t tty_write(struct file *file, const char __user *buf,
printk(KERN_ERR "tty driver %s lacks a write_room method.\n", printk(KERN_ERR "tty driver %s lacks a write_room method.\n",
tty->driver->name); tty->driver->name);
ld = tty_ldisc_ref_wait(tty); ld = tty_ldisc_ref_wait(tty);
if (!ld)
return hung_up_tty_write(file, buf, count, ppos);
if (!ld->ops->write) if (!ld->ops->write)
ret = -EIO; ret = -EIO;
else else
...@@ -2198,6 +2202,8 @@ static unsigned int tty_poll(struct file *filp, poll_table *wait) ...@@ -2198,6 +2202,8 @@ static unsigned int tty_poll(struct file *filp, poll_table *wait)
return 0; return 0;
ld = tty_ldisc_ref_wait(tty); ld = tty_ldisc_ref_wait(tty);
if (!ld)
return hung_up_tty_poll(filp, wait);
if (ld->ops->poll) if (ld->ops->poll)
ret = ld->ops->poll(tty, filp, wait); ret = ld->ops->poll(tty, filp, wait);
tty_ldisc_deref(ld); tty_ldisc_deref(ld);
...@@ -2287,6 +2293,8 @@ static int tiocsti(struct tty_struct *tty, char __user *p) ...@@ -2287,6 +2293,8 @@ static int tiocsti(struct tty_struct *tty, char __user *p)
return -EFAULT; return -EFAULT;
tty_audit_tiocsti(tty, ch); tty_audit_tiocsti(tty, ch);
ld = tty_ldisc_ref_wait(tty); ld = tty_ldisc_ref_wait(tty);
if (!ld)
return -EIO;
ld->ops->receive_buf(tty, &ch, &mbz, 1); ld->ops->receive_buf(tty, &ch, &mbz, 1);
tty_ldisc_deref(ld); tty_ldisc_deref(ld);
return 0; return 0;
...@@ -2679,6 +2687,8 @@ static int tiocgetd(struct tty_struct *tty, int __user *p) ...@@ -2679,6 +2687,8 @@ static int tiocgetd(struct tty_struct *tty, int __user *p)
int ret; int ret;
ld = tty_ldisc_ref_wait(tty); ld = tty_ldisc_ref_wait(tty);
if (!ld)
return -EIO;
ret = put_user(ld->ops->num, p); ret = put_user(ld->ops->num, p);
tty_ldisc_deref(ld); tty_ldisc_deref(ld);
return ret; return ret;
...@@ -2976,6 +2986,8 @@ long tty_ioctl(struct file *file, unsigned int cmd, unsigned long arg) ...@@ -2976,6 +2986,8 @@ long tty_ioctl(struct file *file, unsigned int cmd, unsigned long arg)
return retval; return retval;
} }
ld = tty_ldisc_ref_wait(tty); ld = tty_ldisc_ref_wait(tty);
if (!ld)
return hung_up_tty_ioctl(file, cmd, arg);
retval = -EINVAL; retval = -EINVAL;
if (ld->ops->ioctl) { if (ld->ops->ioctl) {
retval = ld->ops->ioctl(tty, file, cmd, arg); retval = ld->ops->ioctl(tty, file, cmd, arg);
...@@ -3004,6 +3016,8 @@ static long tty_compat_ioctl(struct file *file, unsigned int cmd, ...@@ -3004,6 +3016,8 @@ static long tty_compat_ioctl(struct file *file, unsigned int cmd,
} }
ld = tty_ldisc_ref_wait(tty); ld = tty_ldisc_ref_wait(tty);
if (!ld)
return hung_up_tty_compat_ioctl(file, cmd, arg);
if (ld->ops->compat_ioctl) if (ld->ops->compat_ioctl)
retval = ld->ops->compat_ioctl(tty, file, cmd, arg); retval = ld->ops->compat_ioctl(tty, file, cmd, arg);
else else
......
...@@ -258,8 +258,8 @@ const struct file_operations tty_ldiscs_proc_fops = { ...@@ -258,8 +258,8 @@ const struct file_operations tty_ldiscs_proc_fops = {
* against a discipline change, such as an existing ldisc reference * against a discipline change, such as an existing ldisc reference
* (which we check for) * (which we check for)
* *
* Note: only callable from a file_operations routine (which * Note: a file_operations routine (read/poll/write) should use this
* guarantees tty->ldisc != NULL when the lock is acquired). * function to wait for any ldisc lifetime events to finish.
*/ */
struct tty_ldisc *tty_ldisc_ref_wait(struct tty_struct *tty) struct tty_ldisc *tty_ldisc_ref_wait(struct tty_struct *tty)
......
...@@ -347,6 +347,8 @@ int paste_selection(struct tty_struct *tty) ...@@ -347,6 +347,8 @@ int paste_selection(struct tty_struct *tty)
console_unlock(); console_unlock();
ld = tty_ldisc_ref_wait(tty); ld = tty_ldisc_ref_wait(tty);
if (!ld)
return -EIO; /* ldisc was hung up */
tty_buffer_lock_exclusive(&vc->port); tty_buffer_lock_exclusive(&vc->port);
add_wait_queue(&vc->paste_wait, &wait); add_wait_queue(&vc->paste_wait, &wait);
......
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