mm.rs 9.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339
  1. use core::marker::PhantomData;
  2. use core::ptr::NonNull;
  3. use eonix_hal_traits::mm::Memory;
  4. use eonix_mm::address::{Addr as _, AddrOps, PAddr, PRange, PhysAccess, VAddr};
  5. use eonix_mm::page_table::{
  6. PageAttribute, PageTable, PageTableLevel, PagingMode, RawAttribute,
  7. RawPageTable, TableAttribute, PTE,
  8. };
  9. use eonix_mm::paging::{BasicFolio, Folio, PageAccess, PageBlock, PFN};
  10. use eonix_sync_base::LazyLock;
  11. use fdt::Fdt;
  12. use riscv::asm::{sfence_vma, sfence_vma_all};
  13. use riscv::register::satp;
  14. use super::config::mm::{PHYS_MAP_VIRT, ROOT_PAGE_TABLE_PFN};
  15. use super::fdt::{FdtExt, FDT};
  16. use crate::arch::riscv64::config::mm::KIMAGE_OFFSET;
  17. use crate::extern_symbol_addr;
  18. use crate::mm::BasicPageAlloc;
  19. const PAGE_TABLE_BASE: PFN = PFN::from_val(ROOT_PAGE_TABLE_PFN);
  20. pub const GLOBAL_PAGE_TABLE: BasicFolio = BasicFolio::new(PAGE_TABLE_BASE, 0);
  21. pub const PA_V: u64 = 0b1 << 0;
  22. pub const PA_R: u64 = 0b1 << 1;
  23. pub const PA_W: u64 = 0b1 << 2;
  24. pub const PA_X: u64 = 0b1 << 3;
  25. pub const PA_U: u64 = 0b1 << 4;
  26. pub const PA_G: u64 = 0b1 << 5;
  27. pub const PA_A: u64 = 0b1 << 6;
  28. pub const PA_D: u64 = 0b1 << 7;
  29. // in RSW
  30. pub const PA_COW: u64 = 0b1 << 8;
  31. pub const PA_MMAP: u64 = 0b1 << 9;
  32. #[allow(dead_code)]
  33. pub const PA_SHIFT: u64 = 10;
  34. // Bit 0-9 (V, R, W, X, U, G, A, D, RSW)
  35. #[allow(dead_code)]
  36. pub const PA_FLAGS_MASK: u64 = 0x3FF; // 0b11_1111_1111
  37. #[repr(transparent)]
  38. #[derive(Clone, Copy)]
  39. pub struct PTE64(pub u64);
  40. #[derive(Clone, Copy)]
  41. pub struct PageAttribute64(u64);
  42. pub struct RawPageTableSv48<'a>(NonNull<PTE64>, PhantomData<&'a ()>);
  43. pub struct PagingModeSv48;
  44. pub struct ArchPhysAccess;
  45. pub struct ArchMemory;
  46. #[derive(Clone)]
  47. pub struct PageAccessImpl;
  48. impl PTE for PTE64 {
  49. type Attr = PageAttribute64;
  50. fn set(&mut self, pfn: PFN, attr: Self::Attr) {
  51. self.0 = (usize::from(pfn) << PA_SHIFT) as u64 | attr.0;
  52. }
  53. fn get(&self) -> (PFN, Self::Attr) {
  54. let pfn = PFN::from(self.0 as usize >> PA_SHIFT);
  55. let attr = PageAttribute64(self.0 & PA_FLAGS_MASK);
  56. (pfn, attr)
  57. }
  58. }
  59. impl PagingMode for PagingModeSv48 {
  60. type Entry = PTE64;
  61. type RawTable<'a> = RawPageTableSv48<'a>;
  62. const LEVELS: &'static [PageTableLevel] = &[
  63. PageTableLevel::new(39, 9),
  64. PageTableLevel::new(30, 9),
  65. PageTableLevel::new(21, 9),
  66. PageTableLevel::new(12, 9),
  67. ];
  68. }
  69. pub type ArchPagingMode = PagingModeSv48;
  70. unsafe impl Send for RawPageTableSv48<'_> {}
  71. impl<'a> RawPageTable<'a> for RawPageTableSv48<'a> {
  72. type Entry = PTE64;
  73. fn index(&self, index: u16) -> &'a Self::Entry {
  74. unsafe { self.0.add(index as usize).as_ref() }
  75. }
  76. fn index_mut(&mut self, index: u16) -> &'a mut Self::Entry {
  77. unsafe { self.0.add(index as usize).as_mut() }
  78. }
  79. unsafe fn from_ptr(ptr: NonNull<PageBlock>) -> Self {
  80. Self(ptr.cast(), PhantomData)
  81. }
  82. }
  83. impl RawAttribute for PageAttribute64 {
  84. fn null() -> Self {
  85. Self(0)
  86. }
  87. fn as_table_attr(self) -> Option<TableAttribute> {
  88. let mut table_attr = TableAttribute::empty();
  89. if self.0 & PA_V != 0 {
  90. table_attr |= TableAttribute::PRESENT;
  91. }
  92. if table_attr.contains(TableAttribute::PRESENT)
  93. && self.0 & (PA_R | PA_W | PA_X) != 0
  94. {
  95. return None;
  96. }
  97. if self.0 & PA_G != 0 {
  98. table_attr |= TableAttribute::GLOBAL;
  99. }
  100. if self.0 & PA_U != 0 {
  101. table_attr |= TableAttribute::USER;
  102. }
  103. if self.0 & PA_A != 0 {
  104. table_attr |= TableAttribute::ACCESSED;
  105. }
  106. Some(table_attr)
  107. }
  108. fn as_page_attr(self) -> Option<PageAttribute> {
  109. let mut page_attr = PageAttribute::empty();
  110. if self.0 & PA_V != 0 {
  111. page_attr |= PageAttribute::PRESENT;
  112. }
  113. if page_attr.contains(PageAttribute::PRESENT)
  114. && (self.0 & (PA_R | PA_W | PA_X) == 0)
  115. {
  116. return None;
  117. }
  118. if self.0 & PA_R != 0 {
  119. page_attr |= PageAttribute::READ;
  120. }
  121. if self.0 & PA_W != 0 {
  122. page_attr |= PageAttribute::WRITE;
  123. }
  124. if self.0 & PA_X != 0 {
  125. page_attr |= PageAttribute::EXECUTE;
  126. }
  127. if self.0 & PA_U != 0 {
  128. page_attr |= PageAttribute::USER;
  129. }
  130. if self.0 & PA_A != 0 {
  131. page_attr |= PageAttribute::ACCESSED;
  132. }
  133. if self.0 & PA_D != 0 {
  134. page_attr |= PageAttribute::DIRTY;
  135. }
  136. if self.0 & PA_G != 0 {
  137. page_attr |= PageAttribute::GLOBAL;
  138. }
  139. if self.0 & PA_COW != 0 {
  140. page_attr |= PageAttribute::COPY_ON_WRITE;
  141. }
  142. if self.0 & PA_MMAP != 0 {
  143. page_attr |= PageAttribute::MAPPED;
  144. }
  145. /*if self.0 & PA_ANON != 0 {
  146. page_attr |= PageAttribute::ANONYMOUS;
  147. }*/
  148. Some(page_attr)
  149. }
  150. }
  151. impl From<PageAttribute> for PageAttribute64 {
  152. fn from(page_attr: PageAttribute) -> Self {
  153. let mut raw_attr = 0;
  154. for attr in page_attr.iter() {
  155. match attr {
  156. PageAttribute::PRESENT => raw_attr |= PA_V,
  157. PageAttribute::READ => raw_attr |= PA_R,
  158. PageAttribute::WRITE => raw_attr |= PA_W,
  159. PageAttribute::EXECUTE => raw_attr |= PA_X,
  160. PageAttribute::USER => raw_attr |= PA_U,
  161. PageAttribute::ACCESSED => raw_attr |= PA_A,
  162. PageAttribute::DIRTY => raw_attr |= PA_D,
  163. PageAttribute::GLOBAL => raw_attr |= PA_G,
  164. PageAttribute::COPY_ON_WRITE => raw_attr |= PA_COW,
  165. PageAttribute::MAPPED => raw_attr |= PA_MMAP,
  166. PageAttribute::ANONYMOUS => {}
  167. _ => unreachable!("Invalid page attribute"),
  168. }
  169. }
  170. Self(raw_attr)
  171. }
  172. }
  173. impl From<TableAttribute> for PageAttribute64 {
  174. fn from(table_attr: TableAttribute) -> Self {
  175. let mut raw_attr = 0;
  176. for attr in table_attr.iter() {
  177. match attr {
  178. TableAttribute::PRESENT => raw_attr |= PA_V,
  179. TableAttribute::GLOBAL => raw_attr |= PA_G,
  180. TableAttribute::USER | TableAttribute::ACCESSED => {}
  181. _ => unreachable!("Invalid table attribute"),
  182. }
  183. }
  184. Self(raw_attr)
  185. }
  186. }
  187. impl ArchPhysAccess {
  188. const PHYS_OFFSET: usize = PHYS_MAP_VIRT;
  189. }
  190. impl PhysAccess for ArchPhysAccess {
  191. unsafe fn as_ptr<T>(paddr: PAddr) -> NonNull<T> {
  192. let alignment: usize = align_of::<T>();
  193. assert!(paddr.addr() % alignment == 0, "Alignment error");
  194. unsafe {
  195. // SAFETY: We can assume that we'll never have `self.addr()` equals
  196. // to `-PHYS_OFFSET`. Otherwise, the kernel might be broken.
  197. NonNull::new_unchecked((Self::PHYS_OFFSET + paddr.addr()) as *mut T)
  198. }
  199. }
  200. unsafe fn from_ptr<T>(ptr: NonNull<T>) -> PAddr {
  201. let addr = ptr.addr().get();
  202. assert!(addr % align_of::<T>() == 0, "Alignment error");
  203. assert!(
  204. addr >= Self::PHYS_OFFSET,
  205. "Address is not a valid physical address"
  206. );
  207. PAddr::from_val(addr - Self::PHYS_OFFSET)
  208. }
  209. }
  210. impl PageAccess for PageAccessImpl {
  211. unsafe fn get_ptr_for_pfn(&self, pfn: PFN) -> NonNull<PageBlock> {
  212. unsafe { ArchPhysAccess::as_ptr(PAddr::from(pfn)) }
  213. }
  214. }
  215. impl Memory for ArchMemory {
  216. fn present_ram() -> impl Iterator<Item = PRange> {
  217. FDT.present_ram()
  218. }
  219. fn free_ram() -> impl Iterator<Item = PRange> {
  220. FDT.present_ram().free_ram()
  221. }
  222. }
  223. pub type DefaultPagingMode = PagingModeSv48;
  224. pub trait PresentRam: Iterator<Item = PRange> {}
  225. pub trait FreeRam: PresentRam {
  226. fn free_ram(self) -> impl Iterator<Item = PRange>;
  227. }
  228. impl<T> FreeRam for T
  229. where
  230. T: PresentRam,
  231. {
  232. fn free_ram(self) -> impl Iterator<Item = PRange> {
  233. let kernel_end = extern_symbol_addr!(__kernel_end) - KIMAGE_OFFSET;
  234. let kernel_end = PAddr::from(kernel_end).ceil();
  235. let paddr_after_kimage_aligned = kernel_end.ceil_to(0x200000);
  236. core::iter::once(PRange::new(kernel_end, paddr_after_kimage_aligned))
  237. .chain(
  238. self.filter(move |range| {
  239. range.end() > paddr_after_kimage_aligned
  240. })
  241. .map(move |range| {
  242. if range.start() < paddr_after_kimage_aligned {
  243. let (_, right) =
  244. range.split_at(paddr_after_kimage_aligned);
  245. right
  246. } else {
  247. range
  248. }
  249. }),
  250. )
  251. }
  252. }
  253. #[inline(always)]
  254. pub fn flush_tlb(vaddr: usize) {
  255. sfence_vma(0, vaddr);
  256. }
  257. #[inline(always)]
  258. pub fn flush_tlb_all() {
  259. sfence_vma_all();
  260. }
  261. #[inline(always)]
  262. pub fn get_root_page_table_pfn() -> PFN {
  263. let satp_val = satp::read();
  264. let ppn = satp_val.ppn();
  265. PFN::from(ppn)
  266. }
  267. #[inline(always)]
  268. pub fn set_root_page_table_pfn(pfn: PFN) {
  269. unsafe { satp::set(satp::Mode::Sv48, 0, usize::from(pfn)) };
  270. sfence_vma_all();
  271. }