forked from eeply/ps4-kexec
-
Notifications
You must be signed in to change notification settings - Fork 4
/
x86.h
195 lines (164 loc) · 3.64 KB
/
x86.h
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
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
/*
* ps4-kexec - a kexec() implementation for Orbis OS / FreeBSD
*
* Copyright (C) 2015-2016 shuffle2 <[email protected]>
* Copyright (C) 2015-2016 Hector Martin "marcan" <[email protected]>
*
* This code is licensed to you under the 2-clause BSD license. See the LICENSE
* file for more information.
*/
#ifndef X86_H
#define X86_H
#define FLAGS_IF (1 << 9)
#define CR0_WP (1 << 16)
#define CR0_NW (1 << 29)
#define CR0_CD (1 << 30)
#define PG_V (1 << 0)
#define PG_RW (1 << 1)
#define PG_U (1 << 2)
#define PG_PS (1 << 7)
#define SEG_TYPE_DATA (0 << 3)
#define SEG_TYPE_READ_WRITE (1 << 1)
#define SEG_TYPE_CODE (1 << 3)
#define SEG_TYPE_EXEC_READ (1 << 1)
#define SEG_TYPE_TSS ((1 << 3) | (1 << 0))
static inline u64 cr0_read(void)
{
u64 reg;
asm volatile("mov %0, cr0;" : "=r" (reg));
return reg;
}
static inline void cr0_write(u64 val)
{
asm volatile("mov cr0, %0;" :: "r" (val));
}
static inline u64 write_protect_disable()
{
u64 cr0 = cr0_read();
cr0_write(cr0 & ~CR0_WP);
return cr0;
}
static inline void write_protect_restore(u64 cr0)
{
// Use only WP bit of input
cr0_write(cr0_read() | (cr0 & CR0_WP));
}
static inline u64 cr3_read(void)
{
u64 reg;
asm volatile("mov %0, cr3;" : "=r" (reg));
return reg;
}
static inline void cr3_write(u64 val)
{
asm volatile("mov cr3, %0;" :: "r" (val));
}
typedef struct {
u16 limit;
u64 address;
} __attribute__((packed)) desc_ptr;
static inline desc_ptr gdt_read(void)
{
desc_ptr gdtr;
asm volatile("sgdt %0;" : "=m" (gdtr));
return gdtr;
}
static inline void gdt_write(desc_ptr* val)
{
asm volatile("lgdt %0;" :: "m" (*val));
}
//IDT
typedef struct {
u16 limit;
u64 address;
} __attribute__((packed)) idt_ptr;
static inline idt_ptr idt_read(void)
{
idt_ptr idtr;
asm volatile("sidt %0;" : "=m" (idtr));
return idtr;
}
static inline void idt_write(idt_ptr* val)
{
asm volatile("lidt %0;" :: "m" (*val));
}
static inline void cr4_pge_disable(void)
{
u64 cr4_temp;
asm volatile(
"mov %0, cr4;"
"and %0, ~0x80;"
"mov cr4, %0;"
: "=r" (cr4_temp)
);
}
static inline void wbinvd(void)
{
asm volatile("wbinvd");
}
static inline void cpu_stop(void)
{
for (;;)
asm volatile("cli; hlt;" : : : "memory");
}
static inline void outl(int port, unsigned int data)
{
asm volatile("out %w1, %0" : : "a" (data), "d" (port));
}
static inline void wrmsr(u32 msr_id, u64 msr_value)
{
asm volatile(
"wrmsr"
:
: "c" (msr_id), "a" (msr_value & 0xffffffff), "d" (msr_value >> 32)
);
}
static inline u64 rdmsr(u64 msr_id)
{
u32 low, high;
asm volatile (
"rdmsr"
: "=a"(low), "=d"(high)
: "c"(msr_id)
);
return ((u64)high << 32) | low;
}
static inline u64 rdtsc (void)
{
unsigned int tickl, tickh;
asm volatile(
"rdtsc"
:"=a"(tickl),"=d"(tickh)
);
return ((u64)tickh << 32) | tickl;
}
static inline void udelay(unsigned int usec) {
u64 later = rdtsc() + usec * 1594ULL;
while (((s64)(later - rdtsc())) > 0);
}
static inline void disable_interrupts(void)
{
asm volatile("cli");
}
static inline void enable_interrupts(void)
{
asm volatile("sti");
}
static inline u64 read_flags(void)
{
u64 flags;
asm volatile("pushf; pop %0;" : "=r" (flags));
return flags;
}
static inline u64 intr_disable(void)
{
u64 flags = read_flags();
disable_interrupts();
return flags;
}
static inline void intr_restore(u64 flags)
{
// TODO should only IF be or'd in?
asm volatile("push %0; popf;" : : "rm" (flags) : "memory");
}
#endif