summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--share/man/man4/Makefile2
-rw-r--r--share/man/man4/virtio.41
-rw-r--r--share/man/man4/virtio_console.467
-rw-r--r--sys/amd64/conf/NOTES1
-rw-r--r--sys/conf/files.amd641
-rw-r--r--sys/conf/files.i3861
-rw-r--r--sys/dev/virtio/console/virtio_console.c1409
-rw-r--r--sys/dev/virtio/console/virtio_console.h75
-rw-r--r--sys/i386/conf/NOTES1
-rw-r--r--sys/modules/virtio/Makefile2
-rw-r--r--sys/modules/virtio/console/Makefile36
11 files changed, 1595 insertions, 1 deletions
diff --git a/share/man/man4/Makefile b/share/man/man4/Makefile
index 1b5b982..a3f8c4f 100644
--- a/share/man/man4/Makefile
+++ b/share/man/man4/Makefile
@@ -559,6 +559,7 @@ MAN= aac.4 \
${_virtio.4} \
${_virtio_balloon.4} \
${_virtio_blk.4} \
+ ${_virtio_console.4} \
${_virtio_random.4} \
${_virtio_scsi.4} \
vkbd.4 \
@@ -810,6 +811,7 @@ _nxge.4= nxge.4
_virtio.4= virtio.4
_virtio_balloon.4=virtio_balloon.4
_virtio_blk.4= virtio_blk.4
+_virtio_console.4=virtio_console.4
_virtio_random.4= virtio_random.4
_virtio_scsi.4= virtio_scsi.4
_vmx.4= vmx.4
diff --git a/share/man/man4/virtio.4 b/share/man/man4/virtio.4
index 10ebf20..34516a8 100644
--- a/share/man/man4/virtio.4
+++ b/share/man/man4/virtio.4
@@ -85,6 +85,7 @@ device driver.
.Sh SEE ALSO
.Xr virtio_balloon 4 ,
.Xr virtio_blk 4 ,
+.Xr virtio_console 4 ,
.Xr virtio_scsi 4 ,
.Xr vtnet 4
.Sh HISTORY
diff --git a/share/man/man4/virtio_console.4 b/share/man/man4/virtio_console.4
new file mode 100644
index 0000000..b855f13
--- /dev/null
+++ b/share/man/man4/virtio_console.4
@@ -0,0 +1,67 @@
+.\" Copyright (c) 2014 Bryan Venteicher
+.\" All rights reserved.
+.\"
+.\" Redistribution and use in source and binary forms, with or without
+.\" modification, are permitted provided that the following conditions
+.\" are met:
+.\" 1. Redistributions of source code must retain the above copyright
+.\" notice, this list of conditions and the following disclaimer.
+.\" 2. Redistributions in binary form must reproduce the above copyright
+.\" notice, this list of conditions and the following disclaimer in the
+.\" documentation and/or other materials provided with the distribution.
+.\"
+.\" THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
+.\" ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+.\" IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
+.\" ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
+.\" FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
+.\" DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
+.\" OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
+.\" HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
+.\" LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
+.\" OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
+.\" SUCH DAMAGE.
+.\"
+.\" $FreeBSD$
+.\"
+.Dd October 22, 2014
+.Dt VIRTIO_CONSOLE 4
+.Os
+.Sh NAME
+.Nm virtio_console
+.Nd VirtIO Console driver
+.Sh SYNOPSIS
+To compile this driver into the kernel,
+place the following lines in your
+kernel configuration file:
+.Bd -ragged -offset indent
+.Cd "device virtio_console"
+.Ed
+.Pp
+Alternatively, to load the driver as a
+module at boot time, place the following line in
+.Xr loader.conf 5 :
+.Bd -literal -offset indent
+virtio_console_load="YES"
+.Ed
+.Sh DESCRIPTION
+The
+.Nm
+device driver provides support for VirtIO console devices.
+.Pp
+The console device may have one or more ports.
+Each port is similar to a simple serial interface, and
+each port is accessible through
+.Xr tty 4 .
+.Sh FILES
+.Bl -tag -width ".Pa /dev/ttyV?.??" -compact
+.It Pa /dev/ttyV?.??
+.El
+.Sh SEE ALSO
+.Xr tty 4
+.Xr virtio 4
+.Sh HISTORY
+The
+.Nm
+driver was written by
+.An Bryan Venteicher Aq bryanv@FreeBSD.org .
diff --git a/sys/amd64/conf/NOTES b/sys/amd64/conf/NOTES
index 6db6e43..61f4315 100644
--- a/sys/amd64/conf/NOTES
+++ b/sys/amd64/conf/NOTES
@@ -477,6 +477,7 @@ device virtio_blk # VirtIO Block device
device virtio_scsi # VirtIO SCSI device
device virtio_balloon # VirtIO Memory Balloon device
device virtio_random # VirtIO Entropy device
+device virtio_console # VirtIO Console device
device hyperv # HyperV drivers
diff --git a/sys/conf/files.amd64 b/sys/conf/files.amd64
index 2064219..4f7272d 100644
--- a/sys/conf/files.amd64
+++ b/sys/conf/files.amd64
@@ -469,6 +469,7 @@ dev/virtio/block/virtio_blk.c optional virtio_blk
dev/virtio/balloon/virtio_balloon.c optional virtio_balloon
dev/virtio/scsi/virtio_scsi.c optional virtio_scsi
dev/virtio/random/virtio_random.c optional virtio_random
+dev/virtio/console/virtio_console.c optional virtio_console
isa/syscons_isa.c optional sc
isa/vga_isa.c optional vga
kern/imgact_binmisc.c optional imagact_binmisc
diff --git a/sys/conf/files.i386 b/sys/conf/files.i386
index 2446a7b..d6cd350 100644
--- a/sys/conf/files.i386
+++ b/sys/conf/files.i386
@@ -416,6 +416,7 @@ dev/virtio/block/virtio_blk.c optional virtio_blk
dev/virtio/balloon/virtio_balloon.c optional virtio_balloon
dev/virtio/scsi/virtio_scsi.c optional virtio_scsi
dev/virtio/random/virtio_random.c optional virtio_random
+dev/virtio/console/virtio_console.c optional virtio_console
i386/acpica/acpi_machdep.c optional acpi
acpi_wakecode.o optional acpi \
dependency "$S/i386/acpica/acpi_wakecode.S assym.s" \
diff --git a/sys/dev/virtio/console/virtio_console.c b/sys/dev/virtio/console/virtio_console.c
new file mode 100644
index 0000000..902c85f
--- /dev/null
+++ b/sys/dev/virtio/console/virtio_console.c
@@ -0,0 +1,1409 @@
+/*-
+ * Copyright (c) 2014, Bryan Venteicher <bryanv@FreeBSD.org>
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ * 1. Redistributions of source code must retain the above copyright
+ * notice unmodified, this list of conditions, and the following
+ * disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
+ * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
+ * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
+ * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
+ * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
+ * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+ * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+ * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
+ * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+/* Driver for VirtIO console devices. */
+
+#include <sys/cdefs.h>
+__FBSDID("$FreeBSD$");
+
+#include <sys/param.h>
+#include <sys/systm.h>
+#include <sys/kernel.h>
+#include <sys/malloc.h>
+#include <sys/module.h>
+#include <sys/kdb.h>
+#include <sys/lock.h>
+#include <sys/mutex.h>
+#include <sys/sglist.h>
+#include <sys/sysctl.h>
+#include <sys/taskqueue.h>
+#include <sys/queue.h>
+
+#include <sys/conf.h>
+#include <sys/cons.h>
+#include <sys/tty.h>
+
+#include <machine/bus.h>
+#include <machine/resource.h>
+#include <sys/bus.h>
+
+#include <dev/virtio/virtio.h>
+#include <dev/virtio/virtqueue.h>
+#include <dev/virtio/console/virtio_console.h>
+
+#include "virtio_if.h"
+
+#define VTCON_MAX_PORTS 32
+#define VTCON_TTY_PREFIX "V"
+#define VTCON_BULK_BUFSZ 128
+
+/*
+ * The buffer cannot cross more than one page boundary due to the
+ * size of the sglist segment array used.
+ */
+CTASSERT(VTCON_BULK_BUFSZ <= PAGE_SIZE);
+
+struct vtcon_softc;
+struct vtcon_softc_port;
+
+struct vtcon_port {
+ struct mtx vtcport_mtx;
+ struct vtcon_softc *vtcport_sc;
+ struct vtcon_softc_port *vtcport_scport;
+ struct tty *vtcport_tty;
+ struct virtqueue *vtcport_invq;
+ struct virtqueue *vtcport_outvq;
+ int vtcport_id;
+ int vtcport_flags;
+#define VTCON_PORT_FLAG_GONE 0x01
+#define VTCON_PORT_FLAG_CONSOLE 0x02
+
+#if defined(KDB)
+ int vtcport_alt_break_state;
+#endif
+};
+
+#define VTCON_PORT_LOCK(_port) mtx_lock(&(_port)->vtcport_mtx)
+#define VTCON_PORT_UNLOCK(_port) mtx_unlock(&(_port)->vtcport_mtx)
+
+struct vtcon_softc_port {
+ struct vtcon_softc *vcsp_sc;
+ struct vtcon_port *vcsp_port;
+ struct virtqueue *vcsp_invq;
+ struct virtqueue *vcsp_outvq;
+};
+
+struct vtcon_softc {
+ device_t vtcon_dev;
+ struct mtx vtcon_mtx;
+ uint64_t vtcon_features;
+ uint32_t vtcon_max_ports;
+ uint32_t vtcon_flags;
+#define VTCON_FLAG_DETACHED 0x01
+#define VTCON_FLAG_SIZE 0x02
+#define VTCON_FLAG_MULTIPORT 0x04
+
+ /*
+ * Ports can be added and removed during runtime, but we have
+ * to allocate all the virtqueues during attach. This array is
+ * indexed by the port ID.
+ */
+ struct vtcon_softc_port *vtcon_ports;
+
+ struct task vtcon_ctrl_task;
+ struct virtqueue *vtcon_ctrl_rxvq;
+ struct virtqueue *vtcon_ctrl_txvq;
+ struct mtx vtcon_ctrl_tx_mtx;
+};
+
+#define VTCON_LOCK(_sc) mtx_lock(&(_sc)->vtcon_mtx)
+#define VTCON_UNLOCK(_sc) mtx_unlock(&(_sc)->vtcon_mtx)
+#define VTCON_LOCK_ASSERT(_sc) \
+ mtx_assert(&(_sc)->vtcon_mtx, MA_OWNED)
+#define VTCON_LOCK_ASSERT_NOTOWNED(_sc) \
+ mtx_assert(&(_sc)->vtcon_mtx, MA_NOTOWNED)
+
+#define VTCON_CTRL_TX_LOCK(_sc) mtx_lock(&(_sc)->vtcon_ctrl_tx_mtx)
+#define VTCON_CTRL_TX_UNLOCK(_sc) mtx_unlock(&(_sc)->vtcon_ctrl_tx_mtx)
+
+#define VTCON_ASSERT_VALID_PORTID(_sc, _id) \
+ KASSERT((_id) >= 0 && (_id) < (_sc)->vtcon_max_ports, \
+ ("%s: port ID %d out of range", __func__, _id))
+
+#define VTCON_FEATURES VIRTIO_CONSOLE_F_MULTIPORT
+
+static struct virtio_feature_desc vtcon_feature_desc[] = {
+ { VIRTIO_CONSOLE_F_SIZE, "ConsoleSize" },
+ { VIRTIO_CONSOLE_F_MULTIPORT, "MultiplePorts" },
+ { VIRTIO_CONSOLE_F_EMERG_WRITE, "EmergencyWrite" },
+
+ { 0, NULL }
+};
+
+static int vtcon_modevent(module_t, int, void *);
+static void vtcon_drain_all(void);
+
+static int vtcon_probe(device_t);
+static int vtcon_attach(device_t);
+static int vtcon_detach(device_t);
+static int vtcon_config_change(device_t);
+
+static void vtcon_setup_features(struct vtcon_softc *);
+static void vtcon_negotiate_features(struct vtcon_softc *);
+static int vtcon_alloc_scports(struct vtcon_softc *);
+static int vtcon_alloc_virtqueues(struct vtcon_softc *);
+static void vtcon_read_config(struct vtcon_softc *,
+ struct virtio_console_config *);
+
+static void vtcon_determine_max_ports(struct vtcon_softc *,
+ struct virtio_console_config *);
+static void vtcon_destroy_ports(struct vtcon_softc *);
+static void vtcon_stop(struct vtcon_softc *);
+
+static int vtcon_ctrl_event_enqueue(struct vtcon_softc *,
+ struct virtio_console_control *);
+static int vtcon_ctrl_event_create(struct vtcon_softc *);
+static void vtcon_ctrl_event_requeue(struct vtcon_softc *,
+ struct virtio_console_control *);
+static int vtcon_ctrl_event_populate(struct vtcon_softc *);
+static void vtcon_ctrl_event_drain(struct vtcon_softc *);
+static int vtcon_ctrl_init(struct vtcon_softc *);
+static void vtcon_ctrl_deinit(struct vtcon_softc *);
+static void vtcon_ctrl_port_add_event(struct vtcon_softc *, int);
+static void vtcon_ctrl_port_remove_event(struct vtcon_softc *, int);
+static void vtcon_ctrl_port_console_event(struct vtcon_softc *, int);
+static void vtcon_ctrl_port_open_event(struct vtcon_softc *, int);
+static void vtcon_ctrl_process_event(struct vtcon_softc *,
+ struct virtio_console_control *);
+static void vtcon_ctrl_task_cb(void *, int);
+static void vtcon_ctrl_event_intr(void *);
+static void vtcon_ctrl_poll(struct vtcon_softc *,
+ struct virtio_console_control *control);
+static void vtcon_ctrl_send_control(struct vtcon_softc *, uint32_t,
+ uint16_t, uint16_t);
+
+static int vtcon_port_enqueue_buf(struct vtcon_port *, void *, size_t);
+static int vtcon_port_create_buf(struct vtcon_port *);
+static void vtcon_port_requeue_buf(struct vtcon_port *, void *);
+static int vtcon_port_populate(struct vtcon_port *);
+static void vtcon_port_destroy(struct vtcon_port *);
+static int vtcon_port_create(struct vtcon_softc *, int);
+static void vtcon_port_drain_bufs(struct virtqueue *);
+static void vtcon_port_drain(struct vtcon_port *);
+static void vtcon_port_teardown(struct vtcon_port *);
+static void vtcon_port_change_size(struct vtcon_port *, uint16_t,
+ uint16_t);
+static void vtcon_port_update_console_size(struct vtcon_softc *);
+static void vtcon_port_enable_intr(struct vtcon_port *);
+static void vtcon_port_disable_intr(struct vtcon_port *);
+static void vtcon_port_in(struct vtcon_port *);
+static void vtcon_port_intr(void *);
+static void vtcon_port_out(struct vtcon_port *, void *, int);
+static void vtcon_port_submit_event(struct vtcon_port *, uint16_t,
+ uint16_t);
+
+static int vtcon_tty_open(struct tty *);
+static void vtcon_tty_close(struct tty *);
+static void vtcon_tty_outwakeup(struct tty *);
+static void vtcon_tty_free(void *);
+
+static void vtcon_get_console_size(struct vtcon_softc *, uint16_t *,
+ uint16_t *);
+
+static void vtcon_enable_interrupts(struct vtcon_softc *);
+static void vtcon_disable_interrupts(struct vtcon_softc *);
+
+static int vtcon_pending_free;
+
+static struct ttydevsw vtcon_tty_class = {
+ .tsw_flags = 0,
+ .tsw_open = vtcon_tty_open,
+ .tsw_close = vtcon_tty_close,
+ .tsw_outwakeup = vtcon_tty_outwakeup,
+ .tsw_free = vtcon_tty_free,
+};
+
+static device_method_t vtcon_methods[] = {
+ /* Device methods. */
+ DEVMETHOD(device_probe, vtcon_probe),
+ DEVMETHOD(device_attach, vtcon_attach),
+ DEVMETHOD(device_detach, vtcon_detach),
+
+ /* VirtIO methods. */
+ DEVMETHOD(virtio_config_change, vtcon_config_change),
+
+ DEVMETHOD_END
+};
+
+static driver_t vtcon_driver = {
+ "vtcon",
+ vtcon_methods,
+ sizeof(struct vtcon_softc)
+};
+static devclass_t vtcon_devclass;
+
+DRIVER_MODULE(virtio_console, virtio_pci, vtcon_driver, vtcon_devclass,
+ vtcon_modevent, 0);
+MODULE_VERSION(virtio_console, 1);
+MODULE_DEPEND(virtio_console, virtio, 1, 1, 1);
+
+static int
+vtcon_modevent(module_t mod, int type, void *unused)
+{
+ int error;
+
+ switch (type) {
+ case MOD_LOAD:
+ error = 0;
+ break;
+ case MOD_QUIESCE:
+ error = 0;
+ break;
+ case MOD_UNLOAD:
+ vtcon_drain_all();
+ error = 0;
+ break;
+ case MOD_SHUTDOWN:
+ error = 0;
+ break;
+ default:
+ error = EOPNOTSUPP;
+ break;
+ }
+
+ return (error);
+}
+
+static void
+vtcon_drain_all(void)
+{
+ int first;
+
+ for (first = 1; vtcon_pending_free != 0; first = 0) {
+ if (first != 0) {
+ printf("virtio_console: Waiting for all detached TTY "
+ "devices to have open fds closed.\n");
+ }
+ pause("vtcondra", hz);
+ }
+}
+
+static int
+vtcon_probe(device_t dev)
+{
+
+ if (virtio_get_device_type(dev) != VIRTIO_ID_CONSOLE)
+ return (ENXIO);
+
+ device_set_desc(dev, "VirtIO Console Adapter");
+
+ return (BUS_PROBE_DEFAULT);
+}
+
+static int
+vtcon_attach(device_t dev)
+{
+ struct vtcon_softc *sc;
+ struct virtio_console_config concfg;
+ int error;
+
+ sc = device_get_softc(dev);
+ sc->vtcon_dev = dev;
+
+ mtx_init(&sc->vtcon_mtx, "vtconmtx", NULL, MTX_DEF);
+ mtx_init(&sc->vtcon_ctrl_tx_mtx, "vtconctrlmtx", NULL, MTX_DEF);
+
+ virtio_set_feature_desc(dev, vtcon_feature_desc);
+ vtcon_setup_features(sc);
+
+ vtcon_read_config(sc, &concfg);
+ vtcon_determine_max_ports(sc, &concfg);
+
+ error = vtcon_alloc_scports(sc);
+ if (error) {
+ device_printf(dev, "cannot allocate softc port structures\n");
+ goto fail;
+ }
+
+ error = vtcon_alloc_virtqueues(sc);
+ if (error) {
+ device_printf(dev, "cannot allocate virtqueues\n");
+ goto fail;
+ }
+
+ if (sc->vtcon_flags & VTCON_FLAG_MULTIPORT) {
+ TASK_INIT(&sc->vtcon_ctrl_task, 0, vtcon_ctrl_task_cb, sc);
+ error = vtcon_ctrl_init(sc);
+ if (error)
+ goto fail;
+ } else {
+ error = vtcon_port_create(sc, 0);
+ if (error)
+ goto fail;
+ if (sc->vtcon_flags & VTCON_FLAG_SIZE)
+ vtcon_port_update_console_size(sc);
+ }
+
+ error = virtio_setup_intr(dev, INTR_TYPE_TTY);
+ if (error) {
+ device_printf(dev, "cannot setup virtqueue interrupts\n");
+ goto fail;
+ }
+
+ vtcon_enable_interrupts(sc);
+
+ vtcon_ctrl_send_control(sc, VIRTIO_CONSOLE_BAD_ID,
+ VIRTIO_CONSOLE_DEVICE_READY, 1);
+
+fail:
+ if (error)
+ vtcon_detach(dev);
+
+ return (error);
+}
+
+static int
+vtcon_detach(device_t dev)
+{
+ struct vtcon_softc *sc;
+
+ sc = device_get_softc(dev);
+
+ VTCON_LOCK(sc);
+ sc->vtcon_flags |= VTCON_FLAG_DETACHED;
+ if (device_is_attached(dev))
+ vtcon_stop(sc);
+ VTCON_UNLOCK(sc);
+
+ if (sc->vtcon_flags & VTCON_FLAG_MULTIPORT) {
+ taskqueue_drain(taskqueue_thread, &sc->vtcon_ctrl_task);
+ vtcon_ctrl_deinit(sc);
+ }
+
+ vtcon_destroy_ports(sc);
+ mtx_destroy(&sc->vtcon_mtx);
+ mtx_destroy(&sc->vtcon_ctrl_tx_mtx);
+
+ return (0);
+}
+
+static int
+vtcon_config_change(device_t dev)
+{
+ struct vtcon_softc *sc;
+
+ sc = device_get_softc(dev);
+
+ /*
+ * When the multiport feature is negotiated, all configuration
+ * changes are done through control virtqueue events.
+ */
+ if ((sc->vtcon_flags & VTCON_FLAG_MULTIPORT) == 0) {
+ if (sc->vtcon_flags & VTCON_FLAG_SIZE)
+ vtcon_port_update_console_size(sc);
+ }
+
+ return (0);
+}
+
+static void
+vtcon_negotiate_features(struct vtcon_softc *sc)
+{
+ device_t dev;
+ uint64_t features;
+
+ dev = sc->vtcon_dev;
+ features = VTCON_FEATURES;
+
+ sc->vtcon_features = virtio_negotiate_features(dev, features);
+}
+
+static void
+vtcon_setup_features(struct vtcon_softc *sc)
+{
+ device_t dev;
+
+ dev = sc->vtcon_dev;
+
+ vtcon_negotiate_features(sc);
+
+ if (virtio_with_feature(dev, VIRTIO_CONSOLE_F_SIZE))
+ sc->vtcon_flags |= VTCON_FLAG_SIZE;
+ if (virtio_with_feature(dev, VIRTIO_CONSOLE_F_MULTIPORT))
+ sc->vtcon_flags |= VTCON_FLAG_MULTIPORT;
+}
+
+#define VTCON_GET_CONFIG(_dev, _feature, _field, _cfg) \
+ if (virtio_with_feature(_dev, _feature)) { \
+ virtio_read_device_config(_dev, \
+ offsetof(struct virtio_console_config, _field), \
+ &(_cfg)->_field, sizeof((_cfg)->_field)); \
+ }
+
+static void
+vtcon_read_config(struct vtcon_softc *sc, struct virtio_console_config *concfg)
+{
+ device_t dev;
+
+ dev = sc->vtcon_dev;
+
+ bzero(concfg, sizeof(struct virtio_console_config));
+
+ VTCON_GET_CONFIG(dev, VIRTIO_CONSOLE_F_SIZE, cols, concfg);
+ VTCON_GET_CONFIG(dev, VIRTIO_CONSOLE_F_SIZE, rows, concfg);
+ VTCON_GET_CONFIG(dev, VIRTIO_CONSOLE_F_MULTIPORT, max_nr_ports, concfg);
+}
+
+#undef VTCON_GET_CONFIG
+
+static int
+vtcon_alloc_scports(struct vtcon_softc *sc)
+{
+ struct vtcon_softc_port *scport;
+ int max, i;
+
+ max = sc->vtcon_max_ports;
+
+ sc->vtcon_ports = malloc(sizeof(struct vtcon_softc_port) * max,
+ M_DEVBUF, M_NOWAIT | M_ZERO);
+ if (sc->vtcon_ports == NULL)
+ return (ENOMEM);
+
+ for (i = 0; i < max; i++) {
+ scport = &sc->vtcon_ports[i];
+ scport->vcsp_sc = sc;
+ }
+
+ return (0);
+}
+
+static int
+vtcon_alloc_virtqueues(struct vtcon_softc *sc)
+{
+ device_t dev;
+ struct vq_alloc_info *info;
+ struct vtcon_softc_port *scport;
+ int i, idx, portidx, nvqs, error;
+
+ dev = sc->vtcon_dev;
+
+ nvqs = sc->vtcon_max_ports * 2;
+ if (sc->vtcon_flags & VTCON_FLAG_MULTIPORT)
+ nvqs += 2;
+
+ info = malloc(sizeof(struct vq_alloc_info) * nvqs, M_TEMP, M_NOWAIT);
+ if (info == NULL)
+ return (ENOMEM);
+
+ for (i = 0, idx = 0, portidx = 0; i < nvqs / 2; i++, idx += 2) {
+
+ if (i == 1) {
+ /* The control virtqueues are after the first port. */
+ VQ_ALLOC_INFO_INIT(&info[idx], 0,
+ vtcon_ctrl_event_intr, sc, &sc->vtcon_ctrl_rxvq,
+ "%s-control rx", device_get_nameunit(dev));
+ VQ_ALLOC_INFO_INIT(&info[idx+1], 0,
+ NULL, sc, &sc->vtcon_ctrl_txvq,
+ "%s-control tx", device_get_nameunit(dev));
+ continue;
+ }
+
+ scport = &sc->vtcon_ports[portidx];
+
+ VQ_ALLOC_INFO_INIT(&info[idx], 0, vtcon_port_intr,
+ scport, &scport->vcsp_invq, "%s-port%d in",
+ device_get_nameunit(dev), i);
+ VQ_ALLOC_INFO_INIT(&info[idx+1], 0, NULL,
+ NULL, &scport->vcsp_outvq, "%s-port%d out",
+ device_get_nameunit(dev), i);
+
+ portidx++;
+ }
+
+ error = virtio_alloc_virtqueues(dev, 0, nvqs, info);
+ free(info, M_TEMP);
+
+ return (error);
+}
+
+static void
+vtcon_determine_max_ports(struct vtcon_softc *sc,
+ struct virtio_console_config *concfg)
+{
+
+ if (sc->vtcon_flags & VTCON_FLAG_MULTIPORT) {
+ sc->vtcon_max_ports =
+ min(concfg->max_nr_ports, VTCON_MAX_PORTS);
+ if (sc->vtcon_max_ports == 0)
+ sc->vtcon_max_ports = 1;
+ } else
+ sc->vtcon_max_ports = 1;
+}
+
+static void
+vtcon_destroy_ports(struct vtcon_softc *sc)
+{
+ struct vtcon_softc_port *scport;
+ struct vtcon_port *port;
+ struct virtqueue *vq;
+ int i;
+
+ if (sc->vtcon_ports == NULL)
+ return;
+
+ VTCON_LOCK(sc);
+ for (i = 0; i < sc->vtcon_max_ports; i++) {
+ scport = &sc->vtcon_ports[i];
+
+ port = scport->vcsp_port;
+ if (port != NULL) {
+ scport->vcsp_port = NULL;
+ VTCON_PORT_LOCK(port);
+ VTCON_UNLOCK(sc);
+ vtcon_port_teardown(port);
+ VTCON_LOCK(sc);
+ }
+
+ vq = scport->vcsp_invq;
+ if (vq != NULL)
+ vtcon_port_drain_bufs(vq);
+ }
+ VTCON_UNLOCK(sc);
+
+ free(sc->vtcon_ports, M_DEVBUF);
+ sc->vtcon_ports = NULL;
+}
+
+static void
+vtcon_stop(struct vtcon_softc *sc)
+{
+
+ vtcon_disable_interrupts(sc);
+ virtio_stop(sc->vtcon_dev);
+}
+
+static int
+vtcon_ctrl_event_enqueue(struct vtcon_softc *sc,
+ struct virtio_console_control *control)
+{
+ struct sglist_seg segs[2];
+ struct sglist sg;
+ struct virtqueue *vq;
+ int error;
+
+ vq = sc->vtcon_ctrl_rxvq;
+
+ sglist_init(&sg, 2, segs);
+ error = sglist_append(&sg, control,
+ sizeof(struct virtio_console_control));
+ KASSERT(error == 0, ("%s: error %d adding control to sglist",
+ __func__, error));
+
+ return (virtqueue_enqueue(vq, control, &sg, 0, sg.sg_nseg));
+}
+
+static int
+vtcon_ctrl_event_create(struct vtcon_softc *sc)
+{
+ struct virtio_console_control *control;
+ int error;
+
+ control = malloc(sizeof(struct virtio_console_control), M_DEVBUF,
+ M_ZERO | M_NOWAIT);
+ if (control == NULL)
+ return (ENOMEM);
+
+ error = vtcon_ctrl_event_enqueue(sc, control);
+ if (error)
+ free(control, M_DEVBUF);
+
+ return (error);
+}
+
+static void
+vtcon_ctrl_event_requeue(struct vtcon_softc *sc,
+ struct virtio_console_control *control)
+{
+ int error;
+
+ bzero(control, sizeof(struct virtio_console_control));
+
+ error = vtcon_ctrl_event_enqueue(sc, control);
+ KASSERT(error == 0,
+ ("%s: cannot requeue control buffer %d", __func__, error));
+}
+
+static int
+vtcon_ctrl_event_populate(struct vtcon_softc *sc)
+{
+ struct virtqueue *vq;
+ int nbufs, error;
+
+ vq = sc->vtcon_ctrl_rxvq;
+ error = ENOSPC;
+
+ for (nbufs = 0; !virtqueue_full(vq); nbufs++) {
+ error = vtcon_ctrl_event_create(sc);
+ if (error)
+ break;
+ }
+
+ if (nbufs > 0) {
+ virtqueue_notify(vq);
+ error = 0;
+ }
+
+ return (error);
+}
+
+static void
+vtcon_ctrl_event_drain(struct vtcon_softc *sc)
+{
+ struct virtio_console_control *control;
+ struct virtqueue *vq;
+ int last;
+
+ vq = sc->vtcon_ctrl_rxvq;
+ last = 0;
+
+ if (vq == NULL)
+ return;
+
+ VTCON_LOCK(sc);
+ while ((control = virtqueue_drain(vq, &last)) != NULL)
+ free(control, M_DEVBUF);
+ VTCON_UNLOCK(sc);
+}
+
+static int
+vtcon_ctrl_init(struct vtcon_softc *sc)
+{
+ int error;
+
+ error = vtcon_ctrl_event_populate(sc);
+
+ return (error);
+}
+
+static void
+vtcon_ctrl_deinit(struct vtcon_softc *sc)
+{
+
+ vtcon_ctrl_event_drain(sc);
+}
+
+static void
+vtcon_ctrl_port_add_event(struct vtcon_softc *sc, int id)
+{
+ device_t dev;
+ int error;
+
+ dev = sc->vtcon_dev;
+
+ /* This single thread only way for ports to be created. */
+ if (sc->vtcon_ports[id].vcsp_port != NULL) {
+ device_printf(dev, "%s: adding port %d, but already exists\n",
+ __func__, id);
+ return;
+ }
+
+ error = vtcon_port_create(sc, id);
+ if (error) {
+ device_printf(dev, "%s: cannot create port %d: %d\n",
+ __func__, id, error);
+ vtcon_ctrl_send_control(sc, id, VIRTIO_CONSOLE_PORT_READY, 0);
+ return;
+ }
+}
+
+static void
+vtcon_ctrl_port_remove_event(struct vtcon_softc *sc, int id)
+{
+ device_t dev;
+ struct vtcon_softc_port *scport;
+ struct vtcon_port *port;
+
+ dev = sc->vtcon_dev;
+ scport = &sc->vtcon_ports[id];
+
+ VTCON_LOCK(sc);
+ port = scport->vcsp_port;
+ if (port == NULL) {
+ VTCON_UNLOCK(sc);
+ device_printf(dev, "%s: remove port %d, but does not exist\n",
+ __func__, id);
+ return;
+ }
+
+ scport->vcsp_port = NULL;
+ VTCON_PORT_LOCK(port);
+ VTCON_UNLOCK(sc);
+ vtcon_port_teardown(port);
+}
+
+static void
+vtcon_ctrl_port_console_event(struct vtcon_softc *sc, int id)
+{
+ device_t dev;
+ struct vtcon_softc_port *scport;
+ struct vtcon_port *port;
+
+ dev = sc->vtcon_dev;
+ scport = &sc->vtcon_ports[id];
+
+ VTCON_LOCK(sc);
+ port = scport->vcsp_port;
+ if (port == NULL) {
+ VTCON_UNLOCK(sc);
+ device_printf(dev, "%s: console port %d, but does not exist\n",
+ __func__, id);
+ return;
+ }
+
+ VTCON_PORT_LOCK(port);
+ VTCON_UNLOCK(sc);
+ port->vtcport_flags |= VTCON_PORT_FLAG_CONSOLE;
+ vtcon_port_submit_event(port, VIRTIO_CONSOLE_PORT_OPEN, 1);
+ VTCON_PORT_UNLOCK(port);
+}
+
+static void
+vtcon_ctrl_port_open_event(struct vtcon_softc *sc, int id)
+{
+ device_t dev;
+ struct vtcon_softc_port *scport;
+ struct vtcon_port *port;
+
+ dev = sc->vtcon_dev;
+ scport = &sc->vtcon_ports[id];
+
+ VTCON_LOCK(sc);
+ port = scport->vcsp_port;
+ if (port == NULL) {
+ VTCON_UNLOCK(sc);
+ device_printf(dev, "%s: open port %d, but does not exist\n",
+ __func__, id);
+ return;
+ }
+
+ VTCON_PORT_LOCK(port);
+ VTCON_UNLOCK(sc);
+ vtcon_port_enable_intr(port);
+ VTCON_PORT_UNLOCK(port);
+}
+
+static void
+vtcon_ctrl_process_event(struct vtcon_softc *sc,
+ struct virtio_console_control *control)
+{
+ device_t dev;
+ int id;
+
+ dev = sc->vtcon_dev;
+ id = control->id;
+
+ if (id < 0 || id >= sc->vtcon_max_ports) {
+ device_printf(dev, "%s: invalid port ID %d\n", __func__, id);
+ return;
+ }
+
+ switch (control->event) {
+ case VIRTIO_CONSOLE_PORT_ADD:
+ vtcon_ctrl_port_add_event(sc, id);
+ break;
+
+ case VIRTIO_CONSOLE_PORT_REMOVE:
+ vtcon_ctrl_port_remove_event(sc, id);
+ break;
+
+ case VIRTIO_CONSOLE_CONSOLE_PORT:
+ vtcon_ctrl_port_console_event(sc, id);
+ break;
+
+ case VIRTIO_CONSOLE_RESIZE:
+ break;
+
+ case VIRTIO_CONSOLE_PORT_OPEN:
+ vtcon_ctrl_port_open_event(sc, id);
+ break;
+
+ case VIRTIO_CONSOLE_PORT_NAME:
+ break;
+ }
+}
+
+static void
+vtcon_ctrl_task_cb(void *xsc, int pending)
+{
+ struct vtcon_softc *sc;
+ struct virtqueue *vq;
+ struct virtio_console_control *control;
+ int detached;
+
+ sc = xsc;
+ vq = sc->vtcon_ctrl_rxvq;
+
+ VTCON_LOCK(sc);
+
+ while ((detached = (sc->vtcon_flags & VTCON_FLAG_DETACHED)) == 0) {
+ control = virtqueue_dequeue(vq, NULL);
+ if (control == NULL)
+ break;
+
+ VTCON_UNLOCK(sc);
+ vtcon_ctrl_process_event(sc, control);
+ VTCON_LOCK(sc);
+ vtcon_ctrl_event_requeue(sc, control);
+ }
+
+ if (!detached) {
+ virtqueue_notify(vq);
+ if (virtqueue_enable_intr(vq) != 0)
+ taskqueue_enqueue(taskqueue_thread,
+ &sc->vtcon_ctrl_task);
+ }
+
+ VTCON_UNLOCK(sc);
+}
+
+static void
+vtcon_ctrl_event_intr(void *xsc)
+{
+ struct vtcon_softc *sc;
+
+ sc = xsc;
+
+ /*
+ * Only some events require us to potentially block, but it
+ * easier to just defer all event handling to the taskqueue.
+ */
+ taskqueue_enqueue(taskqueue_thread, &sc->vtcon_ctrl_task);
+}
+
+static void
+vtcon_ctrl_poll(struct vtcon_softc *sc,
+ struct virtio_console_control *control)
+{
+ struct sglist_seg segs[2];
+ struct sglist sg;
+ struct virtqueue *vq;
+ int error;
+
+ vq = sc->vtcon_ctrl_txvq;
+
+ sglist_init(&sg, 2, segs);
+ error = sglist_append(&sg, control,
+ sizeof(struct virtio_console_control));
+ KASSERT(error == 0, ("%s: error %d adding control to sglist",
+ __func__, error));
+
+ /*
+ * We cannot use the softc lock to serialize access to this
+ * virtqueue since this is called from the tty layer with the
+ * port lock held. Acquiring the softc would violate our lock
+ * ordering.
+ */
+ VTCON_CTRL_TX_LOCK(sc);
+ KASSERT(virtqueue_empty(vq),
+ ("%s: virtqueue is not emtpy", __func__));
+ error = virtqueue_enqueue(vq, control, &sg, sg.sg_nseg, 0);
+ if (error == 0) {
+ virtqueue_notify(vq);
+ virtqueue_poll(vq, NULL);
+ }
+ VTCON_CTRL_TX_UNLOCK(sc);
+}
+
+static void
+vtcon_ctrl_send_control(struct vtcon_softc *sc, uint32_t portid,
+ uint16_t event, uint16_t value)
+{
+ struct virtio_console_control control;
+
+ if ((sc->vtcon_flags & VTCON_FLAG_MULTIPORT) == 0)
+ return;
+
+ control.id = portid;
+ control.event = event;
+ control.value = value;
+
+ vtcon_ctrl_poll(sc, &control);
+}
+
+static int
+vtcon_port_enqueue_buf(struct vtcon_port *port, void *buf, size_t len)
+{
+ struct sglist_seg segs[2];
+ struct sglist sg;
+ struct virtqueue *vq;
+ int error;
+
+ vq = port->vtcport_invq;
+
+ sglist_init(&sg, 2, segs);
+ error = sglist_append(&sg, buf, len);
+ KASSERT(error == 0,
+ ("%s: error %d adding buffer to sglist", __func__, error));
+
+ error = virtqueue_enqueue(vq, buf, &sg, 0, sg.sg_nseg);
+
+ return (error);
+}
+
+static int
+vtcon_port_create_buf(struct vtcon_port *port)
+{
+ void *buf;
+ int error;
+
+ buf = malloc(VTCON_BULK_BUFSZ, M_DEVBUF, M_ZERO | M_NOWAIT);
+ if (buf == NULL)
+ return (ENOMEM);
+
+ error = vtcon_port_enqueue_buf(port, buf, VTCON_BULK_BUFSZ);
+ if (error)
+ free(buf, M_DEVBUF);
+
+ return (error);
+}
+
+static void
+vtcon_port_requeue_buf(struct vtcon_port *port, void *buf)
+{
+ int error;
+
+ error = vtcon_port_enqueue_buf(port, buf, VTCON_BULK_BUFSZ);
+ KASSERT(error == 0,
+ ("%s: cannot requeue input buffer %d", __func__, error));
+}
+
+static int
+vtcon_port_populate(struct vtcon_port *port)
+{
+ struct virtqueue *vq;
+ int nbufs, error;
+
+ vq = port->vtcport_invq;
+ error = ENOSPC;
+
+ for (nbufs = 0; !virtqueue_full(vq); nbufs++) {
+ error = vtcon_port_create_buf(port);
+ if (error)
+ break;
+ }
+
+ if (nbufs > 0) {
+ virtqueue_notify(vq);
+ error = 0;
+ }
+
+ return (error);
+}
+
+static void
+vtcon_port_destroy(struct vtcon_port *port)
+{
+
+ port->vtcport_sc = NULL;
+ port->vtcport_scport = NULL;
+ port->vtcport_invq = NULL;
+ port->vtcport_outvq = NULL;
+ port->vtcport_id = -1;
+ mtx_destroy(&port->vtcport_mtx);
+ free(port, M_DEVBUF);
+}
+
+static int
+vtcon_port_init_vqs(struct vtcon_port *port)
+{
+ struct vtcon_softc_port *scport;
+ int error;
+
+ scport = port->vtcport_scport;
+
+ port->vtcport_invq = scport->vcsp_invq;
+ port->vtcport_outvq = scport->vcsp_outvq;
+
+ /*
+ * Free any data left over from when this virtqueue was in use by a
+ * prior port. We have not yet notified the host that the port is
+ * ready, so assume nothing in the virtqueue can be for us.
+ */
+ vtcon_port_drain(port);
+
+ KASSERT(virtqueue_empty(port->vtcport_invq),
+ ("%s: in virtqueue is not empty", __func__));
+ KASSERT(virtqueue_empty(port->vtcport_outvq),
+ ("%s: out virtqueue is not empty", __func__));
+
+ error = vtcon_port_populate(port);
+ if (error)
+ return (error);
+
+ return (0);
+}
+
+static int
+vtcon_port_create(struct vtcon_softc *sc, int id)
+{
+ device_t dev;
+ struct vtcon_softc_port *scport;
+ struct vtcon_port *port;
+ int error;
+
+ dev = sc->vtcon_dev;
+ scport = &sc->vtcon_ports[id];
+
+ VTCON_ASSERT_VALID_PORTID(sc, id);
+ MPASS(scport->vcsp_port == NULL);
+
+ port = malloc(sizeof(struct vtcon_port), M_DEVBUF, M_NOWAIT | M_ZERO);
+ if (port == NULL)
+ return (ENOMEM);
+
+ port->vtcport_sc = sc;
+ port->vtcport_scport = scport;
+ port->vtcport_id = id;
+ mtx_init(&port->vtcport_mtx, "vtcpmtx", NULL, MTX_DEF);
+ port->vtcport_tty = tty_alloc_mutex(&vtcon_tty_class, port,
+ &port->vtcport_mtx);
+
+ error = vtcon_port_init_vqs(port);
+ if (error) {
+ VTCON_PORT_LOCK(port);
+ vtcon_port_teardown(port);
+ return (error);
+ }
+
+ VTCON_LOCK(sc);
+ VTCON_PORT_LOCK(port);
+ scport->vcsp_port = port;
+ vtcon_port_enable_intr(port);
+ vtcon_port_submit_event(port, VIRTIO_CONSOLE_PORT_READY, 1);
+ VTCON_PORT_UNLOCK(port);
+ VTCON_UNLOCK(sc);
+
+ tty_makedev(port->vtcport_tty, NULL, "%s%r.%r", VTCON_TTY_PREFIX,
+ device_get_unit(dev), id);
+
+ return (0);
+}
+
+static void
+vtcon_port_drain_bufs(struct virtqueue *vq)
+{
+ void *buf;
+ int last;
+
+ last = 0;
+
+ while ((buf = virtqueue_drain(vq, &last)) != NULL)
+ free(buf, M_DEVBUF);
+}
+
+static void
+vtcon_port_drain(struct vtcon_port *port)
+{
+
+ vtcon_port_drain_bufs(port->vtcport_invq);
+}
+
+static void
+vtcon_port_teardown(struct vtcon_port *port)
+{
+ struct tty *tp;
+
+ tp = port->vtcport_tty;
+
+ port->vtcport_flags |= VTCON_PORT_FLAG_GONE;
+
+ if (tp != NULL) {
+ atomic_add_int(&vtcon_pending_free, 1);
+ tty_rel_gone(tp);
+ } else
+ vtcon_port_destroy(port);
+}
+
+static void
+vtcon_port_change_size(struct vtcon_port *port, uint16_t cols, uint16_t rows)
+{
+ struct tty *tp;
+ struct winsize sz;
+
+ tp = port->vtcport_tty;
+
+ if (tp == NULL)
+ return;
+
+ bzero(&sz, sizeof(struct winsize));
+ sz.ws_col = cols;
+ sz.ws_row = rows;
+
+ tty_set_winsize(tp, &sz);
+}
+
+static void
+vtcon_port_update_console_size(struct vtcon_softc *sc)
+{
+ struct vtcon_port *port;
+ struct vtcon_softc_port *scport;
+ uint16_t cols, rows;
+
+ vtcon_get_console_size(sc, &cols, &rows);
+
+ /*
+ * For now, assume the first (only) port is the console. Note
+ * QEMU does not implement this feature yet.
+ */
+ scport = &sc->vtcon_ports[0];
+
+ VTCON_LOCK(sc);
+ port = scport->vcsp_port;
+
+ if (port != NULL) {
+ VTCON_PORT_LOCK(port);
+ VTCON_UNLOCK(sc);
+ vtcon_port_change_size(port, cols, rows);
+ VTCON_PORT_UNLOCK(port);
+ } else
+ VTCON_UNLOCK(sc);
+}
+
+static void
+vtcon_port_enable_intr(struct vtcon_port *port)
+{
+
+ /*
+ * NOTE: The out virtqueue is always polled, so its interupt
+ * kept disabled.
+ */
+ virtqueue_enable_intr(port->vtcport_invq);
+}
+
+static void
+vtcon_port_disable_intr(struct vtcon_port *port)
+{
+
+ if (port->vtcport_invq != NULL)
+ virtqueue_disable_intr(port->vtcport_invq);
+ if (port->vtcport_outvq != NULL)
+ virtqueue_disable_intr(port->vtcport_outvq);
+}
+
+static void
+vtcon_port_in(struct vtcon_port *port)
+{
+ struct virtqueue *vq;
+ struct tty *tp;
+ char *buf;
+ uint32_t len;
+ int i, deq;
+
+ tp = port->vtcport_tty;
+ vq = port->vtcport_invq;
+
+again:
+ deq = 0;
+
+ while ((buf = virtqueue_dequeue(vq, &len)) != NULL) {
+ for (i = 0; i < len; i++) {
+#if defined(KDB)
+ if (port->vtcport_flags & VTCON_PORT_FLAG_CONSOLE)
+ kdb_alt_break(buf[i],
+ &port->vtcport_alt_break_state);
+#endif
+ ttydisc_rint(tp, buf[i], 0);
+ }
+ vtcon_port_requeue_buf(port, buf);
+ deq++;
+ }
+ ttydisc_rint_done(tp);
+
+ if (deq > 0)
+ virtqueue_notify(vq);
+
+ if (virtqueue_enable_intr(vq) != 0)
+ goto again;
+}
+
+static void
+vtcon_port_intr(void *scportx)
+{
+ struct vtcon_softc_port *scport;
+ struct vtcon_softc *sc;
+ struct vtcon_port *port;
+
+ scport = scportx;
+ sc = scport->vcsp_sc;
+
+ VTCON_LOCK(sc);
+ port = scport->vcsp_port;
+ if (port == NULL) {
+ VTCON_UNLOCK(sc);
+ return;
+ }
+ VTCON_PORT_LOCK(port);
+ VTCON_UNLOCK(sc);
+ if ((port->vtcport_flags & VTCON_PORT_FLAG_GONE) == 0)
+ vtcon_port_in(port);
+ VTCON_PORT_UNLOCK(port);
+}
+
+static void
+vtcon_port_out(struct vtcon_port *port, void *buf, int bufsize)
+{
+ struct sglist_seg segs[2];
+ struct sglist sg;
+ struct virtqueue *vq;
+ int error;
+
+ vq = port->vtcport_outvq;
+ KASSERT(virtqueue_empty(vq),
+ ("%s: port %p out virtqueue not emtpy", __func__, port));
+
+ sglist_init(&sg, 2, segs);
+ error = sglist_append(&sg, buf, bufsize);
+ KASSERT(error == 0, ("%s: error %d adding buffer to sglist",
+ __func__, error));
+
+ error = virtqueue_enqueue(vq, buf, &sg, sg.sg_nseg, 0);
+ if (error == 0) {
+ virtqueue_notify(vq);
+ virtqueue_poll(vq, NULL);
+ }
+}
+
+static void
+vtcon_port_submit_event(struct vtcon_port *port, uint16_t event,
+ uint16_t value)
+{
+ struct vtcon_softc *sc;
+
+ sc = port->vtcport_sc;
+
+ vtcon_ctrl_send_control(sc, port->vtcport_id, event, value);
+}
+
+static int
+vtcon_tty_open(struct tty *tp)
+{
+ struct vtcon_port *port;
+
+ port = tty_softc(tp);
+
+ if (port->vtcport_flags & VTCON_PORT_FLAG_GONE)
+ return (ENXIO);
+
+ vtcon_port_submit_event(port, VIRTIO_CONSOLE_PORT_OPEN, 1);
+
+ return (0);
+}
+
+static void
+vtcon_tty_close(struct tty *tp)
+{
+ struct vtcon_port *port;
+
+ port = tty_softc(tp);
+
+ if (port->vtcport_flags & VTCON_PORT_FLAG_GONE)
+ return;
+
+ vtcon_port_submit_event(port, VIRTIO_CONSOLE_PORT_OPEN, 0);
+}
+
+static void
+vtcon_tty_outwakeup(struct tty *tp)
+{
+ struct vtcon_port *port;
+ char buf[VTCON_BULK_BUFSZ];
+ int len;
+
+ port = tty_softc(tp);
+
+ if (port->vtcport_flags & VTCON_PORT_FLAG_GONE)
+ return;
+
+ while ((len = ttydisc_getc(tp, buf, sizeof(buf))) != 0)
+ vtcon_port_out(port, buf, len);
+}
+
+static void
+vtcon_tty_free(void *xport)
+{
+ struct vtcon_port *port;
+
+ port = xport;
+
+ vtcon_port_destroy(port);
+ atomic_subtract_int(&vtcon_pending_free, 1);
+}
+
+static void
+vtcon_get_console_size(struct vtcon_softc *sc, uint16_t *cols, uint16_t *rows)
+{
+ struct virtio_console_config concfg;
+
+ KASSERT(sc->vtcon_flags & VTCON_FLAG_SIZE,
+ ("%s: size feature not negotiated", __func__));
+
+ vtcon_read_config(sc, &concfg);
+
+ *cols = concfg.cols;
+ *rows = concfg.rows;
+}
+
+static void
+vtcon_enable_interrupts(struct vtcon_softc *sc)
+{
+ struct vtcon_softc_port *scport;
+ struct vtcon_port *port;
+ int i;
+
+ VTCON_LOCK(sc);
+
+ if (sc->vtcon_flags & VTCON_FLAG_MULTIPORT)
+ virtqueue_enable_intr(sc->vtcon_ctrl_rxvq);
+
+ for (i = 0; i < sc->vtcon_max_ports; i++) {
+ scport = &sc->vtcon_ports[i];
+
+ port = scport->vcsp_port;
+ if (port == NULL)
+ continue;
+
+ VTCON_PORT_LOCK(port);
+ vtcon_port_enable_intr(port);
+ VTCON_PORT_UNLOCK(port);
+ }
+
+ VTCON_UNLOCK(sc);
+}
+
+static void
+vtcon_disable_interrupts(struct vtcon_softc *sc)
+{
+ struct vtcon_softc_port *scport;
+ struct vtcon_port *port;
+ int i;
+
+ VTCON_LOCK_ASSERT(sc);
+
+ if (sc->vtcon_flags & VTCON_FLAG_MULTIPORT)
+ virtqueue_disable_intr(sc->vtcon_ctrl_rxvq);
+
+ for (i = 0; i < sc->vtcon_max_ports; i++) {
+ scport = &sc->vtcon_ports[i];
+
+ port = scport->vcsp_port;
+ if (port == NULL)
+ continue;
+
+ VTCON_PORT_LOCK(port);
+ vtcon_port_disable_intr(port);
+ VTCON_PORT_UNLOCK(port);
+ }
+}
diff --git a/sys/dev/virtio/console/virtio_console.h b/sys/dev/virtio/console/virtio_console.h
new file mode 100644
index 0000000..0efada1
--- /dev/null
+++ b/sys/dev/virtio/console/virtio_console.h
@@ -0,0 +1,75 @@
+/*-
+ * This header is BSD licensed so anyone can use the definitions to implement
+ * compatible drivers/servers.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ * 1. Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ * 3. Neither the name of IBM nor the names of its contributors
+ * may be used to endorse or promote products derived from this software
+ * without specific prior written permission.
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS ``AS IS'' AND
+ * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+ * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
+ * ARE DISCLAIMED. IN NO EVENT SHALL IBM OR CONTRIBUTORS BE LIABLE
+ * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
+ * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
+ * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
+ * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
+ * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
+ * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
+ * SUCH DAMAGE.
+ *
+ * Copyright (C) Red Hat, Inc., 2009, 2010, 2011
+ * Copyright (C) Amit Shah <amit.shah@redhat.com>, 2009, 2010, 2011
+ *
+ * $FreeBSD$
+ */
+
+#ifndef _VIRTIO_CONSOLE_H
+#define _VIRTIO_CONSOLE_H
+
+/* Feature bits */
+#define VIRTIO_CONSOLE_F_SIZE 0x01 /* Console size */
+#define VIRTIO_CONSOLE_F_MULTIPORT 0x02 /* Multiple ports */
+#define VIRTIO_CONSOLE_F_EMERG_WRITE 0x04 /* Emergency write */
+
+#define VIRTIO_CONSOLE_BAD_ID (~(uint32_t)0)
+
+struct virtio_console_config {
+ /* colums of the screens */
+ uint16_t cols;
+ /* rows of the screens */
+ uint16_t rows;
+ /* max. number of ports this device can hold */
+ uint32_t max_nr_ports;
+ /* emergency write register */
+ uint32_t emerg_wr;
+} __packed;
+
+/*
+ * A message that's passed between the Host and the Guest for a
+ * particular port.
+ */
+struct virtio_console_control {
+ uint32_t id; /* Port number */
+ uint16_t event; /* The kind of control event (see below) */
+ uint16_t value; /* Extra information for the key */
+};
+
+/* Some events for control messages */
+#define VIRTIO_CONSOLE_DEVICE_READY 0
+#define VIRTIO_CONSOLE_PORT_ADD 1
+#define VIRTIO_CONSOLE_PORT_REMOVE 2
+#define VIRTIO_CONSOLE_PORT_READY 3
+#define VIRTIO_CONSOLE_CONSOLE_PORT 4
+#define VIRTIO_CONSOLE_RESIZE 5
+#define VIRTIO_CONSOLE_PORT_OPEN 6
+#define VIRTIO_CONSOLE_PORT_NAME 7
+
+#endif /* _VIRTIO_CONSOLE_H */
diff --git a/sys/i386/conf/NOTES b/sys/i386/conf/NOTES
index d60d5db..e201a8a 100644
--- a/sys/i386/conf/NOTES
+++ b/sys/i386/conf/NOTES
@@ -801,6 +801,7 @@ device virtio_blk # VirtIO Block device
device virtio_scsi # VirtIO SCSI device
device virtio_balloon # VirtIO Memory Balloon device
device virtio_random # VirtIO Entropy device
+device virtio_console # VirtIO Console device
device hyperv # HyperV drivers
diff --git a/sys/modules/virtio/Makefile b/sys/modules/virtio/Makefile
index ecf9441..4a823d6 100644
--- a/sys/modules/virtio/Makefile
+++ b/sys/modules/virtio/Makefile
@@ -23,6 +23,6 @@
# SUCH DAMAGE.
#
-SUBDIR= virtio pci network block balloon scsi random
+SUBDIR= virtio pci network block balloon scsi random console
.include <bsd.subdir.mk>
diff --git a/sys/modules/virtio/console/Makefile b/sys/modules/virtio/console/Makefile
new file mode 100644
index 0000000..062e049
--- /dev/null
+++ b/sys/modules/virtio/console/Makefile
@@ -0,0 +1,36 @@
+#
+# $FreeBSD$
+#
+# Redistribution and use in source and binary forms, with or without
+# modification, are permitted provided that the following conditions
+# are met:
+# 1. Redistributions of source code must retain the above copyright
+# notice, this list of conditions and the following disclaimer.
+# 2. Redistributions in binary form must reproduce the above copyright
+# notice, this list of conditions and the following disclaimer in the
+# documentation and/or other materials provided with the distribution.
+#
+# THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
+# ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+# IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
+# ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
+# FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
+# DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
+# OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
+# HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
+# LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
+# OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
+# SUCH DAMAGE.
+#
+
+.PATH: ${.CURDIR}/../../../dev/virtio/console
+
+KMOD= virtio_console
+SRCS= virtio_console.c
+SRCS+= virtio_bus_if.h virtio_if.h
+SRCS+= bus_if.h device_if.h
+
+MFILES= kern/bus_if.m kern/device_if.m \
+ dev/virtio/virtio_bus_if.m dev/virtio/virtio_if.m
+
+.include <bsd.kmod.mk>
OpenPOWER on IntegriCloud