2018-10-19 07:59:38 -05:00
|
|
|
#include "../drivers/vga.h"
|
2018-11-04 16:12:54 -06:00
|
|
|
#include "../drivers/serial.h"
|
2018-10-19 07:59:38 -05:00
|
|
|
#include "../drivers/isr.h"
|
2018-10-20 11:32:14 -05:00
|
|
|
#include "../drivers/gdt.h"
|
2018-10-22 06:44:17 -05:00
|
|
|
#include "../drivers/keyboard.h"
|
|
|
|
#include "../libc/string.h"
|
2018-10-21 17:45:12 -05:00
|
|
|
#include "syscalls.h"
|
2018-11-04 16:12:54 -06:00
|
|
|
#include "multiboot.h"
|
|
|
|
#define MMAP_ENTRIES 5
|
|
|
|
#define VIRT_OFFSET 0xC0000000
|
|
|
|
uint32_t total_mb;
|
|
|
|
uint32_t mem_map[MMAP_ENTRIES+1][2];
|
2018-10-19 07:59:38 -05:00
|
|
|
void switch_to_user_mode() {
|
2018-10-20 11:32:14 -05:00
|
|
|
asm volatile(" \
|
|
|
|
cli; \
|
|
|
|
mov $0x23, %ax; \
|
|
|
|
mov %ax, %ds; \
|
|
|
|
mov %ax, %es; \
|
|
|
|
mov %ax, %fs; \
|
|
|
|
mov %ax, %gs; \
|
|
|
|
\
|
|
|
|
mov %esp, %eax; \
|
|
|
|
pushl $0x23; \
|
|
|
|
pushl %eax; \
|
|
|
|
pushf; \
|
2018-10-21 17:45:12 -05:00
|
|
|
pop %eax; \
|
|
|
|
or $0x200,%eax; \
|
|
|
|
push %eax; \
|
2018-10-20 11:32:14 -05:00
|
|
|
pushl $0x1B; \
|
|
|
|
push $1f; \
|
|
|
|
iret; \
|
|
|
|
1: \
|
|
|
|
");
|
2018-10-19 07:59:38 -05:00
|
|
|
}
|
2018-11-04 16:12:54 -06:00
|
|
|
void halt() {
|
|
|
|
asm volatile("cli;\
|
|
|
|
hltlabel: hlt;\
|
|
|
|
jmp hltlabel");
|
|
|
|
}
|
2018-10-19 07:59:38 -05:00
|
|
|
|
2018-11-04 16:12:54 -06:00
|
|
|
void get_memory(multiboot_info_t* mbd) {
|
|
|
|
if ((mbd->flags&MULTIBOOT_INFO_MEM_MAP)!=0) {
|
|
|
|
uint32_t mmap_length=mbd->mmap_length;
|
|
|
|
struct multiboot_mmap_entry* mmap_addr=(struct multiboot_mmap_entry*)(mbd->mmap_addr+VIRT_OFFSET);
|
|
|
|
uint32_t size;
|
|
|
|
struct multiboot_mmap_entry* mmap_entry=mmap_addr;
|
|
|
|
int i;
|
|
|
|
for (i=0;(uint32_t)mmap_entry<((uint32_t)mmap_addr+mmap_length);mmap_entry=(struct multiboot_mmap_entry*)((uint32_t)mmap_entry+size+4)) {
|
|
|
|
if (i>=MMAP_ENTRIES) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
size=mmap_entry->size;
|
|
|
|
uint32_t start_addr=mmap_entry->addr;
|
|
|
|
uint32_t length=mmap_entry->len;
|
|
|
|
uint32_t type=mmap_entry->type;
|
|
|
|
if (type!=1) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
mem_map[i][0]=start_addr;
|
|
|
|
mem_map[i][1]=start_addr+length-1;
|
|
|
|
i++;
|
|
|
|
}
|
|
|
|
mem_map[i][0]=0;
|
|
|
|
mem_map[i][0]=0;
|
|
|
|
total_mb=0;
|
|
|
|
} else if ((mbd->flags&MULTIBOOT_INFO_MEMORY)!=0) {
|
|
|
|
total_mb=((mbd->mem_upper)/1024)+2;
|
|
|
|
mem_map[0][0]=0;
|
|
|
|
mem_map[0][1]=0;
|
|
|
|
} else {
|
|
|
|
write_string("Cannot detect memory. Halting");
|
|
|
|
halt();
|
|
|
|
}
|
|
|
|
}
|
2018-10-19 07:59:38 -05:00
|
|
|
|
2018-11-04 16:12:54 -06:00
|
|
|
void print_memory() {
|
|
|
|
char str[100];
|
|
|
|
if (total_mb>0) {
|
|
|
|
if (total_mb%1024==0) {
|
|
|
|
int_to_ascii(total_mb/1024,str);
|
|
|
|
} else {
|
|
|
|
int_to_ascii(total_mb,str);
|
|
|
|
}
|
|
|
|
write_string(str);
|
|
|
|
if (total_mb%1024==0) {
|
|
|
|
write_string(" GB ");
|
|
|
|
} else {
|
|
|
|
write_string(" MB ");
|
|
|
|
write_string("of memory detected\n");
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
for (int i=0;i<MMAP_ENTRIES;i++) {
|
|
|
|
if (mem_map[i][0]==0&&mem_map[i][1]==0) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
char str[100];
|
|
|
|
write_string("Memory from ");
|
|
|
|
str[0]='\0';
|
|
|
|
hex_to_ascii(mem_map[i][0],str);
|
|
|
|
write_string(str);
|
|
|
|
write_string(" to ");
|
|
|
|
str[0]='\0';
|
|
|
|
hex_to_ascii(mem_map[i][1],str);
|
|
|
|
write_string(str);
|
|
|
|
write_string("\n");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void main(multiboot_info_t* mbd, uint32_t magic) {
|
|
|
|
uint32_t tmp_mbd=(uint32_t)mbd;
|
|
|
|
tmp_mbd+=VIRT_OFFSET;
|
|
|
|
mbd=(multiboot_info_t*)tmp_mbd;
|
|
|
|
init_vga(WHITE,BLACK);
|
2018-10-19 07:59:38 -05:00
|
|
|
write_string("Initialized VGA\n");
|
2018-11-04 16:12:54 -06:00
|
|
|
if (magic!=MULTIBOOT_BOOTLOADER_MAGIC) {
|
|
|
|
write_string("Multiboot magic number is incorrect. Halting.");
|
|
|
|
halt();
|
|
|
|
}
|
|
|
|
get_memory(mbd);
|
|
|
|
print_memory();
|
|
|
|
serial_full_configure(SERIAL_COM1_BASE,12);
|
|
|
|
write_string("Initialized COM1 at 9600 baud\n");
|
|
|
|
isr_install();
|
|
|
|
asm volatile("sti");
|
2018-10-19 07:59:38 -05:00
|
|
|
write_string("Setup interrupts\n");
|
2018-10-20 11:32:14 -05:00
|
|
|
init_gdt();
|
|
|
|
write_string("Setup new GDT\n");
|
2018-11-04 16:12:54 -06:00
|
|
|
if ((mbd->flags&MULTIBOOT_INFO_MODS)!=0) {
|
|
|
|
uint32_t mods_count=mbd->mods_count;
|
|
|
|
if (mods_count>0) {
|
|
|
|
while (mods_count>0) {
|
|
|
|
multiboot_module_t* mods_addr=(multiboot_module_t*)(mbd->mods_addr+VIRT_OFFSET);
|
|
|
|
write_string("Module:");
|
|
|
|
write_string(mods_addr->cmdline);
|
2018-10-28 13:01:00 -05:00
|
|
|
write_string("\n");
|
2018-11-04 16:12:54 -06:00
|
|
|
mods_addr++;
|
|
|
|
mods_count--;
|
|
|
|
}
|
2018-10-28 13:01:00 -05:00
|
|
|
}
|
|
|
|
}
|
2018-11-04 16:12:54 -06:00
|
|
|
// init_keyboard();
|
|
|
|
// write_string("Keyboard initialized\n");
|
|
|
|
// switch_to_user_mode();
|
|
|
|
// syscall_write_string("MYOS V 1.0\n");
|
|
|
|
// while (1) {};
|
2018-10-21 17:45:12 -05:00
|
|
|
}
|
|
|
|
|
2018-11-04 16:12:54 -06:00
|
|
|
void user_input(char* buf) {};
|
|
|
|
void kgets(char* buf) {};
|