diff options
author | Amit Shah <amit.shah@redhat.com> | 2013-07-29 00:53:21 -0400 |
---|---|---|
committer | Rusty Russell <rusty@rustcorp.com.au> | 2013-07-29 01:13:58 -0400 |
commit | 96f97a83910cdb9d89d127c5ee523f8fc040a804 (patch) | |
tree | 38578e6565dfd1e93ac3556b9c770d5fee37cc51 /drivers/char/virtio_console.c | |
parent | 92d3453815fbe74d539c86b60dab39ecdf01bb99 (diff) |
virtio: console: return -ENODEV on all read operations after unplug
If a port gets unplugged while a user is blocked on read(), -ENODEV is
returned. However, subsequent read()s returned 0, indicating there's no
host-side connection (but not indicating the device went away).
This also happened when a port was unplugged and the user didn't have
any blocking operation pending. If the user didn't monitor the SIGIO
signal, they won't have a chance to find out if the port went away.
Fix by returning -ENODEV on all read()s after the port gets unplugged.
write() already behaves this way.
CC: <stable@vger.kernel.org>
Signed-off-by: Amit Shah <amit.shah@redhat.com>
Signed-off-by: Rusty Russell <rusty@rustcorp.com.au>
Diffstat (limited to 'drivers/char/virtio_console.c')
-rw-r--r-- | drivers/char/virtio_console.c | 6 |
1 files changed, 5 insertions, 1 deletions
diff --git a/drivers/char/virtio_console.c b/drivers/char/virtio_console.c index e4845f1c9a0b..fc45567ad3ac 100644 --- a/drivers/char/virtio_console.c +++ b/drivers/char/virtio_console.c | |||
@@ -749,6 +749,10 @@ static ssize_t port_fops_read(struct file *filp, char __user *ubuf, | |||
749 | 749 | ||
750 | port = filp->private_data; | 750 | port = filp->private_data; |
751 | 751 | ||
752 | /* Port is hot-unplugged. */ | ||
753 | if (!port->guest_connected) | ||
754 | return -ENODEV; | ||
755 | |||
752 | if (!port_has_data(port)) { | 756 | if (!port_has_data(port)) { |
753 | /* | 757 | /* |
754 | * If nothing's connected on the host just return 0 in | 758 | * If nothing's connected on the host just return 0 in |
@@ -765,7 +769,7 @@ static ssize_t port_fops_read(struct file *filp, char __user *ubuf, | |||
765 | if (ret < 0) | 769 | if (ret < 0) |
766 | return ret; | 770 | return ret; |
767 | } | 771 | } |
768 | /* Port got hot-unplugged. */ | 772 | /* Port got hot-unplugged while we were waiting above. */ |
769 | if (!port->guest_connected) | 773 | if (!port->guest_connected) |
770 | return -ENODEV; | 774 | return -ENODEV; |
771 | /* | 775 | /* |