mod.rs 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711
  1. use alloc::sync::Arc;
  2. use system_error::SystemError;
  3. use crate::{arch::MMArch, include::bindings::bindings::PAGE_OFFSET};
  4. use core::{
  5. cmp,
  6. fmt::Debug,
  7. intrinsics::unlikely,
  8. ops::{Add, AddAssign, Sub, SubAssign},
  9. ptr,
  10. sync::atomic::{AtomicBool, Ordering},
  11. };
  12. use self::{
  13. allocator::page_frame::{VirtPageFrame, VirtPageFrameIter},
  14. memblock::MemoryAreaAttr,
  15. page::round_up_to_page_size,
  16. ucontext::{AddressSpace, UserMapper},
  17. };
  18. pub mod allocator;
  19. pub mod c_adapter;
  20. pub mod early_ioremap;
  21. pub mod init;
  22. pub mod kernel_mapper;
  23. pub mod memblock;
  24. pub mod mmio_buddy;
  25. pub mod no_init;
  26. pub mod page;
  27. pub mod percpu;
  28. pub mod syscall;
  29. pub mod ucontext;
  30. /// 内核INIT进程的用户地址空间结构体(仅在process_init中初始化)
  31. static mut __INITIAL_PROCESS_ADDRESS_SPACE: Option<Arc<AddressSpace>> = None;
  32. /// 获取内核INIT进程的用户地址空间结构体
  33. #[allow(non_snake_case)]
  34. #[inline(always)]
  35. pub fn INITIAL_PROCESS_ADDRESS_SPACE() -> Arc<AddressSpace> {
  36. unsafe {
  37. return __INITIAL_PROCESS_ADDRESS_SPACE
  38. .as_ref()
  39. .expect("INITIAL_PROCESS_ADDRESS_SPACE is null")
  40. .clone();
  41. }
  42. }
  43. /// 设置内核INIT进程的用户地址空间结构体全局变量
  44. #[allow(non_snake_case)]
  45. pub unsafe fn set_INITIAL_PROCESS_ADDRESS_SPACE(address_space: Arc<AddressSpace>) {
  46. static INITIALIZED: AtomicBool = AtomicBool::new(false);
  47. if INITIALIZED
  48. .compare_exchange(false, true, Ordering::SeqCst, Ordering::Acquire)
  49. .is_err()
  50. {
  51. panic!("INITIAL_PROCESS_ADDRESS_SPACE is already initialized");
  52. }
  53. __INITIAL_PROCESS_ADDRESS_SPACE = Some(address_space);
  54. }
  55. /// @brief 将内核空间的虚拟地址转换为物理地址
  56. #[inline(always)]
  57. pub fn virt_2_phys(addr: usize) -> usize {
  58. addr - PAGE_OFFSET as usize
  59. }
  60. /// @brief 将物理地址转换为内核空间的虚拟地址
  61. #[inline(always)]
  62. pub fn phys_2_virt(addr: usize) -> usize {
  63. addr + PAGE_OFFSET as usize
  64. }
  65. #[derive(Clone, Copy, Debug, Eq, Ord, PartialEq, PartialOrd, Hash)]
  66. pub enum PageTableKind {
  67. /// 用户可访问的页表
  68. User,
  69. /// 内核页表
  70. Kernel,
  71. /// x86内存虚拟化中使用的EPT
  72. #[cfg(target_arch = "x86_64")]
  73. EPT,
  74. }
  75. /// 物理内存地址
  76. #[derive(Clone, Copy, Eq, Ord, PartialEq, PartialOrd, Hash)]
  77. #[repr(transparent)]
  78. pub struct PhysAddr(usize);
  79. impl PhysAddr {
  80. /// 最大物理地址
  81. pub const MAX: Self = PhysAddr(usize::MAX);
  82. #[inline(always)]
  83. pub const fn new(address: usize) -> Self {
  84. Self(address)
  85. }
  86. /// @brief 获取物理地址的值
  87. #[inline(always)]
  88. pub const fn data(&self) -> usize {
  89. self.0
  90. }
  91. /// @brief 将物理地址加上一个偏移量
  92. #[inline(always)]
  93. pub fn add(self, offset: usize) -> Self {
  94. Self(self.0 + offset)
  95. }
  96. /// @brief 判断物理地址是否按照指定要求对齐
  97. #[inline(always)]
  98. pub fn check_aligned(&self, align: usize) -> bool {
  99. return self.0 & (align - 1) == 0;
  100. }
  101. #[inline(always)]
  102. pub fn is_null(&self) -> bool {
  103. return self.0 == 0;
  104. }
  105. }
  106. impl Debug for PhysAddr {
  107. fn fmt(&self, f: &mut core::fmt::Formatter<'_>) -> core::fmt::Result {
  108. write!(f, "PhysAddr({:#x})", self.0)
  109. }
  110. }
  111. impl core::ops::Add<usize> for PhysAddr {
  112. type Output = Self;
  113. #[inline(always)]
  114. fn add(self, rhs: usize) -> Self::Output {
  115. return Self(self.0 + rhs);
  116. }
  117. }
  118. impl core::ops::AddAssign<usize> for PhysAddr {
  119. #[inline(always)]
  120. fn add_assign(&mut self, rhs: usize) {
  121. self.0 += rhs;
  122. }
  123. }
  124. impl core::ops::Add<PhysAddr> for PhysAddr {
  125. type Output = Self;
  126. #[inline(always)]
  127. fn add(self, rhs: PhysAddr) -> Self::Output {
  128. return Self(self.0 + rhs.0);
  129. }
  130. }
  131. impl core::ops::AddAssign<PhysAddr> for PhysAddr {
  132. #[inline(always)]
  133. fn add_assign(&mut self, rhs: PhysAddr) {
  134. self.0 += rhs.0;
  135. }
  136. }
  137. impl core::ops::BitOrAssign<usize> for PhysAddr {
  138. #[inline(always)]
  139. fn bitor_assign(&mut self, rhs: usize) {
  140. self.0 |= rhs;
  141. }
  142. }
  143. impl core::ops::BitOrAssign<PhysAddr> for PhysAddr {
  144. #[inline(always)]
  145. fn bitor_assign(&mut self, rhs: PhysAddr) {
  146. self.0 |= rhs.0;
  147. }
  148. }
  149. impl core::ops::Sub<usize> for PhysAddr {
  150. type Output = Self;
  151. #[inline(always)]
  152. fn sub(self, rhs: usize) -> Self::Output {
  153. return Self(self.0 - rhs);
  154. }
  155. }
  156. impl core::ops::SubAssign<usize> for PhysAddr {
  157. #[inline(always)]
  158. fn sub_assign(&mut self, rhs: usize) {
  159. self.0 -= rhs;
  160. }
  161. }
  162. impl core::ops::Sub<PhysAddr> for PhysAddr {
  163. type Output = usize;
  164. #[inline(always)]
  165. fn sub(self, rhs: PhysAddr) -> Self::Output {
  166. return self.0 - rhs.0;
  167. }
  168. }
  169. impl core::ops::SubAssign<PhysAddr> for PhysAddr {
  170. #[inline(always)]
  171. fn sub_assign(&mut self, rhs: PhysAddr) {
  172. self.0 -= rhs.0;
  173. }
  174. }
  175. /// 虚拟内存地址
  176. #[derive(Clone, Copy, Eq, Ord, PartialEq, PartialOrd, Hash)]
  177. #[repr(transparent)]
  178. pub struct VirtAddr(usize);
  179. impl VirtAddr {
  180. #[inline(always)]
  181. pub const fn new(address: usize) -> Self {
  182. return Self(address);
  183. }
  184. /// @brief 获取虚拟地址的值
  185. #[inline(always)]
  186. pub const fn data(&self) -> usize {
  187. return self.0;
  188. }
  189. /// @brief 判断虚拟地址的类型
  190. #[inline(always)]
  191. pub fn kind(&self) -> PageTableKind {
  192. if self.check_user() {
  193. return PageTableKind::User;
  194. } else {
  195. return PageTableKind::Kernel;
  196. }
  197. }
  198. /// @brief 判断虚拟地址是否按照指定要求对齐
  199. #[inline(always)]
  200. pub fn check_aligned(&self, align: usize) -> bool {
  201. return self.0 & (align - 1) == 0;
  202. }
  203. /// @brief 判断虚拟地址是否在用户空间
  204. #[inline(always)]
  205. pub fn check_user(&self) -> bool {
  206. if self < &MMArch::USER_END_VADDR {
  207. return true;
  208. } else {
  209. return false;
  210. }
  211. }
  212. #[inline(always)]
  213. pub fn as_ptr<T>(self) -> *mut T {
  214. return self.0 as *mut T;
  215. }
  216. #[inline(always)]
  217. pub fn is_null(&self) -> bool {
  218. return self.0 == 0;
  219. }
  220. }
  221. impl Add<VirtAddr> for VirtAddr {
  222. type Output = Self;
  223. #[inline(always)]
  224. fn add(self, rhs: VirtAddr) -> Self::Output {
  225. return Self(self.0 + rhs.0);
  226. }
  227. }
  228. impl Add<usize> for VirtAddr {
  229. type Output = Self;
  230. #[inline(always)]
  231. fn add(self, rhs: usize) -> Self::Output {
  232. return Self(self.0 + rhs);
  233. }
  234. }
  235. impl Sub<VirtAddr> for VirtAddr {
  236. type Output = usize;
  237. #[inline(always)]
  238. fn sub(self, rhs: VirtAddr) -> Self::Output {
  239. return self.0 - rhs.0;
  240. }
  241. }
  242. impl Sub<usize> for VirtAddr {
  243. type Output = Self;
  244. #[inline(always)]
  245. fn sub(self, rhs: usize) -> Self::Output {
  246. return Self(self.0 - rhs);
  247. }
  248. }
  249. impl AddAssign<usize> for VirtAddr {
  250. #[inline(always)]
  251. fn add_assign(&mut self, rhs: usize) {
  252. self.0 += rhs;
  253. }
  254. }
  255. impl AddAssign<VirtAddr> for VirtAddr {
  256. #[inline(always)]
  257. fn add_assign(&mut self, rhs: VirtAddr) {
  258. self.0 += rhs.0;
  259. }
  260. }
  261. impl SubAssign<usize> for VirtAddr {
  262. #[inline(always)]
  263. fn sub_assign(&mut self, rhs: usize) {
  264. self.0 -= rhs;
  265. }
  266. }
  267. impl SubAssign<VirtAddr> for VirtAddr {
  268. #[inline(always)]
  269. fn sub_assign(&mut self, rhs: VirtAddr) {
  270. self.0 -= rhs.0;
  271. }
  272. }
  273. impl Debug for VirtAddr {
  274. fn fmt(&self, f: &mut core::fmt::Formatter<'_>) -> core::fmt::Result {
  275. write!(f, "VirtAddr({:#x})", self.0)
  276. }
  277. }
  278. /// @brief 物理内存区域
  279. #[derive(Clone, Copy, Debug)]
  280. pub struct PhysMemoryArea {
  281. /// 物理基地址
  282. pub base: PhysAddr,
  283. /// 该区域的物理内存大小
  284. pub size: usize,
  285. pub flags: MemoryAreaAttr,
  286. }
  287. impl PhysMemoryArea {
  288. pub const DEFAULT: Self = Self {
  289. base: PhysAddr::new(0),
  290. size: 0,
  291. flags: MemoryAreaAttr::empty(),
  292. };
  293. pub fn new(base: PhysAddr, size: usize, flags: MemoryAreaAttr) -> Self {
  294. Self { base, size, flags }
  295. }
  296. /// 返回向上页面对齐的区域起始物理地址
  297. pub fn area_base_aligned(&self) -> PhysAddr {
  298. return PhysAddr::new(
  299. (self.base.data() + (MMArch::PAGE_SIZE - 1)) & !(MMArch::PAGE_SIZE - 1),
  300. );
  301. }
  302. /// 返回向下页面对齐的区域截止物理地址
  303. pub fn area_end_aligned(&self) -> PhysAddr {
  304. return PhysAddr::new((self.base.data() + self.size) & !(MMArch::PAGE_SIZE - 1));
  305. }
  306. }
  307. impl Default for PhysMemoryArea {
  308. fn default() -> Self {
  309. return Self::DEFAULT;
  310. }
  311. }
  312. pub trait MemoryManagementArch: Clone + Copy + Debug {
  313. /// 页面大小的shift(假如页面4K,那么这个值就是12,因为2^12=4096)
  314. const PAGE_SHIFT: usize;
  315. /// 每个页表的页表项数目。(以2^n次幂来表示)假如有512个页表项,那么这个值就是9
  316. const PAGE_ENTRY_SHIFT: usize;
  317. /// 页表层级数量
  318. const PAGE_LEVELS: usize;
  319. /// 页表项的有效位的index(假如页表项的第0-51位有效,那么这个值就是52)
  320. const ENTRY_ADDRESS_SHIFT: usize;
  321. /// 页面的页表项的默认值
  322. const ENTRY_FLAG_DEFAULT_PAGE: usize;
  323. /// 页表的页表项的默认值
  324. const ENTRY_FLAG_DEFAULT_TABLE: usize;
  325. /// 页表项的present位被置位之后的值
  326. const ENTRY_FLAG_PRESENT: usize;
  327. /// 页表项为read only时的值
  328. const ENTRY_FLAG_READONLY: usize;
  329. /// 页表项为可读写状态的值
  330. const ENTRY_FLAG_READWRITE: usize;
  331. /// 页面项标记页面为user page的值
  332. const ENTRY_FLAG_USER: usize;
  333. /// 页面项标记页面为write through的值
  334. const ENTRY_FLAG_WRITE_THROUGH: usize;
  335. /// 页面项标记页面为cache disable的值
  336. const ENTRY_FLAG_CACHE_DISABLE: usize;
  337. /// 标记当前页面不可执行的标志位(Execute disable)(也就是说,不能从这段内存里面获取处理器指令)
  338. const ENTRY_FLAG_NO_EXEC: usize;
  339. /// 标记当前页面可执行的标志位(Execute enable)
  340. const ENTRY_FLAG_EXEC: usize;
  341. /// 当该位为1时,标明这是一个脏页
  342. const ENTRY_FLAG_DIRTY: usize;
  343. /// 当该位为1时,代表这个页面被处理器访问过
  344. const ENTRY_FLAG_ACCESSED: usize;
  345. /// 虚拟地址与物理地址的偏移量
  346. const PHYS_OFFSET: usize;
  347. /// 内核在链接时被链接到的偏移量
  348. const KERNEL_LINK_OFFSET: usize;
  349. const KERNEL_VIRT_START: usize = Self::PHYS_OFFSET + Self::KERNEL_LINK_OFFSET;
  350. /// 每个页面的大小
  351. const PAGE_SIZE: usize = 1 << Self::PAGE_SHIFT;
  352. /// 通过这个mask,获取地址的页内偏移量
  353. const PAGE_OFFSET_MASK: usize = Self::PAGE_SIZE - 1;
  354. /// 通过这个mask,获取页的首地址
  355. const PAGE_MASK: usize = !(Self::PAGE_OFFSET_MASK);
  356. /// 页表项的地址、数据部分的shift。
  357. /// 打个比方,如果这个值为52,那么意味着页表项的[0, 52)位,用于表示地址以及其他的标志位
  358. const PAGE_ADDRESS_SHIFT: usize = Self::PAGE_LEVELS * Self::PAGE_ENTRY_SHIFT + Self::PAGE_SHIFT;
  359. /// 最大的虚拟地址(对于不同的架构,由于上述PAGE_ADDRESS_SHIFT可能包括了reserved bits, 事实上能表示的虚拟地址应该比这个值要小)
  360. const PAGE_ADDRESS_SIZE: usize = 1 << Self::PAGE_ADDRESS_SHIFT;
  361. /// 页表项的值与这个常量进行与运算,得到的结果是所填写的物理地址
  362. const PAGE_ADDRESS_MASK: usize = Self::PAGE_ADDRESS_SIZE - Self::PAGE_SIZE;
  363. /// 每个页表项的大小
  364. const PAGE_ENTRY_SIZE: usize = 1 << (Self::PAGE_SHIFT - Self::PAGE_ENTRY_SHIFT);
  365. /// 每个页表的页表项数目
  366. const PAGE_ENTRY_NUM: usize = 1 << Self::PAGE_ENTRY_SHIFT;
  367. /// 该字段用于根据虚拟地址,获取该虚拟地址在对应的页表中是第几个页表项
  368. const PAGE_ENTRY_MASK: usize = Self::PAGE_ENTRY_NUM - 1;
  369. const PAGE_NEGATIVE_MASK: usize = !((Self::PAGE_ADDRESS_SIZE) - 1);
  370. const ENTRY_ADDRESS_SIZE: usize = 1 << Self::ENTRY_ADDRESS_SHIFT;
  371. /// 该mask用于获取页表项中地址字段
  372. const ENTRY_ADDRESS_MASK: usize = Self::ENTRY_ADDRESS_SIZE - Self::PAGE_SIZE;
  373. /// 这个mask用于获取页表项中的flags
  374. const ENTRY_FLAGS_MASK: usize = !Self::ENTRY_ADDRESS_MASK;
  375. /// 用户空间的最高地址
  376. const USER_END_VADDR: VirtAddr;
  377. /// 用户堆的起始地址
  378. const USER_BRK_START: VirtAddr;
  379. /// 用户栈起始地址(向下生长,不包含该值)
  380. const USER_STACK_START: VirtAddr;
  381. /// 内核的固定映射区的起始地址
  382. const FIXMAP_START_VADDR: VirtAddr;
  383. /// 内核的固定映射区的大小
  384. const FIXMAP_SIZE: usize;
  385. /// 内核的固定映射区的结束地址
  386. const FIXMAP_END_VADDR: VirtAddr =
  387. VirtAddr::new(Self::FIXMAP_START_VADDR.data() + Self::FIXMAP_SIZE);
  388. /// @brief 用于初始化内存管理模块与架构相关的信息。
  389. /// 该函数应调用其他模块的接口,把可用内存区域添加到memblock,提供给BumpAllocator使用
  390. unsafe fn init();
  391. /// @brief 读取指定虚拟地址的值,并假设它是类型T的指针
  392. #[inline(always)]
  393. unsafe fn read<T>(address: VirtAddr) -> T {
  394. return ptr::read(address.data() as *const T);
  395. }
  396. /// @brief 将value写入到指定的虚拟地址
  397. #[inline(always)]
  398. unsafe fn write<T>(address: VirtAddr, value: T) {
  399. ptr::write(address.data() as *mut T, value);
  400. }
  401. #[inline(always)]
  402. unsafe fn write_bytes(address: VirtAddr, value: u8, count: usize) {
  403. ptr::write_bytes(address.data() as *mut u8, value, count);
  404. }
  405. /// @brief 刷新TLB中,关于指定虚拟地址的条目
  406. unsafe fn invalidate_page(address: VirtAddr);
  407. /// @brief 刷新TLB中,所有的条目
  408. unsafe fn invalidate_all();
  409. /// @brief 获取顶级页表的物理地址
  410. unsafe fn table(table_kind: PageTableKind) -> PhysAddr;
  411. /// @brief 设置顶级页表的物理地址到处理器中
  412. unsafe fn set_table(table_kind: PageTableKind, table: PhysAddr);
  413. /// @brief 将物理地址转换为虚拟地址.
  414. ///
  415. /// @param phys 物理地址
  416. ///
  417. /// @return 转换后的虚拟地址。如果转换失败,返回None
  418. #[inline(always)]
  419. unsafe fn phys_2_virt(phys: PhysAddr) -> Option<VirtAddr> {
  420. if let Some(vaddr) = phys.data().checked_add(Self::PHYS_OFFSET) {
  421. return Some(VirtAddr::new(vaddr));
  422. } else {
  423. return None;
  424. }
  425. }
  426. /// 将虚拟地址转换为物理地址
  427. ///
  428. /// ## 参数
  429. ///
  430. /// - `virt` 虚拟地址
  431. ///
  432. /// ## 返回值
  433. ///
  434. /// 转换后的物理地址。如果转换失败,返回None
  435. #[inline(always)]
  436. unsafe fn virt_2_phys(virt: VirtAddr) -> Option<PhysAddr> {
  437. if let Some(paddr) = virt.data().checked_sub(Self::PHYS_OFFSET) {
  438. return Some(PhysAddr::new(paddr));
  439. } else {
  440. return None;
  441. }
  442. }
  443. /// @brief 判断指定的虚拟地址是否正确(符合规范)
  444. fn virt_is_valid(virt: VirtAddr) -> bool;
  445. /// 获取内存管理初始化时,创建的第一个内核页表的地址
  446. fn initial_page_table() -> PhysAddr;
  447. /// 初始化新的usermapper,为用户进程创建页表
  448. fn setup_new_usermapper() -> Result<UserMapper, SystemError>;
  449. /// 创建页表项
  450. ///
  451. /// 这是一个低阶api,用于根据物理地址以及指定好的pageflags,创建页表项
  452. ///
  453. /// ## 参数
  454. ///
  455. /// - `paddr` 物理地址
  456. /// - `page_flags` 页表项的flags
  457. ///
  458. /// ## 返回值
  459. ///
  460. /// 页表项的值
  461. fn make_entry(paddr: PhysAddr, page_flags: usize) -> usize;
  462. }
  463. /// @brief 虚拟地址范围
  464. /// 该结构体用于表示一个虚拟地址范围,包括起始地址与大小
  465. ///
  466. /// 请注意与VMA进行区分,该结构体被VMA所包含
  467. #[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)]
  468. pub struct VirtRegion {
  469. start: VirtAddr,
  470. size: usize,
  471. }
  472. #[allow(dead_code)]
  473. impl VirtRegion {
  474. /// # 创建一个新的虚拟地址范围
  475. pub fn new(start: VirtAddr, size: usize) -> Self {
  476. VirtRegion { start, size }
  477. }
  478. /// 获取虚拟地址范围的起始地址
  479. #[inline(always)]
  480. pub fn start(&self) -> VirtAddr {
  481. self.start
  482. }
  483. /// 获取虚拟地址范围的截止地址(不包括返回的地址)
  484. #[inline(always)]
  485. pub fn end(&self) -> VirtAddr {
  486. return self.start().add(self.size);
  487. }
  488. /// # Create a new VirtRegion from a range [start, end)
  489. ///
  490. /// If end <= start, return None
  491. pub fn between(start: VirtAddr, end: VirtAddr) -> Option<Self> {
  492. if unlikely(end.data() <= start.data()) {
  493. return None;
  494. }
  495. let size = end.data() - start.data();
  496. return Some(VirtRegion::new(start, size));
  497. }
  498. /// # 取两个虚拟地址范围的交集
  499. ///
  500. /// 如果两个虚拟地址范围没有交集,返回None
  501. pub fn intersect(&self, other: &VirtRegion) -> Option<VirtRegion> {
  502. let start = self.start.max(other.start);
  503. let end = self.end().min(other.end());
  504. return VirtRegion::between(start, end);
  505. }
  506. /// 设置虚拟地址范围的起始地址
  507. #[inline(always)]
  508. pub fn set_start(&mut self, start: VirtAddr) {
  509. self.start = start;
  510. }
  511. #[inline(always)]
  512. pub fn size(&self) -> usize {
  513. self.size
  514. }
  515. /// 设置虚拟地址范围的大小
  516. #[inline(always)]
  517. pub fn set_size(&mut self, size: usize) {
  518. self.size = size;
  519. }
  520. /// 判断虚拟地址范围是否为空
  521. #[inline(always)]
  522. pub fn is_empty(&self) -> bool {
  523. self.size == 0
  524. }
  525. /// 将虚拟地址区域的大小向上对齐到页大小
  526. #[inline(always)]
  527. pub fn round_up_size_to_page(self) -> Self {
  528. return VirtRegion::new(self.start, round_up_to_page_size(self.size));
  529. }
  530. /// 判断两个虚拟地址范围是否由于具有交集而导致冲突
  531. #[inline(always)]
  532. pub fn collide(&self, other: &VirtRegion) -> bool {
  533. return self.intersect(other).is_some();
  534. }
  535. pub fn iter_pages(&self) -> VirtPageFrameIter {
  536. return VirtPageFrame::iter_range(
  537. VirtPageFrame::new(self.start),
  538. VirtPageFrame::new(self.end()),
  539. );
  540. }
  541. /// 获取[self.start(), region.start())的虚拟地址范围
  542. ///
  543. /// 如果self.start() >= region.start(),返回None
  544. pub fn before(self, region: &VirtRegion) -> Option<Self> {
  545. return Self::between(self.start(), region.start());
  546. }
  547. /// 获取[region.end(),self.end())的虚拟地址范围
  548. ///
  549. /// 如果 self.end() >= region.end() ,返回None
  550. pub fn after(self, region: &VirtRegion) -> Option<Self> {
  551. // if self.end() > region.end() none
  552. return Self::between(region.end(), self.end());
  553. }
  554. /// 把当前虚拟地址范围内的某个虚拟地址,转换为另一个虚拟地址范围内的虚拟地址
  555. ///
  556. /// 如果vaddr不在当前虚拟地址范围内,返回None
  557. ///
  558. /// 如果vaddr在当前虚拟地址范围内,返回vaddr在new_base中的虚拟地址
  559. pub fn rebase(self, vaddr: VirtAddr, new_base: &VirtRegion) -> Option<VirtAddr> {
  560. if !self.contains(vaddr) {
  561. return None;
  562. }
  563. let offset = vaddr.data() - self.start().data();
  564. let new_start = new_base.start().data() + offset;
  565. return Some(VirtAddr::new(new_start));
  566. }
  567. /// 判断虚拟地址范围是否包含指定的虚拟地址
  568. pub fn contains(&self, addr: VirtAddr) -> bool {
  569. return self.start() <= addr && addr < self.end();
  570. }
  571. /// 创建当前虚拟地址范围的页面迭代器
  572. pub fn pages(&self) -> VirtPageFrameIter {
  573. return VirtPageFrame::iter_range(
  574. VirtPageFrame::new(self.start()),
  575. VirtPageFrame::new(self.end()),
  576. );
  577. }
  578. }
  579. impl PartialOrd for VirtRegion {
  580. fn partial_cmp(&self, other: &Self) -> Option<cmp::Ordering> {
  581. return self.start.partial_cmp(&other.start);
  582. }
  583. }
  584. impl Ord for VirtRegion {
  585. fn cmp(&self, other: &Self) -> cmp::Ordering {
  586. return self.start.cmp(&other.start);
  587. }
  588. }
  589. /// ## 判断虚拟地址是否超出了用户空间
  590. ///
  591. /// 如果虚拟地址超出了用户空间,返回Err(SystemError::EFAULT).
  592. /// 如果end < start,返回Err(SystemError::EOVERFLOW)
  593. ///
  594. /// 否则返回Ok(())
  595. pub fn verify_area(addr: VirtAddr, size: usize) -> Result<(), SystemError> {
  596. let end = addr.add(size);
  597. if unlikely(end.data() < addr.data()) {
  598. return Err(SystemError::EOVERFLOW);
  599. }
  600. if !addr.check_user() || !end.check_user() {
  601. return Err(SystemError::EFAULT);
  602. }
  603. return Ok(());
  604. }