toaruos/kernel/main.c

204 lines
6.7 KiB
C
Raw Normal View History

2011-01-29 01:45:27 +03:00
/*
* vim:tabstop=4
* vim:noexpandtab
2011-01-29 01:45:27 +03:00
* Copyright (c) 2011 Kevin Lange. All rights reserved.
*
* Developed by: ToAruOS Kernel Development Team
* http://github.com/klange/osdev
*
* Permission is hereby granted, free of charge, to any person obtaining a copy
* of this software and associated documentation files (the "Software"), to
* deal with the Software without restriction, including without limitation the
* rights to use, copy, modify, merge, publish, distribute, sublicense, and/or
* sell copies of the Software, and to permit persons to whom the Software is
* furnished to do so, subject to the following conditions:
* 1. Redistributions of source code must retain the above copyright notice,
* this list of conditions and the following disclaimers.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimers in the
* documentation and/or other materials provided with the distribution.
* 3. Neither the names of the ToAruOS Kernel Development Team, Kevin Lange,
* nor the names of its contributors may be used to endorse
* or promote products derived from this Software without specific prior
* written permission.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
* CONTRIBUTORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
* FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
* WITH THE SOFTWARE.
*/
2011-01-16 04:01:19 +03:00
#include <system.h>
2011-02-20 04:27:46 +03:00
#include <boot.h>
#include <ext2.h>
2011-01-16 04:01:19 +03:00
/*
* kernel entry point
2011-01-29 01:45:27 +03:00
*
* This is the C entry point for the kernel.
* It is called by the assembly loader and is passed
* multiboot information, if available, from the bootloader.
*
* The kernel boot process does the following:
* - Align the dumb allocator's heap pointer
* - Initialize the x86 descriptor tables (global, interrupts)
* - Initialize the interrupt handlers (ISRS, IRQ)
* - Load up the VGA driver.
* - Initialize the hardware drivers (PIC, keyboard)
* - Set up paging
* - Initialize the kernel heap (klmalloc)
* [Further booting]
*
* After booting, the kernel will display its version and dump the
* multiboot data from the bootloader. It will then proceed to print
* out the contents of the initial ramdisk image.
*/
int main(struct multiboot *mboot_ptr, uint32_t mboot_mag, uintptr_t esp)
2011-01-31 01:59:19 +03:00
{
initial_esp = esp;
2011-02-20 04:27:46 +03:00
enum BOOTMODE boot_mode = unknown; /* Boot Mode */
2011-02-09 23:32:49 +03:00
char * ramdisk = NULL;
2011-02-08 08:51:11 +03:00
if (mboot_mag == MULTIBOOT_EAX_MAGIC) {
2011-03-05 03:18:14 +03:00
/*
* Multiboot (GRUB, native QEMU, PXE)
*/
2011-02-20 04:27:46 +03:00
boot_mode = multiboot;
2011-03-05 03:18:14 +03:00
/*
* Realign memory to the end of the multiboot modules
*/
2011-02-09 23:32:49 +03:00
kmalloc_startat(0x200000);
if (mboot_ptr->flags & (1 << 3)) {
2011-03-05 03:18:14 +03:00
/*
* Mboot modules are available.
*/
2011-02-09 23:32:49 +03:00
if (mboot_ptr->mods_count > 0) {
2011-03-05 03:18:14 +03:00
/*
* Ramdisk image was provided. (hopefully)
*/
uint32_t module_start = *((uint32_t *) mboot_ptr->mods_addr); /* Start address */
uint32_t module_end = *(uint32_t *) (mboot_ptr->mods_addr + 4); /* End address */
ramdisk = (char *)kmalloc(module_end - module_start); /* New chunk of ram for it. */
memcpy(ramdisk, (char *)module_start, module_end - module_start); /* Copy it over. */
2011-02-09 23:32:49 +03:00
}
2011-02-08 08:51:11 +03:00
}
2011-02-20 04:27:46 +03:00
} else {
/*
* This isn't a multiboot attempt. We were probably loaded by
* Mr. Boots, our dedicated boot loader. Verify this...
*/
boot_mode = mrboots;
2011-02-08 08:51:11 +03:00
}
2011-01-29 01:45:27 +03:00
/* Initialize core modules */
2011-01-31 01:59:19 +03:00
gdt_install(); /* Global descriptor table */
idt_install(); /* IDT */
isrs_install(); /* Interrupt service requests */
irq_install(); /* Hardware interrupt requests */
init_video(); /* VGA driver */
2011-01-29 01:45:27 +03:00
/* Hardware drivers */
2011-03-05 03:18:14 +03:00
timer_install(); /* PIC driver */
keyboard_install(); /* Keyboard interrupt handler */
serial_install(); /* Serial console */
2011-01-29 01:45:27 +03:00
/* Memory management */
2011-03-05 03:18:14 +03:00
paging_install(mboot_ptr->mem_upper); /* Paging */
heap_install(); /* Kernel heap */
tasking_install(); /* Multi-tasking */
enable_fpu();
2011-03-30 06:08:56 +04:00
syscalls_install();
2011-01-29 01:45:27 +03:00
/* Kernel Version */
2011-01-22 09:29:04 +03:00
kprintf("[%s %s]\n", KERNEL_UNAME, KERNEL_VERSION_STRING);
2011-01-29 01:45:27 +03:00
2011-02-20 04:27:46 +03:00
if (boot_mode == multiboot) {
2011-02-08 08:51:11 +03:00
/* Print multiboot information */
dump_multiboot(mboot_ptr);
if (mboot_ptr->flags & (1 << 3)) {
2011-03-05 03:18:14 +03:00
/*
* If we have an initial ramdisk, mount it.
*/
2011-02-09 23:32:49 +03:00
if (mboot_ptr->mods_count > 0) {
initrd_mount((uintptr_t)ramdisk, 0);
}
2011-02-08 08:51:11 +03:00
}
/* Parse the command-line arguments */
parse_args((char *)mboot_ptr->cmdline);
2011-02-08 08:51:11 +03:00
}
2011-03-25 08:09:23 +03:00
/*
* All the output we just dumped went to the serial line.
* If you really want it that bad, feel free to remove
* the clear before the rest of the boot process starts.
*
* But really, the kernel shouldn't be outputting all
* sorts of random text on boot up. Only important stuff!
*/
cls();
/*
* Aw man...
*/
2011-03-03 10:39:26 +03:00
fork();
2011-03-25 04:03:52 +03:00
2011-03-30 06:08:56 +04:00
if (getpid() == 1) {
2011-03-29 05:41:17 +04:00
while (1) {
2011-03-25 08:09:23 +03:00
uint16_t hours, minutes, seconds;
get_time(&hours, &minutes, &seconds);
2011-03-25 04:03:52 +03:00
__asm__ __volatile__ ("cli");
2011-03-25 08:50:50 +03:00
/* It would help a lot if I had %.2d */
/* kprintf("[%.2d:%.2d:%.2d]", hours, minutes, seconds); */
if (bochs_resolution_x) {
2011-03-29 05:41:17 +04:00
bochs_write_char('[', bochs_resolution_x - 8 * 10, 0, 0x00FFFFFF, 0x0);
bochs_write_char('0' + hours / 10, bochs_resolution_x - 8 * 9, 0, 0x00FFFFFF, 0x0);
bochs_write_char('0' + hours % 10, bochs_resolution_x - 8 * 8, 0, 0x00FFFFFF, 0x0);
bochs_write_char(':', bochs_resolution_x - 8 * 7, 0, 0x00FFFFFF, 0x0);
bochs_write_char('0' + minutes / 10, bochs_resolution_x - 8 * 6, 0, 0x00FFFFFF, 0x0);
bochs_write_char('0' + minutes % 10, bochs_resolution_x - 8 * 5, 0, 0x00FFFFFF, 0x0);
bochs_write_char(':', bochs_resolution_x - 8 * 4, 0, 0x00FFFFFF, 0x0);
bochs_write_char('0' + seconds / 10, bochs_resolution_x - 8 * 3, 0, 0x00FFFFFF, 0x0);
bochs_write_char('0' + seconds % 10, bochs_resolution_x - 8 * 2, 0, 0x00FFFFFF, 0x0);
bochs_write_char(']', bochs_resolution_x - 8 * 1, 0, 0x00FFFFFF, 0x0);
} else {
store_csr();
set_serial(0);
set_csr(0);
place_csr(70,0);
writech('[');
kprintf("%d", hours / 10);
kprintf("%d", hours % 10);
writech(':');
kprintf("%d", minutes / 10);
kprintf("%d", minutes % 10);
writech(':');
kprintf("%d", seconds / 10);
kprintf("%d", seconds % 10);
writech(']');
restore_csr();
}
2011-03-25 04:03:52 +03:00
__asm__ __volatile__ ("sti");
}
2011-03-25 04:03:52 +03:00
} else {
start_shell();
2011-03-30 06:08:56 +04:00
if (fork()) {
enter_user_mode();
while(1) {
2011-03-30 11:16:30 +04:00
syscall_print("Herp\n");
2011-03-30 06:08:56 +04:00
};
} else {
enter_user_mode();
while(1) {
2011-03-30 11:16:30 +04:00
syscall_print("Derp\n");
2011-03-30 06:08:56 +04:00
};
}
}
2011-01-16 04:01:19 +03:00
return 0;
}