aboutsummaryrefslogtreecommitdiff
path: root/src/kernel
diff options
context:
space:
mode:
Diffstat (limited to 'src/kernel')
-rw-r--r--src/kernel/boot.asm50
-rw-r--r--src/kernel/commands/command.c26
-rw-r--r--src/kernel/commands/command.h6
-rw-r--r--src/kernel/gdt/gdt.asm14
-rw-r--r--src/kernel/gdt/gdt.c51
-rw-r--r--src/kernel/gdt/gdt.h6
-rw-r--r--src/kernel/graphics/graphics.h21
-rw-r--r--src/kernel/graphics/vga.c145
-rw-r--r--src/kernel/grub.cfg6
-rw-r--r--src/kernel/input/input.h10
-rw-r--r--src/kernel/input/ps2/keyboard.c50
-rw-r--r--src/kernel/input/ps2/mouse.c107
-rw-r--r--src/kernel/interrupts/idt.asm6
-rw-r--r--src/kernel/interrupts/idt.c46
-rw-r--r--src/kernel/interrupts/interrupts.h30
-rw-r--r--src/kernel/interrupts/irq.asm157
-rw-r--r--src/kernel/interrupts/irq.c109
-rw-r--r--src/kernel/interrupts/isr.asm277
-rw-r--r--src/kernel/interrupts/isr.c155
-rw-r--r--src/kernel/io/io.asm10
-rw-r--r--src/kernel/io/io.c21
-rw-r--r--src/kernel/io/io.h12
-rw-r--r--src/kernel/kernel.c23
-rw-r--r--src/kernel/lib/lib.h18
-rw-r--r--src/kernel/lib/memory.c26
-rw-r--r--src/kernel/lib/string.c27
-rw-r--r--src/kernel/linker.ld25
-rw-r--r--src/kernel/sound/frequency.c31
-rw-r--r--src/kernel/sound/sound.h6
-rw-r--r--src/kernel/timer/timer.c32
-rw-r--r--src/kernel/timer/timer.h8
31 files changed, 1511 insertions, 0 deletions
diff --git a/src/kernel/boot.asm b/src/kernel/boot.asm
new file mode 100644
index 0000000..eddd3a4
--- /dev/null
+++ b/src/kernel/boot.asm
@@ -0,0 +1,50 @@
+[BITS 32]
+global start
+start:
+ mov esp, _sys_stack ; Points stack to stack area
+ jmp stublet
+
+; Align with 4 Bytes
+ALIGN 4
+mboot:
+ ; Multiboot macros
+ MULTIBOOT_PAGE_ALIGN equ 1<<0
+ MULTIBOOT_MEMORY_INFO equ 1<<1
+ MULTIBOOT_AOUT_KLUDGE equ 1<<16
+ MULTIBOOT_HEADER_MAGIC equ 0x1BADB002
+ MULTIBOOT_HEADER_FLAGS equ MULTIBOOT_PAGE_ALIGN | MULTIBOOT_MEMORY_INFO | MULTIBOOT_AOUT_KLUDGE
+ MULTIBOOT_CHECKSUM equ -(MULTIBOOT_HEADER_MAGIC + MULTIBOOT_HEADER_FLAGS)
+ EXTERN code, bss, end
+
+ ; GRUB Multiboot header
+ dd MULTIBOOT_HEADER_MAGIC
+ dd MULTIBOOT_HEADER_FLAGS
+ dd MULTIBOOT_CHECKSUM
+
+ ; AOUT kludge
+ dd mboot
+ dd code
+ dd bss
+ dd end
+ dd start
+
+; Endless loop
+stublet:
+ extern kernel_main
+ call kernel_main
+ jmp $
+
+%include "src/kernel/gdt/gdt.asm"
+
+%include "src/kernel/interrupts/idt.asm"
+
+%include "src/kernel/interrupts/isr.asm"
+
+%include "src/kernel/interrupts/irq.asm"
+
+%include "src/kernel/io/io.asm"
+
+; Store the stack
+SECTION .bss
+ resb 8192 ; Reserve 8KiB
+_sys_stack: \ No newline at end of file
diff --git a/src/kernel/commands/command.c b/src/kernel/commands/command.c
new file mode 100644
index 0000000..9cbcb07
--- /dev/null
+++ b/src/kernel/commands/command.c
@@ -0,0 +1,26 @@
+#include "../graphics/graphics.h"
+#include "../lib/lib.h"
+#include "../io/io.h"
+
+int32_t starts_with(const char *a, const char *b) {
+ size_t length_pre = strlen(b);
+ size_t length_main = strlen(a);
+ return length_main < length_pre ? 0 : memory_compare(b, a, length_pre) == 0;
+}
+
+extern void shutdown();
+
+void exec_command(char *command) {
+ if (starts_with(command, "ls"))
+ terminal_write_line("Listing files");
+ else if (starts_with(command, "help"))
+ terminal_write_line("I can't help you write now");
+ else if (starts_with(command, "ping"))
+ terminal_write_line("pong!");
+ else if (starts_with(command, "shutdown"))
+ shutdown();
+ else if (starts_with(command, "reboot"))
+ reboot();
+ else
+ terminal_write_line("Command not found!");
+}
diff --git a/src/kernel/commands/command.h b/src/kernel/commands/command.h
new file mode 100644
index 0000000..2a43416
--- /dev/null
+++ b/src/kernel/commands/command.h
@@ -0,0 +1,6 @@
+#ifndef MELVIX_COMMAND_H
+#define MELVIX_COMMAND_H
+
+void exec_command(char *command);
+
+#endif
diff --git a/src/kernel/gdt/gdt.asm b/src/kernel/gdt/gdt.asm
new file mode 100644
index 0000000..c2128e4
--- /dev/null
+++ b/src/kernel/gdt/gdt.asm
@@ -0,0 +1,14 @@
+; GDT flush function
+global gdt_flush
+extern gp
+gdt_flush:
+ lgdt [gp]
+ mov ax, 0x10 ; Data segment offset of GDT
+ mov ds, ax
+ mov es, ax
+ mov fs, ax
+ mov gs, ax
+ mov ss, ax
+ jmp 0x08:flush2 ; Code segment offset
+flush2:
+ ret ; Returns to C code \ No newline at end of file
diff --git a/src/kernel/gdt/gdt.c b/src/kernel/gdt/gdt.c
new file mode 100644
index 0000000..813b432
--- /dev/null
+++ b/src/kernel/gdt/gdt.c
@@ -0,0 +1,51 @@
+struct gdt_entry {
+ unsigned short limit_low;
+ unsigned short base_low;
+ unsigned char base_middle;
+ unsigned char access;
+ unsigned char granularity;
+ unsigned char base_high;
+} __attribute__((packed));
+
+struct gdt_ptr {
+ unsigned short limit;
+ unsigned int base;
+} __attribute__((packed));
+
+struct gdt_entry gdt[3];
+struct gdt_ptr gp;
+
+extern void gdt_flush();
+
+void gdt_set_gate(int num, unsigned long base, unsigned long limit, unsigned char access, unsigned char gran) {
+ // Set descriptor base address
+ gdt[num].base_low = (base & 0xFFFF);
+ gdt[num].base_middle = (base >> 16) & 0xFF;
+ gdt[num].base_high = (base >> 24) & 0xFF;
+
+ // Set descriptor limits
+ gdt[num].limit_low = (limit & 0xFFFF);
+ gdt[num].granularity = ((limit >> 16) & 0x0F);
+
+ // Set granularity and access flags
+ gdt[num].granularity |= (gran & 0xF0);
+ gdt[num].access = access;
+}
+
+void gdt_install() {
+ // Set GDT pointer and limit
+ gp.limit = (sizeof(struct gdt_entry) * 3) - 1;
+ gp.base = &gdt;
+
+ // NULL descriptor
+ gdt_set_gate(0, 0, 0, 0, 0);
+
+ // Code segment
+ gdt_set_gate(1, 0, 0xFFFFFFFF, 0x9A, 0xCF);
+
+ // Data segment
+ gdt_set_gate(2, 0, 0xFFFFFFFF, 0x92, 0xCF);
+
+ // Remove old GDT and install the new changes!
+ gdt_flush();
+} \ No newline at end of file
diff --git a/src/kernel/gdt/gdt.h b/src/kernel/gdt/gdt.h
new file mode 100644
index 0000000..46d80e2
--- /dev/null
+++ b/src/kernel/gdt/gdt.h
@@ -0,0 +1,6 @@
+#ifndef MELVIX_GDT_H
+#define MELVIX_GDT_H
+
+void gdt_install();
+
+#endif
diff --git a/src/kernel/graphics/graphics.h b/src/kernel/graphics/graphics.h
new file mode 100644
index 0000000..a9ed917
--- /dev/null
+++ b/src/kernel/graphics/graphics.h
@@ -0,0 +1,21 @@
+#ifndef MELVIX_VGA_H
+#define MELVIX_VGA_H
+
+#include <stddef.h>
+#include <stdint.h>
+
+enum vga_color;
+
+void terminal_initialize(void);
+
+void terminal_set_color(uint8_t color);
+
+void terminal_clear();
+
+void terminal_write_string(const char *data);
+
+void terminal_put_char(char c);
+
+void terminal_write_line(const char *data);
+
+#endif \ No newline at end of file
diff --git a/src/kernel/graphics/vga.c b/src/kernel/graphics/vga.c
new file mode 100644
index 0000000..86c00ff
--- /dev/null
+++ b/src/kernel/graphics/vga.c
@@ -0,0 +1,145 @@
+#include <stddef.h>
+#include <stdint.h>
+#include "../io/io.h"
+#include "../lib/lib.h"
+#include "../commands/command.h"
+#include "../interrupts/interrupts.h"
+
+// Hardware text mode color constants
+enum vga_color {
+ VGA_COLOR_BLACK = 0,
+ VGA_COLOR_BLUE = 1,
+ VGA_COLOR_GREEN = 2,
+ VGA_COLOR_CYAN = 3,
+ VGA_COLOR_RED = 4,
+ VGA_COLOR_MAGENTA = 5,
+ VGA_COLOR_BROWN = 6,
+ VGA_COLOR_LIGHT_GREY = 7,
+ VGA_COLOR_DARK_GREY = 8,
+ VGA_COLOR_LIGHT_BLUE = 9,
+ VGA_COLOR_LIGHT_GREEN = 10,
+ VGA_COLOR_LIGHT_CYAN = 11,
+ VGA_COLOR_LIGHT_RED = 12,
+ VGA_COLOR_LIGHT_MAGENTA = 13,
+ VGA_COLOR_LIGHT_BROWN = 14,
+ VGA_COLOR_WHITE = 15,
+};
+
+static inline uint8_t vga_entry_color(enum vga_color fg, enum vga_color bg) {
+ return fg | bg << 4;
+}
+
+static inline uint16_t vga_entry(unsigned char uc, uint8_t color) {
+ return (uint16_t) uc | (uint16_t) color << 8;
+}
+
+static const size_t VGA_WIDTH = 80;
+static const size_t VGA_HEIGHT = 25;
+
+size_t terminal_row;
+size_t terminal_column;
+uint8_t terminal_color;
+uint16_t *terminal_buffer;
+
+char text[1024] = {0};
+
+void terminal_clear() {
+ for (size_t y = 0; y < VGA_HEIGHT; y++) {
+ for (size_t x = 0; x < VGA_WIDTH; x++) {
+ const size_t index = y * VGA_WIDTH + x;
+ terminal_buffer[index] = vga_entry(' ', terminal_color);
+ }
+ }
+}
+
+void terminal_enable_cursor(uint8_t cursor_start, uint8_t cursor_end) {
+ send(0x3D4, 0x0A);
+ send(0x3D5, (receive(0x3D5) & 0xC0) | cursor_start);
+ send(0x3D4, 0x0B);
+ send(0x3D5, (receive(0x3D5) & 0xE0) | cursor_end);
+}
+
+void terminal_update_cursor(void) {
+ unsigned temp = terminal_row * VGA_WIDTH + terminal_column;
+ send(0x3D4, 14);
+ send(0x3D5, temp >> 8);
+ send(0x3D4, 15);
+ send(0x3D5, temp);
+}
+
+void terminal_initialize(void) {
+ terminal_enable_cursor(0, 15);
+ terminal_row = 0;
+ terminal_column = 0;
+ terminal_color = vga_entry_color(VGA_COLOR_LIGHT_GREY, VGA_COLOR_BLACK);
+ terminal_buffer = (uint16_t *) 0xB8000;
+ terminal_clear();
+}
+
+void terminal_scroll(void) {
+ if (terminal_row >= VGA_HEIGHT) {
+ terminal_row = VGA_HEIGHT - 1;
+ for (size_t x = 0; x < VGA_WIDTH; x++)
+ for (size_t y = 0; y < VGA_HEIGHT; y++) {
+ uint16_t c = terminal_buffer[y * VGA_WIDTH + x];
+ terminal_buffer[(y - 1) * VGA_WIDTH + x] = c;
+ terminal_buffer[y * VGA_WIDTH + x] = vga_entry(' ', terminal_color);
+ }
+ }
+}
+
+void terminal_set_color(uint8_t color) {
+ terminal_color = color;
+}
+
+void terminal_put_entry_at(char c, uint8_t color, size_t x, size_t y) {
+ const size_t index = y * VGA_WIDTH + x;
+ terminal_buffer[index] = vga_entry(c, color);
+}
+
+void terminal_put_char(char c) {
+ if (c == 0x08) {
+ if (terminal_column != 0) terminal_column--;
+ } else if (c == 0x09) {
+ terminal_column = (terminal_column + 8) & ~(8 - 1);
+ } else if (c == '\r') {
+ terminal_column = 0;
+ } else if (c == '\n') {
+ if (irq_is_installed(1)) exec_command(text);
+ memory_set(text, 0, sizeof(text));
+ terminal_column = 0;
+ terminal_row++;
+ terminal_scroll();
+ terminal_put_entry_at('$', terminal_color, terminal_column, terminal_row);
+ terminal_column = 2;
+ } else if (c >= ' ') { // Any printable character
+ strcat(text, &c);
+ terminal_put_entry_at(c, terminal_color, terminal_column, terminal_row);
+ terminal_column++;
+ }
+
+ // Add new line on overflow
+ if (terminal_column >= VGA_WIDTH) {
+ terminal_column = 0;
+ terminal_row++;
+ }
+
+ terminal_scroll();
+ terminal_update_cursor();
+}
+
+void terminal_write(const char *data, size_t size) {
+ for (size_t i = 0; i < size; i++)
+ terminal_put_char(data[i]);
+}
+
+void terminal_write_string(const char *data) {
+ terminal_write(data, strlen(data));
+}
+
+void terminal_write_line(const char *data) {
+ terminal_row++;
+ terminal_column = 0;
+ terminal_write_string(data);
+ terminal_column = 0;
+}
diff --git a/src/kernel/grub.cfg b/src/kernel/grub.cfg
new file mode 100644
index 0000000..1029549
--- /dev/null
+++ b/src/kernel/grub.cfg
@@ -0,0 +1,6 @@
+set timeout = 0
+set default = 0
+
+menuentry "Melvix" {
+multiboot /boot/melvix.bin
+} \ No newline at end of file
diff --git a/src/kernel/input/input.h b/src/kernel/input/input.h
new file mode 100644
index 0000000..60d2f79
--- /dev/null
+++ b/src/kernel/input/input.h
@@ -0,0 +1,10 @@
+#ifndef MELVIX_INPUT_H
+#define MELVIX_INPUT_H
+
+void mouse_install();
+
+char get_mouse(int n);
+
+void keyboard_install();
+
+#endif
diff --git a/src/kernel/input/ps2/keyboard.c b/src/kernel/input/ps2/keyboard.c
new file mode 100644
index 0000000..ae17675
--- /dev/null
+++ b/src/kernel/input/ps2/keyboard.c
@@ -0,0 +1,50 @@
+#include "../../interrupts/interrupts.h"
+#include "../../io/io.h"
+#include "../../graphics/graphics.h"
+
+unsigned char keymap[128] = {
+ 0 /*E*/, 27, '1', '2', '3', '4', '5', '6', '7', '8', '9', '0', '-', '=', '\b',
+ '\t', 'q', 'w', 'e', 'r', 't', 'y', 'u', 'i', 'o', 'p', '[', ']', '\n',
+ 0 /*C*/, 'a', 's', 'd', 'f', 'g', 'h', 'j', 'k', 'l', ';', '\'', '`', 0 /*LS*/,
+ '\\', 'z', 'x', 'c', 'v', 'b', 'n', 'm', ',', '.', '/', 0 /*RS*/, '*',
+ 0, // Alt key
+ ' ', // Space bar
+ 0, // Caps lock
+ 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, // F keys
+ 0, // Num lock
+ 0, // Scroll lock
+ 0, // Home key
+ 0, // Up arrow
+ 0, // Page up
+ '-',
+ 0, // Left arrow
+ 0,
+ 0, // Right arrow
+ '+',
+ 0, // End key
+ 0, // Down arrow
+ 0, // Page down
+ 0, // Insert key
+ 0, // Delete key
+ 0, 0, 0,
+ 0, // F11
+ 0, // F12
+ 0, // Other keys
+};
+
+void keyboard_handler(struct regs *r) {
+ unsigned char scan_code;
+
+ scan_code = receive(0x60);
+
+ if (scan_code & 0x80) {
+ // Release
+ } else {
+ terminal_put_char(keymap[scan_code]);
+ }
+}
+
+/* Installs the keyboard handler into IRQ1 */
+void keyboard_install() {
+ irq_install_handler(1, keyboard_handler);
+}
diff --git a/src/kernel/input/ps2/mouse.c b/src/kernel/input/ps2/mouse.c
new file mode 100644
index 0000000..49c5a6c
--- /dev/null
+++ b/src/kernel/input/ps2/mouse.c
@@ -0,0 +1,107 @@
+#include "../../io/io.h"
+#include "../../interrupts/interrupts.h"
+
+char mouse_cycle = 0;
+signed char mouse_byte[3], mouse_ex[3];
+signed char mouse_x = 0;
+signed char mouse_y = 0;
+int mouse_but_1 = 0;
+int mouse_but_2 = 0;
+int mm_n[3] = {0, 0, 0,};
+
+void mouse_handler(struct regs *a_r) {
+ switch (mouse_cycle) {
+ case 0:
+ mouse_byte[0] = receive(0x60);
+ mouse_cycle++;
+ break;
+ case 1:
+ mouse_byte[1] = receive(0x60);
+ mouse_cycle++;
+ break;
+ case 2:
+ mouse_byte[2] = receive(0x60);
+ mouse_x = mouse_byte[1];
+ mouse_y = mouse_byte[2];
+ mouse_but_1 = (mouse_byte[0] % 2);
+ mouse_but_2 = ((mouse_byte[0] % 4) - (mouse_byte[0] % 2)) / 2;
+ mouse_cycle = 0;
+ mouse_ex[0] = mouse_byte[0];
+ mm_n[0] = 1;
+ mouse_ex[1] = mouse_byte[1];
+ mm_n[1] = 1;
+ mouse_ex[2] = mouse_byte[2];
+ mm_n[2] = 1;
+ break;
+ default:
+ break;
+ }
+}
+
+inline void mouse_wait(char a_type) {
+ unsigned int _time_out = 100000;
+ if (a_type == 0) {
+ while (_time_out--) {
+ if ((receive(0x64) & 1) == 1) {
+ return;
+ }
+ }
+ return;
+ } else {
+ while (_time_out--) {
+ if ((receive(0x64) & 2) == 0) {
+ return;
+ }
+ }
+ return;
+ }
+}
+
+inline void mouse_write(char a_write) {
+ mouse_wait(1);
+ send(0x64, 0xD4);
+ mouse_wait(1);
+ send(0x60, a_write);
+}
+
+char mouse_read() {
+ mouse_wait(0);
+ return receive(0x60);
+}
+
+void mouse_install() {
+ char _status;
+
+ // Enable auxiliary mouse device
+ mouse_wait(1);
+ send(0x64, 0xA8);
+
+ // Enable interrupts
+ mouse_wait(1);
+ send(0x64, 0x20);
+ mouse_wait(0);
+ _status = (receive(0x60) | 2);
+ mouse_wait(1);
+ send(0x64, 0x60);
+ mouse_wait(1);
+ send(0x60, _status);
+
+ // Use default settings
+ mouse_write(0xF6);
+ mouse_read();
+
+ // Enable mouse
+ mouse_write(0xF4);
+ mouse_read();
+
+ // Setup the mouse handler
+ irq_install_handler(2, mouse_handler);
+}
+
+char get_mouse(int n) {
+ if (mm_n[n] == 1) {
+ mm_n[n] = 0;
+ return mouse_ex[n];
+ } else
+ return 0;
+} \ No newline at end of file
diff --git a/src/kernel/interrupts/idt.asm b/src/kernel/interrupts/idt.asm
new file mode 100644
index 0000000..90eab47
--- /dev/null
+++ b/src/kernel/interrupts/idt.asm
@@ -0,0 +1,6 @@
+; IDT loader
+global idt_load
+extern idtp
+idt_load:
+ lidt [idtp]
+ ret
diff --git a/src/kernel/interrupts/idt.c b/src/kernel/interrupts/idt.c
new file mode 100644
index 0000000..ba71339
--- /dev/null
+++ b/src/kernel/interrupts/idt.c
@@ -0,0 +1,46 @@
+#include "../lib/lib.h"
+
+struct idt_entry {
+ unsigned short base_lo;
+ unsigned short sel; // Kernel segment
+ unsigned char always0; // Always 0
+ unsigned char flags;
+ unsigned short base_hi;
+} __attribute__((packed));
+
+struct idt_ptr {
+ unsigned short limit;
+ unsigned int base;
+} __attribute__((packed));
+
+// Initialize IDT with 256 entries
+struct idt_entry idt[256];
+struct idt_ptr idtp;
+
+// Defined in idt.asm
+extern void idt_load();
+
+void idt_set_gate(unsigned char num, unsigned long base, unsigned short sel, unsigned char flags) {
+ // Specify the interrupt routine's base address
+ idt[num].base_lo = (base & 0xFFFF);
+ idt[num].base_hi = (base >> 16) & 0xFFFF;
+
+ // Set selector/segment of IDT entry
+ idt[num].sel = sel;
+ idt[num].always0 = 0;
+ idt[num].flags = flags;
+}
+
+// Install IDT
+void idt_install() {
+ // Set IDT pointer and limit
+ idtp.limit = (sizeof(struct idt_entry) * 256) - 1;
+ idtp.base = &idt;
+
+ // Clear IDT by setting memory cells to 0
+ memory_set(&idt, 0, sizeof(struct idt_entry) * 256);
+
+ // TODO: Add method to add ISRs to IDT
+
+ idt_load();
+}
diff --git a/src/kernel/interrupts/interrupts.h b/src/kernel/interrupts/interrupts.h
new file mode 100644
index 0000000..755a633
--- /dev/null
+++ b/src/kernel/interrupts/interrupts.h
@@ -0,0 +1,30 @@
+#ifndef MELVIX_INTERRUPTS_H
+#define MELVIX_INTERRUPTS_H
+
+// IDT
+void idt_install();
+
+void idt_set_gate(unsigned char num, unsigned long base, unsigned short sel, unsigned char flags);
+
+// ISRS
+void isrs_install();
+
+struct regs {
+ unsigned int gs, fs, es, ds;
+ unsigned int edi, esi, ebp, esp, ebx, edx, ecx, eax;
+ unsigned int int_no, err_code;
+ unsigned int eip, cs, eflags, useresp, ss;
+};
+
+// IRQ
+void irq_install();
+
+void irq_install_handler(int irq, void (*handler)(struct regs *r));
+
+void irq_uninstall_handler(int irq);
+
+void irq_handler(struct regs *r);
+
+int irq_is_installed(int irq);
+
+#endif
diff --git a/src/kernel/interrupts/irq.asm b/src/kernel/interrupts/irq.asm
new file mode 100644
index 0000000..c485613
--- /dev/null
+++ b/src/kernel/interrupts/irq.asm
@@ -0,0 +1,157 @@
+global irq0
+global irq1
+global irq2
+global irq3
+global irq4
+global irq5
+global irq6
+global irq7
+global irq8
+global irq9
+global irq10
+global irq11
+global irq12
+global irq13
+global irq14
+global irq15
+
+; 32: IRQ0
+irq0:
+ cli
+ push byte 0
+ push byte 32
+ jmp irq_common_stub
+
+; 33: IRQ1
+irq1:
+ cli
+ push byte 0
+ push byte 33
+ jmp irq_common_stub
+
+; 34: IRQ2
+irq2:
+ cli
+ push byte 0
+ push byte 34
+ jmp irq_common_stub
+
+; 35: IRQ3
+irq3:
+ cli
+ push byte 0
+ push byte 35
+ jmp irq_common_stub
+
+; 36: IRQ4
+irq4:
+ cli
+ push byte 0
+ push byte 36
+ jmp irq_common_stub
+
+; 37: IRQ5
+irq5:
+ cli
+ push byte 0
+ push byte 37
+ jmp irq_common_stub
+
+; 38: IRQ6
+irq6:
+ cli
+ push byte 0
+ push byte 38
+ jmp irq_common_stub
+
+; 39: IRQ7
+irq7:
+ cli
+ push byte 0
+ push byte 39
+ jmp irq_common_stub
+
+; 40: IRQ8
+irq8:
+ cli
+ push byte 0
+ push byte 40
+ jmp irq_common_stub
+
+; 41: IRQ9
+irq9:
+ cli
+ push byte 0
+ push byte 41
+ jmp irq_common_stub
+
+; 42: IRQ10
+irq10:
+ cli
+ push byte 0
+ push byte 42
+ jmp irq_common_stub
+
+; 43: IRQ11
+irq11:
+ cli
+ push byte 0
+ push byte 43
+ jmp irq_common_stub
+
+; 44: IRQ12
+irq12:
+ cli
+ push byte 0
+ push byte 44
+ jmp irq_common_stub
+
+; 45: IRQ13
+irq13:
+ cli
+ push byte 0
+ push byte 45
+ jmp irq_common_stub
+
+; 46: IRQ14
+irq14:
+ cli
+ push byte 0
+ push byte 46
+ jmp irq_common_stub
+
+; 47: IRQ15
+irq15:
+ cli
+ push byte 0
+ push byte 47
+ jmp irq_common_stub
+
+extern irq_handler
+
+irq_common_stub:
+ pusha
+ push ds
+ push es
+ push fs
+ push gs
+
+ mov ax, 0x10
+ mov ds, ax
+ mov es, ax
+ mov fs, ax
+ mov gs, ax
+ mov eax, esp
+
+ push eax
+ mov eax, irq_handler
+ call eax
+ pop eax
+
+ pop gs
+ pop fs
+ pop es
+ pop ds
+ popa
+ add esp, 8
+ iret
diff --git a/src/kernel/interrupts/irq.c b/src/kernel/interrupts/irq.c
new file mode 100644
index 0000000..cf9e1fe
--- /dev/null
+++ b/src/kernel/interrupts/irq.c
@@ -0,0 +1,109 @@
+#include "../io/io.h"
+#include "interrupts.h"
+#include "../graphics/graphics.h"
+
+extern void irq0();
+
+extern void irq1();
+
+extern void irq2();
+
+extern void irq3();
+
+extern void irq4();
+
+extern void irq5();
+
+extern void irq6();
+
+extern void irq7();
+
+extern void irq8();
+
+extern void irq9();
+
+extern void irq10();
+
+extern void irq11();
+
+extern void irq12();
+
+extern void irq13();
+
+extern void irq14();
+
+extern void irq15();
+
+// Array to handle custom IRQ handlers
+void *irq_routines[16] = {
+ 0, 0, 0, 0, 0, 0, 0, 0,
+ 0, 0, 0, 0, 0, 0, 0, 0
+};
+
+// Install custom IRQ handler
+void irq_install_handler(int irq, void (*handler)(struct regs *r)) {
+ irq_routines[irq] = handler;
+}
+
+// Removes the custom IRQ handler
+void irq_uninstall_handler(int irq) {
+ irq_routines[irq] = 0;
+}
+
+int irq_is_installed(int irq) {
+ return irq_routines[irq] != 0;
+}
+
+// Remap IRQs for protected mode compatibility via the PIC
+void irq_remap(void) {
+ send(0x20, 0x11);
+ send(0xA0, 0x11);
+ send(0x21, 0x20);
+ send(0xA1, 0x28);
+ send(0x21, 0x04);
+ send(0xA1, 0x02);
+ send(0x21, 0x01);
+ send(0xA1, 0x01);
+ send(0x21, 0x0);
+ send(0xA1, 0x0);
+}
+
+// Map ISRs to the correct entries in the IDT
+void irq_install() {
+ irq_remap();
+ idt_set_gate(32, (unsigned) irq0, 0x08, 0x8E);
+ idt_set_gate(33, (unsigned) irq1, 0x08, 0x8E);
+ idt_set_gate(34, (unsigned) irq2, 0x08, 0x8E);
+ idt_set_gate(35, (unsigned) irq3, 0x08, 0x8E);
+ idt_set_gate(36, (unsigned) irq4, 0x08, 0x8E);
+ idt_set_gate(37, (unsigned) irq5, 0x08, 0x8E);
+ idt_set_gate(38, (unsigned) irq6, 0x08, 0x8E);
+ idt_set_gate(39, (unsigned) irq7, 0x08, 0x8E);
+ idt_set_gate(40, (unsigned) irq8, 0x08, 0x8E);
+ idt_set_gate(41, (unsigned) irq9, 0x08, 0x8E);
+ idt_set_gate(42, (unsigned) irq10, 0x08, 0x8E);
+ idt_set_gate(43, (unsigned) irq11, 0x08, 0x8E);
+ idt_set_gate(44, (unsigned) irq12, 0x08, 0x8E);
+ idt_set_gate(45, (unsigned) irq13, 0x08, 0x8E);
+ idt_set_gate(46, (unsigned) irq14, 0x08, 0x8E);
+ idt_set_gate(47, (unsigned) irq15, 0x08, 0x8E);
+}
+
+// Handle IRQ ISRs
+void irq_handler(struct regs *r) {
+ void (*handler)(struct regs *r);
+
+ // Execute custom handler if exists
+ handler = irq_routines[r->int_no - 32];
+ if (handler) {
+ handler(r);
+ }
+
+ // Send end of interrupt to second (slave) IRQ controller
+ if (r->int_no >= 40) {
+ send(0xA0, 0x20);
+ }
+
+ // Send end of interrupt to master interrupt controller
+ send(0x20, 0x20);
+}
diff --git a/src/kernel/interrupts/isr.asm b/src/kernel/interrupts/isr.asm
new file mode 100644
index 0000000..cf75157
--- /dev/null
+++ b/src/kernel/interrupts/isr.asm
@@ -0,0 +1,277 @@
+global isr0
+global isr1
+global isr2
+global isr3
+global isr4
+global isr5
+global isr6
+global isr7
+global isr8
+global isr9
+global isr10
+global isr11
+global isr12
+global isr13
+global isr14
+global isr15
+global isr16
+global isr17
+global isr18
+global isr19
+global isr20
+global isr21
+global isr22
+global isr23
+global isr24
+global isr25
+global isr26
+global isr27
+global isr28
+global isr29
+global isr30
+global isr31
+
+; 0: Divide By Zero Exception
+isr0:
+ cli
+ push byte 0
+ push byte 0
+ jmp isr_common_stub
+
+; 1: Debug Exception
+isr1:
+ cli
+ push byte 0
+ push byte 1
+ jmp isr_common_stub
+
+; 2: Non Maskable Interrupt Exception
+isr2:
+ cli
+ push byte 0
+ push byte 2
+ jmp isr_common_stub
+
+; 3: Int 3 Exception
+isr3:
+ cli
+ push byte 0
+ push byte 3
+ jmp isr_common_stub
+
+; 4: INTO Exception
+isr4:
+ cli
+ push byte 0
+ push byte 4
+ jmp isr_common_stub
+
+; 5: Out of Bounds Exception
+isr5:
+ cli
+ push byte 0
+ push byte 5
+ jmp isr_common_stub
+
+; 6: Invalid Opcode Exception
+isr6:
+ cli
+ push byte 0
+ push byte 6
+ jmp isr_common_stub
+
+; 7: Coprocessor Not Available Exception
+isr7:
+ cli
+ push byte 0
+ push byte 7
+ jmp isr_common_stub
+
+; 8: Double Fault Exception (With Error Code!)
+isr8:
+ cli
+ push byte 8
+ jmp isr_common_stub
+
+; 9: Coprocessor Segment Overrun Exception
+isr9:
+ cli
+ push byte 0
+ push byte 9
+ jmp isr_common_stub
+
+; 10: Bad TSS Exception (With Error Code!)
+isr10:
+ cli
+ push byte 10
+ jmp isr_common_stub
+
+; 11: Segment Not Present Exception (With Error Code!)
+isr11:
+ cli
+ push byte 11
+ jmp isr_common_stub
+
+; 12: Stack Fault Exception (With Error Code!)
+isr12:
+ cli
+ push byte 12
+ jmp isr_common_stub
+
+; 13: General Protection Fault Exception (With Error Code!)
+isr13:
+ cli
+ push byte 13
+ jmp isr_common_stub
+
+; 14: Page Fault Exception (With Error Code!)
+isr14:
+ cli
+ push byte 14
+ jmp isr_common_stub
+
+; 15: Reserved Exception
+isr15:
+ cli
+ push byte 0
+ push byte 15
+ jmp isr_common_stub
+
+; 16: Floating Point Exception
+isr16:
+ cli
+ push byte 0
+ push byte 16
+ jmp isr_common_stub
+
+; 17: Alignment Check Exception
+isr17:
+ cli
+ push byte 0
+ push byte 17
+ jmp isr_common_stub
+
+; 18: Machine Check Exception
+isr18:
+ cli
+ push byte 0
+ push byte 18
+ jmp isr_common_stub
+
+; 19: Reserved
+isr19:
+ cli
+ push byte 0
+ push byte 19
+ jmp isr_common_stub
+
+; 20: Reserved
+isr20:
+ cli
+ push byte 0
+ push byte 20
+ jmp isr_common_stub
+
+; 21: Reserved
+isr21:
+ cli
+ push byte 0
+ push byte 21
+ jmp isr_common_stub
+
+; 22: Reserved
+isr22:
+ cli
+ push byte 0
+ push byte 22
+ jmp isr_common_stub
+
+; 23: Reserved
+isr23:
+ cli
+ push byte 0
+ push byte 23
+ jmp isr_common_stub
+
+; 24: Reserved
+isr24:
+ cli
+ push byte 0
+ push byte 24
+ jmp isr_common_stub
+
+; 25: Reserved
+isr25:
+ cli
+ push byte 0
+ push byte 25
+ jmp isr_common_stub
+
+; 26: Reserved
+isr26:
+ cli
+ push byte 0
+ push byte 26
+ jmp isr_common_stub
+
+; 27: Reserved
+isr27:
+ cli
+ push byte 0
+ push byte 27
+ jmp isr_common_stub
+
+; 28: Reserved
+isr28:
+ cli
+ push byte 0
+ push byte 28
+ jmp isr_common_stub
+
+; 29: Reserved
+isr29:
+ cli
+ push byte 0
+ push byte 29
+ jmp isr_common_stub
+
+; 30: Reserved
+isr30:
+ cli
+ push byte 0
+ push byte 30
+ jmp isr_common_stub
+
+; 31: Reserved
+isr31:
+ cli
+ push byte 0
+ push byte 31
+ jmp isr_common_stub
+
+extern fault_handler
+
+; Stores the ISR in the stack and calls the C fault handler
+isr_common_stub:
+ pusha
+ push ds
+ push es
+ push fs
+ push gs
+ mov ax, 0x10
+ mov ds, ax
+ mov es, ax
+ mov fs, ax
+ mov gs, ax
+ mov eax, esp
+ push eax
+ mov eax, fault_handler
+ call eax
+ pop eax
+ pop gs
+ pop fs
+ pop es
+ pop ds
+ popa
+ add esp, 8
+ iret
diff --git a/src/kernel/interrupts/isr.c b/src/kernel/interrupts/isr.c
new file mode 100644
index 0000000..8da7919
--- /dev/null
+++ b/src/kernel/interrupts/isr.c
@@ -0,0 +1,155 @@
+#include "../graphics/graphics.h"
+#include "interrupts.h"
+
+// Defined in isr.asm
+extern void isr0();
+
+extern void isr1();
+
+extern void isr2();
+
+extern void isr3();
+
+extern void isr4();
+
+extern void isr5();
+
+extern void isr6();
+
+extern void isr7();
+
+extern void isr8();
+
+extern void isr9();
+
+extern void isr10();
+
+extern void isr11();
+
+extern void isr12();
+
+extern void isr13();
+
+extern void isr14();
+
+extern void isr15();
+
+extern void isr16();
+
+extern void isr17();
+
+extern void isr18();
+
+extern void isr19();
+
+extern void isr20();
+
+extern void isr21();
+
+extern void isr22();
+
+extern void isr23();
+
+extern void isr24();
+
+extern void isr25();
+
+extern void isr26();
+
+extern void isr27();
+
+extern void isr28();
+
+extern void isr29();
+
+extern void isr30();
+
+extern void isr31();
+
+// Install ISRs in IDT
+void isrs_install() {
+ idt_set_gate(0, (unsigned) isr0, 0x08, 0x8E);
+ idt_set_gate(1, (unsigned) isr1, 0x08, 0x8E);
+ idt_set_gate(2, (unsigned) isr2, 0x08, 0x8E);
+ idt_set_gate(3, (unsigned) isr3, 0x08, 0x8E);
+ idt_set_gate(4, (unsigned) isr4, 0x08, 0x8E);
+ idt_set_gate(5, (unsigned) isr5, 0x08, 0x8E);
+ idt_set_gate(6, (unsigned) isr6, 0x08, 0x8E);
+ idt_set_gate(7, (unsigned) isr7, 0x08, 0x8E);
+
+ idt_set_gate(8, (unsigned) isr8, 0x08, 0x8E);
+ idt_set_gate(9, (unsigned) isr9, 0x08, 0x8E);
+ idt_set_gate(10, (unsigned) isr10, 0x08, 0x8E);
+ idt_set_gate(11, (unsigned) isr11, 0x08, 0x8E);
+ idt_set_gate(12, (unsigned) isr12, 0x08, 0x8E);
+ idt_set_gate(13, (unsigned) isr13, 0x08, 0x8E);
+ idt_set_gate(14, (unsigned) isr14, 0x08, 0x8E);
+ idt_set_gate(15, (unsigned) isr15, 0x08, 0x8E);
+
+ idt_set_gate(16, (unsigned) isr16, 0x08, 0x8E);
+ idt_set_gate(17, (unsigned) isr17, 0x08, 0x8E);
+ idt_set_gate(18, (unsigned) isr18, 0x08, 0x8E);
+ idt_set_gate(19, (unsigned) isr19, 0x08, 0x8E);
+ idt_set_gate(20, (unsigned) isr20, 0x08, 0x8E);
+ idt_set_gate(21, (unsigned) isr21, 0x08, 0x8E);
+ idt_set_gate(22, (unsigned) isr22, 0x08, 0x8E);
+ idt_set_gate(23, (unsigned) isr23, 0x08, 0x8E);
+
+ idt_set_gate(24, (unsigned) isr24, 0x08, 0x8E);
+ idt_set_gate(25, (unsigned) isr25, 0x08, 0x8E);
+ idt_set_gate(26, (unsigned) isr26, 0x08, 0x8E);
+ idt_set_gate(27, (unsigned) isr27, 0x08, 0x8E);
+ idt_set_gate(28, (unsigned) isr28, 0x08, 0x8E);
+ idt_set_gate(29, (unsigned) isr29, 0x08, 0x8E);
+ idt_set_gate(30, (unsigned) isr30, 0x08, 0x8E);
+ idt_set_gate(31, (unsigned) isr31, 0x08, 0x8E);
+}
+
+// Error exception messages
+const char *exception_messages[] = {
+ "Division By Zero",
+ "Debug",
+ "Non Maskable Interrupt",
+ "Breakpoint",
+ "Into Detected Overflow",
+ "Out of Bounds",
+ "Invalid Opcode",
+ "No Coprocessor",
+
+ "Double Fault",
+ "Coprocessor Segment Overrun",
+ "Bad TSS",
+ "Segment Not Present",
+ "Stack Fault",
+ "General Protection Fault",
+ "Page Fault",
+ "Unknown Interrupt",
+
+ "Coprocessor Fault",
+ "Alignment Check",
+ "Machine Check",
+ "Reserved",
+ "Reserved",
+ "Reserved",
+ "Reserved",
+ "Reserved",
+
+ "Reserved",
+ "Reserved",
+ "Reserved",
+ "Reserved",
+ "Reserved",
+ "Reserved",
+ "Reserved",
+ "Reserved"
+};
+
+// Master exception handler - halt via endless loop
+void fault_handler(struct regs *r) {
+ if (r->int_no < 32) {
+ terminal_write_string("\n");
+ terminal_write_string(exception_messages[r->int_no]);
+ terminal_write_string(" Exception. System Halted!\n");
+ for (;;);
+ }
+}
diff --git a/src/kernel/io/io.asm b/src/kernel/io/io.asm
new file mode 100644
index 0000000..6ab3707
--- /dev/null
+++ b/src/kernel/io/io.asm
@@ -0,0 +1,10 @@
+global shutdown
+shutdown:
+ mov ax, 0x1000
+ mov ax, ss
+ mov sp, 0xf000
+ mov ax, 0x5307
+ mov bx, 0x0001
+ mov cx, 0x0003
+ int 0x15
+ ret \ No newline at end of file
diff --git a/src/kernel/io/io.c b/src/kernel/io/io.c
new file mode 100644
index 0000000..7bddb13
--- /dev/null
+++ b/src/kernel/io/io.c
@@ -0,0 +1,21 @@
+#include <stdint.h>
+
+unsigned char receive(unsigned short port) {
+ unsigned char value;
+ __asm__ __volatile__ ("inb %1, %0" : "=a" (value) : "dN" (port));
+ return value;
+}
+
+void send(unsigned short port, unsigned char data) {
+ __asm__ __volatile__ ("outb %1, %0" : : "dN" (port), "a" (data));
+}
+
+void reboot() {
+ uint8_t good = 0x02;
+ while (good & 0x02)
+ good = receive(0x64);
+ send(0x64, 0xFE);
+ loop:
+ asm volatile ("hlt");
+ goto loop;
+} \ No newline at end of file
diff --git a/src/kernel/io/io.h b/src/kernel/io/io.h
new file mode 100644
index 0000000..e00a5f0
--- /dev/null
+++ b/src/kernel/io/io.h
@@ -0,0 +1,12 @@
+#ifndef MELVIX_IO_H
+#define MELVIX_IO_H
+
+#include <stdint.h>
+
+unsigned char receive(unsigned short port);
+
+void send(unsigned short port, unsigned char data);
+
+void reboot();
+
+#endif
diff --git a/src/kernel/kernel.c b/src/kernel/kernel.c
new file mode 100644
index 0000000..49fe0b7
--- /dev/null
+++ b/src/kernel/kernel.c
@@ -0,0 +1,23 @@
+#include "graphics/graphics.h"
+#include "gdt/gdt.h"
+#include "interrupts/interrupts.h"
+#include "input/input.h"
+#include "timer/timer.h"
+#include "sound/sound.h"
+
+void kernel_main(void) {
+ gdt_install();
+ idt_install();
+ isrs_install();
+ irq_install();
+
+ __asm__ __volatile__ ("sti");
+
+ terminal_initialize();
+ terminal_write_string("Melvix loaded successfully!\n");
+
+ timer_install();
+ keyboard_install();
+ mouse_install();
+ // __asm__ ("div %0" :: "r"(0)); // Exception testing x/0
+} \ No newline at end of file
diff --git a/src/kernel/lib/lib.h b/src/kernel/lib/lib.h
new file mode 100644
index 0000000..f0d7d1b
--- /dev/null
+++ b/src/kernel/lib/lib.h
@@ -0,0 +1,18 @@
+#ifndef MELVIX_LIB_H
+#define MELVIX_LIB_H
+
+#include <stddef.h>
+
+size_t strlen(const char *str);
+
+size_t strcmp(const char *s1, const char *s2);
+
+char *strcat(char *dst, const char *src);
+
+void *memory_copy(void *dest, const void *src, size_t count);
+
+void *memory_set(void *dest, char val, size_t count);
+
+int memory_compare(const void *a_ptr, const void *b_ptr, size_t size);
+
+#endif
diff --git a/src/kernel/lib/memory.c b/src/kernel/lib/memory.c
new file mode 100644
index 0000000..e306ada
--- /dev/null
+++ b/src/kernel/lib/memory.c
@@ -0,0 +1,26 @@
+#include "../graphics/graphics.h"
+
+void *memory_copy(void *dest, const void *src, size_t count) {
+ const char *sp = (const char *) src;
+ char *dp = (char *) dest;
+ for (; count != 0; count--) *dp++ = *sp++;
+ return dest;
+}
+
+void *memory_set(void *dest, char val, size_t count) {
+ char *temp = (char *) dest;
+ for (; count != 0; count--) *temp++ = val;
+ return dest;
+}
+
+int memory_compare(const void *a_ptr, const void *b_ptr, size_t size) {
+ const unsigned char *a = (const unsigned char *) a_ptr;
+ const unsigned char *b = (const unsigned char *) b_ptr;
+ for (size_t i = 0; i < size; i++) {
+ if (a[i] < b[i])
+ return -1;
+ else if (b[i] < a[i])
+ return 1;
+ }
+ return 0;
+} \ No newline at end of file
diff --git a/src/kernel/lib/string.c b/src/kernel/lib/string.c
new file mode 100644
index 0000000..6ef0316
--- /dev/null
+++ b/src/kernel/lib/string.c
@@ -0,0 +1,27 @@
+#include <stddef.h>
+
+size_t strlen(const char *str) {
+ size_t len = 0;
+ while (str[len])
+ len++;
+ return len;
+}
+
+size_t strcmp(const char *s1, const char *s2) {
+ while (*s1 && (*s1 == *s2)) {
+ s1++;
+ s2++;
+ }
+ return *(const unsigned char *) s1 - *(const unsigned char *) s2;
+}
+
+char *strcat(char *dst, const char *src) {
+ unsigned int i = 0;
+ unsigned int j = 0;
+ for (i = 0; dst[i] != 0; i++) {}
+ for (j = 0; src[j] != 0; j++) {
+ dst[i + j] = src[j];
+ }
+ dst[i + j] = 0;
+ return dst;
+}
diff --git a/src/kernel/linker.ld b/src/kernel/linker.ld
new file mode 100644
index 0000000..6461b2b
--- /dev/null
+++ b/src/kernel/linker.ld
@@ -0,0 +1,25 @@
+OUTPUT_FORMAT("binary")
+ENTRY(start)
+phys = 0x00100000;
+SECTIONS
+{
+ .text phys : AT(phys) {
+ code = .;
+ *(.text)
+ *(.rodata*)
+ . = ALIGN(4096);
+ }
+ .data : AT(phys + (data - code))
+ {
+ data = .;
+ *(.data)
+ . = ALIGN(4096);
+ }
+ .bss : AT(phys + (bss - code))
+ {
+ bss = .;
+ *(.bss)
+ . = ALIGN(4096);
+ }
+ end = .;
+}
diff --git a/src/kernel/sound/frequency.c b/src/kernel/sound/frequency.c
new file mode 100644
index 0000000..6d02690
--- /dev/null
+++ b/src/kernel/sound/frequency.c
@@ -0,0 +1,31 @@
+#include <stdint.h>
+#include "../io/io.h"
+#include "../timer/timer.h"
+
+static void play_sound(uint32_t frequency) {
+ uint32_t divided;
+ uint8_t tmp;
+
+ divided = 1193180 / frequency;
+ send(0x43, 0xb6);
+ send(0x42, (uint8_t) (divided));
+ send(0x42, (uint8_t) (divided >> 8));
+
+ tmp = receive(0x61);
+ if (tmp != (tmp | 3)) {
+ send(0x61, tmp | 3);
+ }
+}
+
+static void shut_up() {
+ uint8_t tmp = receive(0x61) & 0xFC;
+
+ send(0x61, tmp);
+}
+
+//Make a beep
+void beep(uint32_t frequency, uint32_t ticks) {
+ play_sound(frequency);
+ timer_wait(ticks);
+ shut_up();
+} \ No newline at end of file
diff --git a/src/kernel/sound/sound.h b/src/kernel/sound/sound.h
new file mode 100644
index 0000000..baf70b2
--- /dev/null
+++ b/src/kernel/sound/sound.h
@@ -0,0 +1,6 @@
+#ifndef MELVIX_SOUND_H
+#define MELVIX_SOUND_H
+
+void beep(uint32_t frequency, uint32_t ticks);
+
+#endif
diff --git a/src/kernel/timer/timer.c b/src/kernel/timer/timer.c
new file mode 100644
index 0000000..38f5be6
--- /dev/null
+++ b/src/kernel/timer/timer.c
@@ -0,0 +1,32 @@
+#include "../interrupts/interrupts.h"
+#include "../io/io.h"
+
+volatile unsigned int timer_ticks = 0;
+
+void timer_phase(int hz) {
+ int divisor = 1193180 / hz;
+ send(0x43, 0x36); // 01 10 11 0b // CTR, RW, MODE, BCD
+ send(0x40, divisor & 0xFF);
+ send(0x40, divisor >> 8);
+}
+
+// Executed 100 times per second
+void timer_handler(struct regs *r) {
+ timer_ticks++;
+}
+
+// "Delay" function with CPU sleep
+void timer_wait(int ticks) {
+ unsigned int eticks;
+
+ eticks = timer_ticks + ticks;
+ while (timer_ticks < eticks) {
+ __asm__ __volatile__ ("sti//hlt//cli");
+ }
+}
+
+// Install timer handler into IRQ0
+void timer_install() {
+ timer_phase(100);
+ irq_install_handler(0, timer_handler);
+} \ No newline at end of file
diff --git a/src/kernel/timer/timer.h b/src/kernel/timer/timer.h
new file mode 100644
index 0000000..66b3c95
--- /dev/null
+++ b/src/kernel/timer/timer.h
@@ -0,0 +1,8 @@
+#ifndef MELVIX_TIMER_H
+#define MELVIX_TIMER_H
+
+void timer_install();
+
+void timer_wait(int ticks);
+
+#endif