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

std.os.uefi.tables: ziggify boot and runtime services #23441

Open
wants to merge 18 commits into
base: master
Choose a base branch
from

Conversation

dotcarmen
Copy link
Contributor

this PR updates BootServices and RuntimeServices to an idiomatic interface similar to the pattern used in std.os.uefi.protocol. Part of my crusade towards making std.os.uefi more idiomatic

@dotcarmen
Copy link
Contributor Author

@linusg
image

@linusg
Copy link
Collaborator

linusg commented Apr 2, 2025

another-one.png

It is appreciated! :^)

@dotcarmen dotcarmen force-pushed the ziggify-uefi-boot-runtime-services branch from ee41a75 to e4487d5 Compare April 2, 2025 17:22
Copy link
Collaborator

@linusg linusg left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Not a thorough review, LGTM at a high level


/// Returns the current memory map.
getMemoryMap: *const fn (mmap_size: *usize, mmap: ?[*]MemoryDescriptor, map_key: *usize, descriptor_size: *usize, descriptor_version: *u32) callconv(cc) Status,
_getMemoryMap: *const fn (mmap_size: *usize, mmap: [*]MemoryDescriptor, map_key: *MemoryMapKey, descriptor_size: *usize, descriptor_version: *u32) callconv(cc) Status,
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

i figured it's safer to obscure map_key values since as far as i can tell it's meant to be assigned by the system (only retrievable via getMemoryMap)

interfaces: anytype,
) InstallProtocolInterfacesError!Handle {
var hdl: ?Handle = handle;
const args_tuple = protocolInterfaces(&hdl, interfaces);
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

i couldn't find an easier way to manage dynamically constructing a tuple than this helper :/

};

fn protocolInterfaces(
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

this function is kinda gross :/ suggestions are welcome

Copy link
Contributor

@truemedian truemedian left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

A few minor nitpicks, but since we're overhauling boot services it's time we remove [*]MemoryDescriptor and []MemoryDescriptor from std.os.uefi, they are never valid because the size of the zig struct is almost never the size of the descriptor the firmware returns.

Comment on lines -35 to +48
raiseTpl: *const fn (new_tpl: usize) callconv(cc) usize,
raiseTpl: *const fn (new_tpl: TaskPriorityLevel) callconv(cc) TaskPriorityLevel,

/// Restores a task's priority level to its previous value.
restoreTpl: *const fn (old_tpl: usize) callconv(cc) void,
restoreTpl: *const fn (old_tpl: TaskPriorityLevel) callconv(cc) void,
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

doesn't make sense to define wrappers for these functions imo


/// Opens a protocol with a structure as the loaded image for a UEFI application
pub fn openProtocolSt(self: *BootServices, comptime protocol: type, handle: Handle) !*protocol {
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

this function is now basically the handleProtocol function

@dotcarmen dotcarmen force-pushed the ziggify-uefi-boot-runtime-services branch from 50823da to 0733531 Compare April 4, 2025 14:02
@@ -639,7 +639,8 @@ pub fn defaultPanic(
// ExitData buffer must be allocated using boot_services.allocatePool (spec: page 220)
const exit_data: []u16 = uefi.raw_pool_allocator.alloc(u16, exit_msg.len + 1) catch @trap();
@memcpy(exit_data, exit_msg[0..exit_data.len]); // Includes null terminator.
_ = bs.exit(uefi.handle, .aborted, exit_data.len, exit_data.ptr);
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

afaict this was a bug 😬 exit_data.len for number of u16s but the spec says it's the number of bytes

@dotcarmen
Copy link
Contributor Author

dotcarmen commented Apr 4, 2025

i feel like this is ready for a final review :) i haven't tried it with my own project yet though, i'll tackle that later

@@ -134,7 +134,7 @@ pub const BootServices = extern struct {
_setWatchdogTimer: *const fn (timeout: usize, watchdog_code: u64, data_size: usize, watchdog_data: ?[*]const u16) callconv(cc) Status,

/// Connects one or more drives to a controller.
_connectController: *const fn (controller_handle: Handle, driver_image_handle: ?[*:null]Handle, remaining_device_path: ?*const DevicePathProtocol, recursive: bool) callconv(cc) Status,
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

really feels like [*:null]*anyopaque should work :(


services: *const RuntimeServices,
buffer: []u16,
guid: Guid,
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

i'm working on fixing all errors from refAllDeclsRecursive and running into a problem here because guid isn't guaranteed to be 8-aligned. does anybody know if that's a requirement for getNextVariableName??? i can't find anything on the internet anywhere, and it seems like uefi-rs does the same thing but also the spec says that EFI_GUID* must be 8-aligned unless otherwise specified, and it's not specified for getNextVariableName 😭

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I committed the change to getNextVariableName not requiring vendor_guid to be 8-aligned, but if that's not correct i can fix it another way 😅

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

If you make the first field of Guid align(8) you increase its natural alignment to 8 and you no longer have to think about it. The only place that Guid is ever unaligned in the base specification is in a DevicePath (where everything is unaligned).

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

omg you can aligned fields! it's not documented and all the syntax options i'd tried were causing parse errors 😅

i don't want to force Guid to be 8-aligned specifically for the case of DevicePath (so you can cast the pointer and it'll 'just work'). this is fixed by making the guid field 8-aligned :)

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Making the natural alignment 8 does not force it to be 8-aligned, you can always override alignment with *align(1) etc. GUIDs in the device path are already align(1), so changing the natural alignment serves only to make every *align(8) Guid into just a *Guid.

/// A handle to an event structure.
pub const Event = *opaque {};

pub const EventRegistration = *const anyopaque;
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

An event registration is not arbitrary data (as opposed to say, a notify context which is user-controlled), it refers to a specific opaque structure

Suggested change
pub const EventRegistration = *const anyopaque;
pub const EventRegistration = *const opaque{};

self: *const RuntimeServices,
count: u32,
) GetNextHighMonotonicCountError!u32 {
var cnt = count;
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

count is an out-parameter, the input is clobbered and doesn't matter.


/// Returns the first protocol instance that matches the given protocol.
locateProtocol: *const fn (protocol: *align(8) const Guid, registration: ?*const anyopaque, interface: *?*anyopaque) callconv(cc) Status,
_locateProtocol: *const fn (protocol: *align(8) const Guid, registration: ?*const anyopaque, interface: ?*?EventRegistration) callconv(cc) Status,
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Was the type of these parameters flipped?

Comment on lines +852 to +872
/// `data` may be a [*:0]const u16 immediately following additional data of
/// any format. The string is a description that the caller may use to further
/// indicate the reason for the image’s exit. `data` is only valid if `status`
/// is something other than `.success`.
pub fn exit(
self: *BootServices,
handle: Handle,
status: Status,
data: ?[]const u8,
) ExitError!void {
switch (self._exit(
handle,
status,
if (data) |d| d.len else 0,
if (data) |d| @alignCast(@ptrCast(d.ptr)) else null,
)) {
.success => {},
.invalid_parameter => return error.InvalidParameter,
else => |exit_status| return uefi.unexpectedStatus(exit_status),
}
}
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Could this guarantee alignment safety at compile-time?

Suggested change
/// `data` may be a [*:0]const u16 immediately following additional data of
/// any format. The string is a description that the caller may use to further
/// indicate the reason for the image’s exit. `data` is only valid if `status`
/// is something other than `.success`.
pub fn exit(
self: *BootServices,
handle: Handle,
status: Status,
data: ?[]const u8,
) ExitError!void {
switch (self._exit(
handle,
status,
if (data) |d| d.len else 0,
if (data) |d| @alignCast(@ptrCast(d.ptr)) else null,
)) {
.success => {},
.invalid_parameter => return error.InvalidParameter,
else => |exit_status| return uefi.unexpectedStatus(exit_status),
}
}
/// `data` may be a [*:0]const u16 immediately following additional data of
/// any format. The string is a description that the caller may use to further
/// indicate the reason for the image’s exit. `data` is only valid if `status`
/// is something other than `.success`.
pub fn exit(
self: *BootServices,
handle: Handle,
status: Status,
data: ?[]const align(2) u8,
) ExitError!void {
switch (self._exit(
handle,
status,
if (data) |d| d.len else 0,
if (data) |d| @ptrCast(d.ptr) else null,
)) {
.success => {},
.invalid_parameter => return error.InvalidParameter,
else => |exit_status| return uefi.unexpectedStatus(exit_status),
}
}

Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Additionally, I'm not really sure that I like that this interface uses a slice of bytes rather than u16, when the principal use of this is to return the string, since it makes this sort of invocation of this API use a @ptrcast. However, since you can pass binary data after the null-terminated string, I guess that makes sense? The only alternative I can see would be to split into exit (taking [:0]const u16) and exitBytes ([]const align(2) u8), which is not really ideal either.


/// Terminates a loaded EFI image and returns control to boot services.
exit: *const fn (image_handle: Handle, exit_status: Status, exit_data_size: usize, exit_data: ?*const anyopaque) callconv(cc) Status,
_exit: *const fn (image_handle: Handle, exit_status: Status, exit_data_size: usize, exit_data: ?*const u16) callconv(cc) Status,
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Slightly pedantic, but this should really be [*]const u16, no?

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

5 participants