.. _virtio-console: Virtio-Console ############## The Virtio-console is a simple device for data input and output. The console's virtio device ID is ``3`` and can have from 1 to 16 ports. Each port has a pair of input and output virtqueues used to communicate information between the Front End (FE) and Back end (BE) drivers. The size of each virtqueue is 64 (configurable in the source code). The FE driver will place empty buffers for incoming data onto the receiving virtqueue, and enqueue outgoing characters onto the transmitting virtqueue. A Virtio-console device has a pair of control IO virtqueues as well. The control virtqueues are used to communicate information between the device and the driver, including: ports being opened and closed on either side of the connection, indication from the host about whether a particular port is a console port, adding new ports, port hot-plug or unplug, indication from the guest about whether a port or a device was successfully added, or a port opened or closed. The virtio-console architecture diagram in ACRN is shown below. .. figure:: images/virtio-console-arch.png :align: center :width: 700px :name: virtio-console-arch Virtio-console Architecture Diagram Virtio-console is implemented as a virtio legacy device in the ACRN Device Model (DM), and is registered as a PCI virtio device to the guest OS. No changes are required in the frontend Linux virtio-console except that the guest (User VM) kernel should be built with ``CONFIG_VIRTIO_CONSOLE=y``. The virtio console FE driver registers an HVC console to the kernel if the port is configured as console. Otherwise it registers a char device named ``/dev/vportXpY`` to the kernel, and can be read and written from the user space. There are two virtqueues for a port, one is for transmitting and the other is for receiving. The FE driver places empty buffers onto the receiving virtqueue for incoming data, and enqueues outgoing characters onto the transmitting virtqueue. The virtio console BE driver copies data from the FE's transmitting virtqueue when it receives a kick on the virtqueue (implemented as a vmexit). The BE driver then writes the data to the backend, and can be implemented as PTY, TTY, STDIO, and a regular file. BE driver uses mevent to poll the available data from the backend file descriptor. When new data is available, the BE driver reads it to the receiving virtqueue of the FE, followed by an interrupt injection. The feature bits supported by the BE device are: .. list-table:: Feature bits supported by BE drivers :widths: 30 50 :header-rows: 0 * - VTCON_F_SIZE(bit 0) - configuration columns and rows are valid. * - VTCON_F_MULTIPORT(bit 1) - device supports multiple ports, and control virtqueues will be used. * - VTCON_F_EMERG_WRITE(bit 2) - device supports emergency write. Virtio-console supports redirecting guest output to various backend devices. The following backend devices are supported in the ACRN Device Model: STDIO, TTY, PTY and regular file. The Device Model configuration command syntax for virtio-console is:: virtio-console,[@]stdio|tty|pty|file:portname[=portpath]\ [,[@]stdio|tty|pty|file:portname[=portpath][:socket_type]] - Preceding with ``@`` marks the port as a console port, otherwise it is a normal virtio-serial port - The ``portpath`` can be omitted when backend is ``stdio`` or ``pty`` - The ``stdio/tty/pty`` is TTY capable, which means :kbd:`TAB` and :kbd:`BACKSPACE` are supported, as on a regular terminal - When TTY is used, make sure the redirected TTY is sleeping, (e.g., by ``sleep 2d`` command), and will not read input from stdin before it is used by virtio-console to redirect guest output. - When virtio-console socket_type is appointed to client, make sure server VM (socket_type is appointed to server) has started. - Claiming multiple virtio-serial ports as consoles is supported, however the guest Linux OS will only use one of them, through the ``console=hvcN`` kernel parameter. For example, the following command defines two backend ports, which are both console ports, but the frontend driver will only use the second port named ``pty_port`` as its hvc console (specified by ``console=hvc1`` in the kernel command line):: -s n,virtio-console,@tty:tty_port=/dev/pts/0,@pty:pty_port \ -B "root=/dev/vda2 rw rootwait maxcpus=$2 nohpet console=hvc1 console=ttyS0 ..." Console Backend Use Cases ************************* The following sections elaborate on each backend. STDIO ===== 1. Add a PCI slot to the Device Model (``acrn-dm``) command line:: -s n,virtio-console,@stdio:stdio_port #. Add the ``console`` parameter to the guest OS kernel command line:: console=hvc0 PTY === 1. Add a PCI slot to the Device Model (``acrn-dm``) command line:: -s n,virtio-console,@pty:pty_port #. Add the ``console`` parameter to the guest OS kernel command line:: console=hvc0 One line of information, such as shown below, will be printed in the terminal after ``acrn-dm`` is launched (``/dev/pts/0`` may be different, depending on your use case): .. code-block:: console virt-console backend redirected to /dev/pts/0 #. Use a terminal emulator, such as ``minicom`` or ``screen``, to connect to the TTY node: .. code-block:: console # minicom -D /dev/pts/0 or: .. code-block:: console # screen /dev/pts/0 TTY === 1. Identify your TTY that will be used as the User VM console: - If you're connected to your device over the network via ``ssh``, use the Linux ``tty`` command, and it will report the node (may be different in your use case): .. code-block:: console /dev/pts/0 # sleep 2d - If you do not have network access to your device, use screen to create a new TTY: .. code-block:: console # screen # tty you will see (depending on your use case): .. code-block:: console /dev/pts/0 Prevent the TTY from responding by sleeping: .. code-block:: console # sleep 2d and detach the TTY by pressing :kbd:`CTRL-A` :kbd:`d`. #. Add a PCI slot to the Device Model (``acrn-dm``) command line (changing the ``dev/pts/X`` to match your use case):: -s n,virtio-console,@tty:tty_port=/dev/pts/X #. Add the console parameter to the guest OS kernel command line:: console=hvc0 #. Go back to the previous TTY. For example, if you're using ``screen``, use: .. code-block:: console # screen -ls # screen -r FILE ==== The File backend only supports console output to a file (no input). 1. Add a PCI slot to the Device Model (``acrn-dm``) command line, adjusting the ```` to your use case:: -s n,virtio-console,@file:file_port= #. Add the console parameter to the guest OS kernel command line:: console=hvc0 SOCKET ====== The virtio-console socket-type can be set as socket server or client. The Device Model creates a Unix domain socket if appointed the socket_type as server. Then the Service VM or another User VM can bind and listen for communication requirements. If appointed to client, make sure the socket server is ready before launching the Device Model. 1. Add a PCI slot to the Device Model (``acrn-dm``) command line, adjusting the ```` to your use case in the VM1 configuration:: -s n,virtio-console,socket:socket_file_name=:server #. Add a PCI slot to the Device Model (``acrn-dm``) command line, adjusting the ```` to your use case in the VM2 configuration:: -s n,virtio-console,socket:socket_file_name=:client #. Log in to VM1, connect to the virtual port (vport1p0, 1 is decided by the front-end driver): .. code-block:: console # minicom -D /dev/vport1p0 #. Log in to VM2, connect to the virtual port (vport3p0, 3 is decided by the front-end driver): .. code-block:: console # minicom -D /dev/vport3p0 #. Input into ``minicom`` window of VM1 or VM2, the ``minicom`` window of VM1 will indicate the input from VM2, the ``minicom`` window of VM2 will indicate the input from VM1.