Skip to content

Instantly share code, notes, and snippets.

@tdcosta100
Last active October 22, 2025 22:34
Show Gist options
  • Save tdcosta100/7def60bccc8ae32cf9cacb41064b1c0f to your computer and use it in GitHub Desktop.
Save tdcosta100/7def60bccc8ae32cf9cacb41064b1c0f to your computer and use it in GitHub Desktop.
Using full desktop shell in WSL2 using WSLg (Wayland)

Full desktop shell in WSL2 using WSLg (Wayland)

Note

If you want a more complete experience and already tried the Xvnc tutorial, maybe you should try the WSLg (Xwayland) instead.

Warning

This is a work in progress tutorial. Things done here may break existing functionality, so be careful!

In this tutorial, we will install and use a full GNOME Desktop environment in WSL2, without any external software. The only requirement is a working WSLg installation. At the moment, the instructions are only for Ubuntu (20.04, 22.04 and 24.04) distros and GNOME, but you can request me to test other distros and desktop environments.

Current limitations

  • No fullscreen support
  • No complete desktop support (with Display Manager)
  • No support to clipboard (copy & paste) from/to Windows

Ubuntu

GNOME

Installing the required packages

  1. Open a Ubuntu (20.04, 22.04 or 24.04) terminal. First of all, we need to make sure everything is up to date.

    sudo apt update
    sudo apt upgrade
    
  2. WSL doesn't install the app store by default, so we can install it manually (or you can skip this step if you don't need it).

    sudo snap install snap-store
    
  3. Now we can install the required packages. This may take a long time, so be patient.

    For Ubuntu 20.04 and Ubuntu 22.04
    sudo apt install ubuntu-desktop acpi-support-
    
    For Ubuntu 24.04
    sudo apt install ubuntu-desktop
    

Important

Note the minus sign (-) after the package acpi-support. This is necessary because it will be installed by default and installing it will render your distro almost unusable (see microsoft/WSL#10059), so we will tell apt to not install it.

Configuring the environment

  1. Now we have everything installed, we must configure some things before using the GNOME Shell. First, we need to deactivate the GDM service, because unfortunately it doesn't work with WSLg yet (at least I didn't figure it out how to make it work).

    sudo systemctl mask gdm.service
    
  2. Now we need to fix the directory /tmp/.X11-unix/, because it's mounted as read-only by default. We will create a new systemd unit.

    sudo systemctl edit --full --force wslg-fix.service
    
  3. Paste the code below in the editor.

    For Ubuntu 20.04 and Ubuntu 22.04
    [Service]
    Type=oneshot
    ExecStart=-/usr/bin/umount /tmp/.X11-unix
    ExecStart=/usr/bin/rm -rf /tmp/.X11-unix
    ExecStart=/usr/bin/mkdir /tmp/.X11-unix
    ExecStart=/usr/bin/chmod 1777 /tmp/.X11-unix
    ExecStart=/usr/bin/ln -s /mnt/wslg/.X11-unix/X0 /tmp/.X11-unix/X0
    
    [Install]
    WantedBy=multi-user.target
    
    For Ubuntu 24.04
    [Service]
    Type=oneshot
    ExecStart=-/usr/bin/umount /tmp/.X11-unix
    ExecStart=/usr/bin/rm -rf /tmp/.X11-unix
    ExecStart=/usr/bin/mkdir /tmp/.X11-unix
    ExecStart=/usr/bin/chmod 1777 /tmp/.X11-unix
    ExecStart=/usr/bin/ln -s /mnt/wslg/.X11-unix/X0 /tmp/.X11-unix/X0
    ExecStart=/usr/bin/chmod 0777 /mnt/wslg/runtime-dir
    ExecStart=/usr/bin/chmod 0666 /mnt/wslg/runtime-dir/wayland-0.lock
    
    [Install]
    WantedBy=multi-user.target
    
  4. Save the file and close the editor. Now we have to enable this service.

    sudo systemctl enable wslg-fix.service
    
  5. The last step is to configure the GNOME Shell to start in nested mode.

    For Ubuntu 20.04
    sudo mkdir /etc/systemd/user/gnome-shell-wayland.service.d/
    sudo nano /etc/systemd/user/gnome-shell-wayland.service.d/override.conf
    
    For Ubuntu 22.04 and Ubuntu 24.04
    sudo mkdir /etc/systemd/user/[email protected]/
    sudo nano /etc/systemd/user/[email protected]/override.conf
    
  6. Paste the code below in the editor.

    [Service]
    ExecStart=
    ExecStart=/usr/bin/gnome-shell --nested
    
  7. Save the file and close the editor. Then close the distro terminal window.

  8. Finally we will shutdown WSL.

    wsl.exe --shutdown
    

    The configuration is complete.

Starting the GNOME Shell

  1. Now open again your distro terminal, then paste the command below.

    DESKTOP_SESSION=ubuntu \
    GDMSESSION=ubuntu \
    GNOME_SHELL_SESSION_MODE=ubuntu \
    GTK_IM_MODULE=ibus \
    GTK_MODULES=gail:atk-bridge \
    IM_CONFIG_CHECK_ENV=1 \
    IM_CONFIG_PHASE=1 \
    QT_ACCESSIBILITY=1 \
    QT_IM_MODULE=ibus \
    XDG_CURRENT_DESKTOP=ubuntu:GNOME \
    XDG_DATA_DIRS=/usr/share/ubuntu:$XDG_DATA_DIRS \
    XDG_SESSION_TYPE=wayland \
    XMODIFIERS=@im=ibus \
    MUTTER_DEBUG_DUMMY_MODE_SPECS=1366x768 \
    gnome-session
    
  2. A GNOME Shell window will appear.

    Ubuntu 20.04

    GNOME Shell on Ubuntu 20.04

    Ubuntu 22.04

    GNOME Shell on Ubuntu 22.04

    Ubuntu 24.04

    GNOME Shell on Ubuntu 24.04

@tamboril
Copy link

Did you enable the wslg-fix.service in the step 4 of Configuring the environment?

I had not. But I went back and did that. Now the symbolic link is there, but I still get the same results, even after shutting down and restarting wsl. I've probably done bad things to my Ubuntu instance over time. I don't mind nuking it and starting afresh. I'll report after that.

@tdcosta100
Copy link
Author

Nice. Unfortunately it's too easy to mess things up, maybe redoing everything will make it work well.

@tdcosta100
Copy link
Author

So this has been working great so far, and again, I thank you for the effort you put into this tutorial.

Since budgie seems to be making problems, what about pop os? I saw a video of someone rocking it in wsl: https://m.youtube.com/watch?v=SYooHm8bzJU

Hi, @phil9309. Followed the instructions and managed to install Pop_OS over Ubuntu 22.04. After installing it, you need to follow the steps of this tutorial. Then the launch command is:

DESKTOP_SESSION=pop \
GDMSESSION=pop \
GNOME_SHELL_SESSION_MODE=pop \
GTK_IM_MODULE=ibus \
GTK_MODULES=gail:atk-bridge \
NOTIFY_SOCKET=/run/user/`id -u`/systemd/notify \
QT_ACCESSIBILITY=1 \
QT_IM_MODULE=ibus \
XDG_CURRENT_DESKTOP=pop:GNOME \
XDG_DATA_DIRS=/usr/share/pop:$XDG_DATA_DIRS \
XDG_SESSION_TYPE=wayland \
XMODIFIERS=@im=ibus \
MUTTER_DEBUG_DUMMY_MODE_SPECS=1366x768 \
gnome-session

You can see it's very similar to the GNOME one. I hope you will be able to reproduce it.

@jordankoehn
Copy link

This is really impressive, all my previous attempts at launching gnome in wsl2 failed. Will try this later. Also, see my wsl2 sway repo for how I managed to sync clipboard between windows and Wayland. Clipboard-sync may also work, but it didn't for me

@vinberg88
Copy link

Have half screen when i try to start gnome 47 and Manjaro hehe What can be wrong...

half

Regards,
Mattias Vinberg

@tdcosta100
Copy link
Author

Since this tutorial doesn't cover the use of VcxSrv, I cannot help you with your problem.

@wartalski
Copy link

Great tutorial! Do you plan to upgrade it for Ubuntu 24.10?

@tdcosta100
Copy link
Author

@wartalski, when an Ubuntu 24.10 image for WSL gets ready, yes!

@guillaumeportails
Copy link

guillaumeportails commented Dec 30, 2024

So nice ! Thanks, works like a charm.
Ubuntu 24 inside W$10

@markwilber
Copy link

markwilber commented Jan 21, 2025

Is it known whether or not WSLg works for Microsoft Cloud PCs?

I have Ubuntu 24.04 installed,

echo $DISPLAY
# :0

When I try xeyes I get no errors, but nothing shows up. (Same for others, such as xclock.) Most people are saying that WSLg works out-of-the-box, providing DISPLAY is ':0'.

My Cloud PC specs are:

Edition: Windows 11 Enterprise
Version: 23H2
Installed on: 7/23/2024
OS build: 22631.4602
Experience: Windows Feature Experience Pack 1000.22700.1055.0

@tdcosta100
Copy link
Author

Hi, @markwilber. Did you try xdpyinfo? Or any other X11 app?

@markwilber
Copy link

markwilber commented Jan 22, 2025

Hi @tdcosta100. I did try Xming, with identical (non-)results.

As for xdpyinfo, would you be able to find anything useful in the output?

$ xdpyinfo
name of display:    :0
version number:    11.0
vendor string:    Microsoft Corporation
vendor release number:    12010000
maximum request size:  16777212 bytes
motion buffer size:  256
bitmap unit, bit order, padding:    32, LSBFirst, 32
image byte order:    LSBFirst
number of supported pixmap formats:    7
supported pixmap formats:
    depth 1, bits_per_pixel 1, scanline_pad 32
    depth 4, bits_per_pixel 8, scanline_pad 32
    depth 8, bits_per_pixel 8, scanline_pad 32
    depth 15, bits_per_pixel 16, scanline_pad 32
    depth 16, bits_per_pixel 16, scanline_pad 32
    depth 24, bits_per_pixel 32, scanline_pad 32
    depth 32, bits_per_pixel 32, scanline_pad 32
keycode range:    minimum 8, maximum 255
focus:  PointerRoot
number of extensions:    22
    BIG-REQUESTS
    Composite
    DAMAGE
    DOUBLE-BUFFER
    GLX
    Generic Event Extension
    MIT-SHM
    Present
    RANDR
    RECORD
    RENDER
    SHAPE
    SYNC
    X-Resource
    XC-MISC
    XFIXES
    XFree86-VidModeExtension
    XINERAMA
    XInputExtension
    XKEYBOARD
    XTEST
    XVideo
default screen number:    0
number of screens:    1

screen #0:
  dimensions:    640x480 pixels (169x127 millimeters)
  resolution:    96x96 dots per inch
  depths (7):    24, 1, 4, 8, 15, 16, 32
  root window id:    0x390
  depth of root window:    24 planes
  number of colormaps:    minimum 1, maximum 1
  default colormap:    0x22
  default number of colormap cells:    256
  preallocated pixels:    black 0, white 16777215
  options:    backing-store WHEN MAPPED, save-unders NO
  largest cursor:    640x480
  current input event mask:    0x580000
    SubstructureNotifyMask   SubstructureRedirectMask PropertyChangeMask
  number of visuals:    270
  default visual id:  0x23
  visual:
    visual id:    0x23
    class:    TrueColor
    depth:    24 planes
    available colormap entries:    256 per subfield
    red, green, blue masks:    0xff0000, 0xff00, 0xff
    significant bits in color specification:    8 bits
  visual:
    visual id:    0x24
    class:    DirectColor
    depth:    24 planes
    available colormap entries:    256 per subfield
    red, green, blue masks:    0xff0000, 0xff00, 0xff
    significant bits in color specification:    8 bits
            .
            .
            .
    visual:
    visual id:    0x38d
    class:    TrueColor
    depth:    32 planes
    available colormap entries:    256 per subfield
    red, green, blue masks:    0xff0000, 0xff00, 0xff
    significant bits in color specification:    8 bits
  visual:
    visual id:    0x38e
    class:    TrueColor
    depth:    32 planes
    available colormap entries:    256 per subfield
    red, green, blue masks:    0xff0000, 0xff00, 0xff
    significant bits in color specification:    8 bits

@Timbo303
Copy link

Thanks to this and some other github gist I got waydroid running. But there is no sound. I don't know if theres a way to get sound working on waydroid even though gnome shell has sound.

https://www.reddit.com/r/timbo303/comments/1ivzrn8/how_to_run_waydroid_on_windows_11/

@Dustpaw
Copy link

Dustpaw commented Mar 7, 2025

can anyone help me?I couldn't get into the gnome,it shows me something went wrong:(
I had tried many times qwq
Snipaste_2025-03-07_11-20-29

@DhaosEsedess
Copy link

i would like to know if this will work with kali and any of its desktop flavors would prefer kde, xfce, or mate

@Timbo303
Copy link

I figured out that theres sound on flatpaks like firefox but not the snap version of apps like firefox. Waydroid also has no sound too. Thats a problem because waydroid would easily replace bluestacks or mumu on windows with wsl2. I cant figure out how to get sound working for these apps do you have a way to fix that?

@poibear
Copy link

poibear commented Apr 6, 2025

I just wanted to comment that I found a way to hide the mouse cursor on Ubuntu, so there isn't a ghosting effect. It should be showing the Windows cursor in an Ubuntu theme since the desktop GUI is from Linux.

https://askubuntu.com/questions/681269/how-do-i-hide-a-mouse-pointer-in-gnome-3

@Timbo303
Copy link

Can you revisit this to try to get waydroid sound working since wsl 2 is now open source?

@tdcosta100
Copy link
Author

tdcosta100 commented May 22, 2025

Hi guys. It's been a long time since my last tutorial update. I'm sorry for not updating it with the last changes and your requests. There days are being very busy to me. But I will bring some updates as soon as possible.

Regarding the recent Microsoft action in finally opening the WSL2 code, maybe we will have some improvements, but I don't know, I'm not a experienced C/C++/Linux programmer. But I wish they will improve the WSLg backend. My biggest dream is to have a backend change, throwing off the limited RDP seamless connection and bringing a graphics façade for WSL that render windows in 3D surfaces in Windows side. This would bring an enormous performance improvement, I think. It's a dream for now, but who knows the future.

@onomatopellan
Copy link

@Timbo303 Did you try the Weston on Weston approach? Since WSLg uses Weston as a Wayland compositor you can run another Weston window where you can launch Waydroid. You won't need the full Ubuntu desktop and sounds works inside the Weston desktop shell.

I made a quick guide here https://gist.github.com/onomatopellan/c5220c0efddaff69aaff77cca80b7b8e

@Timbo303
Copy link

Timbo303 commented Jun 1, 2025

@Timbo303 Did you try the Weston on Weston approach? Since WSLg uses Weston as a Wayland compositor you can run another Weston window where you can launch Waydroid. You won't need the full Ubuntu desktop and sounds works inside the Weston desktop shell.

I made a quick guide here https://gist.github.com/onomatopellan/c5220c0efddaff69aaff77cca80b7b8e

I tried to follow your instructions. However weston doesn't start up for some reason giving me this error:

Date: 2025-06-01 CDT
[18:39:48.518] weston 14.0.1
https://wayland.freedesktop.org
Bug reports to: https://gitlab.freedesktop.org/wayland/weston/issues/
Build: 14.0.1
[18:39:48.518] Command line: weston
[18:39:48.518] OS: Linux, 6.6.87.1-microsoft-standard-WSL2+, #1 SMP PREEMPT_DYNAMIC Sun Jun 1 17:19:51 CDT 2025, x86_64
[18:39:48.518] Flight recorder: enabled
[18:39:48.518] Starting with no config file.
[18:39:48.519] Output repaint window is 7 ms maximum.
[18:39:48.519] Loading module '/usr/lib/x86_64-linux-gnu/libweston-14/wayland-backend.so'
[18:39:48.523] Error: Failed to connect to parent Wayland compositor: No such file or directory
display option: (none), WAYLAND_DISPLAY=wayland-0
[18:39:48.523] fatal: failed to create compositor backend

@onomatopellan
Copy link

@Timbo303 Since this is somewhat off-topic let's continue in my gist.

@lingmingxi
Copy link

You're an absolute legend! I spent the whole morning digging through outdated guides that had me downloading a bunch of random packages and using old setup methods. Then I found your solution—fresh, clean, and exactly what I needed. Seriously, you're a genius! If my comment takes up space, feel free to remove it. Just had to say, you're amazing!

@VArterJr
Copy link

VArterJr commented Oct 7, 2025

Any chance of getting a delta description for kubuntu? I tried your other xwayland instructions, but nothing ever launches. This one works great, but I'm not a fan of Gnome.

@PuNkYsHuNgRy
Copy link

Can you add instructions for Cinnamon? I've tried all three and this is the only one I can get working.

@kimmydotzip
Copy link

For Gnome 49 (standard now with Ubuntu 25.10 / Questing Quokka):

  1. use --devkit instead of --nested in /etc/systemd/user/[email protected]/override.conf
  2. install the devkit binary: sudo apt install mutter-dev-bin

@PuNkYsHuNgRy
Copy link

For Gnome 49 (standard now with Ubuntu 25.10 / Questing Quokka):

  1. use --devkit instead of --nested in /etc/systemd/user/[email protected]/override.conf
  2. install the devkit binary: sudo apt install mutter-dev-bin

I'll try this out, thanks so much!

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment