Skip to content
GitLab
Explore
Sign in
Register
Primary navigation
Search or go to…
Project
B
bootloader
Manage
Activity
Members
Labels
Plan
Issues
Issue boards
Milestones
Wiki
Code
Merge requests
Repository
Branches
Commits
Tags
Repository graph
Compare revisions
Snippets
Build
Pipelines
Jobs
Pipeline schedules
Artifacts
Deploy
Releases
Container Registry
Model registry
Operate
Environments
Monitor
Incidents
Analyze
Value stream analytics
Contributor analytics
CI/CD analytics
Repository analytics
Model experiments
Help
Help
Support
GitLab documentation
Compare GitLab plans
Community forum
Contribute to GitLab
Provide feedback
Keyboard shortcuts
?
Snippets
Groups
Projects
Show more breadcrumbs
Jacob Lorentzon
bootloader
Commits
91703d9e
Verified
Commit
91703d9e
authored
2 years ago
by
Jeremy Soller
Browse files
Options
Downloads
Patches
Plain Diff
Use aligned buffer if necessary for UEFI disk
parent
bea906d8
No related branches found
Branches containing commit
No related tags found
No related merge requests found
Changes
2
Hide whitespace changes
Inline
Side-by-side
Showing
2 changed files
src/os/uefi/disk.rs
+30
-5
30 additions, 5 deletions
src/os/uefi/disk.rs
src/os/uefi/mod.rs
+32
-23
32 additions, 23 deletions
src/os/uefi/mod.rs
with
62 additions
and
28 deletions
src/os/uefi/disk.rs
+
30
−
5
View file @
91703d9e
use
core
::
ops
::{
ControlFlow
,
Try
};
use
core
::
slice
;
use
redoxfs
::{
BLOCK_SIZE
,
Disk
};
use
syscall
::{
EIO
,
Error
,
Result
};
use
std
::
proto
::
Protocol
;
use
uefi
::
guid
::{
Guid
,
BLOCK_IO_GUID
};
use
uefi
::
block_io
::
BlockIo
as
UefiBlockIo
;
pub
struct
DiskEfi
(
pub
&
'static
mut
UefiBlockIo
);
pub
struct
DiskEfi
(
pub
&
'static
mut
UefiBlockIo
,
&
'static
mut
[
u8
]
);
impl
Protocol
<
UefiBlockIo
>
for
DiskEfi
{
fn
guid
()
->
Guid
{
...
...
@@ -13,7 +14,16 @@ impl Protocol<UefiBlockIo> for DiskEfi {
}
fn
new
(
inner
:
&
'static
mut
UefiBlockIo
)
->
Self
{
Self
(
inner
)
// Hack to get aligned buffer
let
block
=
unsafe
{
let
ptr
=
super
::
alloc_zeroed_page_aligned
(
BLOCK_SIZE
as
usize
);
slice
::
from_raw_parts_mut
(
ptr
,
BLOCK_SIZE
as
usize
,
)
};
Self
(
inner
,
block
)
}
}
...
...
@@ -31,12 +41,27 @@ impl Disk for DiskEfi {
}
}
let
block_size
=
self
.0
.Media.BlockSize
as
u64
;
// Use aligned buffer if necessary
let
mut
ptr
=
buffer
.as_mut_ptr
();
if
self
.0
.Media.IoAlign
!=
0
{
if
(
ptr
as
usize
)
%
(
self
.0
.Media.IoAlign
as
usize
)
!=
0
{
if
buffer
.len
()
==
self
.1
.len
()
{
ptr
=
self
.1
.as_mut_ptr
();
}
}
}
let
block_size
=
self
.0
.Media.BlockSize
as
u64
;
let
lba
=
block
*
BLOCK_SIZE
/
block_size
;
match
(
self
.0
.ReadBlocks
)(
self
.0
,
self
.0
.Media.MediaId
,
lba
,
buffer
.len
(),
buffer
.as_mut_ptr
())
.branch
()
{
ControlFlow
::
Continue
(
_
)
=>
Ok
(
buffer
.len
()),
match
(
self
.0
.ReadBlocks
)(
self
.0
,
self
.0
.Media.MediaId
,
lba
,
buffer
.len
(),
ptr
)
.branch
()
{
ControlFlow
::
Continue
(
_
)
=>
{
// Copy to original buffer if using aligned buffer
if
ptr
!=
buffer
.as_mut_ptr
()
{
buffer
.copy_from_slice
(
&
self
.1
);
}
Ok
(
buffer
.len
())
},
ControlFlow
::
Break
(
err
)
=>
{
println!
(
"DiskEfi::read_at 0x{:X} failed: {:?}"
,
block
,
err
);
Err
(
Error
::
new
(
EIO
))
...
...
This diff is collapsed.
Click to expand it.
src/os/uefi/mod.rs
+
32
−
23
View file @
91703d9e
...
...
@@ -34,6 +34,36 @@ mod dtb;
mod
memory_map
;
mod
video_mode
;
pub
(
crate
)
fn
page_size
()
->
usize
{
// EDK2 always uses 4096 as the page size
4096
}
pub
(
crate
)
fn
alloc_zeroed_page_aligned
(
size
:
usize
)
->
*
mut
u8
{
assert!
(
size
!=
0
);
let
page_size
=
page_size
();
let
pages
=
(
size
+
page_size
-
1
)
/
page_size
;
let
ptr
=
{
// Max address mapped by src/arch paging code (8 GiB)
let
mut
ptr
=
0x2_0000_0000
;
status_to_result
(
(
std
::
system_table
()
.BootServices.AllocatePages
)(
1
,
// AllocateMaxAddress
MemoryType
::
EfiRuntimeServicesData
,
// Keeps this memory out of free space list
pages
,
&
mut
ptr
)
)
.unwrap
();
ptr
as
*
mut
u8
};
assert!
(
!
ptr
.is_null
());
unsafe
{
ptr
::
write_bytes
(
ptr
,
0
,
pages
*
page_size
)
};
ptr
}
pub
struct
OsEfi
{
st
:
&
'static
SystemTable
,
}
...
...
@@ -53,32 +83,11 @@ impl Os<
}
fn
alloc_zeroed_page_aligned
(
&
self
,
size
:
usize
)
->
*
mut
u8
{
assert!
(
size
!=
0
);
let
page_size
=
self
.page_size
();
let
pages
=
(
size
+
page_size
-
1
)
/
page_size
;
let
ptr
=
{
// Max address mapped by src/arch paging code (8 GiB)
let
mut
ptr
=
0x2_0000_0000
;
status_to_result
(
(
self
.st.BootServices.AllocatePages
)(
1
,
// AllocateMaxAddress
MemoryType
::
EfiRuntimeServicesData
,
// Keeps this memory out of free space list
pages
,
&
mut
ptr
)
)
.unwrap
();
ptr
as
*
mut
u8
};
assert!
(
!
ptr
.is_null
());
unsafe
{
ptr
::
write_bytes
(
ptr
,
0
,
pages
*
page_size
)
};
ptr
alloc_zeroed_page_aligned
(
size
)
}
fn
page_size
(
&
self
)
->
usize
{
4096
page_size
()
}
fn
filesystem
(
&
self
,
password_opt
:
Option
<&
[
u8
]
>
)
->
syscall
::
Result
<
redoxfs
::
FileSystem
<
DiskEfi
>>
{
...
...
This diff is collapsed.
Click to expand it.
Preview
0%
Loading
Try again
or
attach a new file
.
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Save comment
Cancel
Please
register
or
sign in
to comment