Skip to content
Merged
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
83 changes: 63 additions & 20 deletions lib/std/os/uefi/protocol/file.zig
Original file line number Diff line number Diff line change
Expand Up @@ -9,14 +9,14 @@ const Error = Status.Error;

pub const File = extern struct {
revision: u64,
_open: *const fn (*const File, **File, [*:0]const u16, u64, u64) callconv(cc) Status,
_open: *const fn (*const File, **File, [*:0]const u16, OpenMode, Attributes) callconv(cc) Status,
_close: *const fn (*File) callconv(cc) Status,
_delete: *const fn (*File) callconv(cc) Status,
_read: *const fn (*File, *usize, [*]u8) callconv(cc) Status,
_write: *const fn (*File, *usize, [*]const u8) callconv(cc) Status,
_get_position: *const fn (*const File, *u64) callconv(cc) Status,
_set_position: *const fn (*File, u64) callconv(cc) Status,
_get_info: *const fn (*const File, *align(8) const Guid, *const usize, [*]u8) callconv(cc) Status,
_get_info: *const fn (*const File, *align(8) const Guid, *usize, ?[*]u8) callconv(cc) Status,
_set_info: *const fn (*File, *align(8) const Guid, usize, [*]const u8) callconv(cc) Status,
_flush: *const fn (*File) callconv(cc) Status,

Expand Down Expand Up @@ -52,11 +52,13 @@ pub const File = extern struct {
AccessDenied,
VolumeFull,
};
pub const GetInfoError = uefi.UnexpectedError || error{
pub const GetInfoSizeError = uefi.UnexpectedError || error{
Unsupported,
NoMedia,
DeviceError,
VolumeCorrupted,
};
pub const GetInfoError = GetInfoSizeError || error{
BufferTooSmall,
};
pub const SetInfoError = uefi.UnexpectedError || error{
Expand Down Expand Up @@ -112,8 +114,8 @@ pub const File = extern struct {
self,
&new,
file_name,
@intFromEnum(mode),
@bitCast(create_attributes),
mode,
create_attributes,
)) {
.success => return new,
.not_found => return Error.NotFound,
Expand Down Expand Up @@ -216,32 +218,65 @@ pub const File = extern struct {
try self.setPosition(pos);
}

pub fn getInfoSize(self: *const File, comptime info: std.meta.Tag(Info)) GetInfoError!usize {
const InfoType = @FieldType(Info, @tagName(info));

var len: usize = 0;
switch (self._get_info(self, &InfoType.guid, &len, null)) {
.success, .buffer_too_small => return len,
.unsupported => return Error.Unsupported,
.no_media => return Error.NoMedia,
.device_error => return Error.DeviceError,
.volume_corrupted => return Error.VolumeCorrupted,
else => |status| return uefi.unexpectedStatus(status),
}
}

/// If `buffer` is too small to contain all of the info, this function returns
/// `Error.BufferTooSmall`. You should call `getInfoSize` first to determine
/// how big the buffer should be to safely call this function.
pub fn getInfo(
self: *const File,
comptime info: std.meta.Tag(Info),
) GetInfoError!std.meta.TagPayload(Info, info) {
const InfoType = std.meta.TagPayload(Info, info);
buffer: []u8,
) GetInfoError!*@FieldType(Info, @tagName(info)) {
const InfoType = @FieldType(Info, @tagName(info));

var val: InfoType = undefined;
var len: usize = @sizeOf(InfoType);
switch (self._get_info(self, &InfoType.guid, &len, @ptrCast(&val))) {
.success => return val,
var len = buffer.len;
switch (self._get_info(
self,
&InfoType.guid,
&len,
buffer.ptr,
)) {
.success => return @as(*InfoType, @ptrCast(buffer.ptr)),
.buffer_too_small => return Error.BufferTooSmall,
.unsupported => return Error.Unsupported,
.no_media => return Error.NoMedia,
.device_error => return Error.DeviceError,
.volume_corrupted => return Error.VolumeCorrupted,
.buffer_too_small => return Error.BufferTooSmall,
else => |status| return uefi.unexpectedStatus(status),
}
}

pub fn setInfo(
self: *File,
comptime info: std.meta.Tag(Info),
data: *const std.meta.TagPayload(Info, info),
data: *const @FieldType(Info, @tagName(info)),
) SetInfoError!void {
const InfoType = @TypeOf(data);
switch (self._set_info(self, &InfoType.guid, @sizeOf(InfoType), @ptrCast(data))) {
const InfoType = @FieldType(Info, @tagName(info));

const attached_str: [*:0]const u16 = switch (info) {
.file => data.getFileName(),
.file_system, .volume_label => data.getVolumeLabel(),
};
const attached_str_len = std.mem.sliceTo(attached_str, 0).len;

// add the length (not +1 for sentinel) because `@sizeOf(InfoType)`
// already contains the first utf16 char
const len = @sizeOf(InfoType) + (attached_str_len * 2);

switch (self._set_info(self, &InfoType.guid, len, @ptrCast(data))) {
.success => {},
.unsupported => return Error.Unsupported,
.no_media => return Error.NoMedia,
Expand All @@ -268,11 +303,19 @@ pub const File = extern struct {
}

pub const OpenMode = enum(u64) {
read = 0x0000000000000001,
// implies read
write = 0x0000000000000002,
// implies read+write
create = 0x8000000000000000,
pub const Bits = packed struct(u64) {
// 0x0000000000000001
read: bool = false,
// 0x0000000000000002
write: bool = false,
_pad: u61 = 0,
// 0x8000000000000000
create: bool = false,
};

read = @bitCast(Bits{ .read = true }),
read_write = @bitCast(Bits{ .read = true, .write = true }),
read_write_create = @bitCast(Bits{ .read = true, .write = true, .create = true }),
};

pub const Attributes = packed struct(u64) {
Expand Down
Loading