OpenXR streaming application
WiVRn lets you run OpenXR applications on a computer and display them on a standalone headset.
WiVRn requires ffmpeg (for AMD/Intel), cuda (Nvidia), x264 (software) for video encoding. Other dependencies include avahi-client, eigen3, libpulse, nlohmann_json.
Note that some distributions don't ship h264/h265 support in their mesa builds. On Fedora this can be solved by replacing mesa:
dnf swap mesa-va-drivers mesa-va-drivers-freeworldFrom your checkout directory
cmake -B build-server . -GNinja -DWIVRN_BUILD_CLIENT=OFF -DCMAKE_BUILD_TYPE=RelWithDebInfo -DWIVRN_USE_VAAPI=ON -DWIVRN_USE_X264=ON -DWIVRN_USE_NVENC=ON
cmake --build build-server
# Set WiVRn as the active OpenXR runtime, delete ~/.config/openxr/1/active_runtime.json after you are done using WiVRn
mkdir -p ~/.config/openxr/1/
ln --relative --symbolic --force build-server/openxr_wivrn-dev.json ~/.config/openxr/1/active_runtime.jsonDownload apk from Releases. Install with adb (headset connected to PC), developer mode must be enabled.
adb install WiVRn.apkAs Arch package names: git git-lfs pkgconf glslang cmake jre17-openjdk librsvg
Download sdkmanager commandline tool and extract it to any directory.
Create your ANDROID_HOME directory, for instance ~/Android.
Review and accept the licenses with
sdkmanager --sdk_root="${HOME}/Android" --licensesFor Pico only: setup git lfs
From the main directory.
export ANDROID_HOME=~/Android
export JAVA_HOME=/usr/lib/jvm/openjdk-bin-17/
./gradlew assembleStandardRelease
# Or for Pico assemblePicoOutputs will be in build/outputs/apk/standard/release/WiVRn-standard-release-unsigned.apk
On the computer, run wivrn-server, from checkout directory
build-server/server/wivrn-serverThen, on headset, launch WiVRn from the App Library, in "unknown sources" section.
You will briefly have a "Waiting for connection" screen, followed by "Waiting for video stream". Now on your computer you can run an OpenXR application, and it will show on your headset, enjoy!
Configuration is done on server side, in $XDG_CONFIG_HOME/wivrn/config.json or if $XDG_CONFIG_HOME is not set, $HOME/.config/wivrn/config.json.
All elements are optional and have default values.
Default value: 1
Controls the size of the video stream, either a number between 0 and 1 or a pair of numbers between 0 and 1. If two numbers are provided the first one is horizontal scale and the second vertical. Scaling is applied in a foveated fashion: the center has a 1:1 ratio and the rest is scaled so that the total number of pixels matches the desired scale.
{
	"scale": 0.5
}The x and y resolution of the streamed video are half the size on the headsed, reduces the required bandwidth and encoding/decoding time by about 4.
{
	"scale": [0.75, 0.5]
}Scales x by a 0.75 factor, and y by a 0.5 factor.
A list of encoders to use.
WiVRn has the ability to split the video in blocks that are processed independently, this may use resources more effectively and reduce latency. All the provided encoders are put into groups, groups are executed concurrently and items within a group are processed sequentially.
Default value: nvenc if Nvidia GPU and compiled with cuda, vaapi for all other GPU when compiled with ffmpeg, else x264.
Identifier of the encoder, one of x264 (software encoding), nvenc (Nvidia hardware encoding), vaapi (AMD/Intel hardware encoding)
Default value: h265
One of h264 or h265. If using x264 encoder, value is ignored and h264 is used.
Default value: 50000000 (50Mb/s)
Bitrate of the video, in bit/s.
Default values: full image (width = 1, height = 1, offset_x = 0, offset_y = 0)
Specifies the portion of the video to encode: all values are in 0, 1 range. Left eye image ranges from x 0 to x 0.5 and y 0 to 1, Rigth eye is x from 0.5 to 1 and y 0 to 1.
Default value: First unused group identifier, if omitted on all encoders, they are executed concurrently.
Identifier (number) of the encoder group. Encoders with the same identifier are executed sequentially, in the order they are defined in the configuration. Encoders with different identifiers are executed concurrently.
- Simple encoder
{
	"encoders": [
		{
			"encoder": "vaapi",
			"bitrate": 50000000,
			"codec": "h265"
		}
	]Creates a single encoder, using vaapi hardware encoding, h265 video codec (HEVC) and 50Mb/s bitrate.
- Hardware + software encoder
{
	"encoders": [
		{
			"encoder": "vaapi",
			"bitrate": 25000000,
			"codec": "h265",
			"width": 0.5,
			"height": 1,
			"offset_x": 0,
			"offset_y": 0
		},
		{
			"encoder": "x264",
			"bitrate": 25000000,
			"codec": "h264",
			"width": 0.5,
			"height": 1,
			"offset_x": 0.5,
			"offset_y": 0
		}
	]
}Creates a hardware encoder for left eye, and a software encoder for right eye.
- 2 Hardware encoders
{
	"encoders": [
		{
			"encoder": "vaapi",
			"bitrate": 25000000,
			"codec": "h265",
			"width": 0.5,
			"height": 1,
			"offset_x": 0,
			"offset_y": 0,
			"group": 0,
		},
		{
			"encoder": "vaapi",
			"bitrate": 25000000,
			"codec": "h264",
			"width": 0.5,
			"height": 1,
			"offset_x": 0.5,
			"offset_y": 0,
			"group": 0,
		}
	]
}Creates two hardware encoders, one for left eye and one for right eye, executed sequentially as they have the same group.
This allows the left eye image to be encoded faster than the full image would be, so network transfer starts earlier, and decoding starts earlier. While the total encoding, transfer and decoding time remain the same or are longer, this can reduce the latency.
Default value: unset
Json object of additional options to pass directly to ffmpeg avcodec_open2's option parameter.
Default value: unset
An application to start when connection with the headset is established, can be a string or an array of strings if parameters need to be provided.
{
	"application": ["steam", "steam://launch/275850/VR"]
}Launch No Man's Sky in VR mode on Steam when connection with headset is established.
- More testing with OpenComposite for OpenVR support
- Latency improvement
- Application launcher
- Support more headsets
WiVRn uses the following software: