os/kernel/cpu/i386/isr.c

269 lines
8.6 KiB
C
Raw Normal View History

2019-02-10 14:11:07 -06:00
#include "isr.h"
#include "idt.h"
2019-02-11 14:13:49 -06:00
#include "gdt.h"
#include <cpu/ports.h>
2019-03-17 12:22:00 -05:00
#include "paging.h"
2019-02-10 14:11:07 -06:00
#include "../halt.h"
#include "../../vga_err.h"
#include "../../kernel.h"
#include "../tasking.h"
2019-03-11 09:32:55 -05:00
#include "interrupt.h"
2019-05-06 08:35:59 -05:00
#include "address_spaces.h"
#include "mailboxes.h"
2019-06-29 09:04:34 -05:00
#include <mailboxes.h>
2019-02-10 14:11:07 -06:00
#include <string.h>
#include <stdint.h>
2019-08-25 13:52:56 -05:00
#include "serial.h"
2019-09-01 13:49:01 -05:00
#include <sys/types.h>
2019-02-10 14:11:07 -06:00
void irq_handler(registers_t r);
2019-03-11 09:32:55 -05:00
static isr_t interrupt_handlers[256];
2019-02-10 14:11:07 -06:00
2019-05-23 17:08:03 -05:00
extern Task* currentTask;
2019-02-10 14:11:07 -06:00
/* Can't do this with a loop because we need the address
* of the function names */
void isr_install() {
2019-03-11 09:32:55 -05:00
idt_set_gate(0,(uint32_t)isr0);
idt_set_gate(1,(uint32_t)isr1);
idt_set_gate(2,(uint32_t)isr2);
idt_set_gate(3,(uint32_t)isr3);
idt_set_gate(4,(uint32_t)isr4);
idt_set_gate(5,(uint32_t)isr5);
idt_set_gate(6,(uint32_t)isr6);
idt_set_gate(7,(uint32_t)isr7);
idt_set_gate(8,(uint32_t)isr8);
idt_set_gate(9,(uint32_t)isr9);
idt_set_gate(10,(uint32_t)isr10);
idt_set_gate(11,(uint32_t)isr11);
idt_set_gate(12,(uint32_t)isr12);
idt_set_gate(13,(uint32_t)isr13);
idt_set_gate(14,(uint32_t)isr14);
idt_set_gate(15,(uint32_t)isr15);
idt_set_gate(16,(uint32_t)isr16);
idt_set_gate(17,(uint32_t)isr17);
idt_set_gate(18,(uint32_t)isr18);
idt_set_gate(19,(uint32_t)isr19);
idt_set_gate(20,(uint32_t)isr20);
idt_set_gate(21,(uint32_t)isr21);
idt_set_gate(22,(uint32_t)isr22);
idt_set_gate(23,(uint32_t)isr23);
idt_set_gate(24,(uint32_t)isr24);
idt_set_gate(25,(uint32_t)isr25);
idt_set_gate(26,(uint32_t)isr26);
idt_set_gate(27,(uint32_t)isr27);
idt_set_gate(28,(uint32_t)isr28);
idt_set_gate(29,(uint32_t)isr29);
idt_set_gate(30,(uint32_t)isr30);
idt_set_gate(31,(uint32_t)isr31);
idt_set_gate(80,(uint32_t)isr80);
2019-02-10 14:11:07 -06:00
// Remap the PIC
port_byte_out(0x20,0x11);
port_byte_out(0xA0,0x11);
port_byte_out(0x21,0x20);
port_byte_out(0xA1,0x28);
port_byte_out(0x21,0x04);
port_byte_out(0xA1,0x02);
port_byte_out(0x21,0x01);
port_byte_out(0xA1,0x01);
port_byte_out(0x21,0x0);
port_byte_out(0xA1,0x0);
// Install the IRQs
2019-03-11 09:32:55 -05:00
idt_set_gate(32,(uint32_t)irq0);
idt_set_gate(33,(uint32_t)irq1);
idt_set_gate(34,(uint32_t)irq2);
idt_set_gate(35,(uint32_t)irq3);
idt_set_gate(36,(uint32_t)irq4);
idt_set_gate(37,(uint32_t)irq5);
idt_set_gate(38,(uint32_t)irq6);
idt_set_gate(39,(uint32_t)irq7);
idt_set_gate(40,(uint32_t)irq8);
idt_set_gate(41,(uint32_t)irq9);
idt_set_gate(42,(uint32_t)irq10);
idt_set_gate(43,(uint32_t)irq11);
idt_set_gate(44,(uint32_t)irq12);
idt_set_gate(45,(uint32_t)irq13);
idt_set_gate(46,(uint32_t)irq14);
idt_set_gate(47,(uint32_t)irq15);
2019-02-10 14:11:07 -06:00
2019-03-11 09:32:55 -05:00
load_idt();
2019-02-10 14:11:07 -06:00
}
/* To print the message which defines every exception */
2019-07-01 15:30:00 -05:00
__attribute__((unused)) static char *exception_messages[] = {
2019-02-10 14:11:07 -06:00
"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"
};
void isr_handler(registers_t r) {
switch (r.int_no) {
case 14: {
uint32_t addr;
asm("movl %%cr2,%0": "=r"(addr));
vga_write_string("In PID ");
char str[11];
int_to_ascii(getPID(),str);
vga_write_string(str);
vga_write_string(" and address ");
str[0]='\0';
hex_to_ascii(r.eip,str);
vga_write_string(str);
2019-02-10 15:04:54 -06:00
if (r.err_code==0) {
vga_write_string(", kernel process tried to read a non-present page entry at address ");
2019-02-10 15:04:54 -06:00
} else if (r.err_code==1) {
vga_write_string(", kernel process tried to read a page and caused a protection fault at address ");
2019-02-10 15:04:54 -06:00
} else if (r.err_code==2) {
vga_write_string(", kernel process tried to write to a non-present page entry at address ");
2019-02-10 15:04:54 -06:00
} else if (r.err_code==3) {
vga_write_string(", kernel process tried to write a page and caused a protection fault at address ");
2019-02-10 15:04:54 -06:00
} else if (r.err_code==4) {
vga_write_string(", user process tried to read a non-present page entry at address ");
2019-02-10 15:04:54 -06:00
} else if (r.err_code==5) {
vga_write_string(", user process tried to read a page and caused a protection fault at address ");
2019-02-10 15:04:54 -06:00
} else if (r.err_code==6) {
vga_write_string(", user process tried to write to a non-present page entry at address ");
2019-02-10 15:04:54 -06:00
} else if (r.err_code==7) {
vga_write_string(", user process tried to write a page and caused a protection fault at address ");
2019-02-10 15:04:54 -06:00
}
str[0]='\0';
hex_to_ascii(addr,str);
vga_write_string(str);
vga_write_string(".");
vga_write_string(" Stack is at ");
str[0]='\0';
hex_to_ascii(r.useresp,str);
vga_write_string(str);
vga_write_string(".\n");
2019-02-10 14:11:07 -06:00
// if ((r.err_code&1)==0) {
// // int dir_entry=(addr&0xFFC00000)>>22;
// // int table_entry=(addr&0x3FF000)>12;
// // if (dir_entry_present(dir_entry)) {
// // set_table_entry(dir_entry,table_entry,((dir_entry*1024)+table_entry)*0x1000,1,1,1);
// // for(int page=0;page<1024;page++) {
// // asm volatile("invlpg (%0)"::"r"(((dir_entry*1024)+page)*0x1000):"memory");
// // }
// // } else {
// // for(int page=0;page<1024;page++) {
// // set_table_entry(dir_entry,page,0x0,1,1,0);
// // }
// // set_table_entry(dir_entry,table_entry,((dir_entry*1024)+table_entry)*0x1000,1,1,1);
// // set_directory_entry(dir_entry,dir_entry,1,1,1);
// // }
// // return;
// }
halt();
break;
case 80:
if (r.eax==1) {
2019-02-11 14:13:49 -06:00
tss_stack_reset();
2019-05-11 10:11:28 -05:00
tasking_yield(r);
2019-05-21 19:16:19 -05:00
// r.ds=0x23;
// r.ss=0x23;
// r.cs=0x1B;
} else if (r.eax==2) {
tasking_createTask((void*)r.ebx);
2019-03-17 12:22:00 -05:00
} else if (r.eax==3) {
r.ebx=(uint32_t)alloc_pages(r.ebx);
} else if (r.eax==4) {
alloc_pages_virt(r.ebx,(void*)r.ecx);
2019-04-08 15:58:30 -05:00
} else if (r.eax==5) {
2019-04-13 09:23:40 -05:00
r.ebx=(uint32_t)tasking_get_errno_address();
2019-05-03 09:33:26 -05:00
} else if (r.eax==6) {
2019-06-29 09:04:34 -05:00
kernel_mailbox_get_msg(r.ebx,(Message*)r.ecx,r.edx);
2019-05-03 09:33:26 -05:00
} else if (r.eax==7) {
kernel_mailbox_send_msg((Message*)r.ebx);
2019-05-05 13:14:14 -05:00
} else if (r.eax==8) {
r.ebx=(uint32_t)paging_new_address_space();
2019-05-05 13:30:01 -05:00
} else if (r.eax==9) {
tasking_createTaskCr3KmodeParam((void*)r.ebx,(void*)r.ecx,0,0,0,0,0);
2019-05-06 08:35:59 -05:00
} else if (r.eax==10) {
address_spaces_copy_data((void*)r.ebx,(void*)r.ecx,r.edx,(void*)r.esi);
2019-05-23 17:08:03 -05:00
} else if (r.eax==11) {
if (!currentTask->priv) {
r.ebx=0;
return;
}
void* virt_addr=find_free_pages(r.ecx);
2019-05-23 17:08:03 -05:00
map_pages(virt_addr,(void*)r.ebx,r.ecx,1,1);
2019-06-29 09:04:34 -05:00
r.ebx=(uint32_t)virt_addr;
2019-05-24 08:51:19 -05:00
} else if (r.eax==12) {
2019-05-23 20:41:33 -05:00
tasking_createTaskCr3KmodeParam((void*)r.ebx,(void*)r.ecx,0,1,r.edx,1,r.esi);
2019-05-24 08:51:19 -05:00
} else if (r.eax==13) {
2019-05-23 20:42:00 -05:00
r.ebx=(uint32_t)address_spaces_put_data((void*)r.ebx,(void*)r.ecx,r.edx);
} else if (r.eax==14) {
2019-09-01 13:52:32 -05:00
r.ebx=kernel_mailbox_new((uint16_t)r.ebx,(char*)r.ecx);
} else if (r.eax==15) {
tasking_yieldToPID(r.ebx);
2019-08-25 13:52:56 -05:00
} else if (r.eax==16) {
serial_write_string((char*)r.ebx);
} else if (r.eax==17) {
tasking_exit((uint8_t)r.ebx);
} else if (r.eax==18) {
serial_printf("Initrd size is %d bytes\n",initrd_sz);
r.ebx=initrd_sz;
} else if (r.eax==19) {
serial_printf("Copying initrd\n");
memcpy((char*)r.ebx,initrd,initrd_sz);
2019-09-01 13:49:01 -05:00
} else if (r.eax==20) {
r.ebx=(pid_t)getPID();
} else if (r.eax==21) {
r.ebx=kernel_mailbox_find_by_name((char*)r.ebx);
}
break;
2019-02-10 14:11:07 -06:00
}
}
}
2019-03-11 09:32:55 -05:00
void isr_register_handler(uint8_t n,isr_t handler) {
2019-02-10 14:11:07 -06:00
interrupt_handlers[n] = handler;
}
void irq_handler(registers_t r) {
/* After every interrupt we need to send an EOI to the PICs
* or they will not send another interrupt again */
if (r.int_no >= 40) port_byte_out(0xA0,0x20); /* slave */
port_byte_out(0x20,0x20); /* master */
/* Handle the interrupt in a more modular way */
if (interrupt_handlers[r.int_no] != 0) {
isr_t handler = interrupt_handlers[r.int_no];
handler(r);
}
}