extensions.rs 3.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126
  1. use serde_device_tree::buildin::NodeSeq;
  2. use crate::riscv_spec::current_hartid;
  3. use crate::sbi::trap_stack::ROOT_STACK;
  4. pub struct HartFeatures {
  5. extension: [bool; Extension::COUNT],
  6. privileged_version: PrivilegedVersion,
  7. }
  8. #[derive(Copy, Clone)]
  9. pub enum Extension {
  10. Sstc = 0,
  11. }
  12. #[derive(Copy, Clone, PartialEq, Eq, PartialOrd, Ord)]
  13. pub enum PrivilegedVersion {
  14. Unknown = 0,
  15. Version1_10 = 1,
  16. Version1_11 = 2,
  17. Version1_12 = 3,
  18. }
  19. impl Extension {
  20. const COUNT: usize = 1;
  21. const ITER: [Self; Extension::COUNT] = [Extension::Sstc];
  22. pub fn as_str(&self) -> &'static str {
  23. match self {
  24. Extension::Sstc => "sstc",
  25. }
  26. }
  27. #[inline]
  28. pub fn index(&self) -> usize {
  29. *self as usize
  30. }
  31. }
  32. pub fn hart_extension_probe(hart_id: usize, ext: Extension) -> bool {
  33. unsafe {
  34. ROOT_STACK
  35. .get_mut(hart_id)
  36. .map(|x| x.hart_context().features.extension[ext.index()])
  37. .unwrap()
  38. }
  39. }
  40. pub fn hart_privileged_version(hart_id: usize) -> PrivilegedVersion {
  41. unsafe {
  42. ROOT_STACK
  43. .get_mut(hart_id)
  44. .map(|x| x.hart_context().features.privileged_version)
  45. .unwrap()
  46. }
  47. }
  48. #[cfg(not(feature = "nemu"))]
  49. pub fn init(cpus: &NodeSeq) {
  50. use crate::dt::Cpu;
  51. for cpu_iter in cpus.iter() {
  52. let cpu = cpu_iter.deserialize::<Cpu>();
  53. let hart_id = cpu.reg.iter().next().unwrap().0.start;
  54. let mut hart_exts = [false; Extension::COUNT];
  55. if cpu.isa_extensions.is_some() {
  56. let isa = cpu.isa_extensions.unwrap();
  57. Extension::ITER.iter().for_each(|ext| {
  58. hart_exts[ext.index()] = isa.iter().any(|e| e == ext.as_str());
  59. });
  60. } else if cpu.isa.is_some() {
  61. let isa_iter = cpu.isa.unwrap();
  62. let isa = isa_iter.iter().next().unwrap_or_default();
  63. Extension::ITER.iter().for_each(|ext| {
  64. hart_exts[ext.index()] = isa.find(ext.as_str()).is_some();
  65. })
  66. }
  67. unsafe {
  68. ROOT_STACK
  69. .get_mut(hart_id)
  70. .map(|stack| stack.hart_context().features.extension = hart_exts)
  71. .unwrap()
  72. }
  73. }
  74. }
  75. pub fn privileged_version_detection() {
  76. let mut current_priv_ver = PrivilegedVersion::Unknown;
  77. {
  78. const CSR_MCOUNTEREN: u64 = 0x306;
  79. const CSR_MCOUNTINHIBIT: u64 = 0x320;
  80. const CSR_MENVCFG: u64 = 0x30a;
  81. if has_csr!(CSR_MCOUNTEREN) {
  82. current_priv_ver = PrivilegedVersion::Version1_10;
  83. if has_csr!(CSR_MCOUNTINHIBIT) {
  84. current_priv_ver = PrivilegedVersion::Version1_11;
  85. if has_csr!(CSR_MENVCFG) {
  86. current_priv_ver = PrivilegedVersion::Version1_12;
  87. }
  88. }
  89. }
  90. }
  91. unsafe {
  92. ROOT_STACK
  93. .get_mut(current_hartid())
  94. .map(|stack| stack.hart_context().features.privileged_version = current_priv_ver)
  95. .unwrap()
  96. }
  97. }
  98. #[cfg(feature = "nemu")]
  99. pub fn init(cpus: &NodeSeq) {
  100. for hart_id in 0..cpus.len() {
  101. let mut hart_exts = [false; Extension::COUNT];
  102. hart_exts[Extension::Sstc.index()] = true;
  103. unsafe {
  104. ROOT_STACK
  105. .get_mut(hart_id)
  106. .map(|stack| stack.hart_context().features = HartFeatures{
  107. extension: hart_exts,
  108. privileged_version: PrivilegedVersion::Version1_12,
  109. })
  110. .unwrap()
  111. }
  112. }
  113. }