/usbkvm

Keyboard, video and mouse over USB

Primary LanguageC

USBKVM

You know this? Your home server won't ping anymore and now you gotta get a monitor and a keyboard to fix it, but can't find anywhere to put them?

Introducing USBKVM, a matchbox-sized keyboard, video monitor and mouse that you can use from the comfort of the screen and input devices you already have. It's just like a KVM over IP console you may find in a server rack, but without the network in between.

A grey, obround 3D-printed 5x5 cm box with a white "USBKVM" label on it. A black HDMI cable an a yellow USB Type-C cable are connected on the left side. Another yellow USB Type-C cable is plugged into the right side.

Diagram showing a server connected to a laptop using USBKVM

Hardware (USBKVM)

3D rendering of a green PCA with an HDMI and a USB Type-C connector on the left edge. On the board itself, there are two QFN ICs as well as a variety of other components. On the right edge, there's another USB Type-C connector. USBKVM is silkscreend in the top right corner.

USBKVM is built around the MS2109 HDMI to USB capture IC that's found on virtually all cheap USB 2.0 HDMI capture sticks. It's not stocked by any of the usual distributors, I got mine from klayers.net. They came with the marking lasered off, which made me a bit suspicious that I've been scammed, but they work just like the original. So I either got the real deal or a perfect clone.

Turns out that this chip has an I²C controller builtin and people reverse engineered to the point that it can be used to communicate with arbitrary I²C devices.

This I²C side channel is used to talk to an STM32F042 MCU that emulates USB keyboard and mouse.

Apart from these two chips, there's not much else on the board other than some voltage regulators, ESD protection diodes and an I²C EEPROM that holds configuration data for the MS2109.

The differential pairs for HDMI and USB are designed for JLCPCB's 7628 stackup.

As usual, this board is made with Horizon EDA. Schematics

The same PCB as above, just as a photo.

The modwire is there because I forgot to order the voltage regulator for the 3.3V USB supply.

Hardware (USBKVM Pro)

3D rendering of a green PCA with a VGA, HDMI and a USB Type-C connector on the left edge. On the board itself, there are thre QFN ICs as well as a variety of other components. On the right edge, there's another USB Type-C connector and four light pipes. USBKVM Pro is silkscreend in the top corner.

USBKVM Pro adds a VGA input since many servers still only provide VGA video output. To accomplish this, the board includes the MS9288C VGA to HDMI converter and an HDMI mux to switch between HDMI and VGA input. Same as the MS2109, the MS9288C is available from klayers.net. The board has provision for connecting an I²C EEPROM to it, but so far I haven't found any documentation on how to make use of it. Anyhow, the chip works just fine without the EEPROM.

The same PCB as above, just as a photo.

For automatically switching between HDMI and VGA input, the MCU senses the +5V applied to the VGA connector with an N-channel MOSFET. It's not directly connected to the MCU since it could reverse-power the MCU if host power isn't connected. For the same reason the I²C lines for the EDID EEPROM aren't connected to the MCU either.

USBKVM Pro uses warm white status LEDs since that's an indicator LED color I'd like to see more often among the sea of cold white and colored LEDs. Unfortunately though there are no decent warm white indicator LEDs, much less side view variants. So I went with LEDs intended for lighting applications running at much lower current and light pipes.

Warm white status LEDs reflecting on a wooden surface

Programming

To be recognized by the client app and show color bars rather than a black screen for no input signal, the EEPROM attached to the MS2109 needs to be programmed. This is done with the cli included in ms-tools: ./cli write-file EEPROM 0 progdata/ms2109.bin. The EEPROM content is based on the one of a capture stick, modified with the "MS21XX&91XXDownloadTool_1.7.0" to set the device string to "USBKVM".

For the Pro variant, there's also the VGA EDID EEPROM that requires programming. I cobbled together its contents by combining the EDID data from an LCD monitor with the one from the MS2109 using the AW EDID Editor. It's intended to be programmed using the VGA connector. To do so, load the i2c-dev kernel module and use the converter script to convert the EDID EEPROM contents to a set of i2ctransfer commands. You probably need to adjust the I²C bus number to the right one for your system.

Apart from the firmware, the MCU needs the BOOT_SEL option bit cleared so that it can jump to the ROM bootloader. Unfortunately, openocd doesn't support this bit(?), so it has to be modified using the STM32Cube Programmer.

Software

The default Raspberry Pi OS desktop with a terminal window showing some dmesg output.

Conceptually, the client is pretty simple as it only has to do these two jobs:

  1. Display the captured video feed. The MS2109 implements the USB video device class, just like a webcam.
  2. Send keyboard and mouse input the the MCU via the aforementioned I²C interface

To get some video on the screen quickly, the client started out as a GStreamer Gtk example, first part done!

The second part however required some more work as the code that makes use of the MS2109's I²C interface is written in Go. After realizing that it's a surprisingly substantial amount code, I quickly scrapped the idea of reimplementing it in C++ and instead figured out how to compile it as a static library that I can integrate into my C++ project.

The protocol layer used for communicating with the MCU firmware is defined in a header file that's both included by the client an the firmware. Some care has to be taken to get the same struct packing on both platforms.

To keep the firmware simple, the client has to send USB HID scan codes, so it has to generate them from the key codes it receives from Gtk. Not wanting to reinvent the wheel, I looked into the sources of software that has has to accomplish a similar task, in particular the SPICE Gtk client, and discovered the keycodemap project. It provides a mapping table from the X11 keycodes used by Gtk to HID scancodes among some other codes used by different platforms.

Since I was too lazy to figure out how to properly capture the mouse to emit relative movements, USBKVM emulates an absolute pointing device. While this works perfectly on Windows and Linux, it's completely broken in less sophisticated environments such as UEFI setup utilities or option ROMs.

See the app's README for how to build it.

Firmware

The firmware uses the excellent USB stack tinyusb to emulate the aforementioned absolute pointing devices as well as a standard USB HID keyboard. After having used the USB device stack included in the STM32 HAL in some previous projects, I can't recommend tinyusb enough. It's really good as far as anything involving USB goes.

On the other end, it implements an I²C peripheral to receive commands from the MS2109. After some fiddling, I got it to work reliably without clock stretching since I don't think the MS2109 supports that. One side-effect of that is that the first byte the MCU has to send in a read transaction has to be loaded into the transmit data register even before the address is received since the time between the SCL edge that latches the last address bit and the first edge that clocks out the first data bye isn't long enough to prepare the data to be sent.

USBKVM and USBKVM Pro hardware variants are supported by one single firmware image. To tell them apart, pin PB8 is grounded on USBKVM Pro and floating on USBKVM.

Firmware updates

For updating itself without special programming adapters, the firmware can activate the built-in ROM bootloader of the STM32. Unfortunately, I picked the wrong pins so only the USB DFU bootloader can be used and not the I²C one. Run make dfu in the firmware directory to build and flash the firmware.

Future improvement: Implement a custom I²C bootloader, that will also make the device less brickable since the activating the ROM bootloaders requires a working firmware.

Case

As usual, the case is of the 3D-printed kind. It consists of the obround main housing and a lid to close it up. The lid attaches with two M2 screws that go into nuts located in the main housing.

Dune 3D screenshot of the case without the lid and some constraints visible.

The case was designed in Dune 3D.

Frequently asked questions

Why doesn't this already exist?

Good question, given that at least three people independently asked the same thing. All of the key components are available off the shelf and USBKVM wasn't exactly hard to put together, so I guess it's mostly a question of time until someone else would have figured it out as well.

While writing this README, I stumbled upon openinterface.com which appears to be pretty much the same thing as USBKVM. Zooming into one of their PCB photos, they're also using the MS2109, what a coincidence! Looking at their subreddit they just went live less than two months ago.

Why is this using such an obscure chip from a Chinese vendor?

I thought about how USBKVM would look like if I had to make it from parts I could get from Digikey. The best solution I could come up with would be using a large MCU with camera input and High-Speed USB and then somehow connecting an HDMI to parallel RGB converter to its camera interface. While this could also have eliminated the MCU used for HID emulation, it's much more complex and expensive than the single QFN-48 IC that is the MS2109.

How can I get one?

I have no plans to commercially offer USBKVM since I don't want to deal with the hurdles of manufacturing and selling hardware products.

If you're in the business of manufacturing and selling hardware, feel free to take the design files and turn USBKVM into a product people can buy! Just keep in mind that the firmware and client app are under GPL when making modifications to it.

What platforms does the client run on?

Right now, the client app builds and runs on Linux. Windows support is experimental as it's significantly more involved to determine the correct video capture device.

Are there plans for clients for other platforms?

Right now, no. You're more than welcome to implement a client for your favourite platform based on the client included in this repo.

Future improvements

While there's not much to improve on the hardware as it is right now, the client app still has a lot of room for improvement.

  • Handle disconnecting the USBKVM more gracefully, potentially allow reconnects
  • Improved feedback for when no USBKVM is found
  • Make sure that it handles non-US keyboard layouts on both ends
  • Relative mouse emulation, requires firmware support
  • Firmware updates via I²C, requires firmware support
  • Screen recording, should be easy to add to the GStreamer pipeline
  • Instant replay to show screen content that just scrolled by
  • Translating text input into keyboard events for pasting text
  • Key macros for repeatedly pressing keys
  • OCR for copying text from the video feed

Licenses

  • Firmware and client: GPLv3
  • Hardware and Case: CERN Open Hardware Licence Version 2 - Permissive