|
@@ -1,13 +1,27 @@
|
|
|
use serde_device_tree::buildin::NodeSeq;
|
|
|
|
|
|
+use crate::riscv_spec::current_hartid;
|
|
|
+use crate::sbi::trap::expected_trap;
|
|
|
use crate::sbi::trap_stack::ROOT_STACK;
|
|
|
-pub struct HartExtensions([bool; Extension::COUNT]);
|
|
|
+
|
|
|
+pub struct HartFeatures {
|
|
|
+ extension: [bool; Extension::COUNT],
|
|
|
+ privileged_version: PrivilegedVersion,
|
|
|
+}
|
|
|
|
|
|
#[derive(Copy, Clone)]
|
|
|
pub enum Extension {
|
|
|
Sstc = 0,
|
|
|
}
|
|
|
|
|
|
+#[derive(Copy, Clone, PartialEq, Eq, PartialOrd, Ord)]
|
|
|
+pub enum PrivilegedVersion {
|
|
|
+ Unknown = 0,
|
|
|
+ Version1_10 = 1,
|
|
|
+ Version1_11 = 2,
|
|
|
+ Version1_12 = 3,
|
|
|
+}
|
|
|
+
|
|
|
impl Extension {
|
|
|
const COUNT: usize = 1;
|
|
|
const ITER: [Self; Extension::COUNT] = [Extension::Sstc];
|
|
@@ -28,7 +42,16 @@ pub fn hart_extension_probe(hart_id: usize, ext: Extension) -> bool {
|
|
|
unsafe {
|
|
|
ROOT_STACK
|
|
|
.get_mut(hart_id)
|
|
|
- .map(|x| x.hart_context().extensions.0[ext.index()])
|
|
|
+ .map(|x| x.hart_context().features.extension[ext.index()])
|
|
|
+ .unwrap()
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+pub fn hart_privileged_version(hart_id: usize) -> PrivilegedVersion {
|
|
|
+ unsafe {
|
|
|
+ ROOT_STACK
|
|
|
+ .get_mut(hart_id)
|
|
|
+ .map(|x| x.hart_context().features.privileged_version)
|
|
|
.unwrap()
|
|
|
}
|
|
|
}
|
|
@@ -56,11 +79,35 @@ pub fn init(cpus: &NodeSeq) {
|
|
|
unsafe {
|
|
|
ROOT_STACK
|
|
|
.get_mut(hart_id)
|
|
|
- .map(|stack| stack.hart_context().extensions = HartExtensions(hart_exts))
|
|
|
+ .map(|stack| stack.hart_context().features.extension = hart_exts)
|
|
|
.unwrap()
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
+pub fn privileged_version_detection() {
|
|
|
+ let mut current_priv_ver = PrivilegedVersion::Unknown;
|
|
|
+ {
|
|
|
+ const CSR_MCOUNTEREN: u64 = 0x306;
|
|
|
+ const CSR_MCOUNTINHIBIT: u64 = 0x320;
|
|
|
+ const CSR_MENVCFG: u64 = 0x30a;
|
|
|
+
|
|
|
+ if csr_test!(CSR_MCOUNTEREN) {
|
|
|
+ current_priv_ver = PrivilegedVersion::Version1_10;
|
|
|
+ if csr_test!(CSR_MCOUNTINHIBIT) {
|
|
|
+ current_priv_ver = PrivilegedVersion::Version1_11;
|
|
|
+ if csr_test!(CSR_MENVCFG) {
|
|
|
+ current_priv_ver = PrivilegedVersion::Version1_12;
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+ unsafe {
|
|
|
+ ROOT_STACK
|
|
|
+ .get_mut(current_hartid())
|
|
|
+ .map(|stack| stack.hart_context().features.privileged_version = current_priv_ver)
|
|
|
+ .unwrap()
|
|
|
+ }
|
|
|
+}
|
|
|
|
|
|
#[cfg(feature = "nemu")]
|
|
|
pub fn init(cpus: &NodeSeq) {
|
|
@@ -70,7 +117,7 @@ pub fn init(cpus: &NodeSeq) {
|
|
|
unsafe {
|
|
|
ROOT_STACK
|
|
|
.get_mut(hart_id)
|
|
|
- .map(|stack| stack.hart_context().extensions = HartExtensions(hart_exts))
|
|
|
+ .map(|stack| stack.hart_context().extensions = HartFeatures(hart_exts))
|
|
|
.unwrap()
|
|
|
}
|
|
|
}
|