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
|
#include <stdint.h>
#include <system.h>
u8 inb(u16 port)
{
u8 value;
asm volatile("inb %1, %0" : "=a"(value) : "Nd"(port));
return value;
}
u16 inw(u16 port)
{
u16 value;
asm volatile("inw %1, %0" : "=a"(value) : "Nd"(port));
return value;
}
u32 inl(u16 port)
{
u32 value;
asm volatile("inl %1, %0" : "=a"(value) : "Nd"(port));
return value;
}
u32 cpu_flags()
{
u32 flags;
asm volatile("pushf\n"
"pop %0\n"
: "=rm"(flags)::"memory");
return flags;
}
int interrupts_enabled()
{
return (cpu_flags() & 0x200) == 0x200;
}
void interrupts_print()
{
if (interrupts_enabled())
log(GRN "Interrupts are enabled!" RES);
else
log(RED "Interrupts are disabled!" RES);
}
void cli()
{
asm volatile("cli");
}
void sti()
{
asm volatile("sti");
}
void hlt()
{
asm volatile("hlt");
}
void outb(u16 port, u8 data)
{
asm volatile("outb %0, %1" ::"a"(data), "Nd"(port));
}
void outw(u16 port, u16 data)
{
asm volatile("outw %0, %1" ::"a"(data), "Nd"(port));
}
void outl(u16 port, u32 data)
{
asm volatile("outl %0, %1" ::"a"(data), "Nd"(port));
}
u32 cr3_get()
{
u32 cr3;
asm volatile("movl %%cr3, %%eax" : "=a"(cr3));
return cr3;
}
void cr3_set(u32 cr3)
{
asm volatile("movl %%eax, %%cr3" ::"a"(cr3));
}
u32 cr0_get()
{
u32 cr0;
asm volatile("movl %%cr0, %%eax" : "=a"(cr0));
return cr0;
}
void cr0_set(u32 cr0)
{
asm volatile("movl %%eax, %%cr0" ::"a"(cr0));
}
void invlpg(u32 addr)
{
asm volatile("invlpg (%0)" ::"r"(addr) : "memory");
}
void serial_install()
{
outb(0x3f8 + 1, 0x00);
outb(0x3f8 + 3, 0x80);
outb(0x3f8 + 0, 0x03);
outb(0x3f8 + 1, 0x00);
outb(0x3f8 + 3, 0x03);
outb(0x3f8 + 2, 0xC7);
outb(0x3f8 + 4, 0x0B);
info("Installed serial connection");
}
int is_transmit_empty()
{
return inb(0x3f8 + 5) & 0x20;
}
void serial_put(char ch)
{
while (is_transmit_empty() == 0)
;
outb(0x3f8, (u8)ch);
}
|