Skip to content

Commit 182bec4

Browse files
wip: Add global system table pointer
1 parent af3c38d commit 182bec4

File tree

13 files changed

+1575
-167
lines changed

13 files changed

+1575
-167
lines changed

uefi-macros/src/lib.rs

+58
Original file line numberDiff line numberDiff line change
@@ -254,6 +254,64 @@ pub fn entry(args: TokenStream, input: TokenStream) -> TokenStream {
254254
result.into()
255255
}
256256

257+
/// TODO
258+
#[proc_macro_attribute]
259+
pub fn main(args: TokenStream, input: TokenStream) -> TokenStream {
260+
let mut errors = TokenStream2::new();
261+
262+
if !args.is_empty() {
263+
errors.append_all(err!(
264+
TokenStream2::from(args),
265+
"Entry attribute accepts no arguments"
266+
));
267+
}
268+
269+
let f = parse_macro_input!(input as ItemFn);
270+
271+
if let Some(ref abi) = f.sig.abi {
272+
errors.append_all(err!(abi, "Entry method must have no ABI modifier"));
273+
}
274+
if let Some(asyncness) = f.sig.asyncness {
275+
errors.append_all(err!(asyncness, "Entry method should not be async"));
276+
}
277+
if let Some(constness) = f.sig.constness {
278+
errors.append_all(err!(constness, "Entry method should not be const"));
279+
}
280+
if !f.sig.generics.params.is_empty() {
281+
errors.append_all(err!(
282+
f.sig.generics.params,
283+
"Entry method should not be generic"
284+
));
285+
}
286+
287+
// TODO: validate no fn args, validate return type
288+
289+
// show most errors at once instead of one by one
290+
if !errors.is_empty() {
291+
return errors.into();
292+
}
293+
294+
let fn_ident = &f.sig.ident;
295+
let result = quote! {
296+
#f
297+
298+
// TODO: non-conflicting name?
299+
#[export_name = "efi_main"]
300+
fn efi_main(image: ::uefi::Handle, system_table: *mut ::core::ffi::c_void) -> ::uefi::Status {
301+
unsafe {
302+
::uefi::boot::set_image_handle(image);
303+
::uefi::system::set_system_table(system_table.cast());
304+
match #fn_ident() {
305+
Ok(()) => Status::SUCCESS,
306+
Err(err) => err.status(),
307+
}
308+
}
309+
}
310+
311+
};
312+
result.into()
313+
}
314+
257315
/// Builds a `CStr8` literal at compile time from a string literal.
258316
///
259317
/// This will throw a compile error if an invalid character is in the passed string.

uefi-services/src/lib.rs

+7-1
Original file line numberDiff line numberDiff line change
@@ -43,7 +43,7 @@ use cfg_if::cfg_if;
4343
use uefi::prelude::*;
4444
use uefi::table::boot::{EventType, Tpl};
4545
use uefi::table::{Boot, SystemTable};
46-
use uefi::{Event, Result};
46+
use uefi::{system, Event, Result};
4747

4848
/// Reference to the system table.
4949
///
@@ -74,6 +74,12 @@ pub fn system_table() -> NonNull<SystemTable<Boot>> {
7474
}
7575
}
7676

77+
/// TODO
78+
pub fn init_v2() -> Result {
79+
let mut st = system::system_table_boot();
80+
init(&mut st)
81+
}
82+
7783
/// Initialize the UEFI utility library.
7884
///
7985
/// This must be called as early as possible,

uefi-test-runner/src/bin/shell_launcher.rs

+20-28
Original file line numberDiff line numberDiff line change
@@ -18,17 +18,14 @@ use uefi::proto::device_path::build::{self, DevicePathBuilder};
1818
use uefi::proto::device_path::{DevicePath, DeviceSubType, DeviceType, LoadedImageDevicePath};
1919
use uefi::proto::loaded_image::LoadedImage;
2020
use uefi::table::boot::LoadImageSource;
21-
use uefi::Status;
21+
use uefi::{boot, Result, Status};
2222

2323
/// Get the device path of the shell app. This is the same as the
2424
/// currently-loaded image's device path, but with the file path part changed.
25-
fn get_shell_app_device_path<'a>(
26-
boot_services: &BootServices,
27-
storage: &'a mut Vec<u8>,
28-
) -> &'a DevicePath {
29-
let loaded_image_device_path = boot_services
30-
.open_protocol_exclusive::<LoadedImageDevicePath>(boot_services.image_handle())
31-
.expect("failed to open LoadedImageDevicePath protocol");
25+
fn get_shell_app_device_path(storage: &mut Vec<u8>) -> &DevicePath {
26+
let loaded_image_device_path =
27+
boot::open_protocol_exclusive::<LoadedImageDevicePath>(boot::image_handle())
28+
.expect("failed to open LoadedImageDevicePath protocol");
3229

3330
let mut builder = DevicePathBuilder::with_vec(storage);
3431
for node in loaded_image_device_path.node_iter() {
@@ -45,29 +42,26 @@ fn get_shell_app_device_path<'a>(
4542
builder.finalize().unwrap()
4643
}
4744

48-
#[entry]
49-
fn efi_main(image: Handle, mut st: SystemTable<Boot>) -> Status {
50-
uefi_services::init(&mut st).unwrap();
51-
let boot_services = st.boot_services();
45+
#[uefi::main]
46+
fn main() -> Result {
47+
uefi_services::init_v2().unwrap();
5248

5349
let mut storage = Vec::new();
54-
let shell_image_path = get_shell_app_device_path(boot_services, &mut storage);
50+
let shell_image_path = get_shell_app_device_path(&mut storage);
5551

5652
// Load the shell app.
57-
let shell_image_handle = boot_services
58-
.load_image(
59-
image,
60-
LoadImageSource::FromDevicePath {
61-
device_path: shell_image_path,
62-
from_boot_manager: false,
63-
},
64-
)
65-
.expect("failed to load shell app");
53+
let shell_image_handle = boot::load_image(
54+
boot::image_handle(),
55+
LoadImageSource::FromDevicePath {
56+
device_path: shell_image_path,
57+
from_boot_manager: false,
58+
},
59+
)
60+
.expect("failed to load shell app");
6661

6762
// Set the command line passed to the shell app so that it will run the
6863
// test-runner app. This automatically turns off the five-second delay.
69-
let mut shell_loaded_image = boot_services
70-
.open_protocol_exclusive::<LoadedImage>(shell_image_handle)
64+
let mut shell_loaded_image = boot::open_protocol_exclusive::<LoadedImage>(shell_image_handle)
7165
.expect("failed to open LoadedImage protocol");
7266
let load_options = cstr16!(r"shell.efi test_runner.efi");
7367
unsafe {
@@ -78,9 +72,7 @@ fn efi_main(image: Handle, mut st: SystemTable<Boot>) -> Status {
7872
}
7973

8074
info!("launching the shell app");
81-
boot_services
82-
.start_image(shell_image_handle)
83-
.expect("failed to launch the shell app");
75+
boot::start_image(shell_image_handle).expect("failed to launch the shell app");
8476

85-
Status::SUCCESS
77+
Ok(())
8678
}

uefi-test-runner/src/boot/memory.rs

+11-37
Original file line numberDiff line numberDiff line change
@@ -1,26 +1,24 @@
1-
use uefi::table::boot::{AllocateType, BootServices, MemoryType};
1+
use uefi::boot;
2+
use uefi::table::boot::{AllocateType, MemoryType};
23

34
use alloc::vec::Vec;
45

5-
pub fn test(bt: &BootServices) {
6+
pub fn test() {
67
info!("Testing memory functions");
78

8-
allocate_pages(bt);
9+
allocate_pages();
910
vec_alloc();
1011
alloc_alignment();
11-
memmove(bt);
1212

13-
memory_map(bt);
13+
memory_map();
1414
}
1515

16-
fn allocate_pages(bt: &BootServices) {
16+
fn allocate_pages() {
1717
info!("Allocating some pages of memory");
1818

1919
let ty = AllocateType::AnyPages;
2020
let mem_ty = MemoryType::LOADER_DATA;
21-
let pgs = bt
22-
.allocate_pages(ty, mem_ty, 1)
23-
.expect("Failed to allocate a page of memory");
21+
let pgs = boot::allocate_pages(ty, mem_ty, 1).expect("Failed to allocate a page of memory");
2422

2523
assert_eq!(pgs % 4096, 0, "Page pointer is not page-aligned");
2624

@@ -32,7 +30,7 @@ fn allocate_pages(bt: &BootServices) {
3230
buf[4095] = 0x23;
3331

3432
// Clean up to avoid memory leaks.
35-
bt.free_pages(pgs, 1).unwrap();
33+
boot::free_pages(pgs, 1).unwrap();
3634
}
3735

3836
// Simple test to ensure our custom allocator works with the `alloc` crate.
@@ -57,43 +55,19 @@ fn alloc_alignment() {
5755
assert_eq!(value.as_ptr() as usize % 0x100, 0, "Wrong alignment");
5856
}
5957

60-
// Test that the `memmove` / `set_mem` functions work.
61-
fn memmove(bt: &BootServices) {
62-
info!("Testing the `memmove` / `set_mem` functions");
63-
64-
let src = [1, 2, 3, 4];
65-
let mut dest = [0u8; 4];
66-
67-
// Fill the buffer with a value
68-
unsafe {
69-
bt.set_mem(dest.as_mut_ptr(), dest.len(), 1);
70-
}
71-
72-
assert_eq!(dest, [1; 4], "Failed to set memory");
73-
74-
// Copy other values on it
75-
unsafe {
76-
bt.memmove(dest.as_mut_ptr(), src.as_ptr(), dest.len());
77-
}
78-
79-
assert_eq!(dest, src, "Failed to copy memory");
80-
}
81-
82-
fn memory_map(bt: &BootServices) {
58+
fn memory_map() {
8359
info!("Testing memory map functions");
8460

8561
// Get the memory descriptor size and an estimate of the memory map size
86-
let sizes = bt.memory_map_size();
62+
let sizes = boot::memory_map_size();
8763

8864
// 2 extra descriptors should be enough.
8965
let buf_sz = sizes.map_size + 2 * sizes.entry_size;
9066

9167
// We will use vectors for convenience.
9268
let mut buffer = vec![0_u8; buf_sz];
9369

94-
let mut memory_map = bt
95-
.memory_map(&mut buffer)
96-
.expect("Failed to retrieve UEFI memory map");
70+
let mut memory_map = boot::memory_map(&mut buffer).expect("Failed to retrieve UEFI memory map");
9771

9872
memory_map.sort();
9973

0 commit comments

Comments
 (0)