A minimalist RISC-V system emulator capable of running Linux the kernel and corresponding userland.
semu implements the following:
- RISC-V instruction set architecture: RV32IMA
- Privilege levels: S and U modes
- Control and status registers (CSR)
- Virtual memory system: RV32 MMU
- UART: 8250/16550
- PLIC (platform-level interrupt controller): 32 interrupts, no priority
- Standard SBI, with the timer extension
- Four types of I/O support using VirtIO standard:
- virtio-blk acquires disk image from the host.
- virtio-net is mapped as TAP interface.
- virtio-snd uses PortAudio for sound playback on the host with one limitations:
- As some unknown issues in guest Linux OS (confirmed in v6.7 and v6.12), you need
to adjust the buffer size to more than four times of period size, or
the program cannot write the PCM frames into guest OS ALSA stack.
- For instance, the following buffer/period size settings on
aplayhas been tested with broken and stutter effects yet complete with no any errors:aplay --buffer-size=32768 --period-size=4096 /usr/share/sounds/alsa/Front_Center.wav.
- For instance, the following buffer/period size settings on
- As some unknown issues in guest Linux OS (confirmed in v6.7 and v6.12), you need
to adjust the buffer size to more than four times of period size, or
the program cannot write the PCM frames into guest OS ALSA stack.
- virtio-input exposes SDL-backed keyboard and mouse devices to the guest.
- You can exit the SDL window by pressing Ctrl+A+G
Device Tree compiler (dtc) is required. To install it on Debian/Ubuntu Linux, enter the following command:
$ sudo apt install device-tree-compilerFor macOS, use the following command:
$ brew install dtcFor demonstration purposes, ext4 is used for file system mounting.
ext4 is a native Linux filesystem, offering stability, high capacity, reliability,
and performance while requiring minimal maintenance. The mkfs.ext4 command can
create an ext4 file system from disk partitions. This command is a symbolic link of
the mke2fs command, and its
usage is the same as the mke2fs command.
For most GNU/Linux distributions, mkfs.ext4 command should be installed in advance.
For macOS, use the following command:
$ brew install e2fsprogsBuild the emulator:
$ makeDownload prebuilt Linux kernel image:
$ make checkPlease be patient while semu is running.
Reference output:
Starting syslogd: OK
Starting klogd: OK
Running sysctl: OK
Starting network: OK
Welcome to Buildroot
buildroot login:
Enter root to access shell.
You can exit the emulator using: <Ctrl-a x>. (press Ctrl+A, leave it, afterwards press X)
./semu -k linux-image [-b dtb-file] [-d disk-image] [-i initrd-image] [-s shared-directory] [-H]linux-imageis the path to the Linux kernelImage.dtb-fileis optional, as it specifies the user-specified device tree blob.disk-imageis the ext4 image exposed as/dev/vdato the guest. The default boot path mounts this as the root filesystem;makebuilds it fromrootfs.cpioviascripts/rootfs_ext4.sh.shared-directoryis optional, as it specifies the path of a directory on the host that will be shared with the guest operating system through virtio-fs, enabling file access from the guest via a virtual filesystem mount.-H(or--headless) skips SDL window creation; useful for CI andmake check.initrd-imageis optional and only used on the legacy boot path. The defaultminimal.dtbbuilt withENABLE_EXTERNAL_ROOT=1does not advertise initrd placement, so-ithere requires eitherENABLE_EXTERNAL_ROOT=0or a custom DTB passed with-b. See Boot mode below.
The default build (make) boots the kernel directly from /dev/vda and
runs /sbin/init from the ext4 root, skipping the initramfs unpack step
entirely. This is faster, avoids the RCU-stall the kernel hits when
unpacking a large cpio, and matches how real systems deploy. The
ext4.img is built from rootfs.cpio via scripts/rootfs_ext4.sh,
which requires fakeroot and mkfs.ext4.
If fakeroot is missing, the build falls back to the legacy initramfs
path (-i rootfs.cpio) automatically and prints a one-line warning. To
force the legacy path explicitly:
$ make ENABLE_EXTERNAL_ROOT=0
$ make ENABLE_EXTERNAL_ROOT=0 checkThe legacy path uses what the flag is still spelled as: -i initrd-image.
That is a runtime choice only when the DTB also carries
linux,initrd-{start,end}. semu's default external-root build emits a DTB
that always boots from /dev/vda, so -i is rejected there unless you
replace the DTB with one that describes the initrd layout.
The classical initrd (a filesystem image mounted as /dev/ram0,
pivoted into via pivot_root) is effectively obsolete -- it required
CONFIG_BLK_DEV_INITRD plus the legacy ramdisk block driver, an
in-kernel filesystem driver to mount the image before any userspace code
ran, and a /linuxrc handoff. Mainstream distros and embedded builds
dropped that path more than a decade ago. Linux 2.6+ kept the flag and
the linux,initrd-{start,end} device-tree properties, but the kernel
inspects the loaded blob: a cpio archive is unpacked into the in-memory
rootfs (initramfs path, runs /init); a filesystem image still falls
back to the legacy initrd path if that driver is configured in. semu
ships and consumes a cpio (rootfs.cpio), so the legacy build is
exercising the initramfs path even though the CLI flag spelling stayed
-i initrd-image.
For detailed networking guidance, see docs/networking.md.
To mount the directory in semu:
$ mount -t virtiofs myfs [shared-directory]shared-directoryis the path of a directory you want to mount in semu.
To unmount the directory in semu:
$ umount [shared-directory]shared-directoryis the path of a directory you want to unmount in semu.
An automated build script is provided to compile the RISC-V cross-compiler, Busybox, and Linux kernel from source. Please note that it only supports the Linux host environment.
To build everything, simply run:
$ make build-imageThis command invokes the underlying script: scripts/build-image.sh, which also offers more flexible usage options.
./scripts/build-image.sh [--buildroot] [--linux] [--all] [--no-ext4] [--clean-build] [--help]
Options:
--buildroot Build Buildroot userland (produces rootfs.cpio and,
unless --no-ext4 is given, ext4.img for vda boot)
--linux Build the Linux kernel
--all Build both Buildroot and Linux
--no-ext4 Skip ext4.img generation; produce only rootfs.cpio
(matches the legacy ENABLE_EXTERNAL_ROOT=0 path)
--clean-build Remove buildroot/ and/or linux/ before building
--help Show this message
Build the Linux kernel only:
$ scripts/build-image.sh --linux
Build Buildroot (produces both rootfs.cpio and ext4.img):
$ scripts/build-image.sh --buildroot
Build Buildroot for the legacy initramfs-only path (no ext4):
$ scripts/build-image.sh --buildroot --no-ext4
Force a clean build:
$ scripts/build-image.sh --all --clean-build
$ scripts/build-image.sh --linux --clean-build
$ scripts/build-image.sh --buildroot --clean-build
semu is released under the MIT License.
Use of this source code is governed by a MIT-style license that can be found in the LICENSE file.