aboutsummaryrefslogtreecommitdiff
path: root/src/kernel/io/io.c
blob: 1ef82b620dc1d67548f782a298c05fc2fc25a9f9 (plain) (blame)
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);
}