ladybird/Documentation/NotesOnWSL.md
Tristan Menke 9f117a79f6 Documentation: Add workaround for blackscreen using WSL+QEMU6.x
This workaround disables the in-kernel interrupt controller.
This impacts the VM performance and should probably be removed
when the workaround is no longer needed.

This workaround was posed by stelar7.
See #7523
2021-07-03 17:32:33 +02:00

7.4 KiB

Notes on WSL

SerenityOS can also be built and run under WSL Version 2. WSL Version 1 is not supported since Version 1 does not support ext2, which is needed for the setup.

WSL Version 2 requires Windows version 2004 or higher, with OS Build 19041 or greater. Here is a guide on how to get WSL2. Once installed, you will need to make sure the distribution you want to use (and the new default) is using Version 2:

  • wsl -l -v will list distros and versions,
  • wsl --set-version <distro> <version> is used to convert a distro to another version, and
  • wsl --set-default-version 2 will set the default version for all new distros (if desired.)

The installation then proceeds as usual.

WSL2 does not natively support graphical applications. You can either install QEMU natively on windows and allow WSL to talk to it, or you can install an X Server for windows.

Setting up an X server with WSL:

  • Install Vcxsrv on Windows.
  • When you start up Vcxsrv, make sure to set the Display number to 0, and to Disable access control.
  • Before actually doing ninja run, you need to set the DISPLAY environmental variable as such:
export DISPLAY=$(cat /etc/resolv.conf | grep nameserver | awk '{print $2}'):0

This is due to a bug in WSL2. For more information, microsoft/WSL#4106.

  • Connect to the window server from WSL.

Now you can finally, ninja run.

Using native QEMU install with WSL:

  • Grab latest QEMU from here and install for Windows.

  • Locate the executable qemu-system-i386.exe in WSL. By default this will be located at /mnt/c/Program Files/qemu/qemu-system-i386.exe.

  • Set the SERENITY_QEMU_BIN environment variable to the location above. For example:
    export SERENITY_QEMU_BIN='/mnt/c/Program Files/qemu/qemu-system-i386.exe'

  • Locate the Windows path to the SerenityOS disk image, as native QEMU will be accessing it via the Windows filesystem. If your build tree is located in the WSL2 partition, this will be accessible under the \\wsl$ network file share (see notes below).

  • Set the SERENITY_KERNEL_CMDLINE environment variable to disable VirtIO support (Because it is currently broken on native windows QEMU): export SERENITY_KERNEL_CMDLINE="disable_virtio"

  • Set the SERENITY_DISK_IMAGE environment variable to the full path of the SerenityOS disk image file from above. For example:
    export SERENITY_DISK_IMAGE='\\wsl$\Ubuntu-20.04\home\username\serenity\Build\i686\_disk_image'

  • ninja run as usual.

Hardware acceleration

The steps above will run QEMU in software virtualisation mode, which is very slow. QEMU supports hardware acceleration on Windows via the Windows Hypervisor Platform (WHPX), a user-mode virtualisation API that can be used alongside Hyper-V. This is important to note as WSL2 itself runs on top of Hyper-V, which conflicts with other acceleration technologies such as Intel HAXM.

To run SerenityOS in a WHPX-enabled QEMU VM:

  • If you have not already done so, enable Windows Hypervisor Platform, either using "Turn Windows features on or off", or by running the following command in an elevated PowerShell session:
    dism /Online /Enable-Feature /All /FeatureName:HypervisorPlatform

  • Specify QEMU acceleration option:
    export SERENITY_EXTRA_QEMU_ARGS="-accel whpx"

  • Disable Virtual Machine eXtensions on the vCPU, otherwise some versions of QEMU will crash out with a "WHPX: Unexpected VP exit code 4" error:
    export SERENITY_QEMU_CPU="max,vmx=off"

  • ninja run as usual.

Known issues with WHPX

Freeze on boot in Scheduler

If Serenity freezes on boot with the log message: Scheduler[0]: idle loop running then you are likely missing some emulated CPU features. Please ensure you have installed the most recent version of QEMU for Windows and that you have followed the step above to enable the maximum feature set available: export SERENITY_QEMU_CPU="max,vmx=off".

If the steps above do not fix the problem, check the boot log for whpx: injection failed, MSI (0, 0) delivery: 0, dest_mode: 0, trigger mode: 0, vector: 0, lost (c0350005). If present, try disabling the KVM in-kernel interrupt handler by extending the above to export SERENITY_EXTRA_QEMU_ARGS="-accel whpx,kernel-irqchip=off". This seems to be an issue for some users with QEMU 6.x.

Illegal instruction on boot

Using SERENITY_QEMU_CPU="max" can trigger a QEMU bug where the OSXSAVE CPUID flag is erroneously set, playing havoc with feature detection logic in libgcc and resulting in this error.

To workaround this, first adjust the SERENITY_QEMU_CPU setting to emulate a more restricted feature set. SERENITY_QEMU_CPU="qemu32" appears to work in some cases, however in others causes the boot freeze issue above. It's worth playing around with various different values here to see if you can find one that works for you. Running qemu-system-i386.exe -cpu ? will list the supported CPU configurations.

If you cannot find a working CPU feature set, the next workaround is to patch libgcc in the Serenity toolchain build to remove the offending instruction. Comment out the if ((ecx & bit_OSXSAVE)) block in Toolchain/Tarballs/gcc-<version>/libgcc/config/i386/cpuinfo.c. In GCC 10.2.0 this is lines 282-297.

Rebuild the toolchain using Toolchain/BuildIt.sh as normal, then rebuild Serenity.

Slow boot on HiDPI systems

On some Windows systems running with >100% scaling, the booting phase of Serenity might slow to a crawl. Changing the zoom settings of the QEMU window will speed up the emulation, but you'll have to squint harder to read the smaller display.

A quick workaround is opening the properties of the QEMU executable at C:\Program Files\qemu\qemu-system-i386.exe, and in the Compatibility tab changing the DPI settings to force the scaling to be performed by the System, by changing the setting at at the bottom of the window. The QEMU window will now render at normal size while retaining acceptable emulation speeds.

This is being tracked as issue #7657.

Note on filesystems

WSL2 filesystem performance for IO heavy tasks (such as compiling a large C++ project) on the host Windows filesystem is terrible. This is because WSL2 runs as a Hyper-V virtual machine and uses the 9P file system protocol to access host windows files, over Hyper-V sockets.

For a more in depth explanation of the technical limitations of their approach, see this issue on the WSL github

The recommendation from the Microsoft team on that issue is:

If it's at all possible, store your projects in the Linux file system in WSL2.

In practice, this means cloning and building the project to somewhere such as /home/username/serenity.

If you're using the native Windows QEMU binary from the above steps, QEMU is not able to access the ext4 root partition of the WSL2 installation without going via the 9P network file share. The root of your WSL2 distro will begin at the network path \\wsl$\{distro-name}.

Alternatively, you may prefer to copy Build/_disk_image and Build/Kernel/Kernel to a native Windows partition (e.g. /mnt/c) before running ninja run, in which case SERENITY_DISK_IMAGE will be a regular Windows path (e.g. 'D:\serenity\_disk_image')