1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
use bitflags::bitflags;
#[derive(Debug)]
pub struct Msr(u32);
impl Msr {
pub const fn new(reg: u32) -> Msr {
Msr(reg)
}
}
#[derive(Debug)]
pub struct Efer;
#[derive(Debug)]
pub struct FsBase;
#[derive(Debug)]
pub struct GsBase;
#[derive(Debug)]
pub struct KernelGsBase;
impl Efer {
pub const MSR: Msr = Msr(0xC0000080);
}
impl FsBase {
pub const MSR: Msr = Msr(0xC000_0100);
}
impl GsBase {
pub const MSR: Msr = Msr(0xC000_0101);
}
impl KernelGsBase {
pub const MSR: Msr = Msr(0xC000_0102);
}
bitflags! {
pub struct EferFlags: u64 {
const SYSTEM_CALL_EXTENSIONS = 1 << 0;
const LONG_MODE_ENABLE = 1 << 8;
const LONG_MODE_ACTIVE = 1 << 10;
const NO_EXECUTE_ENABLE = 1 << 11;
const SECURE_VIRTUAL_MACHINE_ENABLE = 1 << 12;
const LONG_MODE_SEGMENT_LIMIT_ENABLE = 1 << 13;
const FAST_FXSAVE_FXRSTOR = 1 << 14;
const TRANSLATION_CACHE_EXTENSION = 1 << 15;
}
}
#[cfg(target_arch = "x86_64")]
mod x86_64 {
use super::*;
use crate::addr::VirtAddr;
impl Msr {
pub unsafe fn read(&self) -> u64 {
let (high, low): (u32, u32);
asm!("rdmsr" : "={eax}" (low), "={edx}" (high) : "{ecx}" (self.0) : "memory" : "volatile");
((high as u64) << 32) | (low as u64)
}
pub unsafe fn write(&mut self, value: u64) {
let low = value as u32;
let high = (value >> 32) as u32;
asm!("wrmsr" :: "{ecx}" (self.0), "{eax}" (low), "{edx}" (high) : "memory" : "volatile" );
}
}
impl Efer {
pub fn read() -> EferFlags {
EferFlags::from_bits_truncate(Self::read_raw())
}
pub fn read_raw() -> u64 {
unsafe { Self::MSR.read() }
}
pub unsafe fn write(flags: EferFlags) {
let old_value = Self::read_raw();
let reserved = old_value & !(EferFlags::all().bits());
let new_value = reserved | flags.bits();
Self::write_raw(new_value);
}
pub unsafe fn write_raw(flags: u64) {
Self::MSR.write(flags);
}
pub unsafe fn update<F>(f: F)
where
F: FnOnce(&mut EferFlags),
{
let mut flags = Self::read();
f(&mut flags);
Self::write(flags);
}
}
impl FsBase {
pub fn read() -> VirtAddr {
VirtAddr::new(unsafe { Self::MSR.read() })
}
pub fn write(address: VirtAddr) {
unsafe { Self::MSR.write(address.as_u64()) };
}
}
impl GsBase {
pub fn read() -> VirtAddr {
VirtAddr::new(unsafe { Self::MSR.read() })
}
pub fn write(address: VirtAddr) {
unsafe { Self::MSR.write(address.as_u64()) };
}
}
impl KernelGsBase {
pub fn read() -> VirtAddr {
VirtAddr::new(unsafe { Self::MSR.read() })
}
pub fn write(address: VirtAddr) {
unsafe { Self::MSR.write(address.as_u64()) };
}
}
}