Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add support for static X86_64 targets #244

Draft
wants to merge 11 commits into
base: main
Choose a base branch
from
Draft
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
5 changes: 4 additions & 1 deletion README.md
Original file line number Diff line number Diff line change
Expand Up @@ -48,6 +48,7 @@ Please file an issue if additional packages are required.
* xmllint
* qemu-system-aarch64
* qemu-system-riscv64
* qemu-system-x86

To build the documentation you also need
* pandoc
Expand All @@ -68,7 +69,7 @@ On a Debian-like system you can do:
pandoc texlive-latex-base texlive-latex-recommended \
texlive-fonts-recommended texlive-fonts-extra \
python3.9 python3.9-venv \
qemu-system-arm qemu-system-misc \
qemu-system-arm qemu-system-misc qemu-system-x86 \
gcc-riscv64-unknown-elf
$ python3.9 -m venv pyenv
$ ./pyenv/bin/pip install --upgrade pip setuptools wheel
Expand Down Expand Up @@ -210,8 +211,10 @@ The currently supported boards are:
* maaxboard
* odroidc2
* odroidc4
* odroidh2plus
* qemu_virt_aarch64
* qemu_virt_riscv64
* qemu_virt_x86_64
* rockpro64
* star64
* tqma8xqp1gb
Expand Down
49 changes: 49 additions & 0 deletions build_sdk.py
Original file line number Diff line number Diff line change
Expand Up @@ -34,6 +34,7 @@

TOOLCHAIN_AARCH64 = "aarch64-none-elf"
TOOLCHAIN_RISCV = "riscv64-unknown-elf"
TOOLCHAIN_X86_64 = "x86_64-linux-gnu"

KERNEL_CONFIG_TYPE = Union[bool, str]
KERNEL_OPTIONS = Dict[str, Union[bool, str]]
Expand All @@ -42,12 +43,15 @@
class KernelArch(IntEnum):
AARCH64 = 1
RISCV64 = 2
X86_64 = 3

def c_toolchain(self) -> str:
if self == KernelArch.AARCH64:
return TOOLCHAIN_AARCH64
elif self == KernelArch.RISCV64:
return TOOLCHAIN_RISCV
elif self == KernelArch.X86_64:
return TOOLCHAIN_X86_64
else:
raise Exception(f"Unsupported toolchain architecture '{self}'")

Expand All @@ -57,11 +61,16 @@ def is_riscv(self) -> bool:
def is_arm(self) -> bool:
return self == KernelArch.AARCH64

def is_x86_64(self) -> bool:
return self == KernelArch.X86_64

def to_str(self) -> str:
if self == KernelArch.AARCH64:
return "aarch64"
elif self == KernelArch.RISCV64:
return "riscv64"
elif self == KernelArch.X86_64:
return "x86_64"
else:
raise Exception(f"Unsupported arch {self}")

Expand All @@ -74,6 +83,7 @@ class BoardInfo:
loader_link_address: int
kernel_options: KERNEL_OPTIONS
examples: Dict[str, Path]
gcc_march: Optional[str] = None


@dataclass
Expand Down Expand Up @@ -280,6 +290,41 @@ class ConfigInfo:
"hello": Path("example/star64/hello")
}
),
BoardInfo(
name="x86_64_nehalem",
arch=KernelArch.X86_64,
gcc_cpu=None,
gcc_march="nehalem",
loader_link_address=0x10000000, # 256MB
kernel_options = {
"KernelIsMCS": True,
"KernelPlatform": "pc99",
"KernelSel4Arch": "x86_64",
"KernelVTX": True,
"KernelX86MicroArch": "nehalem",
},
examples = {
"hello": Path("example/x86_64_nehalem/hello")
}
),
BoardInfo(
name="x86_64_goldmont_plus",
arch=KernelArch.X86_64,
gcc_cpu=None,
gcc_march="goldmont-plus",
loader_link_address=0x10000000, # 256MB
kernel_options = {
"KernelIsMCS": True,
"KernelPlatform": "pc99",
"KernelSel4Arch": "x86_64",
"KernelVTX": True,
"KernelX86MicroArch": "goldmont-plus",
"KernelSupportPCID": False,
},
examples = {
"hello": Path("example/x86_64_goldmont_plus/hello")
}
),
)

SUPPORTED_CONFIGS = (
Expand Down Expand Up @@ -477,6 +522,8 @@ def build_elf_component(

if board.gcc_cpu is not None:
defines_str += f" GCC_CPU={board.gcc_cpu}"
if board.gcc_march is not None:
defines_str += f" GCC_MARCH={board.gcc_march}"

r = system(
f"{defines_str} make -C {component_name}"
Expand Down Expand Up @@ -523,6 +570,8 @@ def build_lib_component(

if board.gcc_cpu is not None:
defines_str += f" GCC_CPU={board.gcc_cpu}"
if board.gcc_march is not None:
defines_str += f" GCC_MARCH={board.gcc_march}"

r = system(
f"{defines_str} make -C {component_name}"
Expand Down
74 changes: 72 additions & 2 deletions docs/manual.md
Original file line number Diff line number Diff line change
Expand Up @@ -102,6 +102,7 @@ This document attempts to clearly describe all of these terms, however as the co
* [notification](#notification)
* [interrupt](#irq)
* [fault](#fault)
* [ioport](#ioport)

## System {#system}

Expand Down Expand Up @@ -311,6 +312,10 @@ protection domain. The same applies for a virtual machine.
This means that whenever a fault is caused by a child, it will be delivered to the parent PD instead of the system fault
handler via the `fault` entry point. It is then up to the parent to decide how the fault is handled.

## I/O Ports {#ioport}

I/O ports are mechanisms present on x86 hardware to access certain physical devices (e.g. PC serial ports) using the `in` and `out` CPU instructions. On seL4, I/O ports are used by invoking capabilities via the `seL4_X86_IOPort_*` functions. The capabilities must be explicity associated to the protection domain and are accessed by calling `microkit_ioport_cap`.

# SDK {#sdk}

Microkit is distributed as a software development kit (SDK).
Expand Down Expand Up @@ -377,12 +382,15 @@ The format of the system description is described in a subsequent chapter.
Usage:

microkit [-h] [-o OUTPUT] [-r REPORT] --board [BOARD] --config CONFIG
[--search-path [SEARCH_PATH ...]] system
[--search-path [SEARCH_PATH ...]]
[--x86-machine [MACHINE_FILE]] system

The path to the system description file, board to build the system for, and configuration to build for must be provided.

The search paths provided tell the tool where to find any program images specified in the system description file.

The x86 machine file is a json file that describes certain properties of the target hardware. This file is mandatory for x86 targets. It may be produced by booting the [Microkit x86 machine dump](https://github.com/Neutrality-ch/microkit-x86-machine-dump) tool on the target board.

In the case of errors, a diagnostic message shall be output to `stderr` and a non-zero code returned.

In the case of success, a loadable image file and a report shall be produced.
Expand Down Expand Up @@ -439,6 +447,7 @@ If the protection domain has children it must also implement:
seL4_Word microkit_vcpu_arm_read_reg(microkit_child vcpu, seL4_Word reg);
void microkit_vcpu_arm_write_reg(microkit_child vcpu, seL4_Word reg, seL4_Word value);
void microkit_arm_smc_call(seL4_ARM_SMCContext *args, seL4_ARM_SMCContext *response);
seL4_Word microkit_ioport_cap(seL4_Word id);


## `void init(void)`
Expand Down Expand Up @@ -616,6 +625,11 @@ response values will be placed into the `response` structure.

The `seL4_ARM_SMCContext` structure contains fields for registers x0 to x7.

## `seL4_Word microkit_ioport_cap(seL4_Word id)`

This API is available only on X86. The API takes in an I/O port identifier and returns its
associated capability, which can be provided to the `seL4_X86_IOPort_*` functions.

# System Description File {#sysdesc}

This section describes the format of the System Description File (SDF).
Expand Down Expand Up @@ -651,6 +665,7 @@ Additionally, it supports the following child elements:
* `program_image`: (exactly one) Describes the program image for the protection domain.
* `map`: (zero or more) Describes mapping of memory regions into the protection domain.
* `irq`: (zero or more) Describes hardware interrupt associations.
* `ioport`: (zero or more) Describes x86 I/O ports to be exposed to the protection domain.
* `setvar`: (zero or more) Describes variable rewriting.
* `protection_domain`: (zero or more) Describes a child protection domain.
* `virtual_machine`: (zero or one) Describes a child virtual machine.
Expand All @@ -665,12 +680,34 @@ The `map` element has the following attributes:
* `cached`: (optional) Determines if mapped with caching enabled or disabled. Defaults to `true`.
* `setvar_vaddr`: (optional) Specifies a symbol in the program image. This symbol will be rewritten with the virtual address of the memory region.

The `irq` element has the following attributes:
The `irq` element has the following attributes on ARM and RISC-V platforms:

* `irq`: The hardware interrupt number.
* `id`: The channel identifier. Must be at least 0 and less than 63.
* `trigger`: (optional) Whether the IRQ is edge triggered ("edge") or level triggered ("level"). Defaults to "level".

The `irq` element has the following attributes when registering X86_64 IOAPIC interrupts:

* `id`: The channel identifier. Must be at least 0 and less than 63.
* `ioapic`: (optional) Zero based index of the IOAPIC to get the interrupt from. Defaults to 0.
* `pin`: IOAPIC pin that generates the interrupt.
* `level`: (optional) Whether the IRQ is level triggered (1) or edge triggered (0). Defaults to level (1).
* `polarity`: (optional) Whether the line polarity is high (1) or low (0). Defaults to high (1).
* `vector`: The IRQ vector number.

The `irq` element has the following attributes when registering X86_64 MSI interrupts:

* `id`: The channel identifier. Must be at least 0 and less than 63.
* `pcidev`: The PCI device address of the device that will generate the interrupt, in BUS:DEV:FUNC notation (e.g. 01:1f:2).
* `handle`: Value of the handle programmed into the data portion of the MSI.
* `vector`: CPU vector to deliver the interrupt to.

The `ioport` element has the following attributes:

* `id`: The I/O port identifier. Must be at least 0 and less than 63.
* `addr`: The base address of the I/O port.
* `size`: The size in bytes of the I/O port region.

The `setvar` element has the following attributes:

* `symbol`: Name of a symbol in the ELF file.
Expand Down Expand Up @@ -723,6 +760,11 @@ Below are the available page sizes for each architecture that Microkit supports.
* 0x1000 (4KiB)
* 0x200000 (2MiB)

#### X86 64-bit

* 0x1000 (4KiB)
* 0x200000 (2MiB)

## `channel`

The `channel` element has exactly two `end` children elements for specifying the two PDs associated with the channel.
Expand Down Expand Up @@ -786,6 +828,21 @@ Microkit produces a raw binary file, so when using U-Boot you must execute the i

=> go 0x20000000

## Odroid-H2+

The HardKernel Odroid-H2+ is an X86_64 SBC based on the Intel J4115 processor. It should
be noted that the Odroid-H2+ is no longer available for purchase but its successor, the
Odroid-H3, is readily available at the time of writing.

Microkit produces an ELF file that can be directly booted by a multiboot2 bootloader. When
using GNU GRUB you can boot the image using this menu entry:

```
menuentry 'Microkit' {
multiboot2 /loader.img
}
```

## TQMa8XQP 1GB

The TQMa8XQP is a system-on-module designed by TQ-Systems GmbH.
Expand Down Expand Up @@ -892,6 +949,19 @@ QEMU will start the system image using its packaged version of OpenSBI.
You can find more about the QEMU virt platform in the
[QEMU documentation](https://www.qemu.org/docs/master/system/target-riscv.html).


## QEMU virt (X86 64-bit)

Support is available for the virtual X86_64 QEMU platform. This is a platform that is not based
on any specific SoC or hardware platform and is intended for simulating systems for
development or testing.

You can use the `sim` shell script from the `example/qemu_virt_x86_64` directory to
simulate a Microkit system.

You can find more about the QEMU virt platform in the
[QEMU documentation](https://www.qemu.org/docs/master/system/target-i386.html).

## Pine64 ROCKPro64

Microkit produces a raw binary file, so when using U-Boot you must execute the image using:
Expand Down
55 changes: 55 additions & 0 deletions example/x86_64_goldmont_plus/hello/Makefile
Original file line number Diff line number Diff line change
@@ -0,0 +1,55 @@
#
# Copyright 2021, Breakaway Consulting Pty. Ltd.
#
# SPDX-License-Identifier: BSD-2-Clause
#
ifeq ($(strip $(BUILD_DIR)),)
$(error BUILD_DIR must be specified)
endif

ifeq ($(strip $(MICROKIT_SDK)),)
$(error MICROKIT_SDK must be specified)
endif

ifeq ($(strip $(MICROKIT_BOARD)),)
$(error MICROKIT_BOARD must be specified)
endif

ifeq ($(strip $(MICROKIT_CONFIG)),)
$(error MICROKIT_CONFIG must be specified)
endif

TOOLCHAIN := x86_64-linux-gnu

ARCH := nehalem

CC := $(TOOLCHAIN)-gcc
LD := $(TOOLCHAIN)-ld
AS := $(TOOLCHAIN)-as
MICROKIT_TOOL ?= $(MICROKIT_SDK)/bin/microkit

HELLO_OBJS := hello.o

BOARD_DIR := $(MICROKIT_SDK)/board/$(MICROKIT_BOARD)/$(MICROKIT_CONFIG)

IMAGES := hello.elf
CFLAGS := -march=$(ARCH) -DARCH_x86_64 -nostdlib -ffreestanding -g3 -O3 -Wall -Wno-unused-function -Werror -I$(BOARD_DIR)/include
LDFLAGS := -L$(BOARD_DIR)/lib -z noexecstack
LIBS := -lmicrokit -Tmicrokit.ld

IMAGE_FILE = $(BUILD_DIR)/loader.img
REPORT_FILE = $(BUILD_DIR)/report.txt

all: $(IMAGE_FILE)

$(BUILD_DIR)/%.o: %.c Makefile
$(CC) -c $(CFLAGS) $< -o $@

$(BUILD_DIR)/%.o: %.s Makefile
$(AS) -g3 -march=$(ARCH) $< -o $@

$(BUILD_DIR)/hello.elf: $(addprefix $(BUILD_DIR)/, $(HELLO_OBJS))
$(LD) $(LDFLAGS) $^ $(LIBS) -o $@

$(IMAGE_FILE) $(REPORT_FILE): $(addprefix $(BUILD_DIR)/, $(IMAGES)) hello.system
$(MICROKIT_TOOL) hello.system --search-path $(BUILD_DIR) --board $(MICROKIT_BOARD) --config $(MICROKIT_CONFIG) --x86-machine odroidh2plus.json -o $(IMAGE_FILE) -r $(REPORT_FILE)
39 changes: 39 additions & 0 deletions example/x86_64_goldmont_plus/hello/hello.c
Original file line number Diff line number Diff line change
@@ -0,0 +1,39 @@
/*
* Copyright 2021, Breakaway Consulting Pty. Ltd.
*
* SPDX-License-Identifier: BSD-2-Clause
*/
#include <stdint.h>
#include <microkit.h>

static inline void serial_putc(char ch)
{
/* The I/O port capability and address defined here must match the system
* description file (hello.system). */
int cap = microkit_ioport_cap(0);
uint16_t base = 0x3f8;

while ((seL4_X86_IOPort_In8(cap, base + 5).result & 0x20) == 0)
;
seL4_X86_IOPort_Out8(cap, base, ch);
}

static inline void serial_puts(const char *s)
{
while (*s) {
if (*s == '\n')
serial_putc('\r');
serial_putc(*s++);
}
}

void init(void)
{
microkit_dbg_puts("hello, debug port\n");
serial_puts("hello, serial port\n");
}

void notified(microkit_channel ch)
{
(void) ch;
}
12 changes: 12 additions & 0 deletions example/x86_64_goldmont_plus/hello/hello.system
Original file line number Diff line number Diff line change
@@ -0,0 +1,12 @@
<?xml version="1.0" encoding="UTF-8"?>
<!--
Copyright 2021, Breakaway Consulting Pty. Ltd.

SPDX-License-Identifier: BSD-2-Clause
-->
<system>
<protection_domain name="hello" priority="254">
<program_image path="hello.elf" />
<ioport id="0" addr="0x3f8" size="8" />
</protection_domain>
</system>
Loading