-
-
Notifications
You must be signed in to change notification settings - Fork 2.8k
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
base: master
Are you sure you want to change the base?
std.os.uefi.tables: ziggify boot and runtime services #23441
Conversation
It is appreciated! :^) |
ee41a75
to
e4487d5
Compare
There was a problem hiding this 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, |
There was a problem hiding this comment.
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); |
There was a problem hiding this comment.
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( |
There was a problem hiding this comment.
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
There was a problem hiding this 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.
e4487d5
to
5ad806e
Compare
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, |
There was a problem hiding this comment.
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 { |
There was a problem hiding this comment.
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
Co-authored-by: linusg <[email protected]>
50823da
to
0733531
Compare
@@ -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); |
There was a problem hiding this comment.
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
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, |
There was a problem hiding this comment.
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, |
There was a problem hiding this comment.
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
😭
There was a problem hiding this comment.
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 😅
There was a problem hiding this comment.
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).
There was a problem hiding this comment.
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 :)
There was a problem hiding this comment.
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
.
Co-authored-by: truemedian <[email protected]>
/// A handle to an event structure. | ||
pub const Event = *opaque {}; | ||
|
||
pub const EventRegistration = *const anyopaque; |
There was a problem hiding this comment.
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
pub const EventRegistration = *const anyopaque; | |
pub const EventRegistration = *const opaque{}; |
self: *const RuntimeServices, | ||
count: u32, | ||
) GetNextHighMonotonicCountError!u32 { | ||
var cnt = count; |
There was a problem hiding this comment.
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, |
There was a problem hiding this comment.
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?
/// `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), | ||
} | ||
} |
There was a problem hiding this comment.
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?
/// `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), | |
} | |
} |
There was a problem hiding this comment.
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, |
There was a problem hiding this comment.
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?
this PR updates
BootServices
andRuntimeServices
to an idiomatic interface similar to the pattern used instd.os.uefi.protocol
. Part of my crusade towards makingstd.os.uefi
more idiomatic