satp.rs 3.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116
  1. //! satp register
  2. /// satp register
  3. #[derive(Clone, Copy, Debug)]
  4. pub struct Satp {
  5. bits: usize,
  6. }
  7. impl Satp {
  8. /// Returns the contents of the register as raw bits
  9. #[inline]
  10. pub fn bits(&self) -> usize {
  11. self.bits
  12. }
  13. /// Current address-translation scheme
  14. #[inline]
  15. #[cfg(target_pointer_width = "32")]
  16. pub fn mode(&self) -> Mode {
  17. match self.bits & (1 << 31) != 0 {
  18. false => Mode::Bare,
  19. true => Mode::Sv32,
  20. }
  21. }
  22. /// Current address-translation scheme
  23. #[inline]
  24. #[cfg(target_pointer_width = "64")]
  25. pub fn mode(&self) -> Mode {
  26. match self.bits >> 60 {
  27. 0 => Mode::Bare,
  28. 8 => Mode::Sv39,
  29. 9 => Mode::Sv48,
  30. 10 => Mode::Sv57,
  31. 11 => Mode::Sv64,
  32. _ => unreachable!(),
  33. }
  34. }
  35. /// Address space identifier
  36. #[inline]
  37. #[cfg(target_pointer_width = "32")]
  38. pub fn asid(&self) -> usize {
  39. (self.bits >> 22) & 0x1FF // bits 22-30
  40. }
  41. /// Address space identifier
  42. #[inline]
  43. #[cfg(target_pointer_width = "64")]
  44. pub fn asid(&self) -> usize {
  45. self.bits >> 44 & 0xFFFF // bits 44-59
  46. }
  47. /// Physical page number
  48. #[inline]
  49. #[cfg(target_pointer_width = "32")]
  50. pub fn ppn(&self) -> usize {
  51. self.bits & 0x3F_FFFF // bits 0-21
  52. }
  53. /// Physical page number
  54. #[inline]
  55. #[cfg(target_pointer_width = "64")]
  56. pub fn ppn(&self) -> usize {
  57. self.bits & 0xFFF_FFFF_FFFF // bits 0-43
  58. }
  59. }
  60. /// 32-bit satp mode
  61. #[cfg(target_pointer_width = "32")]
  62. #[derive(Clone, Copy, Debug, Eq, PartialEq)]
  63. pub enum Mode {
  64. /// No translation or protection
  65. Bare = 0,
  66. /// Page-based 32-bit virtual addressing
  67. Sv32 = 1,
  68. }
  69. /// 64-bit satp mode
  70. #[cfg(target_pointer_width = "64")]
  71. #[derive(Clone, Copy, Debug, Eq, PartialEq)]
  72. pub enum Mode {
  73. /// No translation or protection
  74. Bare = 0,
  75. /// Page-based 39-bit virtual addressing
  76. Sv39 = 8,
  77. /// Page-based 48-bit virtual addressing
  78. Sv48 = 9,
  79. /// Page-based 57-bit virtual addressing
  80. Sv57 = 10,
  81. /// Page-based 64-bit virtual addressing
  82. Sv64 = 11,
  83. }
  84. read_csr_as!(Satp, 0x180);
  85. write_csr_as_usize!(0x180);
  86. /// Sets the register to corresponding page table mode, physical page number and address space id.
  87. #[inline]
  88. #[cfg(target_pointer_width = "32")]
  89. pub unsafe fn set(mode: Mode, asid: usize, ppn: usize) {
  90. assert_eq!(asid, asid & 0x1FF, "invalid value for asid");
  91. assert_eq!(ppn, ppn & 0x3F_FFFF, "invalid value for ppn");
  92. let bits = (mode as usize) << 31 | (asid << 22) | ppn;
  93. _write(bits);
  94. }
  95. /// Sets the register to corresponding page table mode, physical page number and address space id.
  96. #[inline]
  97. #[cfg(target_pointer_width = "64")]
  98. pub unsafe fn set(mode: Mode, asid: usize, ppn: usize) {
  99. assert_eq!(asid, asid & 0xFFFF, "invalid value for asid");
  100. assert_eq!(ppn, ppn & 0xFFF_FFFF_FFFF, "invalid value for ppn");
  101. let bits = (mode as usize) << 60 | (asid << 44) | ppn;
  102. _write(bits);
  103. }