Building a Linux Kernel Driver using Rust
In this example we're going to build a very basic Linux Kernel driver in Rust, compile a Linux kernel, and load the driver into it.
You will need QEMU installed and in your $PATH. If you have an AArch64 machine
(Mac with Apple Silicon, Raspberry Pi 3 or newer, etc) you should use
qemu-system-aarch64. If you have an x86-64 machine, you should use
qemu-system-x86_64. You can use the 'wrong' one - it'll work just fine, but it
will go much more slowly, and we have a lot of Linux kernel to build.
Task 1 - Fetch a Debian disk image
This example uses Debian Stable 'nocloud' disk image. Other disk images might work but this is the one we tested.
Make yourself a work area on your machine, and download either https://cdimage.debian.org/images/cloud/bookworm/20250316-2053/debian-12-nocloud-arm64-20250316-2053.qcow2 (AArch64 aka Arm64) or https://cdimage.debian.org/images/cloud/bookworm/20250316-2053/debian-12-nocloud-amd64-20250316-2053.qcow2 (x86-64 aka AMD64)
For example:
mkdir linux-rust-demo
cd linux-rust-demo
wget https://cdimage.debian.org/images/cloud/bookworm/20250316-2053/debian-12-nocloud-arm64-20250316-2053.qcow2
(Windows users can use their favourite tool to make the folder, and a web browser to download the disk image)
Task 1a - Fetch the BIOS (AArch64 only)
If you are going to use AArch64, you'll need a UEFI boot-loader. On macOS,
homebrew installs a copy of EDK2, which is
fine for our use-case. On my machine it was installed into
/opt/homebrew/Cellar/qemu/9.2.2/share/qemu/edk2-aarch64-code.fd. You'll need
to have a look in your QEMU installation directory to find where your copy is.
Once you have found it, copy it to ./QEMU_EFI.fd, which is what the following
qemu-system-aarch64 command lines expect.
When emulating x86-64, QEMU uses a copy of SeaBIOS automatically.
Task 2 - Resize the disk image
If your downloads take a while, you may want to make a backup copy of the disk image, because as soon as you boot up the VM, the disk will be changed.
Then we're going to resize the disk image because it's too small (this applies to both the AArch64 one and the x86-64 one). We'll deal with making the partition larger a bit later once the VM has booted.
qemu-img resize debian-12-nocloud-arm64-20250316-2053.qcow2 +32G
Task 3 - Boot it up
We're now going to boot the VM.
For AArch64 on Apple Silicon macOS:
qemu-system-aarch64 -m 8G -M virt -cpu host -accel hvf -smp 8 -bios QEMU_EFI.fd -drive if=none,file=debian-12-nocloud-arm64-20250316-2053.qcow2,id=hd0 -device virtio-blk-device,drive=hd0 -device e1000,netdev=net0 -netdev user,id=net0,hostfwd=tcp:127.0.0.1:5555-:22 -nographic -serial telnet:localhost:4321,server,wait
For AArch64 on Arm Linux:
qemu-system-aarch64 -m 8G -M virt -cpu host -accel kvm -smp 8 -bios QEMU_EFI.fd -drive if=none,file=debian-12-nocloud-arm64-20250316-2053.qcow2,id=hd0 -device virtio-blk-device,drive=hd0 -device e1000,netdev=net0 -netdev user,id=net0,hostfwd=tcp:127.0.0.1:5555-:22 -nographic -serial telnet:localhost:4321,server,wait
AArch64 otherwise:
qemu-system-aarch64 -m 8G -M virt -cpu cortex-a53 -smp 8 -bios QEMU_EFI.fd -drive if=none,file=debian-12-nocloud-arm64-20250316-2053.qcow2,id=hd0 -device virtio-blk-device,drive=hd0 -device e1000,netdev=net0 -netdev user,id=net0,hostfwd=tcp:127.0.0.1:5555-:22 -nographic -serial telnet:localhost:4321,server,wait
For x86-64 on x86-64 Windows:
qemu-system-x86_64 -m 8G -M q35 -accel whpx -smp 8 -hda debian-12-nocloud-amd64-20250316-2053.qcow2 -device e1000,netdev=net0 -netdev user,id=net0,hostfwd=tcp:127.0.0.1:5555-:22 -nographic -serial telnet:localhost:4321,server,wait
For x86-64 on x86-64 Linux:
qemu-system-x86_64 -m 8G -M q35 -accel kvm -smp 8 -hda debian-12-nocloud-amd64-20250316-2053.qcow2 -device e1000,netdev=net0 -netdev user,id=net0,hostfwd=tcp:127.0.0.1:5555-:22 -nographic -serial telnet:localhost:4321,server,wait
x86-64 otherwise:
qemu-system-x86_64 -m 8G -M q35 -smp 8 -hda debian-12-nocloud-amd64-20250316-2053.qcow2 -device e1000,netdev=net0 -netdev user,id=net0,hostfwd=tcp:127.0.0.1:5555-:22 -nographic -serial telnet:localhost:4321,server,wait
In all cases I gave the machine 8 GiB of RAM and 8 CPU cores. You may want to tweak that to suit your needs.
Connect to your virtual machine over Telnet (you can use PuTTY, or your
favourite telnet client) on localhost:4321 You should end up at a login
prompt. The user is root and there is no password.
Task 4 - Resize the partition
The Debian image we downloaded has quite a small root partition, and we're going to need a lot more space to build the Linux kernel. We already made the virtual disk (the QCOW file) larger, so now let's go in and resize it the partition to use the extra space.
Inside your VM, run:
apt update
apt install fdisk
cfdisk /dev/*da
Once in cfdisk, use the arrow keys to select [ Sort ], then select the root
filesystem (the bottom one) and pick [ Resize ], then [ Write ], and
[ Quit ]. You'll need to type yes in response to [ Write ].
Now run reboot and connect to the VM again. Running df -h should show that
/ is about 35 GiB in size - Debian's start-up scripts automatically resized
the root filesystem to fill our newly enlarged the partition.
Task 5 - Pre-requisites
If you have an SSH key, you can install it now and start an SSH server.
apt install openssh-server
nano .ssh/authorized_keys # paste your key into this file
If you do that, you can SSH into the VM using localhost:5555 right away. You
might prefer that to whatever telnet client you were using. Or, keep using
telnet - either is fine.
Now install some more tools:
apt install build-essential libssl-dev python3 flex bison bc libncurses-dev gawk openssl libssl-dev libelf-dev libudev-dev libpci-dev libiberty-dev autoconf llvm clang lld git 
curl https://sh.rustup.rs | bash
source $HOME/.cargo/env
cargo install --locked bindgen-cli
rustup component add rust-src
Task 6 - Build a kernel
Let's grab Linux 6.14 and build it.
curl -O https://cdn.kernel.org/pub/linux/kernel/v6.x/linux-6.14.tar.xz
tar xvf linux-6.14.tar.xz
cd linux-6.14
make LLVM=1 rustavailable
make LLVM=1 defconfig 
make LLVM=1 menuconfig  # General setup / [*] Rust support
In the menuconfig interface you will need to enter General Support and
scroll down to Rust support. Press Y to enable Rust support. Then use the
arrow keys to select < Exit > and < Exit > again, then < Yes > to save.
Now we can build and install our kernel.
make LLVM=1 -j8
make LLVM=1 modules_install
make LLVM=1 install
reboot
This takes around 12 minutes or so on a fast laptop (using Hypervisor acceleration). It'll be much longer if you're on the 'opposite' architecture and are having to fully emulate the guest processor.
Task 7 - Build a kernel module
OK! Now we have Linux 6.14 with Rust support enabled. Let's build an out-of-tree kernel module.
git clone https://github.com/Rust-for-Linux/rust-out-of-tree-module
cd rust-out-of-tree-module
git checkout 15de8569df46e16f4940b52c91ee8f6bfbe5ab22
make KDIR=../linux-6.14 LLVM=1
The kernel module has been compiled as rust_out_of_tree.ko. Let's load it.
insmod ./rust_out_of_tree.ko
As it loaded, you should see a message from the kernel - if you're on SSH, check
the telnet window. If you don't have telnet connected, you can review the kernel
logs with dmesg.
root@localhost:~/rust-out-of-tree-module# insmod rust_out_of_tree.ko
[   13.933513] rust_out_of_tree: loading out-of-tree module taints kernel.
[   13.938155] rust_out_of_tree: Rust out-of-tree sample (init)
root@localhost:~/rust-out-of-tree-module# dmesg
...
[   13.933513] rust_out_of_tree: loading out-of-tree module taints kernel.
[   13.938155] rust_out_of_tree: Rust out-of-tree sample (init)
root@localhost:~/rust-out-of-tree-module#
Now let's unload it.
rmmod rust_out_of_tree
Again, you should see some output:
root@localhost:~/rust-out-of-tree-module# rmmod rust_out_of_tree
[   72.677287] rust_out_of_tree: My numbers are [72, 108, 200]
[   72.677835] rust_out_of_tree: Rust out-of-tree sample (exit)
root@localhost:~/rust-out-of-tree-module# dmesg
...
[   13.933513] rust_out_of_tree: loading out-of-tree module taints kernel.
[   13.938155] rust_out_of_tree: Rust out-of-tree sample (init)
[   72.677287] rust_out_of_tree: My numbers are [72, 108, 200]
[   72.677835] rust_out_of_tree: Rust out-of-tree sample (exit)
root@localhost:~/rust-out-of-tree-module#
Task 8 - Create a device
As of Linux 6.14, Rust for Linux has Rust APIs for:
- Block Devices
- Miscellaneous Devices
- Network Devices
A Miscellaneous Device has an entry like /dev/foobar and we can open it,
close it, and send it ioctl requests.
Looking at the documentation for the MiscDevice::register method we can see
that we get an opaque object that implements PinInit rather than a concrete
type. So, we're going to need to make a bunch of changes, step by step.
First, let's remove that example Vec and hold MiscDeviceRegistration
instead. We mark the struct with #[pin_data(PinnedDrop)] to promise that we're
not going to be moving things around in memory whilst the module is loaded, and
mark the _miscdev field with #[pin]:
#[pin_data(PinnedDrop)]
struct RustOutOfTree {
    #[pin]
    _miscdev: kernel::miscdevice::MiscDeviceRegistration<RustOutOfTreeDevice>,
}Now instead of implementing kernel::Module, let's implement kernel::InPlaceModule:
impl kernel::InPlaceModule for RustOutOfTree {
    fn init(_module: &'static ThisModule) -> impl PinInit<Self, Error> {
        pr_info!("Rust out-of-tree sample (init)\n");
        let options = kernel::miscdevice::MiscDeviceOptions {
            name: kernel::c_str!("rust-misc-device"),
        };
        try_pin_init!(Self {
            _miscdev <- kernel::miscdevice::MiscDeviceRegistration::register(options),
        })
    }
}Instead of a plain Result we're returning something that implements PinInit.
The try_pin_init! macro will do this for us. The name in our
kernel::miscdevice::MiscDeviceOptions sets the name of our device in /dev.
Pick something else if you like!
We need to adjust our Drop impl, to deal with our newly pinned data structure.
#[pinned_drop]
impl PinnedDrop for RustOutOfTree {
    fn drop(self: Pin<&mut Self>) {
        pr_info!("Rust out-of-tree sample (exit)\n");
    }
}(We've also removed the bit that prints the Vec we removed).
Finally, let's make the RustOutOfTreeDevice we referenced earlier in our
RustOutOfTree structure. It's as basic as we can get away with.
struct RustOutOfTreeDevice {}
#[vtable]
impl kernel::miscdevice::MiscDevice for RustOutOfTreeDevice {
    type Ptr = Pin<KBox<Self>>;
    fn open(
        _file: &kernel::fs::File,
        _misc: &kernel::miscdevice::MiscDeviceRegistration<Self>,
    ) -> Result<Pin<KBox<Self>>> {
        return Err(ENOTTY);
    }
}Our full file looks like this
// SPDX-License-Identifier: GPL-2.0
//! Rust out-of-tree sample
use kernel::prelude::*;
module! {
    type: RustOutOfTree,
    name: "rust_out_of_tree",
    author: "Rust for Linux Contributors",
    description: "Rust out-of-tree sample",
    license: "GPL",
}
#[pin_data(PinnedDrop)]
struct RustOutOfTree {
    #[pin]
    _miscdev: kernel::miscdevice::MiscDeviceRegistration<RustOutOfTreeDevice>,
}
impl kernel::InPlaceModule for RustOutOfTree {
    fn init(_module: &'static ThisModule) -> impl PinInit<Self, Error> {
        pr_info!("Rust out-of-tree sample (init)\n");
        let options = kernel::miscdevice::MiscDeviceOptions {
            name: kernel::c_str!("rust-misc-device"),
        };
        try_pin_init!(Self {
            _miscdev <- kernel::miscdevice::MiscDeviceRegistration::register(options),
        })
    }
}
#[pinned_drop]
impl PinnedDrop for RustOutOfTree {
    fn drop(self: Pin<&mut Self>) {
        pr_info!("Rust out-of-tree sample (exit)\n");
    }
}
struct RustOutOfTreeDevice {}
#[vtable]
impl kernel::miscdevice::MiscDevice for RustOutOfTreeDevice {
    type Ptr = Pin<KBox<Self>>;
    fn open(
        _file: &kernel::fs::File,
        _misc: &kernel::miscdevice::MiscDeviceRegistration<Self>,
    ) -> Result<Pin<KBox<Self>>> {
        return Err(ENOTTY);
    }
}Let's load it and see if we get a device:
$ make KDIR=../linux-6.14 LLVM=1
$ insmod ./rust_out_of_tree.ko
[ 2337.507487] rust_out_of_tree: Rust out-of-tree sample (init)
$ ls /dev/rust*
crw------- 1 root root 10, 124 Apr  2 16:29 /dev/rust-misc-device
$ rmmod rust_out_of_tree
[ 2345.938810] rust_out_of_tree: Rust out-of-tree sample (exit)
Nice, we got a device!
Task 9 - Implement open
Let's implement the open function for our RustOutOfTreeDevice.
Our RustOutOfTreeDevice will need to hold onto a reference to our open Device:
#[pin_data]
struct RustOutOfTreeDevice {
    dev: kernel::types::ARef<kernel::device::Device>,
}Yes, more pinning was required. The clue was the return type of the open function in the MiscDevice trait: Result<Pin<KBox<Self>>>.
Let's re-write that open function to actually open our device.
#[vtable]
impl kernel::miscdevice::MiscDevice for RustOutOfTreeDevice {
    type Ptr = Pin<KBox<Self>>;
    fn open(
        file: &kernel::fs::File,
        misc: &kernel::miscdevice::MiscDeviceRegistration<Self>,
    ) -> Result<Pin<KBox<Self>>> {
        let dev = kernel::types::ARef::from(misc.device());
        dev_info!(
            dev,
            "Opening Rust Misc Device Sample (uid = {})\n",
            file.cred().euid().into_uid_in_current_ns()
        );
        KBox::try_pin_init(
            try_pin_init! {
                RustOutOfTreeDevice {
                    dev: dev,
                }
            },
            GFP_KERNEL,
        )
    }
}Our full file looks like this:
// SPDX-License-Identifier: GPL-2.0
//! Rust out-of-tree sample
use kernel::prelude::*;
module! {
    type: RustOutOfTree,
    name: "rust_out_of_tree",
    author: "Rust for Linux Contributors",
    description: "Rust out-of-tree sample",
    license: "GPL",
}
#[pin_data(PinnedDrop)]
struct RustOutOfTree {
    #[pin]
    _miscdev: kernel::miscdevice::MiscDeviceRegistration<RustOutOfTreeDevice>,
}
impl kernel::InPlaceModule for RustOutOfTree {
    fn init(_module: &'static ThisModule) -> impl PinInit<Self, Error> {
        pr_info!("Rust out-of-tree sample (init)\n");
        let options = kernel::miscdevice::MiscDeviceOptions {
            name: kernel::c_str!("rust-misc-device"),
        };
        try_pin_init!(Self {
            _miscdev <- kernel::miscdevice::MiscDeviceRegistration::register(options),
        })
    }
}
#[pinned_drop]
impl PinnedDrop for RustOutOfTree {
    fn drop(self: Pin<&mut Self>) {
        pr_info!("Rust out-of-tree sample (exit)\n");
    }
}
#[pin_data]
struct RustOutOfTreeDevice {
    dev: kernel::types::ARef<kernel::device::Device>,
}
#[vtable]
impl kernel::miscdevice::MiscDevice for RustOutOfTreeDevice {
    type Ptr = Pin<KBox<Self>>;
    fn open(
        file: &kernel::fs::File,
        misc: &kernel::miscdevice::MiscDeviceRegistration<Self>,
    ) -> Result<Pin<KBox<Self>>> {
        let dev = kernel::types::ARef::from(misc.device());
        dev_info!(
            dev,
            "Opening Rust Misc Device Sample (uid = {})\n",
            file.cred().euid().into_uid_in_current_ns()
        );
        KBox::try_pin_init(
            try_pin_init! {
                RustOutOfTreeDevice {
                    dev: dev,
                }
            },
            GFP_KERNEL,
        )
    }
}Now we should be able to see a log when we try and open our device.
$ make KDIR=../linux-6.14 LLVM=1
$ insmod ./rust_out_of_tree.ko
[ 3918.696311] rust_out_of_tree: Rust out-of-tree sample (init)
$ cat /dev/rust-misc-device
cat: /dev/rust-misc-device: Invalid argument
[ 3990.836103] misc rust-misc-device: Opening Rust Misc Device Sample (uid = 0)
Great, we can see that the device has been opened by cat. As of Linux 6.14
there's no support for Read operations - only ioctl operations - so cat gets
an error from the Kernel. That's expected.
Task 10 - implement ioctl
You've got the hang of this now, so as a bonus exercise, why not implement
ioctl. See
https://rust.docs.kernel.org/kernel/miscdevice/trait.MiscDevice.html for
details.
You'll need some IOCTL numbers to use. Try creating a "Hello" ioctl, with no
argument (no data read and no data written):
const RUST_MISC_DEV_HELLO: u32 = _IO('|' as u32, 0x80);I chose | as the ioctl type for Miscellaneous Devices, because that's what
is in the example code. If you need help with this step, that's a great place
to look.
To send an ioctl to your device, you can use this Rust program. You'll need to
put it in a package (cargo new --bin openfile) and add the nix crate with
ioctl feature enabled (cargo add -F ioctl nix).
use std::os::fd::AsRawFd;
const HELLO: u8 = 0x80;
nix::ioctl_none!(hello_ioctl, '|', HELLO);
fn main() -> Result<(), Box<dyn std::error::Error>> {
    let f = std::fs::File::open("/dev/rust-misc-device")?;
    let fd = f.as_raw_fd();
    let result = unsafe { hello_ioctl(fd) };
    println!("ioctl returned {:?}", result);
    Ok(())
}Or you could try and write the equivalent in Rust (you'll probably need the
nix crate and the libc crate).
Here's an example `ioctl` method if you need one
fn ioctl(
    me: Pin<&RustOutOfTreeDevice>,
    _file: &kernel::fs::File,
    cmd: u32,
    arg: usize,
) -> Result<isize> {
    dev_info!(me.dev, "IOCTLing Rust Out Of Tree Device\n");
    let size = kernel::ioctl::_IOC_SIZE(cmd);
    match cmd {
        RUST_MISC_DEV_HELLO => {
            dev_info!(me.dev, "-> hello received (size {}, arg {})\n", size, arg);
            Ok(100)
        }
        _ => {
            dev_err!(me.dev, "-> IOCTL not recognised: {}\n", cmd);
            Err(ENOTTY)
        }
    }
}Here's what your output might look like if we run that example:
$ make KDIR=../linux-6.14 LLVM=1
$ insmod ./rust_out_of_tree.ko
[12147.696311] rust_out_of_tree: Rust out-of-tree sample (init)
$ cd ../openfile
$ cargo run
    Finished `dev` profile [unoptimized + debuginfo] target(s) in 0.01s
     Running `target/debug/openfile`
ioctl returned Ok(100)
$ dmesg
[12150.540427] misc rust-misc-device: Opening Rust Misc Device Sample (uid = 0)
[12150.541318] misc rust-misc-device: IOCTLing Rust Misc Device Sample
[12150.541606] misc rust-misc-device: -> hello received (size 0, arg 0)
Task 11 - Keep going!
OK! If you still want more, try implementing 'read' and 'write' ioctls, so you
can communicate with your driver. Or look at the Kernel mailing list for the
patches that will let you do ordinary read and write syscalls on your
device, rather than just ioctls. Happy kernel hacking in Rust!