16f744ddeSBryan Venteicher /*-
26f744ddeSBryan Venteicher * Copyright (c) 2014, Bryan Venteicher <bryanv@FreeBSD.org>
36f744ddeSBryan Venteicher * All rights reserved.
46f744ddeSBryan Venteicher *
56f744ddeSBryan Venteicher * Redistribution and use in source and binary forms, with or without
66f744ddeSBryan Venteicher * modification, are permitted provided that the following conditions
76f744ddeSBryan Venteicher * are met:
86f744ddeSBryan Venteicher * 1. Redistributions of source code must retain the above copyright
96f744ddeSBryan Venteicher * notice unmodified, this list of conditions, and the following
106f744ddeSBryan Venteicher * disclaimer.
116f744ddeSBryan Venteicher * 2. Redistributions in binary form must reproduce the above copyright
126f744ddeSBryan Venteicher * notice, this list of conditions and the following disclaimer in the
136f744ddeSBryan Venteicher * documentation and/or other materials provided with the distribution.
146f744ddeSBryan Venteicher *
156f744ddeSBryan Venteicher * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
166f744ddeSBryan Venteicher * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
176f744ddeSBryan Venteicher * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
186f744ddeSBryan Venteicher * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
196f744ddeSBryan Venteicher * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
206f744ddeSBryan Venteicher * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
216f744ddeSBryan Venteicher * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
226f744ddeSBryan Venteicher * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
236f744ddeSBryan Venteicher * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
246f744ddeSBryan Venteicher * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
256f744ddeSBryan Venteicher */
266f744ddeSBryan Venteicher
276f744ddeSBryan Venteicher /* Driver for VirtIO console devices. */
286f744ddeSBryan Venteicher
296f744ddeSBryan Venteicher #include <sys/param.h>
30a019e26cSBryan Venteicher #include <sys/ctype.h>
316f744ddeSBryan Venteicher #include <sys/systm.h>
326f744ddeSBryan Venteicher #include <sys/kernel.h>
336f744ddeSBryan Venteicher #include <sys/malloc.h>
346f744ddeSBryan Venteicher #include <sys/module.h>
35b84b3efdSBryan Venteicher #include <sys/kdb.h>
366f744ddeSBryan Venteicher #include <sys/lock.h>
376f744ddeSBryan Venteicher #include <sys/mutex.h>
386f744ddeSBryan Venteicher #include <sys/sglist.h>
396f744ddeSBryan Venteicher #include <sys/sysctl.h>
406f744ddeSBryan Venteicher #include <sys/taskqueue.h>
416f744ddeSBryan Venteicher #include <sys/queue.h>
426f744ddeSBryan Venteicher
436f744ddeSBryan Venteicher #include <sys/conf.h>
446f744ddeSBryan Venteicher #include <sys/cons.h>
456f744ddeSBryan Venteicher #include <sys/tty.h>
466f744ddeSBryan Venteicher
476f744ddeSBryan Venteicher #include <machine/bus.h>
486f744ddeSBryan Venteicher #include <machine/resource.h>
496f744ddeSBryan Venteicher #include <sys/bus.h>
506f744ddeSBryan Venteicher
516f744ddeSBryan Venteicher #include <dev/virtio/virtio.h>
526f744ddeSBryan Venteicher #include <dev/virtio/virtqueue.h>
536f744ddeSBryan Venteicher #include <dev/virtio/console/virtio_console.h>
546f744ddeSBryan Venteicher
556f744ddeSBryan Venteicher #include "virtio_if.h"
566f744ddeSBryan Venteicher
5704434c94SBryan Venteicher #define VTCON_MAX_PORTS 32
586f744ddeSBryan Venteicher #define VTCON_TTY_PREFIX "V"
59a019e26cSBryan Venteicher #define VTCON_TTY_ALIAS_PREFIX "vtcon"
606f744ddeSBryan Venteicher #define VTCON_BULK_BUFSZ 128
61a019e26cSBryan Venteicher #define VTCON_CTRL_BUFSZ 128
626f744ddeSBryan Venteicher
6304434c94SBryan Venteicher /*
64a019e26cSBryan Venteicher * The buffers cannot cross more than one page boundary due to the
6504434c94SBryan Venteicher * size of the sglist segment array used.
6604434c94SBryan Venteicher */
6704434c94SBryan Venteicher CTASSERT(VTCON_BULK_BUFSZ <= PAGE_SIZE);
68a019e26cSBryan Venteicher CTASSERT(VTCON_CTRL_BUFSZ <= PAGE_SIZE);
69a019e26cSBryan Venteicher
70a019e26cSBryan Venteicher CTASSERT(sizeof(struct virtio_console_config) <= VTCON_CTRL_BUFSZ);
7104434c94SBryan Venteicher
726f744ddeSBryan Venteicher struct vtcon_softc;
7304434c94SBryan Venteicher struct vtcon_softc_port;
746f744ddeSBryan Venteicher
756f744ddeSBryan Venteicher struct vtcon_port {
766f744ddeSBryan Venteicher struct mtx vtcport_mtx;
7704434c94SBryan Venteicher struct vtcon_softc *vtcport_sc;
7804434c94SBryan Venteicher struct vtcon_softc_port *vtcport_scport;
796f744ddeSBryan Venteicher struct tty *vtcport_tty;
806f744ddeSBryan Venteicher struct virtqueue *vtcport_invq;
816f744ddeSBryan Venteicher struct virtqueue *vtcport_outvq;
8204434c94SBryan Venteicher int vtcport_id;
8304434c94SBryan Venteicher int vtcport_flags;
8404434c94SBryan Venteicher #define VTCON_PORT_FLAG_GONE 0x01
85b84b3efdSBryan Venteicher #define VTCON_PORT_FLAG_CONSOLE 0x02
86a019e26cSBryan Venteicher #define VTCON_PORT_FLAG_ALIAS 0x04
87b84b3efdSBryan Venteicher
88b84b3efdSBryan Venteicher #if defined(KDB)
89b84b3efdSBryan Venteicher int vtcport_alt_break_state;
90b84b3efdSBryan Venteicher #endif
916f744ddeSBryan Venteicher };
926f744ddeSBryan Venteicher
9304434c94SBryan Venteicher #define VTCON_PORT_LOCK(_port) mtx_lock(&(_port)->vtcport_mtx)
9404434c94SBryan Venteicher #define VTCON_PORT_UNLOCK(_port) mtx_unlock(&(_port)->vtcport_mtx)
9504434c94SBryan Venteicher
9604434c94SBryan Venteicher struct vtcon_softc_port {
9704434c94SBryan Venteicher struct vtcon_softc *vcsp_sc;
9804434c94SBryan Venteicher struct vtcon_port *vcsp_port;
9904434c94SBryan Venteicher struct virtqueue *vcsp_invq;
10004434c94SBryan Venteicher struct virtqueue *vcsp_outvq;
10104434c94SBryan Venteicher };
1026f744ddeSBryan Venteicher
1036f744ddeSBryan Venteicher struct vtcon_softc {
1046f744ddeSBryan Venteicher device_t vtcon_dev;
1056f744ddeSBryan Venteicher struct mtx vtcon_mtx;
1066f744ddeSBryan Venteicher uint64_t vtcon_features;
1076f744ddeSBryan Venteicher uint32_t vtcon_max_ports;
108b84b3efdSBryan Venteicher uint32_t vtcon_flags;
109b84b3efdSBryan Venteicher #define VTCON_FLAG_DETACHED 0x01
110b84b3efdSBryan Venteicher #define VTCON_FLAG_SIZE 0x02
111b84b3efdSBryan Venteicher #define VTCON_FLAG_MULTIPORT 0x04
1126f744ddeSBryan Venteicher
1136f744ddeSBryan Venteicher /*
1146f744ddeSBryan Venteicher * Ports can be added and removed during runtime, but we have
1156f744ddeSBryan Venteicher * to allocate all the virtqueues during attach. This array is
1166f744ddeSBryan Venteicher * indexed by the port ID.
1176f744ddeSBryan Venteicher */
11804434c94SBryan Venteicher struct vtcon_softc_port *vtcon_ports;
119b84b3efdSBryan Venteicher
120b84b3efdSBryan Venteicher struct task vtcon_ctrl_task;
121b84b3efdSBryan Venteicher struct virtqueue *vtcon_ctrl_rxvq;
122b84b3efdSBryan Venteicher struct virtqueue *vtcon_ctrl_txvq;
123b84b3efdSBryan Venteicher struct mtx vtcon_ctrl_tx_mtx;
1246f744ddeSBryan Venteicher };
1256f744ddeSBryan Venteicher
12604434c94SBryan Venteicher #define VTCON_LOCK(_sc) mtx_lock(&(_sc)->vtcon_mtx)
12704434c94SBryan Venteicher #define VTCON_UNLOCK(_sc) mtx_unlock(&(_sc)->vtcon_mtx)
12804434c94SBryan Venteicher #define VTCON_LOCK_ASSERT(_sc) \
12904434c94SBryan Venteicher mtx_assert(&(_sc)->vtcon_mtx, MA_OWNED)
1306f744ddeSBryan Venteicher #define VTCON_LOCK_ASSERT_NOTOWNED(_sc) \
13104434c94SBryan Venteicher mtx_assert(&(_sc)->vtcon_mtx, MA_NOTOWNED)
13204434c94SBryan Venteicher
13304434c94SBryan Venteicher #define VTCON_CTRL_TX_LOCK(_sc) mtx_lock(&(_sc)->vtcon_ctrl_tx_mtx)
13404434c94SBryan Venteicher #define VTCON_CTRL_TX_UNLOCK(_sc) mtx_unlock(&(_sc)->vtcon_ctrl_tx_mtx)
1356f744ddeSBryan Venteicher
1366f744ddeSBryan Venteicher #define VTCON_ASSERT_VALID_PORTID(_sc, _id) \
1376f744ddeSBryan Venteicher KASSERT((_id) >= 0 && (_id) < (_sc)->vtcon_max_ports, \
1386f744ddeSBryan Venteicher ("%s: port ID %d out of range", __func__, _id))
1396f744ddeSBryan Venteicher
14004434c94SBryan Venteicher #define VTCON_FEATURES VIRTIO_CONSOLE_F_MULTIPORT
1416f744ddeSBryan Venteicher
1426f744ddeSBryan Venteicher static struct virtio_feature_desc vtcon_feature_desc[] = {
1436f744ddeSBryan Venteicher { VIRTIO_CONSOLE_F_SIZE, "ConsoleSize" },
1446f744ddeSBryan Venteicher { VIRTIO_CONSOLE_F_MULTIPORT, "MultiplePorts" },
145b84b3efdSBryan Venteicher { VIRTIO_CONSOLE_F_EMERG_WRITE, "EmergencyWrite" },
1466f744ddeSBryan Venteicher { 0, NULL }
1476f744ddeSBryan Venteicher };
1486f744ddeSBryan Venteicher
1496f744ddeSBryan Venteicher static int vtcon_modevent(module_t, int, void *);
15004434c94SBryan Venteicher static void vtcon_drain_all(void);
1516f744ddeSBryan Venteicher
1526f744ddeSBryan Venteicher static int vtcon_probe(device_t);
1536f744ddeSBryan Venteicher static int vtcon_attach(device_t);
1546f744ddeSBryan Venteicher static int vtcon_detach(device_t);
1556f744ddeSBryan Venteicher static int vtcon_config_change(device_t);
1566f744ddeSBryan Venteicher
157e6cc42f1SBryan Venteicher static int vtcon_setup_features(struct vtcon_softc *);
158e6cc42f1SBryan Venteicher static int vtcon_negotiate_features(struct vtcon_softc *);
15904434c94SBryan Venteicher static int vtcon_alloc_scports(struct vtcon_softc *);
1606f744ddeSBryan Venteicher static int vtcon_alloc_virtqueues(struct vtcon_softc *);
1616f744ddeSBryan Venteicher static void vtcon_read_config(struct vtcon_softc *,
1626f744ddeSBryan Venteicher struct virtio_console_config *);
1636f744ddeSBryan Venteicher
1646f744ddeSBryan Venteicher static void vtcon_determine_max_ports(struct vtcon_softc *,
1656f744ddeSBryan Venteicher struct virtio_console_config *);
16604434c94SBryan Venteicher static void vtcon_destroy_ports(struct vtcon_softc *);
1676f744ddeSBryan Venteicher static void vtcon_stop(struct vtcon_softc *);
1686f744ddeSBryan Venteicher
16904434c94SBryan Venteicher static int vtcon_ctrl_event_enqueue(struct vtcon_softc *,
1706f744ddeSBryan Venteicher struct virtio_console_control *);
17104434c94SBryan Venteicher static int vtcon_ctrl_event_create(struct vtcon_softc *);
17204434c94SBryan Venteicher static void vtcon_ctrl_event_requeue(struct vtcon_softc *,
1736f744ddeSBryan Venteicher struct virtio_console_control *);
17404434c94SBryan Venteicher static int vtcon_ctrl_event_populate(struct vtcon_softc *);
17504434c94SBryan Venteicher static void vtcon_ctrl_event_drain(struct vtcon_softc *);
1766f744ddeSBryan Venteicher static int vtcon_ctrl_init(struct vtcon_softc *);
1776f744ddeSBryan Venteicher static void vtcon_ctrl_deinit(struct vtcon_softc *);
1786f744ddeSBryan Venteicher static void vtcon_ctrl_port_add_event(struct vtcon_softc *, int);
1796f744ddeSBryan Venteicher static void vtcon_ctrl_port_remove_event(struct vtcon_softc *, int);
1806f744ddeSBryan Venteicher static void vtcon_ctrl_port_console_event(struct vtcon_softc *, int);
1816f744ddeSBryan Venteicher static void vtcon_ctrl_port_open_event(struct vtcon_softc *, int);
182cb981343SJakub Wojciech Klama static void vtcon_ctrl_port_name_event(struct vtcon_softc *, int,
183cb981343SJakub Wojciech Klama const char *, size_t);
18404434c94SBryan Venteicher static void vtcon_ctrl_process_event(struct vtcon_softc *,
185cb981343SJakub Wojciech Klama struct virtio_console_control *, void *, size_t);
1866f744ddeSBryan Venteicher static void vtcon_ctrl_task_cb(void *, int);
18704434c94SBryan Venteicher static void vtcon_ctrl_event_intr(void *);
18804434c94SBryan Venteicher static void vtcon_ctrl_poll(struct vtcon_softc *,
18904434c94SBryan Venteicher struct virtio_console_control *control);
19004434c94SBryan Venteicher static void vtcon_ctrl_send_control(struct vtcon_softc *, uint32_t,
19104434c94SBryan Venteicher uint16_t, uint16_t);
1926f744ddeSBryan Venteicher
19304434c94SBryan Venteicher static int vtcon_port_enqueue_buf(struct vtcon_port *, void *, size_t);
19404434c94SBryan Venteicher static int vtcon_port_create_buf(struct vtcon_port *);
19504434c94SBryan Venteicher static void vtcon_port_requeue_buf(struct vtcon_port *, void *);
1966f744ddeSBryan Venteicher static int vtcon_port_populate(struct vtcon_port *);
1976f744ddeSBryan Venteicher static void vtcon_port_destroy(struct vtcon_port *);
19804434c94SBryan Venteicher static int vtcon_port_create(struct vtcon_softc *, int);
199a019e26cSBryan Venteicher static void vtcon_port_dev_alias(struct vtcon_port *, const char *,
200a019e26cSBryan Venteicher size_t);
20104434c94SBryan Venteicher static void vtcon_port_drain_bufs(struct virtqueue *);
20204434c94SBryan Venteicher static void vtcon_port_drain(struct vtcon_port *);
20304434c94SBryan Venteicher static void vtcon_port_teardown(struct vtcon_port *);
2046f744ddeSBryan Venteicher static void vtcon_port_change_size(struct vtcon_port *, uint16_t,
2056f744ddeSBryan Venteicher uint16_t);
20604434c94SBryan Venteicher static void vtcon_port_update_console_size(struct vtcon_softc *);
2076f744ddeSBryan Venteicher static void vtcon_port_enable_intr(struct vtcon_port *);
2086f744ddeSBryan Venteicher static void vtcon_port_disable_intr(struct vtcon_port *);
20904434c94SBryan Venteicher static void vtcon_port_in(struct vtcon_port *);
21004434c94SBryan Venteicher static void vtcon_port_intr(void *);
21104434c94SBryan Venteicher static void vtcon_port_out(struct vtcon_port *, void *, int);
21204434c94SBryan Venteicher static void vtcon_port_submit_event(struct vtcon_port *, uint16_t,
2136f744ddeSBryan Venteicher uint16_t);
2146f744ddeSBryan Venteicher
2156f744ddeSBryan Venteicher static int vtcon_tty_open(struct tty *);
2166f744ddeSBryan Venteicher static void vtcon_tty_close(struct tty *);
2176f744ddeSBryan Venteicher static void vtcon_tty_outwakeup(struct tty *);
2186f744ddeSBryan Venteicher static void vtcon_tty_free(void *);
2196f744ddeSBryan Venteicher
2206f744ddeSBryan Venteicher static void vtcon_get_console_size(struct vtcon_softc *, uint16_t *,
2216f744ddeSBryan Venteicher uint16_t *);
2226f744ddeSBryan Venteicher
2236f744ddeSBryan Venteicher static void vtcon_enable_interrupts(struct vtcon_softc *);
2246f744ddeSBryan Venteicher static void vtcon_disable_interrupts(struct vtcon_softc *);
2256f744ddeSBryan Venteicher
226edf7c8ddSBryan Venteicher #define vtcon_modern(_sc) (((_sc)->vtcon_features & VIRTIO_F_VERSION_1) != 0)
227edf7c8ddSBryan Venteicher #define vtcon_htog16(_sc, _val) virtio_htog16(vtcon_modern(_sc), _val)
228edf7c8ddSBryan Venteicher #define vtcon_htog32(_sc, _val) virtio_htog32(vtcon_modern(_sc), _val)
229edf7c8ddSBryan Venteicher #define vtcon_htog64(_sc, _val) virtio_htog64(vtcon_modern(_sc), _val)
230edf7c8ddSBryan Venteicher #define vtcon_gtoh16(_sc, _val) virtio_gtoh16(vtcon_modern(_sc), _val)
231edf7c8ddSBryan Venteicher #define vtcon_gtoh32(_sc, _val) virtio_gtoh32(vtcon_modern(_sc), _val)
232edf7c8ddSBryan Venteicher #define vtcon_gtoh64(_sc, _val) virtio_gtoh64(vtcon_modern(_sc), _val)
233edf7c8ddSBryan Venteicher
2346f744ddeSBryan Venteicher static int vtcon_pending_free;
2356f744ddeSBryan Venteicher
2366f744ddeSBryan Venteicher static struct ttydevsw vtcon_tty_class = {
2376f744ddeSBryan Venteicher .tsw_flags = 0,
2386f744ddeSBryan Venteicher .tsw_open = vtcon_tty_open,
2396f744ddeSBryan Venteicher .tsw_close = vtcon_tty_close,
2406f744ddeSBryan Venteicher .tsw_outwakeup = vtcon_tty_outwakeup,
2416f744ddeSBryan Venteicher .tsw_free = vtcon_tty_free,
2426f744ddeSBryan Venteicher };
2436f744ddeSBryan Venteicher
2446f744ddeSBryan Venteicher static device_method_t vtcon_methods[] = {
2456f744ddeSBryan Venteicher /* Device methods. */
2466f744ddeSBryan Venteicher DEVMETHOD(device_probe, vtcon_probe),
2476f744ddeSBryan Venteicher DEVMETHOD(device_attach, vtcon_attach),
2486f744ddeSBryan Venteicher DEVMETHOD(device_detach, vtcon_detach),
2496f744ddeSBryan Venteicher
2506f744ddeSBryan Venteicher /* VirtIO methods. */
2516f744ddeSBryan Venteicher DEVMETHOD(virtio_config_change, vtcon_config_change),
2526f744ddeSBryan Venteicher
2536f744ddeSBryan Venteicher DEVMETHOD_END
2546f744ddeSBryan Venteicher };
2556f744ddeSBryan Venteicher
2566f744ddeSBryan Venteicher static driver_t vtcon_driver = {
2576f744ddeSBryan Venteicher "vtcon",
2586f744ddeSBryan Venteicher vtcon_methods,
2596f744ddeSBryan Venteicher sizeof(struct vtcon_softc)
2606f744ddeSBryan Venteicher };
2616f744ddeSBryan Venteicher
2625c4c96d3SJohn Baldwin VIRTIO_DRIVER_MODULE(virtio_console, vtcon_driver, vtcon_modevent, NULL);
2636f744ddeSBryan Venteicher MODULE_VERSION(virtio_console, 1);
2646f744ddeSBryan Venteicher MODULE_DEPEND(virtio_console, virtio, 1, 1, 1);
2656f744ddeSBryan Venteicher
266633218eeSJessica Clarke VIRTIO_SIMPLE_PNPINFO(virtio_console, VIRTIO_ID_CONSOLE,
2670f6040f0SConrad Meyer "VirtIO Console Adapter");
2680f6040f0SConrad Meyer
2696f744ddeSBryan Venteicher static int
vtcon_modevent(module_t mod,int type,void * unused)2706f744ddeSBryan Venteicher vtcon_modevent(module_t mod, int type, void *unused)
2716f744ddeSBryan Venteicher {
2726f744ddeSBryan Venteicher int error;
2736f744ddeSBryan Venteicher
2746f744ddeSBryan Venteicher switch (type) {
2756f744ddeSBryan Venteicher case MOD_LOAD:
2766f744ddeSBryan Venteicher error = 0;
2776f744ddeSBryan Venteicher break;
2786f744ddeSBryan Venteicher case MOD_QUIESCE:
27904434c94SBryan Venteicher error = 0;
28004434c94SBryan Venteicher break;
2816f744ddeSBryan Venteicher case MOD_UNLOAD:
28204434c94SBryan Venteicher vtcon_drain_all();
28304434c94SBryan Venteicher error = 0;
2846f744ddeSBryan Venteicher break;
2856f744ddeSBryan Venteicher case MOD_SHUTDOWN:
2866f744ddeSBryan Venteicher error = 0;
2876f744ddeSBryan Venteicher break;
2886f744ddeSBryan Venteicher default:
2896f744ddeSBryan Venteicher error = EOPNOTSUPP;
2906f744ddeSBryan Venteicher break;
2916f744ddeSBryan Venteicher }
2926f744ddeSBryan Venteicher
2936f744ddeSBryan Venteicher return (error);
2946f744ddeSBryan Venteicher }
2956f744ddeSBryan Venteicher
29604434c94SBryan Venteicher static void
vtcon_drain_all(void)29704434c94SBryan Venteicher vtcon_drain_all(void)
29804434c94SBryan Venteicher {
29904434c94SBryan Venteicher int first;
30004434c94SBryan Venteicher
30104434c94SBryan Venteicher for (first = 1; vtcon_pending_free != 0; first = 0) {
30204434c94SBryan Venteicher if (first != 0) {
30304434c94SBryan Venteicher printf("virtio_console: Waiting for all detached TTY "
30404434c94SBryan Venteicher "devices to have open fds closed.\n");
30504434c94SBryan Venteicher }
30604434c94SBryan Venteicher pause("vtcondra", hz);
30704434c94SBryan Venteicher }
30804434c94SBryan Venteicher }
30904434c94SBryan Venteicher
3106f744ddeSBryan Venteicher static int
vtcon_probe(device_t dev)3116f744ddeSBryan Venteicher vtcon_probe(device_t dev)
3126f744ddeSBryan Venteicher {
3130f6040f0SConrad Meyer return (VIRTIO_SIMPLE_PROBE(dev, virtio_console));
3146f744ddeSBryan Venteicher }
3156f744ddeSBryan Venteicher
3166f744ddeSBryan Venteicher static int
vtcon_attach(device_t dev)3176f744ddeSBryan Venteicher vtcon_attach(device_t dev)
3186f744ddeSBryan Venteicher {
3196f744ddeSBryan Venteicher struct vtcon_softc *sc;
3206f744ddeSBryan Venteicher struct virtio_console_config concfg;
3216f744ddeSBryan Venteicher int error;
3226f744ddeSBryan Venteicher
3236f744ddeSBryan Venteicher sc = device_get_softc(dev);
3246f744ddeSBryan Venteicher sc->vtcon_dev = dev;
325e6cc42f1SBryan Venteicher virtio_set_feature_desc(dev, vtcon_feature_desc);
3266f744ddeSBryan Venteicher
32704434c94SBryan Venteicher mtx_init(&sc->vtcon_mtx, "vtconmtx", NULL, MTX_DEF);
32804434c94SBryan Venteicher mtx_init(&sc->vtcon_ctrl_tx_mtx, "vtconctrlmtx", NULL, MTX_DEF);
3296f744ddeSBryan Venteicher
330e6cc42f1SBryan Venteicher error = vtcon_setup_features(sc);
331e6cc42f1SBryan Venteicher if (error) {
332e6cc42f1SBryan Venteicher device_printf(dev, "cannot setup features\n");
333e6cc42f1SBryan Venteicher goto fail;
334e6cc42f1SBryan Venteicher }
3356f744ddeSBryan Venteicher
3366f744ddeSBryan Venteicher vtcon_read_config(sc, &concfg);
3376f744ddeSBryan Venteicher vtcon_determine_max_ports(sc, &concfg);
3386f744ddeSBryan Venteicher
33904434c94SBryan Venteicher error = vtcon_alloc_scports(sc);
34004434c94SBryan Venteicher if (error) {
34104434c94SBryan Venteicher device_printf(dev, "cannot allocate softc port structures\n");
34204434c94SBryan Venteicher goto fail;
34304434c94SBryan Venteicher }
34404434c94SBryan Venteicher
3456f744ddeSBryan Venteicher error = vtcon_alloc_virtqueues(sc);
3466f744ddeSBryan Venteicher if (error) {
3476f744ddeSBryan Venteicher device_printf(dev, "cannot allocate virtqueues\n");
3486f744ddeSBryan Venteicher goto fail;
3496f744ddeSBryan Venteicher }
3506f744ddeSBryan Venteicher
35104434c94SBryan Venteicher if (sc->vtcon_flags & VTCON_FLAG_MULTIPORT) {
35204434c94SBryan Venteicher TASK_INIT(&sc->vtcon_ctrl_task, 0, vtcon_ctrl_task_cb, sc);
3536f744ddeSBryan Venteicher error = vtcon_ctrl_init(sc);
354b84b3efdSBryan Venteicher if (error)
355b84b3efdSBryan Venteicher goto fail;
356b84b3efdSBryan Venteicher } else {
35704434c94SBryan Venteicher error = vtcon_port_create(sc, 0);
3586f744ddeSBryan Venteicher if (error)
3596f744ddeSBryan Venteicher goto fail;
360b84b3efdSBryan Venteicher if (sc->vtcon_flags & VTCON_FLAG_SIZE)
361b84b3efdSBryan Venteicher vtcon_port_update_console_size(sc);
362b84b3efdSBryan Venteicher }
3636f744ddeSBryan Venteicher
3646f744ddeSBryan Venteicher error = virtio_setup_intr(dev, INTR_TYPE_TTY);
3656f744ddeSBryan Venteicher if (error) {
3666f744ddeSBryan Venteicher device_printf(dev, "cannot setup virtqueue interrupts\n");
3676f744ddeSBryan Venteicher goto fail;
3686f744ddeSBryan Venteicher }
3696f744ddeSBryan Venteicher
3706f744ddeSBryan Venteicher vtcon_enable_interrupts(sc);
3716f744ddeSBryan Venteicher
37204434c94SBryan Venteicher vtcon_ctrl_send_control(sc, VIRTIO_CONSOLE_BAD_ID,
3736f744ddeSBryan Venteicher VIRTIO_CONSOLE_DEVICE_READY, 1);
3746f744ddeSBryan Venteicher
3756f744ddeSBryan Venteicher fail:
3766f744ddeSBryan Venteicher if (error)
3776f744ddeSBryan Venteicher vtcon_detach(dev);
3786f744ddeSBryan Venteicher
3796f744ddeSBryan Venteicher return (error);
3806f744ddeSBryan Venteicher }
3816f744ddeSBryan Venteicher
3826f744ddeSBryan Venteicher static int
vtcon_detach(device_t dev)3836f744ddeSBryan Venteicher vtcon_detach(device_t dev)
3846f744ddeSBryan Venteicher {
3856f744ddeSBryan Venteicher struct vtcon_softc *sc;
3866f744ddeSBryan Venteicher
3876f744ddeSBryan Venteicher sc = device_get_softc(dev);
3886f744ddeSBryan Venteicher
3896f744ddeSBryan Venteicher VTCON_LOCK(sc);
3906f744ddeSBryan Venteicher sc->vtcon_flags |= VTCON_FLAG_DETACHED;
3916f744ddeSBryan Venteicher if (device_is_attached(dev))
3926f744ddeSBryan Venteicher vtcon_stop(sc);
3936f744ddeSBryan Venteicher VTCON_UNLOCK(sc);
3946f744ddeSBryan Venteicher
39504434c94SBryan Venteicher if (sc->vtcon_flags & VTCON_FLAG_MULTIPORT) {
3966f744ddeSBryan Venteicher taskqueue_drain(taskqueue_thread, &sc->vtcon_ctrl_task);
3976f744ddeSBryan Venteicher vtcon_ctrl_deinit(sc);
39804434c94SBryan Venteicher }
3996f744ddeSBryan Venteicher
40004434c94SBryan Venteicher vtcon_destroy_ports(sc);
40104434c94SBryan Venteicher mtx_destroy(&sc->vtcon_mtx);
40204434c94SBryan Venteicher mtx_destroy(&sc->vtcon_ctrl_tx_mtx);
4036f744ddeSBryan Venteicher
4046f744ddeSBryan Venteicher return (0);
4056f744ddeSBryan Venteicher }
4066f744ddeSBryan Venteicher
4076f744ddeSBryan Venteicher static int
vtcon_config_change(device_t dev)4086f744ddeSBryan Venteicher vtcon_config_change(device_t dev)
4096f744ddeSBryan Venteicher {
4106f744ddeSBryan Venteicher struct vtcon_softc *sc;
4116f744ddeSBryan Venteicher
4126f744ddeSBryan Venteicher sc = device_get_softc(dev);
4136f744ddeSBryan Venteicher
4146f744ddeSBryan Venteicher /*
41504434c94SBryan Venteicher * When the multiport feature is negotiated, all configuration
41604434c94SBryan Venteicher * changes are done through control virtqueue events.
4176f744ddeSBryan Venteicher */
41804434c94SBryan Venteicher if ((sc->vtcon_flags & VTCON_FLAG_MULTIPORT) == 0) {
41904434c94SBryan Venteicher if (sc->vtcon_flags & VTCON_FLAG_SIZE)
42004434c94SBryan Venteicher vtcon_port_update_console_size(sc);
4216f744ddeSBryan Venteicher }
4226f744ddeSBryan Venteicher
4236f744ddeSBryan Venteicher return (0);
4246f744ddeSBryan Venteicher }
4256f744ddeSBryan Venteicher
426e6cc42f1SBryan Venteicher static int
vtcon_negotiate_features(struct vtcon_softc * sc)4276f744ddeSBryan Venteicher vtcon_negotiate_features(struct vtcon_softc *sc)
4286f744ddeSBryan Venteicher {
4296f744ddeSBryan Venteicher device_t dev;
4306f744ddeSBryan Venteicher uint64_t features;
4316f744ddeSBryan Venteicher
4326f744ddeSBryan Venteicher dev = sc->vtcon_dev;
4336f744ddeSBryan Venteicher features = VTCON_FEATURES;
4346f744ddeSBryan Venteicher
4356f744ddeSBryan Venteicher sc->vtcon_features = virtio_negotiate_features(dev, features);
436e6cc42f1SBryan Venteicher return (virtio_finalize_features(dev));
4376f744ddeSBryan Venteicher }
4386f744ddeSBryan Venteicher
439e6cc42f1SBryan Venteicher static int
vtcon_setup_features(struct vtcon_softc * sc)44004434c94SBryan Venteicher vtcon_setup_features(struct vtcon_softc *sc)
44104434c94SBryan Venteicher {
44204434c94SBryan Venteicher device_t dev;
443e6cc42f1SBryan Venteicher int error;
44404434c94SBryan Venteicher
44504434c94SBryan Venteicher dev = sc->vtcon_dev;
44604434c94SBryan Venteicher
447e6cc42f1SBryan Venteicher error = vtcon_negotiate_features(sc);
448e6cc42f1SBryan Venteicher if (error)
449e6cc42f1SBryan Venteicher return (error);
45004434c94SBryan Venteicher
45104434c94SBryan Venteicher if (virtio_with_feature(dev, VIRTIO_CONSOLE_F_SIZE))
45204434c94SBryan Venteicher sc->vtcon_flags |= VTCON_FLAG_SIZE;
45304434c94SBryan Venteicher if (virtio_with_feature(dev, VIRTIO_CONSOLE_F_MULTIPORT))
45404434c94SBryan Venteicher sc->vtcon_flags |= VTCON_FLAG_MULTIPORT;
455e6cc42f1SBryan Venteicher
456e6cc42f1SBryan Venteicher return (0);
45704434c94SBryan Venteicher }
45804434c94SBryan Venteicher
4596f744ddeSBryan Venteicher #define VTCON_GET_CONFIG(_dev, _feature, _field, _cfg) \
4606f744ddeSBryan Venteicher if (virtio_with_feature(_dev, _feature)) { \
4616f744ddeSBryan Venteicher virtio_read_device_config(_dev, \
4626f744ddeSBryan Venteicher offsetof(struct virtio_console_config, _field), \
4636f744ddeSBryan Venteicher &(_cfg)->_field, sizeof((_cfg)->_field)); \
4646f744ddeSBryan Venteicher }
4656f744ddeSBryan Venteicher
4666f744ddeSBryan Venteicher static void
vtcon_read_config(struct vtcon_softc * sc,struct virtio_console_config * concfg)4676f744ddeSBryan Venteicher vtcon_read_config(struct vtcon_softc *sc, struct virtio_console_config *concfg)
4686f744ddeSBryan Venteicher {
4696f744ddeSBryan Venteicher device_t dev;
4706f744ddeSBryan Venteicher
4716f744ddeSBryan Venteicher dev = sc->vtcon_dev;
4726f744ddeSBryan Venteicher
4736f744ddeSBryan Venteicher bzero(concfg, sizeof(struct virtio_console_config));
4746f744ddeSBryan Venteicher
4756f744ddeSBryan Venteicher VTCON_GET_CONFIG(dev, VIRTIO_CONSOLE_F_SIZE, cols, concfg);
4766f744ddeSBryan Venteicher VTCON_GET_CONFIG(dev, VIRTIO_CONSOLE_F_SIZE, rows, concfg);
4776f744ddeSBryan Venteicher VTCON_GET_CONFIG(dev, VIRTIO_CONSOLE_F_MULTIPORT, max_nr_ports, concfg);
4786f744ddeSBryan Venteicher }
4796f744ddeSBryan Venteicher
4806f744ddeSBryan Venteicher #undef VTCON_GET_CONFIG
4816f744ddeSBryan Venteicher
4826f744ddeSBryan Venteicher static int
vtcon_alloc_scports(struct vtcon_softc * sc)48304434c94SBryan Venteicher vtcon_alloc_scports(struct vtcon_softc *sc)
48404434c94SBryan Venteicher {
48504434c94SBryan Venteicher struct vtcon_softc_port *scport;
486ac2fffa4SPedro F. Giffuni int max, i;
48704434c94SBryan Venteicher
48804434c94SBryan Venteicher max = sc->vtcon_max_ports;
48904434c94SBryan Venteicher
490ac2fffa4SPedro F. Giffuni sc->vtcon_ports = malloc(sizeof(struct vtcon_softc_port) * max,
49104434c94SBryan Venteicher M_DEVBUF, M_NOWAIT | M_ZERO);
49204434c94SBryan Venteicher if (sc->vtcon_ports == NULL)
49304434c94SBryan Venteicher return (ENOMEM);
49404434c94SBryan Venteicher
49504434c94SBryan Venteicher for (i = 0; i < max; i++) {
49604434c94SBryan Venteicher scport = &sc->vtcon_ports[i];
49704434c94SBryan Venteicher scport->vcsp_sc = sc;
49804434c94SBryan Venteicher }
49904434c94SBryan Venteicher
50004434c94SBryan Venteicher return (0);
50104434c94SBryan Venteicher }
50204434c94SBryan Venteicher
50304434c94SBryan Venteicher static int
vtcon_alloc_virtqueues(struct vtcon_softc * sc)5046f744ddeSBryan Venteicher vtcon_alloc_virtqueues(struct vtcon_softc *sc)
5056f744ddeSBryan Venteicher {
5066f744ddeSBryan Venteicher device_t dev;
5076f744ddeSBryan Venteicher struct vq_alloc_info *info;
50804434c94SBryan Venteicher struct vtcon_softc_port *scport;
509ac2fffa4SPedro F. Giffuni int i, idx, portidx, nvqs, error;
5106f744ddeSBryan Venteicher
5116f744ddeSBryan Venteicher dev = sc->vtcon_dev;
5126f744ddeSBryan Venteicher
5136f744ddeSBryan Venteicher nvqs = sc->vtcon_max_ports * 2;
5146f744ddeSBryan Venteicher if (sc->vtcon_flags & VTCON_FLAG_MULTIPORT)
5156f744ddeSBryan Venteicher nvqs += 2;
5166f744ddeSBryan Venteicher
517ac2fffa4SPedro F. Giffuni info = malloc(sizeof(struct vq_alloc_info) * nvqs, M_TEMP, M_NOWAIT);
5186f744ddeSBryan Venteicher if (info == NULL)
5196f744ddeSBryan Venteicher return (ENOMEM);
5206f744ddeSBryan Venteicher
5216f744ddeSBryan Venteicher for (i = 0, idx = 0, portidx = 0; i < nvqs / 2; i++, idx += 2) {
5226f744ddeSBryan Venteicher if (i == 1) {
5236f744ddeSBryan Venteicher /* The control virtqueues are after the first port. */
5246f744ddeSBryan Venteicher VQ_ALLOC_INFO_INIT(&info[idx], 0,
52504434c94SBryan Venteicher vtcon_ctrl_event_intr, sc, &sc->vtcon_ctrl_rxvq,
5266f744ddeSBryan Venteicher "%s-control rx", device_get_nameunit(dev));
5276f744ddeSBryan Venteicher VQ_ALLOC_INFO_INIT(&info[idx+1], 0,
5286f744ddeSBryan Venteicher NULL, sc, &sc->vtcon_ctrl_txvq,
5296f744ddeSBryan Venteicher "%s-control tx", device_get_nameunit(dev));
5306f744ddeSBryan Venteicher continue;
5316f744ddeSBryan Venteicher }
5326f744ddeSBryan Venteicher
53304434c94SBryan Venteicher scport = &sc->vtcon_ports[portidx];
5346f744ddeSBryan Venteicher
53504434c94SBryan Venteicher VQ_ALLOC_INFO_INIT(&info[idx], 0, vtcon_port_intr,
53604434c94SBryan Venteicher scport, &scport->vcsp_invq, "%s-port%d in",
53704434c94SBryan Venteicher device_get_nameunit(dev), i);
5386f744ddeSBryan Venteicher VQ_ALLOC_INFO_INIT(&info[idx+1], 0, NULL,
53904434c94SBryan Venteicher NULL, &scport->vcsp_outvq, "%s-port%d out",
54004434c94SBryan Venteicher device_get_nameunit(dev), i);
5416f744ddeSBryan Venteicher
5426f744ddeSBryan Venteicher portidx++;
5436f744ddeSBryan Venteicher }
5446f744ddeSBryan Venteicher
545180c0240SMina Galić error = virtio_alloc_virtqueues(dev, nvqs, info);
5466f744ddeSBryan Venteicher free(info, M_TEMP);
5476f744ddeSBryan Venteicher
5486f744ddeSBryan Venteicher return (error);
5496f744ddeSBryan Venteicher }
5506f744ddeSBryan Venteicher
5516f744ddeSBryan Venteicher static void
vtcon_determine_max_ports(struct vtcon_softc * sc,struct virtio_console_config * concfg)5526f744ddeSBryan Venteicher vtcon_determine_max_ports(struct vtcon_softc *sc,
5536f744ddeSBryan Venteicher struct virtio_console_config *concfg)
5546f744ddeSBryan Venteicher {
5556f744ddeSBryan Venteicher
5566f744ddeSBryan Venteicher if (sc->vtcon_flags & VTCON_FLAG_MULTIPORT) {
5576f744ddeSBryan Venteicher sc->vtcon_max_ports =
5586f744ddeSBryan Venteicher min(concfg->max_nr_ports, VTCON_MAX_PORTS);
5596f744ddeSBryan Venteicher if (sc->vtcon_max_ports == 0)
5606f744ddeSBryan Venteicher sc->vtcon_max_ports = 1;
5616f744ddeSBryan Venteicher } else
5626f744ddeSBryan Venteicher sc->vtcon_max_ports = 1;
5636f744ddeSBryan Venteicher }
5646f744ddeSBryan Venteicher
5656f744ddeSBryan Venteicher static void
vtcon_destroy_ports(struct vtcon_softc * sc)56604434c94SBryan Venteicher vtcon_destroy_ports(struct vtcon_softc *sc)
5676f744ddeSBryan Venteicher {
56804434c94SBryan Venteicher struct vtcon_softc_port *scport;
56904434c94SBryan Venteicher struct vtcon_port *port;
57004434c94SBryan Venteicher struct virtqueue *vq;
57104434c94SBryan Venteicher int i;
5726f744ddeSBryan Venteicher
57304434c94SBryan Venteicher if (sc->vtcon_ports == NULL)
57404434c94SBryan Venteicher return;
57504434c94SBryan Venteicher
57604434c94SBryan Venteicher VTCON_LOCK(sc);
57704434c94SBryan Venteicher for (i = 0; i < sc->vtcon_max_ports; i++) {
57804434c94SBryan Venteicher scport = &sc->vtcon_ports[i];
57904434c94SBryan Venteicher
58004434c94SBryan Venteicher port = scport->vcsp_port;
58104434c94SBryan Venteicher if (port != NULL) {
58204434c94SBryan Venteicher scport->vcsp_port = NULL;
58304434c94SBryan Venteicher VTCON_PORT_LOCK(port);
58404434c94SBryan Venteicher VTCON_UNLOCK(sc);
58504434c94SBryan Venteicher vtcon_port_teardown(port);
58604434c94SBryan Venteicher VTCON_LOCK(sc);
5876f744ddeSBryan Venteicher }
5886f744ddeSBryan Venteicher
58904434c94SBryan Venteicher vq = scport->vcsp_invq;
59004434c94SBryan Venteicher if (vq != NULL)
59104434c94SBryan Venteicher vtcon_port_drain_bufs(vq);
5926f744ddeSBryan Venteicher }
59304434c94SBryan Venteicher VTCON_UNLOCK(sc);
59404434c94SBryan Venteicher
59504434c94SBryan Venteicher free(sc->vtcon_ports, M_DEVBUF);
59604434c94SBryan Venteicher sc->vtcon_ports = NULL;
5976f744ddeSBryan Venteicher }
5986f744ddeSBryan Venteicher
5996f744ddeSBryan Venteicher static void
vtcon_stop(struct vtcon_softc * sc)6006f744ddeSBryan Venteicher vtcon_stop(struct vtcon_softc *sc)
6016f744ddeSBryan Venteicher {
6026f744ddeSBryan Venteicher
6036f744ddeSBryan Venteicher vtcon_disable_interrupts(sc);
6046f744ddeSBryan Venteicher virtio_stop(sc->vtcon_dev);
6056f744ddeSBryan Venteicher }
6066f744ddeSBryan Venteicher
6076f744ddeSBryan Venteicher static int
vtcon_ctrl_event_enqueue(struct vtcon_softc * sc,struct virtio_console_control * control)60804434c94SBryan Venteicher vtcon_ctrl_event_enqueue(struct vtcon_softc *sc,
6096f744ddeSBryan Venteicher struct virtio_console_control *control)
6106f744ddeSBryan Venteicher {
61104434c94SBryan Venteicher struct sglist_seg segs[2];
6126f744ddeSBryan Venteicher struct sglist sg;
6136f744ddeSBryan Venteicher struct virtqueue *vq;
6144d78ef3eSMateusz Guzik int error __diagused;
6156f744ddeSBryan Venteicher
6166f744ddeSBryan Venteicher vq = sc->vtcon_ctrl_rxvq;
6176f744ddeSBryan Venteicher
61804434c94SBryan Venteicher sglist_init(&sg, 2, segs);
619a019e26cSBryan Venteicher error = sglist_append(&sg, control, VTCON_CTRL_BUFSZ);
62004434c94SBryan Venteicher KASSERT(error == 0, ("%s: error %d adding control to sglist",
62104434c94SBryan Venteicher __func__, error));
6226f744ddeSBryan Venteicher
62304434c94SBryan Venteicher return (virtqueue_enqueue(vq, control, &sg, 0, sg.sg_nseg));
6246f744ddeSBryan Venteicher }
6256f744ddeSBryan Venteicher
6266f744ddeSBryan Venteicher static int
vtcon_ctrl_event_create(struct vtcon_softc * sc)62704434c94SBryan Venteicher vtcon_ctrl_event_create(struct vtcon_softc *sc)
6286f744ddeSBryan Venteicher {
6296f744ddeSBryan Venteicher struct virtio_console_control *control;
6306f744ddeSBryan Venteicher int error;
6316f744ddeSBryan Venteicher
632a019e26cSBryan Venteicher control = malloc(VTCON_CTRL_BUFSZ, M_DEVBUF, M_ZERO | M_NOWAIT);
6336f744ddeSBryan Venteicher if (control == NULL)
6346f744ddeSBryan Venteicher return (ENOMEM);
6356f744ddeSBryan Venteicher
63604434c94SBryan Venteicher error = vtcon_ctrl_event_enqueue(sc, control);
6376f744ddeSBryan Venteicher if (error)
6386f744ddeSBryan Venteicher free(control, M_DEVBUF);
6396f744ddeSBryan Venteicher
6406f744ddeSBryan Venteicher return (error);
6416f744ddeSBryan Venteicher }
6426f744ddeSBryan Venteicher
6436f744ddeSBryan Venteicher static void
vtcon_ctrl_event_requeue(struct vtcon_softc * sc,struct virtio_console_control * control)64404434c94SBryan Venteicher vtcon_ctrl_event_requeue(struct vtcon_softc *sc,
6456f744ddeSBryan Venteicher struct virtio_console_control *control)
6466f744ddeSBryan Venteicher {
6474d78ef3eSMateusz Guzik int error __diagused;
6486f744ddeSBryan Venteicher
649a019e26cSBryan Venteicher bzero(control, VTCON_CTRL_BUFSZ);
6506f744ddeSBryan Venteicher
65104434c94SBryan Venteicher error = vtcon_ctrl_event_enqueue(sc, control);
6526f744ddeSBryan Venteicher KASSERT(error == 0,
6536f744ddeSBryan Venteicher ("%s: cannot requeue control buffer %d", __func__, error));
6546f744ddeSBryan Venteicher }
6556f744ddeSBryan Venteicher
6566f744ddeSBryan Venteicher static int
vtcon_ctrl_event_populate(struct vtcon_softc * sc)65704434c94SBryan Venteicher vtcon_ctrl_event_populate(struct vtcon_softc *sc)
6586f744ddeSBryan Venteicher {
6596f744ddeSBryan Venteicher struct virtqueue *vq;
6606f744ddeSBryan Venteicher int nbufs, error;
6616f744ddeSBryan Venteicher
6626f744ddeSBryan Venteicher vq = sc->vtcon_ctrl_rxvq;
6636f744ddeSBryan Venteicher error = ENOSPC;
6646f744ddeSBryan Venteicher
6656f744ddeSBryan Venteicher for (nbufs = 0; !virtqueue_full(vq); nbufs++) {
66604434c94SBryan Venteicher error = vtcon_ctrl_event_create(sc);
6676f744ddeSBryan Venteicher if (error)
6686f744ddeSBryan Venteicher break;
6696f744ddeSBryan Venteicher }
6706f744ddeSBryan Venteicher
6716f744ddeSBryan Venteicher if (nbufs > 0) {
6726f744ddeSBryan Venteicher virtqueue_notify(vq);
6736f744ddeSBryan Venteicher error = 0;
6746f744ddeSBryan Venteicher }
6756f744ddeSBryan Venteicher
6766f744ddeSBryan Venteicher return (error);
6776f744ddeSBryan Venteicher }
6786f744ddeSBryan Venteicher
6796f744ddeSBryan Venteicher static void
vtcon_ctrl_event_drain(struct vtcon_softc * sc)68004434c94SBryan Venteicher vtcon_ctrl_event_drain(struct vtcon_softc *sc)
6816f744ddeSBryan Venteicher {
6826f744ddeSBryan Venteicher struct virtio_console_control *control;
6836f744ddeSBryan Venteicher struct virtqueue *vq;
6846f744ddeSBryan Venteicher int last;
6856f744ddeSBryan Venteicher
6866f744ddeSBryan Venteicher vq = sc->vtcon_ctrl_rxvq;
6876f744ddeSBryan Venteicher last = 0;
6886f744ddeSBryan Venteicher
6896f744ddeSBryan Venteicher if (vq == NULL)
6906f744ddeSBryan Venteicher return;
6916f744ddeSBryan Venteicher
69204434c94SBryan Venteicher VTCON_LOCK(sc);
6936f744ddeSBryan Venteicher while ((control = virtqueue_drain(vq, &last)) != NULL)
6946f744ddeSBryan Venteicher free(control, M_DEVBUF);
69504434c94SBryan Venteicher VTCON_UNLOCK(sc);
69604434c94SBryan Venteicher }
69704434c94SBryan Venteicher
69804434c94SBryan Venteicher static int
vtcon_ctrl_init(struct vtcon_softc * sc)69904434c94SBryan Venteicher vtcon_ctrl_init(struct vtcon_softc *sc)
70004434c94SBryan Venteicher {
70104434c94SBryan Venteicher int error;
70204434c94SBryan Venteicher
70304434c94SBryan Venteicher error = vtcon_ctrl_event_populate(sc);
70404434c94SBryan Venteicher
70504434c94SBryan Venteicher return (error);
7066f744ddeSBryan Venteicher }
7076f744ddeSBryan Venteicher
7086f744ddeSBryan Venteicher static void
vtcon_ctrl_deinit(struct vtcon_softc * sc)7096f744ddeSBryan Venteicher vtcon_ctrl_deinit(struct vtcon_softc *sc)
7106f744ddeSBryan Venteicher {
7116f744ddeSBryan Venteicher
71204434c94SBryan Venteicher vtcon_ctrl_event_drain(sc);
7136f744ddeSBryan Venteicher }
7146f744ddeSBryan Venteicher
7156f744ddeSBryan Venteicher static void
vtcon_ctrl_port_add_event(struct vtcon_softc * sc,int id)7166f744ddeSBryan Venteicher vtcon_ctrl_port_add_event(struct vtcon_softc *sc, int id)
7176f744ddeSBryan Venteicher {
7186f744ddeSBryan Venteicher device_t dev;
7196f744ddeSBryan Venteicher int error;
7206f744ddeSBryan Venteicher
7216f744ddeSBryan Venteicher dev = sc->vtcon_dev;
7226f744ddeSBryan Venteicher
72304434c94SBryan Venteicher /* This single thread only way for ports to be created. */
72404434c94SBryan Venteicher if (sc->vtcon_ports[id].vcsp_port != NULL) {
7256f744ddeSBryan Venteicher device_printf(dev, "%s: adding port %d, but already exists\n",
7266f744ddeSBryan Venteicher __func__, id);
7276f744ddeSBryan Venteicher return;
7286f744ddeSBryan Venteicher }
7296f744ddeSBryan Venteicher
73004434c94SBryan Venteicher error = vtcon_port_create(sc, id);
7316f744ddeSBryan Venteicher if (error) {
7326f744ddeSBryan Venteicher device_printf(dev, "%s: cannot create port %d: %d\n",
7336f744ddeSBryan Venteicher __func__, id, error);
734b84b3efdSBryan Venteicher vtcon_ctrl_send_control(sc, id, VIRTIO_CONSOLE_PORT_READY, 0);
7356f744ddeSBryan Venteicher return;
7366f744ddeSBryan Venteicher }
7376f744ddeSBryan Venteicher }
7386f744ddeSBryan Venteicher
7396f744ddeSBryan Venteicher static void
vtcon_ctrl_port_remove_event(struct vtcon_softc * sc,int id)7406f744ddeSBryan Venteicher vtcon_ctrl_port_remove_event(struct vtcon_softc *sc, int id)
7416f744ddeSBryan Venteicher {
7426f744ddeSBryan Venteicher device_t dev;
74304434c94SBryan Venteicher struct vtcon_softc_port *scport;
7446f744ddeSBryan Venteicher struct vtcon_port *port;
7456f744ddeSBryan Venteicher
7466f744ddeSBryan Venteicher dev = sc->vtcon_dev;
74704434c94SBryan Venteicher scport = &sc->vtcon_ports[id];
7486f744ddeSBryan Venteicher
74904434c94SBryan Venteicher VTCON_LOCK(sc);
75004434c94SBryan Venteicher port = scport->vcsp_port;
7516f744ddeSBryan Venteicher if (port == NULL) {
75204434c94SBryan Venteicher VTCON_UNLOCK(sc);
7536f744ddeSBryan Venteicher device_printf(dev, "%s: remove port %d, but does not exist\n",
7546f744ddeSBryan Venteicher __func__, id);
7556f744ddeSBryan Venteicher return;
7566f744ddeSBryan Venteicher }
7576f744ddeSBryan Venteicher
75804434c94SBryan Venteicher scport->vcsp_port = NULL;
75904434c94SBryan Venteicher VTCON_PORT_LOCK(port);
76004434c94SBryan Venteicher VTCON_UNLOCK(sc);
76104434c94SBryan Venteicher vtcon_port_teardown(port);
7626f744ddeSBryan Venteicher }
7636f744ddeSBryan Venteicher
7646f744ddeSBryan Venteicher static void
vtcon_ctrl_port_console_event(struct vtcon_softc * sc,int id)7656f744ddeSBryan Venteicher vtcon_ctrl_port_console_event(struct vtcon_softc *sc, int id)
7666f744ddeSBryan Venteicher {
767b84b3efdSBryan Venteicher device_t dev;
768b84b3efdSBryan Venteicher struct vtcon_softc_port *scport;
769b84b3efdSBryan Venteicher struct vtcon_port *port;
7706f744ddeSBryan Venteicher
771b84b3efdSBryan Venteicher dev = sc->vtcon_dev;
772b84b3efdSBryan Venteicher scport = &sc->vtcon_ports[id];
773b84b3efdSBryan Venteicher
774b84b3efdSBryan Venteicher VTCON_LOCK(sc);
775b84b3efdSBryan Venteicher port = scport->vcsp_port;
776b84b3efdSBryan Venteicher if (port == NULL) {
777b84b3efdSBryan Venteicher VTCON_UNLOCK(sc);
778b84b3efdSBryan Venteicher device_printf(dev, "%s: console port %d, but does not exist\n",
77904434c94SBryan Venteicher __func__, id);
780b84b3efdSBryan Venteicher return;
781b84b3efdSBryan Venteicher }
782b84b3efdSBryan Venteicher
783b84b3efdSBryan Venteicher VTCON_PORT_LOCK(port);
784b84b3efdSBryan Venteicher VTCON_UNLOCK(sc);
785b84b3efdSBryan Venteicher port->vtcport_flags |= VTCON_PORT_FLAG_CONSOLE;
786b84b3efdSBryan Venteicher vtcon_port_submit_event(port, VIRTIO_CONSOLE_PORT_OPEN, 1);
787b84b3efdSBryan Venteicher VTCON_PORT_UNLOCK(port);
7886f744ddeSBryan Venteicher }
7896f744ddeSBryan Venteicher
7906f744ddeSBryan Venteicher static void
vtcon_ctrl_port_open_event(struct vtcon_softc * sc,int id)7916f744ddeSBryan Venteicher vtcon_ctrl_port_open_event(struct vtcon_softc *sc, int id)
7926f744ddeSBryan Venteicher {
7936f744ddeSBryan Venteicher device_t dev;
79404434c94SBryan Venteicher struct vtcon_softc_port *scport;
7956f744ddeSBryan Venteicher struct vtcon_port *port;
7966f744ddeSBryan Venteicher
7976f744ddeSBryan Venteicher dev = sc->vtcon_dev;
79804434c94SBryan Venteicher scport = &sc->vtcon_ports[id];
7996f744ddeSBryan Venteicher
80004434c94SBryan Venteicher VTCON_LOCK(sc);
80104434c94SBryan Venteicher port = scport->vcsp_port;
8026f744ddeSBryan Venteicher if (port == NULL) {
80304434c94SBryan Venteicher VTCON_UNLOCK(sc);
8046f744ddeSBryan Venteicher device_printf(dev, "%s: open port %d, but does not exist\n",
8056f744ddeSBryan Venteicher __func__, id);
8066f744ddeSBryan Venteicher return;
8076f744ddeSBryan Venteicher }
8086f744ddeSBryan Venteicher
80904434c94SBryan Venteicher VTCON_PORT_LOCK(port);
81004434c94SBryan Venteicher VTCON_UNLOCK(sc);
8116f744ddeSBryan Venteicher vtcon_port_enable_intr(port);
81204434c94SBryan Venteicher VTCON_PORT_UNLOCK(port);
8136f744ddeSBryan Venteicher }
8146f744ddeSBryan Venteicher
8156f744ddeSBryan Venteicher static void
vtcon_ctrl_port_name_event(struct vtcon_softc * sc,int id,const char * name,size_t len)816cb981343SJakub Wojciech Klama vtcon_ctrl_port_name_event(struct vtcon_softc *sc, int id, const char *name,
817cb981343SJakub Wojciech Klama size_t len)
818cb981343SJakub Wojciech Klama {
819cb981343SJakub Wojciech Klama device_t dev;
820cb981343SJakub Wojciech Klama struct vtcon_softc_port *scport;
821cb981343SJakub Wojciech Klama struct vtcon_port *port;
822cb981343SJakub Wojciech Klama
823cb981343SJakub Wojciech Klama dev = sc->vtcon_dev;
824cb981343SJakub Wojciech Klama scport = &sc->vtcon_ports[id];
825cb981343SJakub Wojciech Klama
826a019e26cSBryan Venteicher /*
827a019e26cSBryan Venteicher * The VirtIO specification says the NUL terminator is not included in
828a019e26cSBryan Venteicher * the length, but QEMU includes it. Adjust the length if needed.
829a019e26cSBryan Venteicher */
830a019e26cSBryan Venteicher if (name == NULL || len == 0)
831a019e26cSBryan Venteicher return;
832a019e26cSBryan Venteicher if (name[len - 1] == '\0') {
833a019e26cSBryan Venteicher len--;
834a019e26cSBryan Venteicher if (len == 0)
835a019e26cSBryan Venteicher return;
836a019e26cSBryan Venteicher }
837a019e26cSBryan Venteicher
838a019e26cSBryan Venteicher VTCON_LOCK(sc);
839cb981343SJakub Wojciech Klama port = scport->vcsp_port;
840cb981343SJakub Wojciech Klama if (port == NULL) {
841a019e26cSBryan Venteicher VTCON_UNLOCK(sc);
842cb981343SJakub Wojciech Klama device_printf(dev, "%s: name port %d, but does not exist\n",
843cb981343SJakub Wojciech Klama __func__, id);
844cb981343SJakub Wojciech Klama return;
845cb981343SJakub Wojciech Klama }
846cb981343SJakub Wojciech Klama
847a019e26cSBryan Venteicher VTCON_PORT_LOCK(port);
848a019e26cSBryan Venteicher VTCON_UNLOCK(sc);
849a019e26cSBryan Venteicher vtcon_port_dev_alias(port, name, len);
850a019e26cSBryan Venteicher VTCON_PORT_UNLOCK(port);
851cb981343SJakub Wojciech Klama }
852cb981343SJakub Wojciech Klama
853cb981343SJakub Wojciech Klama static void
vtcon_ctrl_process_event(struct vtcon_softc * sc,struct virtio_console_control * control,void * data,size_t data_len)85404434c94SBryan Venteicher vtcon_ctrl_process_event(struct vtcon_softc *sc,
855a019e26cSBryan Venteicher struct virtio_console_control *control, void *data, size_t data_len)
8566f744ddeSBryan Venteicher {
8576f744ddeSBryan Venteicher device_t dev;
858edf7c8ddSBryan Venteicher uint32_t id;
859edf7c8ddSBryan Venteicher uint16_t event;
8606f744ddeSBryan Venteicher
8616f744ddeSBryan Venteicher dev = sc->vtcon_dev;
862edf7c8ddSBryan Venteicher id = vtcon_htog32(sc, control->id);
863edf7c8ddSBryan Venteicher event = vtcon_htog16(sc, control->event);
8646f744ddeSBryan Venteicher
865edf7c8ddSBryan Venteicher if (id >= sc->vtcon_max_ports) {
866edf7c8ddSBryan Venteicher device_printf(dev, "%s: event %d invalid port ID %d\n",
867edf7c8ddSBryan Venteicher __func__, event, id);
8686f744ddeSBryan Venteicher return;
8696f744ddeSBryan Venteicher }
8706f744ddeSBryan Venteicher
871edf7c8ddSBryan Venteicher switch (event) {
8726f744ddeSBryan Venteicher case VIRTIO_CONSOLE_PORT_ADD:
8736f744ddeSBryan Venteicher vtcon_ctrl_port_add_event(sc, id);
8746f744ddeSBryan Venteicher break;
8756f744ddeSBryan Venteicher
8766f744ddeSBryan Venteicher case VIRTIO_CONSOLE_PORT_REMOVE:
8776f744ddeSBryan Venteicher vtcon_ctrl_port_remove_event(sc, id);
8786f744ddeSBryan Venteicher break;
8796f744ddeSBryan Venteicher
8806f744ddeSBryan Venteicher case VIRTIO_CONSOLE_CONSOLE_PORT:
8816f744ddeSBryan Venteicher vtcon_ctrl_port_console_event(sc, id);
8826f744ddeSBryan Venteicher break;
8836f744ddeSBryan Venteicher
8846f744ddeSBryan Venteicher case VIRTIO_CONSOLE_RESIZE:
8856f744ddeSBryan Venteicher break;
8866f744ddeSBryan Venteicher
8876f744ddeSBryan Venteicher case VIRTIO_CONSOLE_PORT_OPEN:
8886f744ddeSBryan Venteicher vtcon_ctrl_port_open_event(sc, id);
8896f744ddeSBryan Venteicher break;
8906f744ddeSBryan Venteicher
8916f744ddeSBryan Venteicher case VIRTIO_CONSOLE_PORT_NAME:
892a019e26cSBryan Venteicher vtcon_ctrl_port_name_event(sc, id, (const char *)data, data_len);
8936f744ddeSBryan Venteicher break;
8946f744ddeSBryan Venteicher }
8956f744ddeSBryan Venteicher }
8966f744ddeSBryan Venteicher
8976f744ddeSBryan Venteicher static void
vtcon_ctrl_task_cb(void * xsc,int pending)8986f744ddeSBryan Venteicher vtcon_ctrl_task_cb(void *xsc, int pending)
8996f744ddeSBryan Venteicher {
9006f744ddeSBryan Venteicher struct vtcon_softc *sc;
9016f744ddeSBryan Venteicher struct virtqueue *vq;
9026f744ddeSBryan Venteicher struct virtio_console_control *control;
903a019e26cSBryan Venteicher void *data;
904a019e26cSBryan Venteicher size_t data_len;
90504434c94SBryan Venteicher int detached;
906cb981343SJakub Wojciech Klama uint32_t len;
9076f744ddeSBryan Venteicher
9086f744ddeSBryan Venteicher sc = xsc;
9096f744ddeSBryan Venteicher vq = sc->vtcon_ctrl_rxvq;
9106f744ddeSBryan Venteicher
9116f744ddeSBryan Venteicher VTCON_LOCK(sc);
91204434c94SBryan Venteicher
91304434c94SBryan Venteicher while ((detached = (sc->vtcon_flags & VTCON_FLAG_DETACHED)) == 0) {
914cb981343SJakub Wojciech Klama control = virtqueue_dequeue(vq, &len);
9156f744ddeSBryan Venteicher if (control == NULL)
9166f744ddeSBryan Venteicher break;
9176f744ddeSBryan Venteicher
918a019e26cSBryan Venteicher if (len > sizeof(struct virtio_console_control)) {
919a019e26cSBryan Venteicher data = (void *) &control[1];
920a019e26cSBryan Venteicher data_len = len - sizeof(struct virtio_console_control);
921a019e26cSBryan Venteicher } else {
922a019e26cSBryan Venteicher data = NULL;
923a019e26cSBryan Venteicher data_len = 0;
924cb981343SJakub Wojciech Klama }
925cb981343SJakub Wojciech Klama
9266f744ddeSBryan Venteicher VTCON_UNLOCK(sc);
927a019e26cSBryan Venteicher vtcon_ctrl_process_event(sc, control, data, data_len);
9286f744ddeSBryan Venteicher VTCON_LOCK(sc);
92904434c94SBryan Venteicher vtcon_ctrl_event_requeue(sc, control);
9306f744ddeSBryan Venteicher }
9316f744ddeSBryan Venteicher
93204434c94SBryan Venteicher if (!detached) {
93304434c94SBryan Venteicher virtqueue_notify(vq);
9346f744ddeSBryan Venteicher if (virtqueue_enable_intr(vq) != 0)
93504434c94SBryan Venteicher taskqueue_enqueue(taskqueue_thread,
93604434c94SBryan Venteicher &sc->vtcon_ctrl_task);
93704434c94SBryan Venteicher }
93804434c94SBryan Venteicher
93904434c94SBryan Venteicher VTCON_UNLOCK(sc);
94004434c94SBryan Venteicher }
94104434c94SBryan Venteicher
94204434c94SBryan Venteicher static void
vtcon_ctrl_event_intr(void * xsc)94304434c94SBryan Venteicher vtcon_ctrl_event_intr(void *xsc)
94404434c94SBryan Venteicher {
94504434c94SBryan Venteicher struct vtcon_softc *sc;
94604434c94SBryan Venteicher
94704434c94SBryan Venteicher sc = xsc;
94804434c94SBryan Venteicher
94904434c94SBryan Venteicher /*
95004434c94SBryan Venteicher * Only some events require us to potentially block, but it
95104434c94SBryan Venteicher * easier to just defer all event handling to the taskqueue.
95204434c94SBryan Venteicher */
9536f744ddeSBryan Venteicher taskqueue_enqueue(taskqueue_thread, &sc->vtcon_ctrl_task);
9546f744ddeSBryan Venteicher }
9556f744ddeSBryan Venteicher
95604434c94SBryan Venteicher static void
vtcon_ctrl_poll(struct vtcon_softc * sc,struct virtio_console_control * control)95704434c94SBryan Venteicher vtcon_ctrl_poll(struct vtcon_softc *sc,
95804434c94SBryan Venteicher struct virtio_console_control *control)
9596f744ddeSBryan Venteicher {
96004434c94SBryan Venteicher struct sglist_seg segs[2];
96104434c94SBryan Venteicher struct sglist sg;
96204434c94SBryan Venteicher struct virtqueue *vq;
96304434c94SBryan Venteicher int error;
96404434c94SBryan Venteicher
96504434c94SBryan Venteicher vq = sc->vtcon_ctrl_txvq;
96604434c94SBryan Venteicher
96704434c94SBryan Venteicher sglist_init(&sg, 2, segs);
96804434c94SBryan Venteicher error = sglist_append(&sg, control,
96904434c94SBryan Venteicher sizeof(struct virtio_console_control));
97004434c94SBryan Venteicher KASSERT(error == 0, ("%s: error %d adding control to sglist",
97104434c94SBryan Venteicher __func__, error));
97204434c94SBryan Venteicher
97304434c94SBryan Venteicher /*
97404434c94SBryan Venteicher * We cannot use the softc lock to serialize access to this
97504434c94SBryan Venteicher * virtqueue since this is called from the tty layer with the
97604434c94SBryan Venteicher * port lock held. Acquiring the softc would violate our lock
97704434c94SBryan Venteicher * ordering.
97804434c94SBryan Venteicher */
97904434c94SBryan Venteicher VTCON_CTRL_TX_LOCK(sc);
98004434c94SBryan Venteicher KASSERT(virtqueue_empty(vq),
981*018a361fSGordon Bergling ("%s: virtqueue is not empty", __func__));
98204434c94SBryan Venteicher error = virtqueue_enqueue(vq, control, &sg, sg.sg_nseg, 0);
98304434c94SBryan Venteicher if (error == 0) {
98404434c94SBryan Venteicher virtqueue_notify(vq);
98504434c94SBryan Venteicher virtqueue_poll(vq, NULL);
98604434c94SBryan Venteicher }
98704434c94SBryan Venteicher VTCON_CTRL_TX_UNLOCK(sc);
98804434c94SBryan Venteicher }
98904434c94SBryan Venteicher
99004434c94SBryan Venteicher static void
vtcon_ctrl_send_control(struct vtcon_softc * sc,uint32_t portid,uint16_t event,uint16_t value)99104434c94SBryan Venteicher vtcon_ctrl_send_control(struct vtcon_softc *sc, uint32_t portid,
99204434c94SBryan Venteicher uint16_t event, uint16_t value)
99304434c94SBryan Venteicher {
99404434c94SBryan Venteicher struct virtio_console_control control;
99504434c94SBryan Venteicher
99604434c94SBryan Venteicher if ((sc->vtcon_flags & VTCON_FLAG_MULTIPORT) == 0)
99704434c94SBryan Venteicher return;
99804434c94SBryan Venteicher
999edf7c8ddSBryan Venteicher control.id = vtcon_gtoh32(sc, portid);
1000edf7c8ddSBryan Venteicher control.event = vtcon_gtoh16(sc, event);
1001edf7c8ddSBryan Venteicher control.value = vtcon_gtoh16(sc, value);
100204434c94SBryan Venteicher
100304434c94SBryan Venteicher vtcon_ctrl_poll(sc, &control);
100404434c94SBryan Venteicher }
100504434c94SBryan Venteicher
100604434c94SBryan Venteicher static int
vtcon_port_enqueue_buf(struct vtcon_port * port,void * buf,size_t len)100704434c94SBryan Venteicher vtcon_port_enqueue_buf(struct vtcon_port *port, void *buf, size_t len)
100804434c94SBryan Venteicher {
100904434c94SBryan Venteicher struct sglist_seg segs[2];
10106f744ddeSBryan Venteicher struct sglist sg;
10116f744ddeSBryan Venteicher struct virtqueue *vq;
10126f744ddeSBryan Venteicher int error;
10136f744ddeSBryan Venteicher
10146f744ddeSBryan Venteicher vq = port->vtcport_invq;
10156f744ddeSBryan Venteicher
101604434c94SBryan Venteicher sglist_init(&sg, 2, segs);
10176f744ddeSBryan Venteicher error = sglist_append(&sg, buf, len);
101804434c94SBryan Venteicher KASSERT(error == 0,
10196f744ddeSBryan Venteicher ("%s: error %d adding buffer to sglist", __func__, error));
10206f744ddeSBryan Venteicher
102104434c94SBryan Venteicher error = virtqueue_enqueue(vq, buf, &sg, 0, sg.sg_nseg);
102204434c94SBryan Venteicher
102304434c94SBryan Venteicher return (error);
10246f744ddeSBryan Venteicher }
10256f744ddeSBryan Venteicher
10266f744ddeSBryan Venteicher static int
vtcon_port_create_buf(struct vtcon_port * port)102704434c94SBryan Venteicher vtcon_port_create_buf(struct vtcon_port *port)
10286f744ddeSBryan Venteicher {
10296f744ddeSBryan Venteicher void *buf;
10306f744ddeSBryan Venteicher int error;
10316f744ddeSBryan Venteicher
10326f744ddeSBryan Venteicher buf = malloc(VTCON_BULK_BUFSZ, M_DEVBUF, M_ZERO | M_NOWAIT);
10336f744ddeSBryan Venteicher if (buf == NULL)
10346f744ddeSBryan Venteicher return (ENOMEM);
10356f744ddeSBryan Venteicher
103604434c94SBryan Venteicher error = vtcon_port_enqueue_buf(port, buf, VTCON_BULK_BUFSZ);
10376f744ddeSBryan Venteicher if (error)
10386f744ddeSBryan Venteicher free(buf, M_DEVBUF);
10396f744ddeSBryan Venteicher
10406f744ddeSBryan Venteicher return (error);
10416f744ddeSBryan Venteicher }
10426f744ddeSBryan Venteicher
10436f744ddeSBryan Venteicher static void
vtcon_port_requeue_buf(struct vtcon_port * port,void * buf)104404434c94SBryan Venteicher vtcon_port_requeue_buf(struct vtcon_port *port, void *buf)
10456f744ddeSBryan Venteicher {
10464d78ef3eSMateusz Guzik int error __diagused;
10476f744ddeSBryan Venteicher
104804434c94SBryan Venteicher error = vtcon_port_enqueue_buf(port, buf, VTCON_BULK_BUFSZ);
10496f744ddeSBryan Venteicher KASSERT(error == 0,
10506f744ddeSBryan Venteicher ("%s: cannot requeue input buffer %d", __func__, error));
10516f744ddeSBryan Venteicher }
10526f744ddeSBryan Venteicher
10536f744ddeSBryan Venteicher static int
vtcon_port_populate(struct vtcon_port * port)10546f744ddeSBryan Venteicher vtcon_port_populate(struct vtcon_port *port)
10556f744ddeSBryan Venteicher {
10566f744ddeSBryan Venteicher struct virtqueue *vq;
10576f744ddeSBryan Venteicher int nbufs, error;
10586f744ddeSBryan Venteicher
10596f744ddeSBryan Venteicher vq = port->vtcport_invq;
10606f744ddeSBryan Venteicher error = ENOSPC;
10616f744ddeSBryan Venteicher
10626f744ddeSBryan Venteicher for (nbufs = 0; !virtqueue_full(vq); nbufs++) {
106304434c94SBryan Venteicher error = vtcon_port_create_buf(port);
10646f744ddeSBryan Venteicher if (error)
10656f744ddeSBryan Venteicher break;
10666f744ddeSBryan Venteicher }
10676f744ddeSBryan Venteicher
10686f744ddeSBryan Venteicher if (nbufs > 0) {
10696f744ddeSBryan Venteicher virtqueue_notify(vq);
10706f744ddeSBryan Venteicher error = 0;
10716f744ddeSBryan Venteicher }
10726f744ddeSBryan Venteicher
10736f744ddeSBryan Venteicher return (error);
10746f744ddeSBryan Venteicher }
10756f744ddeSBryan Venteicher
10766f744ddeSBryan Venteicher static void
vtcon_port_destroy(struct vtcon_port * port)10776f744ddeSBryan Venteicher vtcon_port_destroy(struct vtcon_port *port)
10786f744ddeSBryan Venteicher {
10796f744ddeSBryan Venteicher
10806f744ddeSBryan Venteicher port->vtcport_sc = NULL;
108104434c94SBryan Venteicher port->vtcport_scport = NULL;
108204434c94SBryan Venteicher port->vtcport_invq = NULL;
108304434c94SBryan Venteicher port->vtcport_outvq = NULL;
10846f744ddeSBryan Venteicher port->vtcport_id = -1;
108504434c94SBryan Venteicher mtx_destroy(&port->vtcport_mtx);
10866f744ddeSBryan Venteicher free(port, M_DEVBUF);
10876f744ddeSBryan Venteicher }
10886f744ddeSBryan Venteicher
10896f744ddeSBryan Venteicher static int
vtcon_port_init_vqs(struct vtcon_port * port)109004434c94SBryan Venteicher vtcon_port_init_vqs(struct vtcon_port *port)
109104434c94SBryan Venteicher {
109204434c94SBryan Venteicher struct vtcon_softc_port *scport;
109304434c94SBryan Venteicher int error;
109404434c94SBryan Venteicher
109504434c94SBryan Venteicher scport = port->vtcport_scport;
109604434c94SBryan Venteicher
109704434c94SBryan Venteicher port->vtcport_invq = scport->vcsp_invq;
109804434c94SBryan Venteicher port->vtcport_outvq = scport->vcsp_outvq;
109904434c94SBryan Venteicher
110004434c94SBryan Venteicher /*
110104434c94SBryan Venteicher * Free any data left over from when this virtqueue was in use by a
110204434c94SBryan Venteicher * prior port. We have not yet notified the host that the port is
110304434c94SBryan Venteicher * ready, so assume nothing in the virtqueue can be for us.
110404434c94SBryan Venteicher */
110504434c94SBryan Venteicher vtcon_port_drain(port);
110604434c94SBryan Venteicher
110704434c94SBryan Venteicher KASSERT(virtqueue_empty(port->vtcport_invq),
110804434c94SBryan Venteicher ("%s: in virtqueue is not empty", __func__));
110904434c94SBryan Venteicher KASSERT(virtqueue_empty(port->vtcport_outvq),
111004434c94SBryan Venteicher ("%s: out virtqueue is not empty", __func__));
111104434c94SBryan Venteicher
111204434c94SBryan Venteicher error = vtcon_port_populate(port);
111304434c94SBryan Venteicher if (error)
111404434c94SBryan Venteicher return (error);
111504434c94SBryan Venteicher
111604434c94SBryan Venteicher return (0);
111704434c94SBryan Venteicher }
111804434c94SBryan Venteicher
111904434c94SBryan Venteicher static int
vtcon_port_create(struct vtcon_softc * sc,int id)112004434c94SBryan Venteicher vtcon_port_create(struct vtcon_softc *sc, int id)
11216f744ddeSBryan Venteicher {
11226f744ddeSBryan Venteicher device_t dev;
112304434c94SBryan Venteicher struct vtcon_softc_port *scport;
11246f744ddeSBryan Venteicher struct vtcon_port *port;
11256f744ddeSBryan Venteicher int error;
11266f744ddeSBryan Venteicher
11276f744ddeSBryan Venteicher dev = sc->vtcon_dev;
112804434c94SBryan Venteicher scport = &sc->vtcon_ports[id];
11296f744ddeSBryan Venteicher
113004434c94SBryan Venteicher VTCON_ASSERT_VALID_PORTID(sc, id);
113104434c94SBryan Venteicher MPASS(scport->vcsp_port == NULL);
11326f744ddeSBryan Venteicher
11336f744ddeSBryan Venteicher port = malloc(sizeof(struct vtcon_port), M_DEVBUF, M_NOWAIT | M_ZERO);
11346f744ddeSBryan Venteicher if (port == NULL)
11356f744ddeSBryan Venteicher return (ENOMEM);
11366f744ddeSBryan Venteicher
11376f744ddeSBryan Venteicher port->vtcport_sc = sc;
113804434c94SBryan Venteicher port->vtcport_scport = scport;
11396f744ddeSBryan Venteicher port->vtcport_id = id;
114004434c94SBryan Venteicher mtx_init(&port->vtcport_mtx, "vtcpmtx", NULL, MTX_DEF);
11416f744ddeSBryan Venteicher port->vtcport_tty = tty_alloc_mutex(&vtcon_tty_class, port,
11426f744ddeSBryan Venteicher &port->vtcport_mtx);
11436f744ddeSBryan Venteicher
114404434c94SBryan Venteicher error = vtcon_port_init_vqs(port);
11456f744ddeSBryan Venteicher if (error) {
114604434c94SBryan Venteicher VTCON_PORT_LOCK(port);
114704434c94SBryan Venteicher vtcon_port_teardown(port);
11486f744ddeSBryan Venteicher return (error);
11496f744ddeSBryan Venteicher }
11506f744ddeSBryan Venteicher
11516f744ddeSBryan Venteicher VTCON_LOCK(sc);
115204434c94SBryan Venteicher VTCON_PORT_LOCK(port);
115304434c94SBryan Venteicher scport->vcsp_port = port;
115446822c48SBryan Venteicher vtcon_port_enable_intr(port);
115504434c94SBryan Venteicher vtcon_port_submit_event(port, VIRTIO_CONSOLE_PORT_READY, 1);
115604434c94SBryan Venteicher VTCON_PORT_UNLOCK(port);
11576f744ddeSBryan Venteicher VTCON_UNLOCK(sc);
11586f744ddeSBryan Venteicher
115946822c48SBryan Venteicher tty_makedev(port->vtcport_tty, NULL, "%s%r.%r", VTCON_TTY_PREFIX,
116046822c48SBryan Venteicher device_get_unit(dev), id);
116146822c48SBryan Venteicher
11626f744ddeSBryan Venteicher return (0);
11636f744ddeSBryan Venteicher }
11646f744ddeSBryan Venteicher
11656f744ddeSBryan Venteicher static void
vtcon_port_dev_alias(struct vtcon_port * port,const char * name,size_t len)1166a019e26cSBryan Venteicher vtcon_port_dev_alias(struct vtcon_port *port, const char *name, size_t len)
1167a019e26cSBryan Venteicher {
1168a019e26cSBryan Venteicher struct vtcon_softc *sc;
1169a019e26cSBryan Venteicher struct cdev *pdev;
1170a019e26cSBryan Venteicher struct tty *tp;
1171a019e26cSBryan Venteicher int i, error;
1172a019e26cSBryan Venteicher
1173a019e26cSBryan Venteicher sc = port->vtcport_sc;
1174a019e26cSBryan Venteicher tp = port->vtcport_tty;
1175a019e26cSBryan Venteicher
1176a019e26cSBryan Venteicher if (port->vtcport_flags & VTCON_PORT_FLAG_ALIAS)
1177a019e26cSBryan Venteicher return;
1178a019e26cSBryan Venteicher
1179a019e26cSBryan Venteicher /* Port name is UTF-8, but we can only handle ASCII. */
1180a019e26cSBryan Venteicher for (i = 0; i < len; i++) {
1181a019e26cSBryan Venteicher if (!isascii(name[i]))
1182a019e26cSBryan Venteicher return;
1183a019e26cSBryan Venteicher }
1184a019e26cSBryan Venteicher
1185a019e26cSBryan Venteicher /*
1186a019e26cSBryan Venteicher * Port name may not conform to the devfs requirements so we cannot use
1187a019e26cSBryan Venteicher * tty_makealias() because the MAKEDEV_CHECKNAME flag must be specified.
1188a019e26cSBryan Venteicher */
1189a019e26cSBryan Venteicher error = make_dev_alias_p(MAKEDEV_NOWAIT | MAKEDEV_CHECKNAME, &pdev,
1190a019e26cSBryan Venteicher tp->t_dev, "%s/%*s", VTCON_TTY_ALIAS_PREFIX, (int)len, name);
1191a019e26cSBryan Venteicher if (error) {
1192a019e26cSBryan Venteicher device_printf(sc->vtcon_dev,
1193a019e26cSBryan Venteicher "%s: cannot make dev alias (%s/%*s) error %d\n", __func__,
1194a019e26cSBryan Venteicher VTCON_TTY_ALIAS_PREFIX, (int)len, name, error);
1195a019e26cSBryan Venteicher } else
1196a019e26cSBryan Venteicher port->vtcport_flags |= VTCON_PORT_FLAG_ALIAS;
1197a019e26cSBryan Venteicher }
1198a019e26cSBryan Venteicher
1199a019e26cSBryan Venteicher static void
vtcon_port_drain_bufs(struct virtqueue * vq)120004434c94SBryan Venteicher vtcon_port_drain_bufs(struct virtqueue *vq)
12016f744ddeSBryan Venteicher {
12026f744ddeSBryan Venteicher void *buf;
12036f744ddeSBryan Venteicher int last;
12046f744ddeSBryan Venteicher
12056f744ddeSBryan Venteicher last = 0;
12066f744ddeSBryan Venteicher
12076f744ddeSBryan Venteicher while ((buf = virtqueue_drain(vq, &last)) != NULL)
12086f744ddeSBryan Venteicher free(buf, M_DEVBUF);
12096f744ddeSBryan Venteicher }
12106f744ddeSBryan Venteicher
12116f744ddeSBryan Venteicher static void
vtcon_port_drain(struct vtcon_port * port)121204434c94SBryan Venteicher vtcon_port_drain(struct vtcon_port *port)
12136f744ddeSBryan Venteicher {
12146f744ddeSBryan Venteicher
121504434c94SBryan Venteicher vtcon_port_drain_bufs(port->vtcport_invq);
121604434c94SBryan Venteicher }
121704434c94SBryan Venteicher
121804434c94SBryan Venteicher static void
vtcon_port_teardown(struct vtcon_port * port)121904434c94SBryan Venteicher vtcon_port_teardown(struct vtcon_port *port)
122004434c94SBryan Venteicher {
122104434c94SBryan Venteicher struct tty *tp;
122204434c94SBryan Venteicher
12236f744ddeSBryan Venteicher tp = port->vtcport_tty;
12246f744ddeSBryan Venteicher
122504434c94SBryan Venteicher port->vtcport_flags |= VTCON_PORT_FLAG_GONE;
12266f744ddeSBryan Venteicher
12276f744ddeSBryan Venteicher if (tp != NULL) {
12286f744ddeSBryan Venteicher atomic_add_int(&vtcon_pending_free, 1);
12296f744ddeSBryan Venteicher tty_rel_gone(tp);
12306f744ddeSBryan Venteicher } else
12316f744ddeSBryan Venteicher vtcon_port_destroy(port);
12326f744ddeSBryan Venteicher }
12336f744ddeSBryan Venteicher
12346f744ddeSBryan Venteicher static void
vtcon_port_change_size(struct vtcon_port * port,uint16_t cols,uint16_t rows)12356f744ddeSBryan Venteicher vtcon_port_change_size(struct vtcon_port *port, uint16_t cols, uint16_t rows)
12366f744ddeSBryan Venteicher {
12376f744ddeSBryan Venteicher struct tty *tp;
12386f744ddeSBryan Venteicher struct winsize sz;
12396f744ddeSBryan Venteicher
12406f744ddeSBryan Venteicher tp = port->vtcport_tty;
12416f744ddeSBryan Venteicher
12426f744ddeSBryan Venteicher if (tp == NULL)
12436f744ddeSBryan Venteicher return;
12446f744ddeSBryan Venteicher
12456f744ddeSBryan Venteicher bzero(&sz, sizeof(struct winsize));
12466f744ddeSBryan Venteicher sz.ws_col = cols;
12476f744ddeSBryan Venteicher sz.ws_row = rows;
12486f744ddeSBryan Venteicher
12496f744ddeSBryan Venteicher tty_set_winsize(tp, &sz);
125004434c94SBryan Venteicher }
125104434c94SBryan Venteicher
125204434c94SBryan Venteicher static void
vtcon_port_update_console_size(struct vtcon_softc * sc)125304434c94SBryan Venteicher vtcon_port_update_console_size(struct vtcon_softc *sc)
125404434c94SBryan Venteicher {
125504434c94SBryan Venteicher struct vtcon_port *port;
125604434c94SBryan Venteicher struct vtcon_softc_port *scport;
125704434c94SBryan Venteicher uint16_t cols, rows;
125804434c94SBryan Venteicher
125904434c94SBryan Venteicher vtcon_get_console_size(sc, &cols, &rows);
126004434c94SBryan Venteicher
126104434c94SBryan Venteicher /*
126204434c94SBryan Venteicher * For now, assume the first (only) port is the console. Note
126304434c94SBryan Venteicher * QEMU does not implement this feature yet.
126404434c94SBryan Venteicher */
126504434c94SBryan Venteicher scport = &sc->vtcon_ports[0];
126604434c94SBryan Venteicher
126704434c94SBryan Venteicher VTCON_LOCK(sc);
126804434c94SBryan Venteicher port = scport->vcsp_port;
126904434c94SBryan Venteicher
127004434c94SBryan Venteicher if (port != NULL) {
127104434c94SBryan Venteicher VTCON_PORT_LOCK(port);
127204434c94SBryan Venteicher VTCON_UNLOCK(sc);
127304434c94SBryan Venteicher vtcon_port_change_size(port, cols, rows);
12746f744ddeSBryan Venteicher VTCON_PORT_UNLOCK(port);
127504434c94SBryan Venteicher } else
127604434c94SBryan Venteicher VTCON_UNLOCK(sc);
12776f744ddeSBryan Venteicher }
12786f744ddeSBryan Venteicher
12796f744ddeSBryan Venteicher static void
vtcon_port_enable_intr(struct vtcon_port * port)12806f744ddeSBryan Venteicher vtcon_port_enable_intr(struct vtcon_port *port)
12816f744ddeSBryan Venteicher {
12826f744ddeSBryan Venteicher
12836f744ddeSBryan Venteicher /*
1284453130d9SPedro F. Giffuni * NOTE: The out virtqueue is always polled, so its interrupt
128504434c94SBryan Venteicher * kept disabled.
12866f744ddeSBryan Venteicher */
12876f744ddeSBryan Venteicher virtqueue_enable_intr(port->vtcport_invq);
12886f744ddeSBryan Venteicher }
12896f744ddeSBryan Venteicher
12906f744ddeSBryan Venteicher static void
vtcon_port_disable_intr(struct vtcon_port * port)12916f744ddeSBryan Venteicher vtcon_port_disable_intr(struct vtcon_port *port)
12926f744ddeSBryan Venteicher {
12936f744ddeSBryan Venteicher
12946f744ddeSBryan Venteicher if (port->vtcport_invq != NULL)
12956f744ddeSBryan Venteicher virtqueue_disable_intr(port->vtcport_invq);
12966f744ddeSBryan Venteicher if (port->vtcport_outvq != NULL)
12976f744ddeSBryan Venteicher virtqueue_disable_intr(port->vtcport_outvq);
12986f744ddeSBryan Venteicher }
12996f744ddeSBryan Venteicher
13006f744ddeSBryan Venteicher static void
vtcon_port_in(struct vtcon_port * port)130104434c94SBryan Venteicher vtcon_port_in(struct vtcon_port *port)
13026f744ddeSBryan Venteicher {
13036f744ddeSBryan Venteicher struct virtqueue *vq;
130404434c94SBryan Venteicher struct tty *tp;
13056f744ddeSBryan Venteicher char *buf;
13066f744ddeSBryan Venteicher uint32_t len;
13076f744ddeSBryan Venteicher int i, deq;
13086f744ddeSBryan Venteicher
13096f744ddeSBryan Venteicher tp = port->vtcport_tty;
13106f744ddeSBryan Venteicher vq = port->vtcport_invq;
13116f744ddeSBryan Venteicher
13126f744ddeSBryan Venteicher again:
13136f744ddeSBryan Venteicher deq = 0;
13146f744ddeSBryan Venteicher
13156f744ddeSBryan Venteicher while ((buf = virtqueue_dequeue(vq, &len)) != NULL) {
1316b84b3efdSBryan Venteicher for (i = 0; i < len; i++) {
1317b84b3efdSBryan Venteicher #if defined(KDB)
1318b84b3efdSBryan Venteicher if (port->vtcport_flags & VTCON_PORT_FLAG_CONSOLE)
1319b84b3efdSBryan Venteicher kdb_alt_break(buf[i],
1320b84b3efdSBryan Venteicher &port->vtcport_alt_break_state);
1321b84b3efdSBryan Venteicher #endif
13226f744ddeSBryan Venteicher ttydisc_rint(tp, buf[i], 0);
1323b84b3efdSBryan Venteicher }
132404434c94SBryan Venteicher vtcon_port_requeue_buf(port, buf);
132504434c94SBryan Venteicher deq++;
13266f744ddeSBryan Venteicher }
13276f744ddeSBryan Venteicher ttydisc_rint_done(tp);
13286f744ddeSBryan Venteicher
13296f744ddeSBryan Venteicher if (deq > 0)
13306f744ddeSBryan Venteicher virtqueue_notify(vq);
13316f744ddeSBryan Venteicher
13326f744ddeSBryan Venteicher if (virtqueue_enable_intr(vq) != 0)
13336f744ddeSBryan Venteicher goto again;
13346f744ddeSBryan Venteicher }
13356f744ddeSBryan Venteicher
13366f744ddeSBryan Venteicher static void
vtcon_port_intr(void * scportx)133704434c94SBryan Venteicher vtcon_port_intr(void *scportx)
13386f744ddeSBryan Venteicher {
133904434c94SBryan Venteicher struct vtcon_softc_port *scport;
134004434c94SBryan Venteicher struct vtcon_softc *sc;
13416f744ddeSBryan Venteicher struct vtcon_port *port;
13426f744ddeSBryan Venteicher
134304434c94SBryan Venteicher scport = scportx;
134404434c94SBryan Venteicher sc = scport->vcsp_sc;
13456f744ddeSBryan Venteicher
134604434c94SBryan Venteicher VTCON_LOCK(sc);
134704434c94SBryan Venteicher port = scport->vcsp_port;
134804434c94SBryan Venteicher if (port == NULL) {
134904434c94SBryan Venteicher VTCON_UNLOCK(sc);
135004434c94SBryan Venteicher return;
135104434c94SBryan Venteicher }
135204434c94SBryan Venteicher VTCON_PORT_LOCK(port);
135304434c94SBryan Venteicher VTCON_UNLOCK(sc);
135404434c94SBryan Venteicher if ((port->vtcport_flags & VTCON_PORT_FLAG_GONE) == 0)
135504434c94SBryan Venteicher vtcon_port_in(port);
135604434c94SBryan Venteicher VTCON_PORT_UNLOCK(port);
13576f744ddeSBryan Venteicher }
13586f744ddeSBryan Venteicher
13596f744ddeSBryan Venteicher static void
vtcon_port_out(struct vtcon_port * port,void * buf,int bufsize)136004434c94SBryan Venteicher vtcon_port_out(struct vtcon_port *port, void *buf, int bufsize)
13616f744ddeSBryan Venteicher {
136204434c94SBryan Venteicher struct sglist_seg segs[2];
13636f744ddeSBryan Venteicher struct sglist sg;
13646f744ddeSBryan Venteicher struct virtqueue *vq;
13656f744ddeSBryan Venteicher int error;
13666f744ddeSBryan Venteicher
13676f744ddeSBryan Venteicher vq = port->vtcport_outvq;
136804434c94SBryan Venteicher KASSERT(virtqueue_empty(vq),
1369*018a361fSGordon Bergling ("%s: port %p out virtqueue not empty", __func__, port));
13706f744ddeSBryan Venteicher
137104434c94SBryan Venteicher sglist_init(&sg, 2, segs);
13726f744ddeSBryan Venteicher error = sglist_append(&sg, buf, bufsize);
137304434c94SBryan Venteicher KASSERT(error == 0, ("%s: error %d adding buffer to sglist",
137404434c94SBryan Venteicher __func__, error));
13756f744ddeSBryan Venteicher
137604434c94SBryan Venteicher error = virtqueue_enqueue(vq, buf, &sg, sg.sg_nseg, 0);
137704434c94SBryan Venteicher if (error == 0) {
13786f744ddeSBryan Venteicher virtqueue_notify(vq);
13796f744ddeSBryan Venteicher virtqueue_poll(vq, NULL);
13806f744ddeSBryan Venteicher }
13816f744ddeSBryan Venteicher }
13826f744ddeSBryan Venteicher
13836f744ddeSBryan Venteicher static void
vtcon_port_submit_event(struct vtcon_port * port,uint16_t event,uint16_t value)138404434c94SBryan Venteicher vtcon_port_submit_event(struct vtcon_port *port, uint16_t event,
13856f744ddeSBryan Venteicher uint16_t value)
13866f744ddeSBryan Venteicher {
13876f744ddeSBryan Venteicher struct vtcon_softc *sc;
13886f744ddeSBryan Venteicher
13896f744ddeSBryan Venteicher sc = port->vtcport_sc;
13906f744ddeSBryan Venteicher
139104434c94SBryan Venteicher vtcon_ctrl_send_control(sc, port->vtcport_id, event, value);
13926f744ddeSBryan Venteicher }
13936f744ddeSBryan Venteicher
13946f744ddeSBryan Venteicher static int
vtcon_tty_open(struct tty * tp)13956f744ddeSBryan Venteicher vtcon_tty_open(struct tty *tp)
13966f744ddeSBryan Venteicher {
13976f744ddeSBryan Venteicher struct vtcon_port *port;
13986f744ddeSBryan Venteicher
13996f744ddeSBryan Venteicher port = tty_softc(tp);
14006f744ddeSBryan Venteicher
140104434c94SBryan Venteicher if (port->vtcport_flags & VTCON_PORT_FLAG_GONE)
140204434c94SBryan Venteicher return (ENXIO);
140304434c94SBryan Venteicher
140404434c94SBryan Venteicher vtcon_port_submit_event(port, VIRTIO_CONSOLE_PORT_OPEN, 1);
14056f744ddeSBryan Venteicher
14066f744ddeSBryan Venteicher return (0);
14076f744ddeSBryan Venteicher }
14086f744ddeSBryan Venteicher
14096f744ddeSBryan Venteicher static void
vtcon_tty_close(struct tty * tp)14106f744ddeSBryan Venteicher vtcon_tty_close(struct tty *tp)
14116f744ddeSBryan Venteicher {
14126f744ddeSBryan Venteicher struct vtcon_port *port;
14136f744ddeSBryan Venteicher
14146f744ddeSBryan Venteicher port = tty_softc(tp);
14156f744ddeSBryan Venteicher
141604434c94SBryan Venteicher if (port->vtcport_flags & VTCON_PORT_FLAG_GONE)
141704434c94SBryan Venteicher return;
141804434c94SBryan Venteicher
141904434c94SBryan Venteicher vtcon_port_submit_event(port, VIRTIO_CONSOLE_PORT_OPEN, 0);
14206f744ddeSBryan Venteicher }
14216f744ddeSBryan Venteicher
14226f744ddeSBryan Venteicher static void
vtcon_tty_outwakeup(struct tty * tp)14236f744ddeSBryan Venteicher vtcon_tty_outwakeup(struct tty *tp)
14246f744ddeSBryan Venteicher {
14256f744ddeSBryan Venteicher struct vtcon_port *port;
14266f744ddeSBryan Venteicher char buf[VTCON_BULK_BUFSZ];
14276f744ddeSBryan Venteicher int len;
14286f744ddeSBryan Venteicher
14296f744ddeSBryan Venteicher port = tty_softc(tp);
14306f744ddeSBryan Venteicher
143104434c94SBryan Venteicher if (port->vtcport_flags & VTCON_PORT_FLAG_GONE)
143204434c94SBryan Venteicher return;
143304434c94SBryan Venteicher
14346f744ddeSBryan Venteicher while ((len = ttydisc_getc(tp, buf, sizeof(buf))) != 0)
143504434c94SBryan Venteicher vtcon_port_out(port, buf, len);
14366f744ddeSBryan Venteicher }
14376f744ddeSBryan Venteicher
14386f744ddeSBryan Venteicher static void
vtcon_tty_free(void * xport)14396f744ddeSBryan Venteicher vtcon_tty_free(void *xport)
14406f744ddeSBryan Venteicher {
14416f744ddeSBryan Venteicher struct vtcon_port *port;
14426f744ddeSBryan Venteicher
14436f744ddeSBryan Venteicher port = xport;
14446f744ddeSBryan Venteicher
14456f744ddeSBryan Venteicher vtcon_port_destroy(port);
14466f744ddeSBryan Venteicher atomic_subtract_int(&vtcon_pending_free, 1);
14476f744ddeSBryan Venteicher }
14486f744ddeSBryan Venteicher
14496f744ddeSBryan Venteicher static void
vtcon_get_console_size(struct vtcon_softc * sc,uint16_t * cols,uint16_t * rows)14506f744ddeSBryan Venteicher vtcon_get_console_size(struct vtcon_softc *sc, uint16_t *cols, uint16_t *rows)
14516f744ddeSBryan Venteicher {
14526f744ddeSBryan Venteicher struct virtio_console_config concfg;
14536f744ddeSBryan Venteicher
14546f744ddeSBryan Venteicher KASSERT(sc->vtcon_flags & VTCON_FLAG_SIZE,
14556f744ddeSBryan Venteicher ("%s: size feature not negotiated", __func__));
14566f744ddeSBryan Venteicher
14576f744ddeSBryan Venteicher vtcon_read_config(sc, &concfg);
14586f744ddeSBryan Venteicher
14596f744ddeSBryan Venteicher *cols = concfg.cols;
14606f744ddeSBryan Venteicher *rows = concfg.rows;
14616f744ddeSBryan Venteicher }
14626f744ddeSBryan Venteicher
14636f744ddeSBryan Venteicher static void
vtcon_enable_interrupts(struct vtcon_softc * sc)14646f744ddeSBryan Venteicher vtcon_enable_interrupts(struct vtcon_softc *sc)
14656f744ddeSBryan Venteicher {
146604434c94SBryan Venteicher struct vtcon_softc_port *scport;
14676f744ddeSBryan Venteicher struct vtcon_port *port;
146804434c94SBryan Venteicher int i;
146904434c94SBryan Venteicher
147004434c94SBryan Venteicher VTCON_LOCK(sc);
14716f744ddeSBryan Venteicher
14726f744ddeSBryan Venteicher if (sc->vtcon_flags & VTCON_FLAG_MULTIPORT)
14736f744ddeSBryan Venteicher virtqueue_enable_intr(sc->vtcon_ctrl_rxvq);
14746f744ddeSBryan Venteicher
147504434c94SBryan Venteicher for (i = 0; i < sc->vtcon_max_ports; i++) {
147604434c94SBryan Venteicher scport = &sc->vtcon_ports[i];
147704434c94SBryan Venteicher
147804434c94SBryan Venteicher port = scport->vcsp_port;
147904434c94SBryan Venteicher if (port == NULL)
148004434c94SBryan Venteicher continue;
148104434c94SBryan Venteicher
148204434c94SBryan Venteicher VTCON_PORT_LOCK(port);
14836f744ddeSBryan Venteicher vtcon_port_enable_intr(port);
148404434c94SBryan Venteicher VTCON_PORT_UNLOCK(port);
148504434c94SBryan Venteicher }
148604434c94SBryan Venteicher
148704434c94SBryan Venteicher VTCON_UNLOCK(sc);
14886f744ddeSBryan Venteicher }
14896f744ddeSBryan Venteicher
14906f744ddeSBryan Venteicher static void
vtcon_disable_interrupts(struct vtcon_softc * sc)14916f744ddeSBryan Venteicher vtcon_disable_interrupts(struct vtcon_softc *sc)
14926f744ddeSBryan Venteicher {
149304434c94SBryan Venteicher struct vtcon_softc_port *scport;
14946f744ddeSBryan Venteicher struct vtcon_port *port;
149504434c94SBryan Venteicher int i;
149604434c94SBryan Venteicher
149704434c94SBryan Venteicher VTCON_LOCK_ASSERT(sc);
14986f744ddeSBryan Venteicher
14996f744ddeSBryan Venteicher if (sc->vtcon_flags & VTCON_FLAG_MULTIPORT)
15006f744ddeSBryan Venteicher virtqueue_disable_intr(sc->vtcon_ctrl_rxvq);
15016f744ddeSBryan Venteicher
150204434c94SBryan Venteicher for (i = 0; i < sc->vtcon_max_ports; i++) {
150304434c94SBryan Venteicher scport = &sc->vtcon_ports[i];
150404434c94SBryan Venteicher
150504434c94SBryan Venteicher port = scport->vcsp_port;
150604434c94SBryan Venteicher if (port == NULL)
150704434c94SBryan Venteicher continue;
150804434c94SBryan Venteicher
150904434c94SBryan Venteicher VTCON_PORT_LOCK(port);
15106f744ddeSBryan Venteicher vtcon_port_disable_intr(port);
151104434c94SBryan Venteicher VTCON_PORT_UNLOCK(port);
151204434c94SBryan Venteicher }
15136f744ddeSBryan Venteicher }
1514