[wip] extract phys_frame code
This commit is contained in:
parent
fa10d649e4
commit
35097458b2
|
@ -332,193 +332,6 @@ impl fmt::Debug for PageTableEntry {
|
||||||
}
|
}
|
||||||
}*/
|
}*/
|
||||||
|
|
||||||
// Verbatim from https://github.com/rust-osdev/x86_64/blob/aa9ae54657beb87c2a491f2ab2140b2332afa6ba/src/structures/paging/frame.rs
|
|
||||||
// Abstractions for default-sized and huge physical memory frames.
|
|
||||||
|
|
||||||
/// A physical memory frame.
|
|
||||||
/// Frame is an addressable unit of the physical address space.
|
|
||||||
#[derive(Clone, Copy, PartialEq, Eq, PartialOrd, Ord)]
|
|
||||||
#[repr(C)]
|
|
||||||
pub struct PhysFrame<S: PageSize = Size4KiB> {
|
|
||||||
start_address: PhysAddr,
|
|
||||||
size: PhantomData<S>,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<S: PageSize> From<u64> for PhysFrame<S> {
|
|
||||||
fn from(address: u64) -> PhysFrame<S> {
|
|
||||||
PhysFrame::containing_address(PhysAddr::new(address))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<S: PageSize> From<PhysFrame<S>> for u64 {
|
|
||||||
fn from(frame: PhysFrame<S>) -> u64 {
|
|
||||||
frame.start_address.as_u64()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<S: PageSize> PhysFrame<S> {
|
|
||||||
/// Returns the frame that starts at the given virtual address.
|
|
||||||
///
|
|
||||||
/// Returns an error if the address is not correctly aligned (i.e. is not a valid frame start).
|
|
||||||
pub fn from_start_address(address: PhysAddr) -> Result<Self, ()> {
|
|
||||||
if !address.is_aligned(S::SIZE) {
|
|
||||||
return Err(());
|
|
||||||
}
|
|
||||||
Ok(PhysFrame::containing_address(address))
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Returns the frame that contains the given physical address.
|
|
||||||
pub fn containing_address(address: PhysAddr) -> Self {
|
|
||||||
PhysFrame {
|
|
||||||
start_address: address.align_down(S::SIZE),
|
|
||||||
size: PhantomData,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Returns the start address of the frame.
|
|
||||||
pub fn start_address(&self) -> PhysAddr {
|
|
||||||
self.start_address
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Returns the size the frame (4KB, 2MB or 1GB).
|
|
||||||
pub fn size(&self) -> u64 {
|
|
||||||
S::SIZE
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Returns a range of frames, exclusive `end`.
|
|
||||||
pub fn range(start: PhysFrame<S>, end: PhysFrame<S>) -> PhysFrameRange<S> {
|
|
||||||
PhysFrameRange { start, end }
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Returns a range of frames, inclusive `end`.
|
|
||||||
pub fn range_inclusive(start: PhysFrame<S>, end: PhysFrame<S>) -> PhysFrameRangeInclusive<S> {
|
|
||||||
PhysFrameRangeInclusive { start, end }
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<S: PageSize> fmt::Debug for PhysFrame<S> {
|
|
||||||
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
|
||||||
f.write_fmt(format_args!(
|
|
||||||
"PhysFrame[{}]({:#x})",
|
|
||||||
S::SIZE_AS_DEBUG_STR,
|
|
||||||
self.start_address().as_u64()
|
|
||||||
))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<S: PageSize> Add<u64> for PhysFrame<S> {
|
|
||||||
type Output = Self;
|
|
||||||
fn add(self, rhs: u64) -> Self::Output {
|
|
||||||
PhysFrame::containing_address(self.start_address() + rhs * u64::from(S::SIZE))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<S: PageSize> AddAssign<u64> for PhysFrame<S> {
|
|
||||||
fn add_assign(&mut self, rhs: u64) {
|
|
||||||
*self = self.clone() + rhs;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<S: PageSize> Sub<u64> for PhysFrame<S> {
|
|
||||||
type Output = Self;
|
|
||||||
fn sub(self, rhs: u64) -> Self::Output {
|
|
||||||
PhysFrame::containing_address(self.start_address() - rhs * u64::from(S::SIZE))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<S: PageSize> SubAssign<u64> for PhysFrame<S> {
|
|
||||||
fn sub_assign(&mut self, rhs: u64) {
|
|
||||||
*self = self.clone() - rhs;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<S: PageSize> Sub<PhysFrame<S>> for PhysFrame<S> {
|
|
||||||
type Output = u64;
|
|
||||||
fn sub(self, rhs: PhysFrame<S>) -> Self::Output {
|
|
||||||
(self.start_address - rhs.start_address) / S::SIZE
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// An range of physical memory frames, exclusive the upper bound.
|
|
||||||
#[derive(Clone, Copy, PartialEq, Eq)]
|
|
||||||
#[repr(C)]
|
|
||||||
pub struct PhysFrameRange<S: PageSize = Size4KiB> {
|
|
||||||
/// The start of the range, inclusive.
|
|
||||||
pub start: PhysFrame<S>,
|
|
||||||
/// The end of the range, exclusive.
|
|
||||||
pub end: PhysFrame<S>,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<S: PageSize> PhysFrameRange<S> {
|
|
||||||
/// Returns whether the range contains no frames.
|
|
||||||
pub fn is_empty(&self) -> bool {
|
|
||||||
!(self.start < self.end)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<S: PageSize> Iterator for PhysFrameRange<S> {
|
|
||||||
type Item = PhysFrame<S>;
|
|
||||||
|
|
||||||
fn next(&mut self) -> Option<Self::Item> {
|
|
||||||
if self.start < self.end {
|
|
||||||
let frame = self.start.clone();
|
|
||||||
self.start += 1;
|
|
||||||
Some(frame)
|
|
||||||
} else {
|
|
||||||
None
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<S: PageSize> fmt::Debug for PhysFrameRange<S> {
|
|
||||||
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
|
||||||
f.debug_struct("PhysFrameRange")
|
|
||||||
.field("start", &self.start)
|
|
||||||
.field("end", &self.end)
|
|
||||||
.finish()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// An range of physical memory frames, inclusive the upper bound.
|
|
||||||
#[derive(Clone, Copy, PartialEq, Eq)]
|
|
||||||
#[repr(C)]
|
|
||||||
pub struct PhysFrameRangeInclusive<S: PageSize = Size4KiB> {
|
|
||||||
/// The start of the range, inclusive.
|
|
||||||
pub start: PhysFrame<S>,
|
|
||||||
/// The start of the range, exclusive.
|
|
||||||
pub end: PhysFrame<S>,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<S: PageSize> PhysFrameRangeInclusive<S> {
|
|
||||||
/// Returns whether the range contains no frames.
|
|
||||||
pub fn is_empty(&self) -> bool {
|
|
||||||
!(self.start <= self.end)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<S: PageSize> Iterator for PhysFrameRangeInclusive<S> {
|
|
||||||
type Item = PhysFrame<S>;
|
|
||||||
|
|
||||||
fn next(&mut self) -> Option<Self::Item> {
|
|
||||||
if self.start <= self.end {
|
|
||||||
let frame = self.start.clone();
|
|
||||||
self.start += 1;
|
|
||||||
Some(frame)
|
|
||||||
} else {
|
|
||||||
None
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<S: PageSize> fmt::Debug for PhysFrameRangeInclusive<S> {
|
|
||||||
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
|
||||||
f.debug_struct("PhysFrameRangeInclusive")
|
|
||||||
.field("start", &self.start)
|
|
||||||
.field("end", &self.end)
|
|
||||||
.finish()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Verbatim from https://github.com/rust-osdev/x86_64/blob/aa9ae54657beb87c2a491f2ab2140b2332afa6ba/src/structures/paging/page.rs
|
// Verbatim from https://github.com/rust-osdev/x86_64/blob/aa9ae54657beb87c2a491f2ab2140b2332afa6ba/src/structures/paging/page.rs
|
||||||
// Abstractions for default-sized and huge virtual memory pages.
|
// Abstractions for default-sized and huge virtual memory pages.
|
||||||
|
|
||||||
|
|
|
@ -12,6 +12,7 @@ use {
|
||||||
|
|
||||||
mod addr;
|
mod addr;
|
||||||
pub mod mmu;
|
pub mod mmu;
|
||||||
|
mod phys_frame;
|
||||||
|
|
||||||
pub mod mmu_experimental;
|
pub mod mmu_experimental;
|
||||||
pub use mmu_experimental::*;
|
pub use mmu_experimental::*;
|
||||||
|
@ -23,6 +24,7 @@ pub use mmu_experimental::*;
|
||||||
|
|
||||||
pub use addr::PhysAddr;
|
pub use addr::PhysAddr;
|
||||||
pub use addr::VirtAddr;
|
pub use addr::VirtAddr;
|
||||||
|
pub use phys_frame::PhysFrame;
|
||||||
|
|
||||||
use mmu_experimental::PhysFrame;
|
use mmu_experimental::PhysFrame;
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,186 @@
|
||||||
|
// Verbatim from https://github.com/rust-osdev/x86_64/blob/aa9ae54657beb87c2a491f2ab2140b2332afa6ba/src/structures/paging/frame.rs
|
||||||
|
// Abstractions for default-sized and huge physical memory frames.
|
||||||
|
|
||||||
|
/// A physical memory frame.
|
||||||
|
/// Frame is an addressable unit of the physical address space.
|
||||||
|
#[derive(Clone, Copy, PartialEq, Eq, PartialOrd, Ord)]
|
||||||
|
#[repr(C)]
|
||||||
|
pub struct PhysFrame<S: PageSize = Size4KiB> {
|
||||||
|
start_address: PhysAddr,
|
||||||
|
size: PhantomData<S>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<S: PageSize> From<u64> for PhysFrame<S> {
|
||||||
|
fn from(address: u64) -> PhysFrame<S> {
|
||||||
|
PhysFrame::containing_address(PhysAddr::new(address))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<S: PageSize> From<PhysFrame<S>> for u64 {
|
||||||
|
fn from(frame: PhysFrame<S>) -> u64 {
|
||||||
|
frame.start_address.as_u64()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<S: PageSize> PhysFrame<S> {
|
||||||
|
/// Returns the frame that starts at the given virtual address.
|
||||||
|
///
|
||||||
|
/// Returns an error if the address is not correctly aligned (i.e. is not a valid frame start).
|
||||||
|
pub fn from_start_address(address: PhysAddr) -> Result<Self, ()> {
|
||||||
|
if !address.is_aligned(S::SIZE) {
|
||||||
|
return Err(());
|
||||||
|
}
|
||||||
|
Ok(PhysFrame::containing_address(address))
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Returns the frame that contains the given physical address.
|
||||||
|
pub fn containing_address(address: PhysAddr) -> Self {
|
||||||
|
PhysFrame {
|
||||||
|
start_address: address.align_down(S::SIZE),
|
||||||
|
size: PhantomData,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Returns the start address of the frame.
|
||||||
|
pub fn start_address(&self) -> PhysAddr {
|
||||||
|
self.start_address
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Returns the size the frame (4KB, 2MB or 1GB).
|
||||||
|
pub fn size(&self) -> u64 {
|
||||||
|
S::SIZE
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Returns a range of frames, exclusive `end`.
|
||||||
|
pub fn range(start: PhysFrame<S>, end: PhysFrame<S>) -> PhysFrameRange<S> {
|
||||||
|
PhysFrameRange { start, end }
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Returns a range of frames, inclusive `end`.
|
||||||
|
pub fn range_inclusive(start: PhysFrame<S>, end: PhysFrame<S>) -> PhysFrameRangeInclusive<S> {
|
||||||
|
PhysFrameRangeInclusive { start, end }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<S: PageSize> fmt::Debug for PhysFrame<S> {
|
||||||
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
||||||
|
f.write_fmt(format_args!(
|
||||||
|
"PhysFrame[{}]({:#x})",
|
||||||
|
S::SIZE_AS_DEBUG_STR,
|
||||||
|
self.start_address().as_u64()
|
||||||
|
))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<S: PageSize> Add<u64> for PhysFrame<S> {
|
||||||
|
type Output = Self;
|
||||||
|
fn add(self, rhs: u64) -> Self::Output {
|
||||||
|
PhysFrame::containing_address(self.start_address() + rhs * u64::from(S::SIZE))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<S: PageSize> AddAssign<u64> for PhysFrame<S> {
|
||||||
|
fn add_assign(&mut self, rhs: u64) {
|
||||||
|
*self = self.clone() + rhs;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<S: PageSize> Sub<u64> for PhysFrame<S> {
|
||||||
|
type Output = Self;
|
||||||
|
fn sub(self, rhs: u64) -> Self::Output {
|
||||||
|
PhysFrame::containing_address(self.start_address() - rhs * u64::from(S::SIZE))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<S: PageSize> SubAssign<u64> for PhysFrame<S> {
|
||||||
|
fn sub_assign(&mut self, rhs: u64) {
|
||||||
|
*self = self.clone() - rhs;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<S: PageSize> Sub<PhysFrame<S>> for PhysFrame<S> {
|
||||||
|
type Output = u64;
|
||||||
|
fn sub(self, rhs: PhysFrame<S>) -> Self::Output {
|
||||||
|
(self.start_address - rhs.start_address) / S::SIZE
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// An range of physical memory frames, exclusive the upper bound.
|
||||||
|
#[derive(Clone, Copy, PartialEq, Eq)]
|
||||||
|
#[repr(C)]
|
||||||
|
pub struct PhysFrameRange<S: PageSize = Size4KiB> {
|
||||||
|
/// The start of the range, inclusive.
|
||||||
|
pub start: PhysFrame<S>,
|
||||||
|
/// The end of the range, exclusive.
|
||||||
|
pub end: PhysFrame<S>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<S: PageSize> PhysFrameRange<S> {
|
||||||
|
/// Returns whether the range contains no frames.
|
||||||
|
pub fn is_empty(&self) -> bool {
|
||||||
|
!(self.start < self.end)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<S: PageSize> Iterator for PhysFrameRange<S> {
|
||||||
|
type Item = PhysFrame<S>;
|
||||||
|
|
||||||
|
fn next(&mut self) -> Option<Self::Item> {
|
||||||
|
if self.start < self.end {
|
||||||
|
let frame = self.start.clone();
|
||||||
|
self.start += 1;
|
||||||
|
Some(frame)
|
||||||
|
} else {
|
||||||
|
None
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<S: PageSize> fmt::Debug for PhysFrameRange<S> {
|
||||||
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
||||||
|
f.debug_struct("PhysFrameRange")
|
||||||
|
.field("start", &self.start)
|
||||||
|
.field("end", &self.end)
|
||||||
|
.finish()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// An range of physical memory frames, inclusive the upper bound.
|
||||||
|
#[derive(Clone, Copy, PartialEq, Eq)]
|
||||||
|
#[repr(C)]
|
||||||
|
pub struct PhysFrameRangeInclusive<S: PageSize = Size4KiB> {
|
||||||
|
/// The start of the range, inclusive.
|
||||||
|
pub start: PhysFrame<S>,
|
||||||
|
/// The start of the range, exclusive.
|
||||||
|
pub end: PhysFrame<S>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<S: PageSize> PhysFrameRangeInclusive<S> {
|
||||||
|
/// Returns whether the range contains no frames.
|
||||||
|
pub fn is_empty(&self) -> bool {
|
||||||
|
!(self.start <= self.end)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<S: PageSize> Iterator for PhysFrameRangeInclusive<S> {
|
||||||
|
type Item = PhysFrame<S>;
|
||||||
|
|
||||||
|
fn next(&mut self) -> Option<Self::Item> {
|
||||||
|
if self.start <= self.end {
|
||||||
|
let frame = self.start.clone();
|
||||||
|
self.start += 1;
|
||||||
|
Some(frame)
|
||||||
|
} else {
|
||||||
|
None
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<S: PageSize> fmt::Debug for PhysFrameRangeInclusive<S> {
|
||||||
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
||||||
|
f.debug_struct("PhysFrameRangeInclusive")
|
||||||
|
.field("start", &self.start)
|
||||||
|
.field("end", &self.end)
|
||||||
|
.finish()
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in New Issue