ACK: [SRU][Xenial][v2][PATCH 1/1] tty: Prepare for destroying line discipline on hangup
Kleber Souza
kleber.souza at canonical.com
Mon Oct 16 16:40:14 UTC 2017
On 10/13/17 21:52, Joseph Salisbury wrote:
> From: Peter Hurley <peter at hurleysoftware.com>
>
> 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: Peter Hurley <peter at hurleysoftware.com>
> Signed-off-by: Greg Kroah-Hartman <gregkh at linuxfoundation.org>
> (cherry picked from commit e55afd11a48354c810caf6b6ad4c103016a88230)
> Signed-off-by: Joseph Salisbury <joseph.salisbury at canonical.com>
Acked-by: Kleber Sacilotto de Souza <kleber.souza at canonical.com>
> ---
> drivers/tty/tty_io.c | 14 ++++++++++++++
> drivers/tty/tty_ldisc.c | 4 ++--
> drivers/tty/vt/selection.c | 2 ++
> 3 files changed, 18 insertions(+), 2 deletions(-)
>
> diff --git a/drivers/tty/tty_io.c b/drivers/tty/tty_io.c
> index 8c0f013..6e3935f 100644
> --- a/drivers/tty/tty_io.c
> +++ b/drivers/tty/tty_io.c
> @@ -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
> situation */
> ld = tty_ldisc_ref_wait(tty);
> + if (!ld)
> + return hung_up_tty_read(file, buf, count, ppos);
> if (ld->ops->read)
> i = ld->ops->read(tty, file, buf, count);
> else
> @@ -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",
> tty->driver->name);
> ld = tty_ldisc_ref_wait(tty);
> + if (!ld)
> + return hung_up_tty_write(file, buf, count, ppos);
> if (!ld->ops->write)
> ret = -EIO;
> else
> @@ -2198,6 +2202,8 @@ static unsigned int tty_poll(struct file *filp, poll_table *wait)
> return 0;
>
> ld = tty_ldisc_ref_wait(tty);
> + if (!ld)
> + return hung_up_tty_poll(filp, wait);
> if (ld->ops->poll)
> ret = ld->ops->poll(tty, filp, wait);
> tty_ldisc_deref(ld);
> @@ -2287,6 +2293,8 @@ static int tiocsti(struct tty_struct *tty, char __user *p)
> return -EFAULT;
> tty_audit_tiocsti(tty, ch);
> ld = tty_ldisc_ref_wait(tty);
> + if (!ld)
> + return -EIO;
> ld->ops->receive_buf(tty, &ch, &mbz, 1);
> tty_ldisc_deref(ld);
> return 0;
> @@ -2679,6 +2687,8 @@ static int tiocgetd(struct tty_struct *tty, int __user *p)
> int ret;
>
> ld = tty_ldisc_ref_wait(tty);
> + if (!ld)
> + return -EIO;
> ret = put_user(ld->ops->num, p);
> tty_ldisc_deref(ld);
> return ret;
> @@ -2976,6 +2986,8 @@ long tty_ioctl(struct file *file, unsigned int cmd, unsigned long arg)
> return retval;
> }
> ld = tty_ldisc_ref_wait(tty);
> + if (!ld)
> + return hung_up_tty_ioctl(file, cmd, arg);
> retval = -EINVAL;
> if (ld->ops->ioctl) {
> retval = ld->ops->ioctl(tty, file, cmd, arg);
> @@ -3004,6 +3016,8 @@ static long tty_compat_ioctl(struct file *file, unsigned int cmd,
> }
>
> ld = tty_ldisc_ref_wait(tty);
> + if (!ld)
> + return hung_up_tty_compat_ioctl(file, cmd, arg);
> if (ld->ops->compat_ioctl)
> retval = ld->ops->compat_ioctl(tty, file, cmd, arg);
> else
> diff --git a/drivers/tty/tty_ldisc.c b/drivers/tty/tty_ldisc.c
> index 539dc3c..bc26e48 100644
> --- a/drivers/tty/tty_ldisc.c
> +++ b/drivers/tty/tty_ldisc.c
> @@ -258,8 +258,8 @@ const struct file_operations tty_ldiscs_proc_fops = {
> * against a discipline change, such as an existing ldisc reference
> * (which we check for)
> *
> - * Note: only callable from a file_operations routine (which
> - * guarantees tty->ldisc != NULL when the lock is acquired).
> + * Note: a file_operations routine (read/poll/write) should use this
> + * function to wait for any ldisc lifetime events to finish.
> */
>
> struct tty_ldisc *tty_ldisc_ref_wait(struct tty_struct *tty)
> diff --git a/drivers/tty/vt/selection.c b/drivers/tty/vt/selection.c
> index 381a2b1..4dd9dd2 100644
> --- a/drivers/tty/vt/selection.c
> +++ b/drivers/tty/vt/selection.c
> @@ -347,6 +347,8 @@ int paste_selection(struct tty_struct *tty)
> console_unlock();
>
> ld = tty_ldisc_ref_wait(tty);
> + if (!ld)
> + return -EIO; /* ldisc was hung up */
> tty_buffer_lock_exclusive(&vc->port);
>
> add_wait_queue(&vc->paste_wait, &wait);
>
More information about the kernel-team
mailing list