Skip to content

Commit 31b6ef1

Browse files
authored
Merge pull request #73 from 64/parse-toml
Parse bootloader configuration from kernel's Cargo.toml
2 parents cda5eb6 + fcc50da commit 31b6ef1

File tree

5 files changed

+162
-41
lines changed

5 files changed

+162
-41
lines changed

Cargo.lock

+16
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

Cargo.toml

+2-1
Original file line numberDiff line numberDiff line change
@@ -26,10 +26,11 @@ optional = true
2626

2727
[build-dependencies]
2828
llvm-tools = { version = "0.1", optional = true }
29+
toml = { version = "0.5.1", optional = true }
2930

3031
[features]
3132
default = []
32-
binary = ["xmas-elf", "x86_64", "usize_conversions", "fixedvec", "llvm-tools"]
33+
binary = ["xmas-elf", "x86_64", "usize_conversions", "fixedvec", "llvm-tools", "toml"]
3334
vga_320x200 = ["font8x8"]
3435
recursive_page_table = []
3536
map_physical_memory = []

README.md

+30-5
Original file line numberDiff line numberDiff line change
@@ -10,18 +10,43 @@ Written for the [second edition](https://github.com/phil-opp/blog_os/issues/360)
1010

1111
TODO
1212

13+
## Configuration
14+
15+
The bootloader exposes a few variables which can be configured through the `Cargo.toml` of your kernel:
16+
17+
```toml
18+
[package.metadata.bootloader]
19+
# The address at which the kernel stack is placed. If not provided, the bootloader
20+
# dynamically searches for a location.
21+
kernel-stack-address = "0xFFFFFF8000000000"
22+
23+
# The size of the kernel stack, given in number of 4KiB pages. Defaults to 512.
24+
kernel-stack-size = 128
25+
26+
# The virtual address offset from which physical memory is mapped, as described in
27+
# https://os.phil-opp.com/paging-implementation/#map-the-complete-physical-memory
28+
# Only applies if the `map_physical_memory` feature of the crate is enabled.
29+
# If not provided, the bootloader dynamically searches for a location.
30+
physical-memory-offset = "0xFFFF800000000000"
31+
```
32+
33+
Note that the addresses **must** be given as strings (in either hex or decimal format), as [TOML](https://github.com/toml-lang/toml) does not support unsigned 64-bit integers.
34+
1335
## Requirements
1436

1537
You need a nightly [Rust](https://www.rust-lang.org) compiler and [cargo xbuild](https://github.com/rust-osdev/cargo-xbuild). You also need the `llvm-tools-preview` component, which can be installed through `rustup component add llvm-tools-preview`.
1638

1739
## Build
1840

19-
The simplest way to use the bootloader is in combination with the [bootimage](https://github.com/rust-osdev/bootimage) tool. This crate **requires at least bootimage 0.7.6**. With the tool installed, you can add a normal cargo dependency on the `bootloader` crate to your kernel and then run `bootimage build` to create a bootable disk image. You can also execute `bootimage run` to run your kernel in [QEMU](https://www.qemu.org/) (needs to be installed).
41+
The simplest way to use the bootloader is in combination with the [bootimage](https://github.com/rust-osdev/bootimage) tool. This crate **requires at least bootimage 0.7.7**. With the tool installed, you can add a normal cargo dependency on the `bootloader` crate to your kernel and then run `bootimage build` to create a bootable disk image. You can also execute `bootimage run` to run your kernel in [QEMU](https://www.qemu.org/) (needs to be installed).
2042

21-
To compile the bootloader manually, you need to invoke `cargo xbuild` with a `KERNEL` environment variable that points to your kernel executable (in the ELF format):
43+
To compile the bootloader manually, you need to invoke `cargo xbuild` with two environment variables:
44+
* `KERNEL`: points to your kernel executable (in the ELF format)
45+
* `KERNEL_MANIFEST`: points to the `Cargo.toml` describing your kernel
2246

47+
For example:
2348
```
24-
KERNEL=/path/to/your/kernel/target/debug/your_kernel cargo xbuild
49+
KERNEL=/path/to/your/kernel/target/debug/your_kernel KERNEL_MANIFEST=/path/to/your/kernel/Cargo.toml cargo xbuild
2550
```
2651

2752
As an example, you can build the bootloader with example kernel from the `example-kernel` directory with the following commands:
@@ -30,7 +55,7 @@ As an example, you can build the bootloader with example kernel from the `exampl
3055
cd example-kernel
3156
cargo xbuild
3257
cd ..
33-
KERNEL=example-kernel/target/x86_64-example-kernel/debug/example-kernel cargo xbuild --release --features binary
58+
KERNEL=example-kernel/target/x86_64-example-kernel/debug/example-kernel KERNEL_MANIFEST=example-kernel/Cargo.toml cargo xbuild --release --features binary
3459
```
3560

3661
The `binary` feature is required to enable the dependencies required for compiling the bootloader executable. The command results in a bootloader executable at `target/x86_64-bootloader.json/release/bootloader`. This executable is still an ELF file, which can't be run directly.
@@ -64,7 +89,7 @@ The bootloader crate can be configured through some cargo features:
6489
- `vga_320x200`: This feature switches the VGA hardware to mode 0x13, a graphics mode with resolution 320x200 and 256 colors per pixel. The framebuffer is linear and lives at address `0xa0000`.
6590
- `recursive_page_table`: Maps the level 4 page table recursively and adds the [`recursive_page_table_address`](https://docs.rs/bootloader/0.4.0/bootloader/bootinfo/struct.BootInfo.html#structfield.recursive_page_table_addr) field to the passed `BootInfo`.
6691
- `map_physical_memory`: Maps the complete physical memory in the virtual address space and passes a [`physical_memory_offset`](https://docs.rs/bootloader/0.4.0/bootloader/bootinfo/struct.BootInfo.html#structfield.physical_memory_offset) field in the `BootInfo`.
67-
- The virtual address where the physical memory should be mapped is configurable by setting the `BOOTLOADER_PHYSICAL_MEMORY_OFFSET` environment variable (supports decimal and hex numbers (prefixed with `0x`)).
92+
- The virtual address where the physical memory should be mapped is configurable by setting the `physical-memory-offset` field in the kernel's `Cargo.toml`, as explained in [Configuration](#Configuration).
6893

6994
## Advanced Documentation
7095
See these guides for advanced usage of this crate:

azure-pipelines.yml

+3-1
Original file line numberDiff line numberDiff line change
@@ -96,7 +96,9 @@ steps:
9696

9797
- script: cargo xbuild --bin bootloader --features binary --release
9898
displayName: 'Build Bootloader'
99-
env: { KERNEL: "test-kernel/target/x86_64-test-kernel/debug/test-kernel" }
99+
env:
100+
KERNEL: "test-kernel/target/x86_64-test-kernel/debug/test-kernel"
101+
KERNEL_MANIFEST: "test-kernel/Cargo.toml"
100102

101103
- script: cargo objcopy -- -I elf64-x86-64 -O binary --binary-architecture=i386:x86-64 target/x86_64-bootloader/release/bootloader target/x86_64-bootloader/release/bootloader.bin
102104
displayName: 'Convert Bootloader ELF to Binary'

build.rs

+111-34
Original file line numberDiff line numberDiff line change
@@ -11,34 +11,79 @@ fn main() {
1111
}
1212

1313
#[cfg(feature = "binary")]
14-
fn num_from_env(env: &'static str, aligned: bool) -> Option<u64> {
15-
use std::env;
16-
match env::var(env) {
17-
Err(env::VarError::NotPresent) => None,
18-
Err(env::VarError::NotUnicode(_)) => {
19-
panic!("The `{}` environment variable must be valid unicode", env,)
20-
}
21-
Ok(s) => {
22-
let num = if s.starts_with("0x") {
23-
u64::from_str_radix(&s[2..], 16)
24-
} else {
25-
u64::from_str_radix(&s, 10)
26-
};
27-
28-
let num = num.expect(&format!(
29-
"The `{}` environment variable must be an integer\
30-
(is `{}`).",
31-
env, s
32-
));
14+
#[derive(Default)]
15+
struct BootloaderConfig {
16+
physical_memory_offset: Option<u64>,
17+
kernel_stack_address: Option<u64>,
18+
kernel_stack_size: Option<u64>,
19+
}
20+
21+
#[cfg(feature = "binary")]
22+
fn parse_aligned_addr(key: &str, value: &str) -> u64 {
23+
let num = if value.starts_with("0x") {
24+
u64::from_str_radix(&value[2..], 16)
25+
} else {
26+
u64::from_str_radix(&value, 10)
27+
};
3328

34-
if aligned && num % 0x1000 != 0 {
29+
let num = num.expect(&format!(
30+
"`{}` in the kernel manifest must be an integer (is `{}`)",
31+
key, value
32+
));
33+
34+
if num % 0x1000 != 0 {
35+
panic!(
36+
"`{}` in the kernel manifest must be aligned to 4KiB (is `{}`)",
37+
key, value
38+
);
39+
} else {
40+
num
41+
}
42+
}
43+
44+
#[cfg(feature = "binary")]
45+
fn parse_to_config(cfg: &mut BootloaderConfig, table: &toml::value::Table) {
46+
use toml::Value;
47+
48+
for (key, value) in table {
49+
match (key.as_str(), value.clone()) {
50+
("kernel-stack-address", Value::Integer(i))
51+
| ("physical-memory-offset", Value::Integer(i)) => {
3552
panic!(
36-
"The `{}` environment variable must be aligned to 0x1000 (is `{:#x}`).",
37-
env, num
53+
"`{0}` in the kernel manifest must be given as a string, \
54+
as toml does not support unsigned 64-bit integers (try `{0} = \"{1}\"`)",
55+
key.as_str(),
56+
i
57+
);
58+
}
59+
("kernel-stack-address", Value::String(s)) => {
60+
cfg.kernel_stack_address = Some(parse_aligned_addr(key.as_str(), &s));
61+
}
62+
#[cfg(not(feature = "map_physical_memory"))]
63+
("physical-memory-offset", Value::String(_)) => {
64+
panic!(
65+
"`physical-memory-offset` is only supported when the `map_physical_memory` \
66+
feature of the crate is enabled"
67+
);
68+
}
69+
#[cfg(feature = "map_physical_memory")]
70+
("physical-memory-offset", Value::String(s)) => {
71+
cfg.physical_memory_offset = Some(parse_aligned_addr(key.as_str(), &s));
72+
}
73+
("kernel-stack-size", Value::Integer(i)) => {
74+
if i <= 0 {
75+
panic!("`kernel-stack-size` in kernel manifest must be positive");
76+
} else {
77+
cfg.kernel_stack_size = Some(i as u64);
78+
}
79+
}
80+
(s, _) => {
81+
panic!(
82+
"unknown key '{}' in kernel manifest \
83+
- you may need to update the bootloader crate",
84+
s
3885
);
3986
}
40-
41-
Some(num)
4287
}
4388
}
4489
}
@@ -47,11 +92,12 @@ fn num_from_env(env: &'static str, aligned: bool) -> Option<u64> {
4792
fn main() {
4893
use std::{
4994
env,
50-
fs::File,
95+
fs::{self, File},
5196
io::Write,
5297
path::{Path, PathBuf},
5398
process::{self, Command},
5499
};
100+
use toml::Value;
55101

56102
let target = env::var("TARGET").expect("TARGET not set");
57103
if Path::new(&target)
@@ -185,22 +231,55 @@ fn main() {
185231
process::exit(1);
186232
}
187233

234+
// Parse the kernel's Cargo.toml which is given to us by bootimage
235+
let mut bootloader_config = BootloaderConfig::default();
236+
237+
match env::var("KERNEL_MANIFEST") {
238+
Err(env::VarError::NotPresent) => {
239+
panic!("The KERNEL_MANIFEST environment variable must be set for building the bootloader.\n\n\
240+
If you use `bootimage` for building you need at least version 0.7.7. You can \
241+
update `bootimage` by running `cargo install bootimage --force`.");
242+
}
243+
Err(env::VarError::NotUnicode(_)) => {
244+
panic!("The KERNEL_MANIFEST environment variable contains invalid unicode")
245+
}
246+
Ok(path) => {
247+
println!("cargo:rerun-if-changed={}", path);
248+
249+
let contents = fs::read_to_string(&path).expect(&format!(
250+
"failed to read kernel manifest file (path: {})",
251+
path
252+
));
253+
254+
let manifest = contents
255+
.parse::<Value>()
256+
.expect("failed to parse kernel's Cargo.toml");
257+
258+
let table = manifest
259+
.get("package")
260+
.and_then(|table| table.get("metadata"))
261+
.and_then(|table| table.get("bootloader"))
262+
.and_then(|table| table.as_table());
263+
264+
if let Some(table) = table {
265+
parse_to_config(&mut bootloader_config, table);
266+
}
267+
}
268+
}
269+
188270
// Configure constants for the bootloader
189271
// We leave some variables as Option<T> rather than hardcoding their defaults so that they
190272
// can be calculated dynamically by the bootloader.
191273
let file_path = out_dir.join("bootloader_config.rs");
192274
let mut file = File::create(file_path).expect("failed to create bootloader_config.rs");
193-
let physical_memory_offset = num_from_env("BOOTLOADER_PHYSICAL_MEMORY_OFFSET", true);
194-
let kernel_stack_address = num_from_env("BOOTLOADER_KERNEL_STACK_ADDRESS", true);
195-
let kernel_stack_size = num_from_env("BOOTLOADER_KERNEL_STACK_SIZE", false);
196275
file.write_all(
197276
format!(
198277
"const PHYSICAL_MEMORY_OFFSET: Option<u64> = {:?};
199278
const KERNEL_STACK_ADDRESS: Option<u64> = {:?};
200279
const KERNEL_STACK_SIZE: u64 = {};",
201-
physical_memory_offset,
202-
kernel_stack_address,
203-
kernel_stack_size.unwrap_or(512), // size is in number of pages
280+
bootloader_config.physical_memory_offset,
281+
bootloader_config.kernel_stack_address,
282+
bootloader_config.kernel_stack_size.unwrap_or(512), // size is in number of pages
204283
)
205284
.as_bytes(),
206285
)
@@ -214,9 +293,7 @@ fn main() {
214293
);
215294

216295
println!("cargo:rerun-if-env-changed=KERNEL");
217-
println!("cargo:rerun-if-env-changed=BOOTLOADER_PHYSICAL_MEMORY_OFFSET");
218-
println!("cargo:rerun-if-env-changed=BOOTLOADER_KERNEL_STACK_ADDRESS");
219-
println!("cargo:rerun-if-env-changed=BOOTLOADER_KERNEL_STACK_SIZE");
296+
println!("cargo:rerun-if-env-changed=KERNEL_MANIFEST");
220297
println!("cargo:rerun-if-changed={}", kernel.display());
221298
println!("cargo:rerun-if-changed=build.rs");
222299
}

0 commit comments

Comments
 (0)