major backup 8.24.21

This commit is contained in:
Brett Weiland 2021-08-24 14:09:29 -05:00
parent bad4b0e9bd
commit 9b22a69655
53 changed files with 3119 additions and 1413 deletions

View File

@ -1,256 +1,256 @@
next
print buddy_size
print pmap->next
print pmap
print *pmap
print *pmap
print pmap
c
info threads
quit
quit
quit
break page.c:198
c
watch pmap
c
print pmap
next
print buddy_size
next
c
print *pmap
quitquit
quit
break page.c
c
quit
break init_pmap
c
next
watch pmap_i
c
print pmap_i
next
quit
break page.c:220
c
quit
break init_pmap
c
next
watch pmap_i
c
print pmap_i
c
print pmap_i
next
print zones[zone_i]
next
break page.c:137
c
next
info b
c
print pmap_i
c
print pmap_i
c
print pmap_i
next
print *pmap
print pmap
quit
break page.c:220
c
print pmap
print *pmap
quit
break debug_pmap
c
c
quit
break page.c:179
c
next
print buddy_bit
print budorder
c
quit
quit
break page.c:147
c
next
print threshold_bitsize
print (uint64_t)threshold_bitsize
print pmap_i
print *pmap
print threshold_bitsize
print buddy_bitsize
next
print buddy_bit
print buddy_bit
print buddy_bitlen
print buddy_bitlen[0][7]
print buddy_bitlen[0][7]
next
quit
break panic
c
next
x ebp
print $ebp
info reg ebp
x 0x1ffff8
stack 50
quit
break debug_pmap
c
info reg rbp
info reg rbp
x 0x1fffe8
print (void*)&debug_pmap
next
stack
stack 50
info reg rbp
x 0x1fffe8
x/20 0x1fffe8
quit
quit
break panic
c
next
info stack
stack
next
info rsp
x 0x1fff28
x *0x1fff28
stack 50
quit
break panic
c
c
stack
stack 50
c
stack 50
info reg rbp
quit
break panic
c
next
print stack
stack 50
stack --help
stack 0x20
stack 100 0x20
info reg rbp
info reg rbp + 1
info reg rbp
x 0x1fff58+8
x 0x1fff58-8
x 0x1fff58+1
quit
break main
c
next
stepi
next
stepi
x 0x1fffe8 - 8
x 0x1fffe8 + 8
x 0x1fffe8
x 0x1fffc8-8
x 0x1fffe8-8
next
info threads
step
print rsp
stack
x $rbp-8
next
x $rbp-8
x $rbp+8
stack 50
stack 100
stack 90
stack 90 90
stack 90
x $ebp
x $ebp-8x $eb
x $ebp-8
x $ebp+8
c
print cores
print cores_active
info threads
c
info threads
quit
break panic.c:27
c
info threads
c
info threads
quit
b smp_boot
c
qit
quit
b smp.c:59
c
next
print *Frame
print frame
print *frame
print frame->function_base
print/x frame->function_base
x 0x103ef2
x frame->function_base
x frame->lastframe->function_base
info threads
next
info threads
next
info threads
info threads
info threads
quit
break panic.c:29
b smp.c:65
c
print *frame
next
info threads
info thread 2
quit
b timer.c:57
c
quit
break page.c:30
b timer.c:57
c
quit
break panic.c:30
b smp.c:57
c
print *(uint32_t *)&icr
print/x *(uint32_t *)&icr
info threads
next
info threads
next
info threads
next
info threads
next
info threads
next
info threads
next
info threads
print/x *(uint32_t *)&icr
pirnt (void *)0x8000
prit (void *)0x8000
print (void *)0x8000
print *(void *)0x8000
print __load_start_smp_bootloader
print &__load_start_smp_bootloader
print (void *)&__load_start_smp_bootloader
print (void *)&__load_stop_smp_bootloader
quit
print &__load_start_smp_bootloader
print &__load_stop_smp_bootloader
print &__load_stop_smp_bootloader
print smp_bootstrap
print smp_bootloader
print &__load_stop_smp_bootloader
print __load_stop_smp_bootloader
print &__load_stop_smp_bootloader
print smp_bootstrap_corecount
b smp_trampoline
hb smp_trampoline
c
pirnt *frame
print *frame
d
d
hb smp_trampoline
c
print *frame
c
print *frame
c
print *frame
c
print *frame
x 0x1ffff8
info reg rip
quit
hb smp_trampoline
c
quit
c
x 831e
x 0x831e
x 0x103f58
x/i 0x831e
x/i 0x831e-8
x/i 0x831e-16
x/10i 0x831e
x/10i 0x831e-16
x/10i 0x831e-1
x/10i 0x831e-
x/10i 0x831e
x/10i 0x831e-8
x/10i 0x831e+8
x/10i 0x831e
quit
break 0x831e
hb smp_trampoline
c
info threads
hexdump 0
c
c
c
c
c
c
c
c
c
c
c
c
c
c
c
cc
c
c
c
c
c
c
c
d
c
quit
b *0x831e
b smp_trampoline
c
hb smp_trampoline
d 2
c
c
c
c
c
c
c
c
c
c
c
c
c
c
c
c
c
c
c
c
c
c
c
c
c
c
c
c
c
c
c
c
c
c
c
quit
c
info threads
quit
c
print *frame
print *frame
x *frame
x/1 *frame
*frame
info threads
quit
break main.c
info threads
next
next
print 0x8000 / 0x1000
next
quit
quit
next
info threads
next
info threads
next
info threads
quit
next
info threads
next
info threads
next
info threads
next
info threads
next
info threads
next
info threads
quit
hb 0
c
quit
break main
c
stack
c
next
next
next
print icr
next
next
info threads
quit
next
info threads
next
info threads
next
info threads
next
info threads
quit
b smp.c:60
c
next
info threads
next
info threads
quit
b memcpy
c
print dest
x 0xffff800000008000
print 0xffff800000008000
print/x 0xffff800000008000
print n
b 54
c
next
info reg
info reg rax
info reg rbx
inf reg rdi
x/i 0xffff800000000000
x/i 0xffff800000008000
smp_trampoline
hexdump 0xffff800000008000
quit
b memcpy
c
print dest
hexdump 0xffff800000008000
b 64
c
hexdump 0xffff800000008000
print __load_start_smp_bootloader
hexdump 0xffff800000000800
quit
info reg rsp
quit

View File

@ -1,437 +0,0 @@
#include <printf.h>
#include <paging.h>
#include <stdint.h>
#include <libc.h>
#include <limits.h>
#include <kernel.h>
void debug_print_memory() {
struct memory_table *memtable = (struct memory_table *)&_meminfo_loc;
printf(" __________________________________________________________________________\n");
printf("| type\tstart\t\t\tend\t\t\tsize\t\t |\n");
printf("|--------------------------------------------------------------------------|\n");
for(unsigned int i = 0; memtable[i].length > 0; i++) {
printf("| %u %u\t0x%p\t0x%p\t0x%p |\n", memtable[i].type, memtable[i].ACPI, memtable[i].base, (memtable[i].base + memtable[i].length), memtable[i].length);
}
printf("----------------------------------------------------------------------------\n");
}
void debug_pmap() {
struct phys_map* pmap;
int pmap_i = 0, order;
uint64_t buddy_size, blong_i, bbit_i, buddy_chunksize, omit_cnt;
printf("Maximum buddy order: %u (up to %#x sized chunks)\n", MAX_BUDDY_ORDER, (0x1000 << MAX_BUDDY_ORDER - 1));
for(pmap = (struct phys_map*)&_stage2_pagetable; pmap != 0; pmap = pmap->next) {
printf("Table %u:\n"
"\tPhysical Start:\t%#p\n"
"\tTable location:\t%#p\n", pmap_i, pmap->zone_paddr, pmap);
for(order = 0; order <= MAX_BUDDY_ORDER - 1; order++) {
buddy_chunksize = (0x1000 << order); //TODO just put it in the for loop
buddy_size = (((order == MAX_BUDDY_ORDER - 1)
? (uint64_t *)pmap->next : pmap->buddy[order + 1]) - pmap->buddy[order]);
printf("\tbuddy[%u]:\n"
"\t\tAddress:\t%#x\n"
"\t\tSize:\t\t%u\n"
"\t\tBuddies:\t\t\n", order, pmap->buddy[order], buddy_size);
omit_cnt = 0;
for(blong_i = 0; blong_i < buddy_size; blong_i++) {
for(bbit_i = 0; bbit_i < 64; bbit_i++) {
if(*(pmap->buddy[order] + blong_i) & ((uint64_t)1 << bbit_i)) {
if((omit_cnt < 20) || (blong_i == buddy_size - 1)) {
printf("address %#x\tbit %u: %x\t is free\n",
pmap->buddy[order] + blong_i, bbit_i, pmap->zone_paddr + ((blong_i * 64) + bbit_i) * buddy_chunksize);
}
omit_cnt++;
if(omit_cnt == 20) {
printf("\t\t\t[more entries ommited]\n");
}
}
}
}
print_next_buddy: ;
}
pmap_i++;
}
}
/*
* part 1:
* init tables (complete)
*
* part 2: setting the actual entires
* if entry contains table, set 0, set threshold
* else:
* if entry:
*/
// init_memory revision
// rules:
/**
void init_pmap() {
struct memory_table *memtable = (struct memory_table *)&_meminfo_loc;
struct phys_map *pmap = (struct phys_map*)&_stage2_pagetable;
struct phys_map *last_pmap;
unsigned int i, x;
uint64_t budentry_len, pmap_chunksize, total_pmap_len = 0;
void *y;
void *paged_mem = &_stage2_pagetable + 0x200000;
map_page(&_stage2_pagetable, &_stage2_pagetable, PAGE_SIZE_2M);
for(i = 0; memtable[i].length > 0; i++) {
if((memtable[i].type == MEM_AVAILABLE) && (memtable[i].ACPI & 1)) {
total_pmap_len += 88; //it'd be nice to find a cleaner way
//make sure we don't overwrite what we have so far of the kernel
if((memtable[i].base <= (void*)&_stage2_pagetable) && (memtable[i].base + memtable[i].length >= (void *)&_stage2_pagetable)) {
pmap->chunk_start = &_stage2_pagetable;
pmap->chunk_size = memtable[i].length - (pmap->chunk_start - memtable[i].base);
}
else {
pmap->chunk_start = memtable[i].base;
pmap->chunk_size = memtable[i].length;
}
for(x = 0; x < 8; x++) {
pmap->bsize[x] = ceil((pmap->chunk_size / (0x1000 * (1 << x))) / (double)64);
total_pmap_len += pmap->bsize[x] * 8;
}
pmap->next = (void*)&_stage2_pagetable + total_pmap_len;
while((void*)pmap->next + sizeof(struct phys_map) >= paged_mem) {
//do check here if nessesary
map_page(paged_mem, paged_mem, PAGE_SIZE_2M);
paged_mem += 0x200000;
}
last_pmap = pmap;
pmap = pmap->next;
}
}
last_pmap->next = 0; //I wonder if there's a better way to do this
pmap = (struct phys_map*)&_stage2_pagetable;
}
**/
void init_pmap() {
struct memory_table *zones = (struct memory_table *)&_meminfo_loc;
struct phys_map *pmap = (struct phys_map*)&_stage2_pagetable;
struct phys_map *last_pmap = (struct phys_map*)&_stage2_pagetable;
unsigned int zone_i, pmap_i = 0;
int budorder;
//we keep this info out of the struct because we won't need it after setting up
uint64_t zone_len[MAX_ZONE_CNT], buddy_bitlen[MAX_ZONE_CNT][MAX_BUDDY_ORDER];
uint64_t *buddy_end;
uint64_t pmap_size, pmap_bbitsize, pmap_blongsize, buddy_size, pmap_bit;
uint64_t threshold_longsize = 0, threshold_bit, threshold_end, threshold_bitsize;
uint64_t deleteme_last_p_bits;
void *paged_mem = (void *)&_stage2_pagetable + 0x200000;
map_page(&_stage2_pagetable, &_stage2_pagetable, PAGE_SIZE_2M);
for(zone_i = 0; zones[zone_i].length > 0; zone_i++) {
if((zones[zone_i].type == MEM_AVAILABLE) && (zones[zone_i].ACPI & 1)) {
//hopefully this should never happen... but x86 is routy.
//I should dig into the docs to check before removing this.
//We also could forget about MAX_ZONE_CNT if we did.
if(zone_i >= MAX_ZONE_CNT) {
printf("Only %u zones can be used! Modify MAX_ZONE_CNT in paging.h to use all memory.\n", MAX_ZONE_CNT);
break;
}
if((zones[zone_i].base <= (void*)&_stage2_pagetable) &&
(zones[zone_i].base + zones[zone_i].length >= (void *)&_stage2_pagetable)) {
pmap->zone_paddr = &_stage2_pagetable;
zone_len[pmap_i] = zones[zone_i].length - (pmap->zone_paddr - zones[zone_i].base);
}
else {
pmap->zone_paddr = zones[zone_i].base;
zone_len[pmap_i] = zones[zone_i].length;
}
pmap->buddy[0] = (void *)pmap + sizeof(*pmap);
for(budorder = 1; budorder < MAX_BUDDY_ORDER; budorder++) {
buddy_bitlen[pmap_i][budorder - 1] = GET_BUDDY_BITLEN(zone_len[pmap_i], budorder - 1);
pmap->buddy[budorder] = (uint64_t *)pmap->buddy[budorder - 1] +
LSIZE_FROM_BITLEN(buddy_bitlen[pmap_i][budorder - 1]);
}
buddy_bitlen[pmap_i][MAX_BUDDY_ORDER - 1] = GET_BUDDY_BITLEN(zone_len[pmap_i], MAX_BUDDY_ORDER - 1);
pmap->next = (void *)pmap->buddy[MAX_BUDDY_ORDER - 1] +
(LSIZE_FROM_BITLEN(buddy_bitlen[pmap_i][MAX_BUDDY_ORDER - 1]) * 8);
pmap = pmap->next;
pmap_i++;
last_pmap = pmap;
//allocates by an extra sizeof(struct phys_map),
//but were about to discard anyway
while((void *)pmap + sizeof(*pmap) >= paged_mem) {
map_page(paged_mem, paged_mem, PAGE_SIZE_2M);
paged_mem += 0x200000;
}
}
}
pmap_size = (void*)(pmap) - (void*)&_stage2_pagetable;
if(pmap_size >= zone_len[0]) panic(); //TODO debugging
pmap = (struct phys_map*)&_stage2_pagetable;
// Honestly, this section makes me feel like Yandere Dev. It's ugly.
// I've been rewriting this function forever, so I'm deciding to slam it out start to finish.
// I know there's a lot of repeated if statements. I know it hurts.
// But fear not.
// I'll come back and rewrite this part when I've gotten a break from memory management.
for(pmap = (struct phys_map*)&_stage2_pagetable; pmap != 0; pmap++) {
for(budorder = MAX_BUDDY_ORDER - 1; budorder >= 0; budorder--) {
pmap_bbitsize = ceil((float)pmap_size / ((uint64_t)0x1000 << budorder));
pmap_blongsize = pmap_bbitsize / 64;
if(budorder == MAX_BUDDY_ORDER - 1) {
buddy_end = (uint64_t *)pmap->next - 1;
threshold_bitsize = ((pmap_blongsize * 64) + pmap_bbitsize) * 2;
threshold_bitsize = UINT64_MAX;
}
else {
buddy_end = pmap->buddy[budorder + 1] - 1;
threshold_longsize = threshold_bitsize / 64;
threshold_end = threshold_longsize + 1;
}
pmap_bit = pmap_bbitsize & 63;
buddy_size = buddy_end - pmap->buddy[budorder];
if(pmap_bbitsize >= BITLEN_FROM_LSIZE(buddy_size)) {
//is this nessesary?
bzero(pmap->buddy[budorder], buddy_size * 8);
}
else {
if(budorder == MAX_BUDDY_ORDER - 1) {
if(pmap_blongsize) bzero(pmap->buddy[budorder], (pmap_blongsize - 1) * 8);
if(pmap_bit) {
*(pmap->buddy[budorder] + pmap_blongsize) = ~(((uint64_t)1 << pmap_bit) - 1);
}
else {
*(pmap->buddy[budorder] + pmap_blongsize) = UINT64_MAX;
}
if(pmap_blongsize + 1 == buddy_size) {
//TODO why did I have this conditional? Do I need it later? Check on desktop before removing
if(buddy_bitlen[0][budorder]) {
*(pmap->buddy[budorder] + pmap_blongsize) &=
((uint64_t)1 << (buddy_bitlen[0][budorder] & 63)) - 1;
}
}
else {
memset(pmap->buddy[budorder] + pmap_blongsize + 1, UINT8_MAX,
(void *)buddy_end - (void *)pmap->buddy[budorder] - 8);
*buddy_end = ((uint64_t)1 << (buddy_bitlen[0][budorder] & 63)) - 1;
}
}
else {
if(threshold_longsize) bzero(pmap->buddy[budorder], (threshold_longsize - 1) * 8);
if(threshold_bitsize > pmap_bbitsize)
*(pmap->buddy[budorder] + threshold_longsize) = ((uint64_t)1 << ((threshold_bitsize - 1) & 63));
if(buddy_size - threshold_longsize)
bzero(pmap->buddy[budorder] + threshold_longsize + 1, buddy_size - threshold_longsize);
*buddy_end = ((uint64_t)1 << ((buddy_bitlen[0][budorder] & 63) - 1));
}
threshold_bitsize = ((pmap_blongsize * 64) + pmap_bbitsize) * 2;
}
}
}
// for(pmap = pmap->next; pmap != 0; pmap++) {
// }
}
// for(budlong_ptr = pmap->buddy[budorder]; budlong_ptr < (uint64_t *)pmap->next - 1; budlong_ptr++) {
/**
//uses buddy system allocation
void init_memory() {
struct memory_table *memtable = (struct memory_table *)&_meminfo_loc;
struct phys_map *map = (struct phys_map*)&_stage2_pagetable;
struct phys_map *lasttable;
unsigned int i, buddy_chunksize;
uint64_t buddy_bitsize, prev_buddy_bsize, ppt_size;
uint64_t *buddy_ptr, *budentry_end, *threshold;
uint64_t buddy_size64;
map_page((void*)&_stage2_pagetable, (void*)&_stage2_pagetable, PAGE_SIZE_2M);
void *next_page = (void *)0x400000;
// at this point, we are declaring our header and kernel itself as free (so don't forget to fix that!)
//TODO all these for loops are unsanitary.
//We hae a lot of branches in the for loops,
//And we re-do work to allocate the page-table.
//I'll clean this up when I'm certain my buddy system works.
for(i = 0; memtable[i].length > 0; i++) {
if((memtable[i].type == MEM_AVAILABLE) && (memtable[i].ACPI & 1)) {
map->chunk_start = (void*)memtable[i].base;
map->chunk_size = memtable[i].length;
buddy_ptr = (void*)&map->buddies;
for(buddy_chunksize = 0x1000; buddy_chunksize < 0x100000; buddy_chunksize *= 2) {
buddy_bitsize = memtable[i].length / buddy_chunksize;
buddy_size64 = ceil(buddy_bitsize / (double)64);
if((void*)&buddy_ptr[buddy_size64] + 24 >= next_page) {
map_page(next_page, next_page, PAGE_SIZE_2M);
next_page += 0x200000;
}
printf("buddy\t%x\theader bitsize\t%u\theader longsize\t%u\tbuddy start\t%p",\
buddy_chunksize, buddy_bitsize, buddy_size64, buddy_ptr);
if(((buddy_bitsize * 2) != prev_buddy_bsize) && (buddy_chunksize != 0x1000)) {
buddy_ptr[-1] |= ((uint64_t)1 << ((prev_buddy_bsize & 63) - 1));
printf("\tlast:\t%lx", buddy_ptr[-1]);
}
printf("\n");
if(buddy_chunksize < 0x80000) {
bzero(buddy_ptr, buddy_size64 * 8);
prev_buddy_bsize = buddy_bitsize;
}
else if(buddy_size64 % buddy_bitsize) {
memset(buddy_ptr, 0xff, (buddy_size64 - 1) * 8);
buddy_ptr[buddy_size64] |= 1 << buddy_bitsize;
}
else memset(buddy_ptr, 0xff, buddy_size64);
buddy_ptr += buddy_size64;
}
//this feels kind of gross
lasttable = map;
map->next = (struct phys_map*)buddy_ptr;
map = (struct phys_map*)buddy_ptr;
}
}
lasttable->next = (void*)0;
map = (struct phys_map*)&_stage2_pagetable;
//now we will allocate the table out of itself so we don't mess things up.
//We can't just call palloc(), we need to allocate out of the first available pages (where the pages are)
//we should clean this whole gross function
/**
ppt_size = (uint64_t)((void *)buddy_ptr - (void *)&_stage2_pagetable);
threshold = (void*)UINT64_MAX;
int buddy_bit;
for(buddy_chunksize = 0x80000; buddy_chunksize >= 0x1000; buddy_chunksize /= 2) {
//this means that our table must fit into the first page table. Fixme later, low priotrity
buddy_size64 = ceil((map->chunk_size / buddy_chunksize) / (double)64);
budentry_end = buddy_ptr;
for(buddy_ptr -= buddy_size64; buddy_ptr <= budentry_end; buddy_ptr++) {
if(buddy_ptr > threshold) {
buddy_ptr = budentry_end;
continue;
}
if(
}
}
}
**/
//TODO this function was deleted due to it being wrong.
//I'll create it once I have the physical paging prerequisite set up.
void create_pagetable_stage2(uint64_t free_mem) {
}
/**
* BIG TODO:
* Paging turned out to be simpler then I thought. I've temporarily fixed the code, but needs to be rewritten/simplified.
* Let's get rid of those nasty GOTOs if we can.
* Also, once we get physical memory allocator up and running, impliment that in this function.
**/
bool map_page(void *virtual_addr, void *physical_addr, uint8_t size) {
//printf("map page called\n");
uintptr_t va_ptr = (uintptr_t)virtual_addr;
uintptr_t pa_ptr = (uintptr_t)physical_addr;
if((va_ptr % (1 << size)) || (pa_ptr % (1 << size))) {
return 0;
}
page_table *table = (page_table *)PAGEMAP_LOCATION;
int pte_i = (va_ptr >> 12) & 0x1ff;
int pde_i = (va_ptr >> 21) & 0x1ff;
int pdpe_i = (va_ptr >> 30) & 0x1ff;
int pml4e_i = (va_ptr >> 39) & 0x1ff;
if(table->pml4e[pml4e_i].present) {
if(table->pml4e[pml4e_i].base_ptr != (uintptr_t)&table->pdpe[pdpe_i] >> 12) goto error;
if(table->pdpe[pdpe_i].present) {
if(size == PAGE_SIZE_1G) {
if(table->pdpe[pdpe_i].base_ptr == ((uintptr_t)pa_ptr >> 30 & 0x1ff))
return true;
goto error;
}
if(table->pdpe[pdpe_i].base_ptr != (uintptr_t)&table->pde[pde_i] >> 12) goto error;
if(table->pde[pde_i].present) {
if(size == PAGE_SIZE_2M) {
if(table->pde[pde_i].base_ptr == ((uintptr_t)pa_ptr >> 21 & 0x1ff))
return true;
goto error;
}
if(table->pde[pde_i].base_ptr != (uintptr_t)&table->pte[pte_i] >> 12) goto error;
if(table->pte[pte_i].present) {
if(table->pte[pte_i].base_ptr != ((pa_ptr >> 12) & 0x1ff)) goto error;
return true;
}
else goto mod_page_pte;
}
else goto mod_page_pde;
}
else goto mod_page_pdpe;
}
else {
table->pml4e[pml4e_i].base_ptr = (uintptr_t)&table->pdpe[pdpe_i] >> 12;
table->pdpe[pml4e_i].read_write = 1;
table->pml4e[pml4e_i].present = 1;
mod_page_pdpe:
table->pdpe[pdpe_i].read_write = 1;
//TODO you just found out things are a lot more simple then you thought!
if(size == PAGE_SIZE_1G) {
table->pdpe[pdpe_i].size = 1;
table->pdpe[pdpe_i].base_ptr = pa_ptr >> 12;
table->pdpe[pdpe_i].present = 1;
return true;
}
table->pdpe[pdpe_i].base_ptr = (uintptr_t)&table->pde[pde_i] >> 12;
table->pdpe[pdpe_i].present = 1;
mod_page_pde:
table->pde[pde_i].read_write = 1;
if(size == PAGE_SIZE_2M) {
table->pde[pde_i].size = 1;
table->pde[pde_i].base_ptr = pa_ptr >> 12;
table->pde[pde_i].present = 1;
return true;
}
table->pde[pde_i].base_ptr = (uintptr_t)&table->pte[pte_i] >> 12;
table->pde[pde_i].present = 1;
mod_page_pte:
table->pte[pte_i].base_ptr = pa_ptr >> 12;
table->pte[pte_i].read_write = 1;
table->pte[pte_i].present = 1;
return true;
}
error:
printf("Page allocation error!\n");
return false;
}

View File

@ -0,0 +1,149 @@
bios_disk:
.init:
mov [boot_device], dl
mov ah, 0x8
int 0x13
;TODO defaults
jc .init_use_defaults
mov [max_head], dh
mov [max_sector], cl
and WORD [max_sector], 0x3f
mov [max_cylinder], cl
shl WORD [max_cylinder], 8
or [max_cylinder], ch
and WORD [max_cylinder], 0x3ff
ret
.init_use_defaults:
mov BYTE [max_sector], 63
mov WORD [max_head], 15
mov BYTE [max_cylinder], 1023
ret
; Return:
; CH = low eight bits of maximum cylinder number
; CL = maximum sector number (bits 5-0), high two bits of maximum cylinder number (bits 7-6)
; DH = maximum head number
.load_sectors_v2:
;AH = 02h
;AL = number of sectors to read (must be nonzero)
;CH = low eight bits of cylinder number
;CL = sector number 1-63 (bits 0-5) high two bits of cylinder (bits 6-7)
;DH = head number
;DL = drive number (bit 7 set for hard disk)
;ES:BX -> data buffer
; I don't understand i8086. The base pointer needs to be above the stack pointer,
; yet the stack grows down,
; and we can't address using sp, or by subtraction.
; it's like it's forcing us to use the whole push/pop thing.
sub sp, 16
mov bp, sp
; sector: rbp + 0 (uint8_t)
; head: rbp + 1 (uint8_t)
; cylinder: rbp + 2 (uint16_t)
; sector_cnt: rbp + 4 (uint16_t)
; destination: rbp + 6 (uint32_t)
; set up variables ______________________________________________
;destination to stack
mov [bp + 6], edi
; find init chs ________________________________________________
; divide sector_start / max_sectors
; TODO test for overflow
mov ax, si
shr esi, 16
mov dx, si
;find sector
div DWORD [max_sector]
inc dx
mov [bp], edx ; remainder is sector variable
;find head and cylinder
mov dx, 0
div DWORD [max_head]
mov [bp + 1], dx ; head
mov [bp + 2], ax ; cylinder
.sector_for_loop:
mov [bp + 4], cx
; do bios call ___________________________________________________
mov ah, 2
mov al, 1
; sets up ch and cl
mov cx, [bp + 2]
xchg ch, cl
shl cl, 6
or cl, [bp]
mov dh, [bp + 1]
mov dl, [boot_device]
mov bx, 0x500 ; where we store the sector temporarily
int 0x13
jc .read_failed
; mempcy __________________________________________________________
; we move it to the place at edi, which I don't think changes
mov esi, 0x500
mov ecx, 512
a32 rep movsb
; recalc chs ______________________________________________________
; need to test for more overflows I think
inc WORD [bp]
mov al, [bp]; increase the sector we're on
cmp al, [max_sector]
jbe .continue_loop
mov WORD [bp], 1
inc WORD [bp + 1]
mov al, [bp + 1]
cmp al, [max_head]
jbe .continue_loop
mov WORD [bp + 1], 0
inc DWORD [bp + 2]
; loopin back ____________________________________________________
.continue_loop:
mov cx, [bp + 4]
loop .sector_for_loop
add bp, 16
mov sp, bp
ret
.read_failed:
mov bx, .loadsectors_error
call bios_print
cli
hlt
.loadsectors_error: db "Error loading required sector!", 0
boot_device: db 0
max_sector: db 0
max_cylinder: dw 0
max_head: db 0

View File

@ -1,40 +0,0 @@
bios_disk:
.load_sectors:
pusha
mov es, dx
; TODO when our kernel gets to big to fit in one track, we need to just switch over to LHS with extended bios functions.
mov ah, 0x02 ; read disc sectors
mov ch, 0x00 ; track 0
mov dh, 0x00 ; head 0
mov dl, [stage0.boot_device]
int 0x13
jc .failed
mov ah, 0
popa
ret
.failed:
debug:
mov bx, .loadsectors_error
mov cx, 0
call bios_print
push 0
mov al, ah
mov ah, 0 ; you need to clean up the bios print function!
push ax
mov cx, 1
mov bx, sp
call bios_print
mov ah, 1
popa
ret
.loadsectors_error: db "Error loading sectors: ", 0

View File

@ -1,58 +0,0 @@
;TODO fix null problem, allow passing value insted of pointer
bios_print:
pusha
mov ah, 0x0e
.print_loop:
mov al, [bx]
cmp al, 0
je .fini
test cx, cx ; if cx is zero, ascii, otherwise hex string
jne .print_hex
int 0x10
.print_hex_ret:
inc bx
jmp .print_loop
.fini:
mov al, 0xd
int 0x10
mov al, 0xa
int 0x10
popa
ret
.print_hex:
mov al, '0'
int 0x10
mov al, 'x'
int 0x10
mov al, [bx]
shr al, 4
call .bios_print_nibble
mov al, [bx]
and al, 0x0f
call .bios_print_nibble
mov al, ' '
int 0x10
jmp .print_hex_ret
.bios_print_nibble:
cmp al, 9
ja .print_hex_letter
add al, 0x30
int 0x10
ret
.print_hex_letter:
add al, 0x57
int 0x10
ret

View File

@ -1,11 +1,11 @@
[bits 16]
[extern _kernel_size]
[extern _kernel_sector_size]
[extern _bootloader_stage1_size]
[extern _kernel_loc]
[extern _kernel_s1_loc]
jmp stage0
times 3-($-$$) db 0x90 ; a temporary dirty fix to emulate a floppy disk insted of a hard risk
times 59 db 0 ; (TODO support hard disks)
times 3-($-$$) db 0x90
times 59 db 0
stage0:
@ -13,44 +13,49 @@ jmp 0:.entry
.entry:
mov ax, 0
mov ds, ax
mov bp, ax
mov ds, ax
mov es, ax
mov ax, 0x8fc0
;stack segments
mov ax, 0x7000
mov ss, ax
mov ax, 0xfffe
mov ax, 0xfff0
mov sp, ax
mov bp, 0
mov al, 0x92
or al, 2
out 0x92, al
mov [.boot_device], dl
call bios_disk.init
mov bx, .loadstage2_msg
mov cx, 0
call bios_print
; TODO put in an error message and a maximum fail count
.load_stage1:
mov al, _bootloader_stage1_size
mov cl, 0x2 ; read sector 2
mov dx, 0x0 ; dest segment 0
mov bx, 0x7e00 ; dest offst 0
call bios_disk.load_sectors
;TODO LOAD STAGE ONE HERE
; we start on 0x100000
; esi: sector start
; cx: sector count
; edi: memory desination
mov esi, 1 ; 2nt sector
mov ecx, _bootloader_stage1_size
mov edi, 0x7e00 ; desination
call bios_disk.load_sectors_v2
jmp mbr_end.entry
.boot_device: db 0
.loadstage2_msg: db "Loading (stage 2) bootloader...", 0
%include "bootloader/bios_functions/bios_disk.asm"
%include "bootloader/bios_functions/print.asm"
%include "bootloader/bios_disk.asm"
%include "bootloader/print.asm"
times 510 - ($-$$) db 0
dw 0xaa55
@ -66,15 +71,12 @@ boot_msg:
mbr_end:
.entry:
cli
; entering unreal mode
mov bx, boot_msg.stage2_loaded
mov cx, 0
call bios_print
cli
push ds
lgdt [protected_gdt.descriptor]
mov eax, cr0
@ -83,52 +85,29 @@ mov cr0, eax
jmp $+2
mov bx, 0x10 ; descriptor 2, the data descriptor
mov ds, bx ; put it into the segment register
mov es, bx ;
mov bx, 0x10
mov ds, bx
mov es, bx
and al, 0xfe
mov cr0, eax
pop ds
mov bx, 0
mov ds, bx
mov es, bx
sti
;we are now in unreal mode
mov cl, 5
mov edi, _kernel_loc
.loop:
mov al, 0x1 ; sector count
mov dx, 0x0 ; dest addr segment
mov bx, 0x500 ; dest addr offset
call bios_disk.load_sectors
inc cl
push cx
mov esi, 0x500
mov ecx, 512
a32 rep movsb
nop
pop cx
cmp cl, _kernel_size+2
je .loop_end
jmp .loop
.loop_end:
mov esi, _bootloader_stage1_size
mov edi, 0x100000
mov ecx, _kernel_sector_size
call bios_disk.load_sectors_v2
mov bx, boot_msg.kernel_loaded
mov cx, 0
call bios_print
call detect_arch
call vbe_init
;call vbe_init
done:
call enter_longmode

View File

@ -1,4 +1,4 @@
[extern _meminfo_loc]
[extern _meminfo]
detect_arch:
; detect CPUID
@ -19,6 +19,7 @@ cpuid
and edx, 0x0020000000
jz .print_long_error
; APIC
mov eax, 0x00000001
cpuid
@ -31,14 +32,13 @@ jz .print_sse_error
mov bx, .no_error
mov cx, 0
call bios_print
; get memory mappings
mov ax, 0
mov es, ax
mov ax, _meminfo_loc
mov ax, _meminfo
mov di, ax
mov eax, 0xe820
@ -64,50 +64,50 @@ je .mem_detected
jmp .mem_detect_loop
.mem_detected:
add di,24
mov ecx, 24
mov al, 0
rep stosb
ret
.print_cpuid_error:
mov bx, .cpuid_error
mov cx, 0
call bios_print
jmp $
.print_long_error:
mov bx, .arch_error
mov cx, 0
call bios_print
jmp $
.print_apic_error:
mov bx, .apic_error
mov cx, 0
call bios_print
jmp $
.print_mem_error:
mov bx, .mem_error
mov cx, 0
call bios_print
jmp $
.print_sse_error:
mov bx, .sse_error
mov cx, 0
call bios_print
jmp $
.cpuid_error:
db "No CPUID capabilitity", 0x0
db "Somehow, your CPU doesn't even support the ability to detect its abilities. I'm gonna go out on a limb and say your cpu is too old to run this OS.", 0x0
.arch_error:
db "This operating system was compiled to run in 64 bit mode!", 0x0
db "This operating system requires a x64 processor.", 0x0
.apic_error:
db "No apic support", 0x0
db "No APIC support.", 0x0
.mem_error:
db "Could not get information on memory!", 0x0
db "Couldn't obtain memory map.", 0x0
.sse_error:
db "This OS requires SSE", 0x0
db "This OS requires the SSE instruction set.", 0x0
.no_error:
db "CPU info gathered!", 0x0
db "CPU meets requirements to boot!", 0x0

View File

@ -1,14 +1,14 @@
[extern main]
[extern _kernel_stack_loc]
[extern _stage1_pagetable]
[extern _kernel_page_size]
enter_longmode:
cli
; TODO check if a20 is already set
mov al, 0x92
or al, 2
out 0x92, al
;enter 32 bit mode
lgdt [protected_gdt.descriptor]
mov eax, cr0
or eax, 0x1
@ -30,44 +30,56 @@ mov es, ax
mov fs, ax
mov gs, ax
mov edi, _stage1_pagetable ; 0x3000
mov cr3, edi
mov eax, 0
mov ecx, 0xc00 ; 0x1000
rep stosd
mov edi, cr3
mov edi, 0x10000
mov cr3, edi
mov eax, 0
mov ecx, 0x1800 ; bzero 6 pages
rep stosd
mov DWORD [0x10000], 0x11003 ; pml4e[0] = pdpe
mov DWORD [0x11000], 0x12003 ; pdpe[0] = pde
mov DWORD [0x12000], 0x83 ; pde[0] = pte
mov DWORD [0x10ff8], 0x13003
mov DWORD [0x13ff0], 0x14003
mov DWORD [0x14000], 0x15003
mov eax, 0x100003
mov ebx, 0
mov ecx, _kernel_page_size
.kernel_load_loop:
mov DWORD [0x15000 + ebx], eax
add ebx, 8
add eax, 0x1000
loop .kernel_load_loop
mov DWORD [edi], _stage1_pagetable + 0x1003 ; pml4e[0] = pdpe
add edi, 0x1000
mov DWORD [edi], _stage1_pagetable + 0x2003 ; pdpe[0] = pde
add edi, 0x1000
mov DWORD [edi], 0x83 ; pde[0] = pte
mov eax, cr4
or eax, 0x620
; PAE, OSFXSR, OSXMMEXCPT
or eax, 1 << 5 | 1 << 9 | 1 << 10
mov cr4, eax
;end of setting up pages
;testing to see if NX bit is available.
;If it's not and we enable it, it will cause pagefaults on read
mov eax, 0x80000001
cpuid
and edx, 1 << 20
shr edx, 9
mov ecx, 0xc0000080
rdmsr
or eax, 1 << 8
or eax, 1 << 8 | 1 << 11
or eax, edx
wrmsr
mov eax, cr0
or eax, 1 << 31 | 1 << 0 ; this is where we set paging and protected mode (respectively)!
mov cr0, eax
mov ecx, 0xc0000080
rdmsr
or eax, 1 << 8
wrmsr
mov eax, cr0
and ax, 0xfffb
or eax, 0x80000002
or eax, 1 << 31 | 1 << 0;
and ax, ~(1 << 2)
mov cr0, eax
@ -76,8 +88,10 @@ lgdt [long_gdt.descriptor]
jmp LONG_CODE_SEGMENT:enter_kernel
enter_kernel:
bits 64
mov rbp, _kernel_stack_loc
mov rbp, 0
mov rsp, _kernel_stack_loc
call main ; where we actually call the kernel
mov rax, QWORD main
;push QWORD 0
jmp rax
jmp $
ret

View File

@ -45,8 +45,8 @@ db 0
dw 0 ; segment limit 15:00 (I don't think this matters in 64 bit mode!)
dw 0 ; base address 15:00
db 0 ; base address 23:16
db 10011010b ;1st flags and type. The first four bits (1010) are type, and the last are flags. See https://www.cs.bham.ac.uk/~exr/lectures/opsys/10_11/lectures/os-dev.pdf
db 10101111b ;1111 is segment limit continued. 0: available, 0: 64 bit (change?), 1: 32 bit segment, 1: granularity (shifts 3 hex didgets to get all of memory)
db 10011010b
db 00100000b
db 0
.gdt_data:
@ -54,7 +54,7 @@ dw 0
dw 0
db 0
db 10010010b
db 00000000b
db 00100000b
db 0
.gdt_end: ; later calculates offset in defs below

21
src/bootloader/print.asm Normal file
View File

@ -0,0 +1,21 @@
;TODO fix null problem, allow passing value insted of pointer
bios_print:
pusha
mov ah, 0x0e
.print_loop:
mov al, [bx]
cmp al, 0
je .fini
int 0x10
inc bx
jmp .print_loop
.fini:
mov al, 0xd
int 0x10
mov al, 0xa
int 0x10
popa
ret

View File

@ -151,25 +151,21 @@ ret
.vbe_unsupported:
mov bx, .unsupported_msg
mov cx, 0
call bios_print
jmp $
.vbe_generic_error:
mov bx, .lazyerror_msg
mov cx, 0
call bios_print
jmp $
.edid_error:
mov bx, .ediderror_msg
mov cx, 0
call bios_print
jmp $
.res_unsupported:
mov bx, .res_unsupported_msg
mov cx, 0
call bios_print
jmp $

View File

@ -1,4 +1,11 @@
target remote localhost:1234
symbol-file debug/debug_syms.o
break *0x7c00
hb smp_boot
define cs2bs
print (1 << (5 + $arg0))
end
continue

28
src/debug/gdbinit.py Executable file
View File

@ -0,0 +1,28 @@
class NextInstructionAddress(gdb.Command):
"""
Run until Next Instruction address.
Usage: nia
Put a temporary breakpoint at the address of the next instruction, and continue.
Useful to step over int interrupts.
See also: http://stackoverflow.com/questions/24491516/how-to-step-over-interrupt-calls-when-debugging-a-bootloader-bios-with-gdb-and-q
"""
def __init__(self):
super().__init__(
'nia',
gdb.COMMAND_BREAKPOINTS,
gdb.COMPLETE_NONE,
False
)
def invoke(self, arg, from_tty):
frame = gdb.selected_frame()
arch = frame.architecture()
pc = gdb.selected_frame().pc()
length = arch.disassemble(pc)[0]['length']
gdb.Breakpoint('*' + str(pc + length), temporary = True)
gdb.execute('continue')
NextInstructionAddress()

113
src/debug/i8086.xml Normal file
View File

@ -0,0 +1,113 @@
<?xml version="1.0"?>
<!-- Copyright (C) 2010-2017 Free Software Foundation, Inc.
Copying and distribution of this file, with or without modification,
are permitted in any medium without royalty provided the copyright
notice and this notice are preserved. -->
<!DOCTYPE target SYSTEM "gdb-target.dtd">
<target>
<architecture>i8086</architecture>
<feature name="org.gnu.gdb.i386.core">
<flags id="i386_eflags" size="4">
<field name="CF" start="0" end="0"/>
<field name="" start="1" end="1"/>
<field name="PF" start="2" end="2"/>
<field name="AF" start="4" end="4"/>
<field name="ZF" start="6" end="6"/>
<field name="SF" start="7" end="7"/>
<field name="TF" start="8" end="8"/>
<field name="IF" start="9" end="9"/>
<field name="DF" start="10" end="10"/>
<field name="OF" start="11" end="11"/>
<field name="NT" start="14" end="14"/>
<field name="RF" start="16" end="16"/>
<field name="VM" start="17" end="17"/>
<field name="AC" start="18" end="18"/>
<field name="VIF" start="19" end="19"/>
<field name="VIP" start="20" end="20"/>
<field name="ID" start="21" end="21"/>
</flags>
<reg name="eax" bitsize="32" type="int32"/>
<reg name="ecx" bitsize="32" type="int32"/>
<reg name="edx" bitsize="32" type="int32"/>
<reg name="ebx" bitsize="32" type="int32"/>
<reg name="esp" bitsize="32" type="data_ptr"/>
<reg name="ebp" bitsize="32" type="data_ptr"/>
<reg name="esi" bitsize="32" type="int32"/>
<reg name="edi" bitsize="32" type="int32"/>
<reg name="eip" bitsize="32" type="code_ptr"/>
<reg name="eflags" bitsize="32" type="i386_eflags"/>
<reg name="cs" bitsize="32" type="int32"/>
<reg name="ss" bitsize="32" type="int32"/>
<reg name="ds" bitsize="32" type="int32"/>
<reg name="es" bitsize="32" type="int32"/>
<reg name="fs" bitsize="32" type="int32"/>
<reg name="gs" bitsize="32" type="int32"/>
<reg name="st0" bitsize="80" type="i387_ext"/>
<reg name="st1" bitsize="80" type="i387_ext"/>
<reg name="st2" bitsize="80" type="i387_ext"/>
<reg name="st3" bitsize="80" type="i387_ext"/>
<reg name="st4" bitsize="80" type="i387_ext"/>
<reg name="st5" bitsize="80" type="i387_ext"/>
<reg name="st6" bitsize="80" type="i387_ext"/>
<reg name="st7" bitsize="80" type="i387_ext"/>
<reg name="fctrl" bitsize="32" type="int" group="float"/>
<reg name="fstat" bitsize="32" type="int" group="float"/>
<reg name="ftag" bitsize="32" type="int" group="float"/>
<reg name="fiseg" bitsize="32" type="int" group="float"/>
<reg name="fioff" bitsize="32" type="int" group="float"/>
<reg name="foseg" bitsize="32" type="int" group="float"/>
<reg name="fooff" bitsize="32" type="int" group="float"/>
<reg name="fop" bitsize="32" type="int" group="float"/>
</feature>
<feature name="org.gnu.gdb.i386.32bit.sse">
<vector id="v4f" type="ieee_single" count="4"/>
<vector id="v2d" type="ieee_double" count="2"/>
<vector id="v16i8" type="int8" count="16"/>
<vector id="v8i16" type="int16" count="8"/>
<vector id="v4i32" type="int32" count="4"/>
<vector id="v2i64" type="int64" count="2"/>
<union id="vec128">
<field name="v4_float" type="v4f"/>
<field name="v2_double" type="v2d"/>
<field name="v16_int8" type="v16i8"/>
<field name="v8_int16" type="v8i16"/>
<field name="v4_int32" type="v4i32"/>
<field name="v2_int64" type="v2i64"/>
<field name="uint128" type="uint128"/>
</union>
<flags id="i386_mxcsr" size="4">
<field name="IE" start="0" end="0"/>
<field name="DE" start="1" end="1"/>
<field name="ZE" start="2" end="2"/>
<field name="OE" start="3" end="3"/>
<field name="UE" start="4" end="4"/>
<field name="PE" start="5" end="5"/>
<field name="DAZ" start="6" end="6"/>
<field name="IM" start="7" end="7"/>
<field name="DM" start="8" end="8"/>
<field name="ZM" start="9" end="9"/>
<field name="OM" start="10" end="10"/>
<field name="UM" start="11" end="11"/>
<field name="PM" start="12" end="12"/>
<field name="FZ" start="15" end="15"/>
</flags>
<reg name="xmm0" bitsize="128" type="vec128" regnum="32"/>
<reg name="xmm1" bitsize="128" type="vec128"/>
<reg name="xmm2" bitsize="128" type="vec128"/>
<reg name="xmm3" bitsize="128" type="vec128"/>
<reg name="xmm4" bitsize="128" type="vec128"/>
<reg name="xmm5" bitsize="128" type="vec128"/>
<reg name="xmm6" bitsize="128" type="vec128"/>
<reg name="xmm7" bitsize="128" type="vec128"/>
<reg name="mxcsr" bitsize="32" type="i386_mxcsr" group="vector"/>
</feature>
</target>

View File

@ -1,30 +1,20 @@
#ifndef acpi_included
#define acpi_included
#include <stdint.h>
#include <libc.h>
struct rsdp_v1 {
char sig[8];
uint8_t checksum;
char OEMID[6];
uint8_t version;
uint32_t rsdt_addr;
} __attribute__((packed));
struct rsdp_v2 {
struct rsdp_v1 v1;
uint32_t len;
uint64_t xsdt_addr;
uint8_t extended_checksum;
uint8_t reserved[3];
} __attribute__((packed));
#include <stdbool.h>
#include <int.h>
typedef union rsdp_t {
struct rsdp_v1 v1;
struct rsdp_v2 v2;
} rsdp_t;
//sdt types
#define SDT_MADT 0
#define SDT_HPET 1
struct acpi_header {
void find_root_sdp();
void *find_sdt(int type);
void debug_acpi();
typedef struct sdt {
char sig[4];
uint32_t length;
uint8_t revision;
@ -34,6 +24,7 @@ struct acpi_header {
uint32_t OEMRevision;
uint32_t creator_id;
uint32_t creator_revision;
} __attribute__((packed));
} __attribute__((packed)) sdt_head;
rsdp_t *find_RSDP();
#endif

5
src/include/heap.h Normal file
View File

@ -0,0 +1,5 @@
void *malloc(size_t size);
void *realloc(void *old_chunk, size_t size);
void malloc_init();
void debug_heap();
void free(void *size);

84
src/include/int.h Normal file
View File

@ -0,0 +1,84 @@
#ifndef int_header
#define int_header
#include <stdint.h>
#include <stddef.h>
void usleep(unsigned int us);
void init_interrupts();
typedef uint32_t *lapic_t;
//this is only for madt; do we really need it here?
struct ioapic_t {
void *address;
unsigned int gsi_base;
unsigned int gsi_count;
};
//future: if we start to do things like this more then once,
struct ioapic_fixedlen {
size_t count;
struct ioapic_t *ioapic;
};
//TODO change name
struct apic_vt {
uint8_t vector;
unsigned int delivery_mode:3;
unsigned int dest_mode:1;
unsigned int delivery_status:1;
unsigned int polarity:1; //0 = high, 1 = low.
unsigned int remote_irr:1;
unsigned int trigger_mode:1;
unsigned int mask:1;
unsigned long int reserved:40;
unsigned int destination:7;
} __attribute__((packed));
//simplified idt_descriptor makes things easier
struct idt_entry {
void *addr;
unsigned int ist:3;
unsigned int type:4;
unsigned int priv:2;
};
#define IOAPIC_REDIRECT_FIXED(vector, destination) \
((struct apic_vt) { \
.vector = vector, \
.delivery_mode = 0, \
.dest_mode = 0, \
.polarity = 0, \
.remote_irr = 0, \
.trigger_mode = 0, \
.mask = 0, \
.destination = destination \
})
#define KERNEL_IDT_GATE(isr) \
((struct idt_entry) { \
.addr = isr, \
.ist = 0, \
.type = INT_GATE_64, \
.priv = 0 \
})
#define TASK_GATE_64 0x5
#define INT_GATE_64 0xe
#define TRAP_GATE_64 0xf
#define SPURRIOUS_VECTOR 255
void create_fixed_interrupt(unsigned int irq, struct apic_vt *redirect);
void clear_int();
unsigned int alloc_idt(struct idt_entry *entry);
void free_idt(unsigned int entry);
void modify_idt(struct idt_entry *entry, unsigned int vector);
#endif

38
src/include/io.h Normal file
View File

@ -0,0 +1,38 @@
#ifndef io_header
#define io_header
#include <stdint.h>
#include <printf.h>
static inline void outb(uint16_t port, uint8_t value) {
asm volatile("outb %1, %0" :: "a"(value), "Nd"(port));
}
static inline uint8_t inb(uint16_t port) {
uint8_t ret;
asm volatile("inb %0, %1" : "=a"(ret) : "Nd"(port));
return(ret);
}
static inline void io_wait() {
asm volatile("outb 0x80, %%al"::"a"(0));
}
static inline uint64_t read_tsc() {
uint64_t time;
asm volatile("xor rax, rax\n"
"rdtsc\n"
"shl rdx, 32\n"
"or rax, rdx\n"
"mov %0, rax\n"
:"=r"(time):);
return time;
}
void outb_wait(uint16_t port, uint8_t value);
void read_msr(uint32_t addr, uint64_t *value);
void write_msr(uint32_t addr, uint64_t value);
void hlt();
uint64_t get_rip();
#endif

25
src/include/isv.h Normal file
View File

@ -0,0 +1,25 @@
#ifndef isv_defined
#define isv_defined
//TODO move to int.h
struct int_frame {
void *rip;
uint64_t segment_selector;
uint64_t rflags;
void *rsp;
uint64_t stack_segment;
} __attribute__((packed));
struct exception_frame {
uint64_t err;
struct int_frame frame;
} __attribute__((packed));
__attribute__((interrupt)) void spurrious_int(void *unused);
__attribute__((interrupt)) void kernel_block(void *unused);
__attribute__((interrupt)) void fatal_hwexception(struct int_frame *frame);
__attribute__((interrupt)) void fatal_hwexception_errcode(struct exception_frame *frame);
__attribute__((interrupt)) void lapic_timer_racefixer(void *unused);
#endif

View File

@ -1,5 +1,13 @@
#define EASTEREGG_BLOATWARE //TODO move to some kind of global config file
void panic();
#ifndef KERNEL
#define KERNEL
#define KERNEL_PANIC_INVOKED 0
//more to come
#define PA_OFFSET 0xffff800000000000
#define TXT_OFFSET 0xffffffff80000000
#define PHYS_TO_VIRT(addr) ((void *)(addr) + PA_OFFSET)
#define DEV_EMAIL "brett_weiland@bpcspace.com"
#endif

View File

@ -1,13 +1,17 @@
#ifndef _STRING_H_
#define _STRING_H_
#include <stddef.h>
#include <stdint.h>
void *strcpy(char *dest, char *src);
void *memcpy(void *dest, void *src, size_t n); //TODO
void *bzero(const void *dest, size_t size);
void *memset(void *s, char c, size_t n);
size_t strlen(const char *s);
int strcmp(const char *str1, const char *str2);
int strncmp(const char *str1, const char *str2, size_t n);
int memcmp(const void *s1, const void *s2, size_t n);
int ceil(double n1);
int round(double n1);
#endif

20
src/include/madt.h Normal file
View File

@ -0,0 +1,20 @@
#ifndef madt_header
#define madt_header
#include <int.h>
struct cores_info {
unsigned int corecount;
uint8_t apic_id[256];
uint8_t bsp;
};
void init_madt();
void get_ioapic(struct ioapic_fixedlen *ret);
void get_coreinfo(struct cores_info *cores);
void debug_madt();
unsigned int irq_to_gsi(unsigned int irq);
lapic_t get_lapic();
void boot_multicore();
#endif

6
src/include/math.h Normal file
View File

@ -0,0 +1,6 @@
#ifndef MATH_INCLUDED
#define MATH_INCLUDED
#define DIV_ROUND_UP(number, factor) (((number) + ((factor) - 1)) / factor)
#endif

View File

@ -1,96 +1,23 @@
#include <stdbool.h>
//paging errors
#define PAGEMAP_LOCATION 0x4000
#ifndef _PAGE_H_
#define _PAGE_H_
#include <stdbool.h>
#include <stdint.h>
#define PAGE_VIRT_UNALIGNED 0x01
#define PAGE_PHYS_UNALIGNED 0x02
#define PAGE_PHYS_INVALID 0x03
#define PAGE_VIRT_INVALID 0x04
//*
typedef struct __attribute__((packed)) {
unsigned int present : 1; // present, must be one when accessed.
unsigned int read_write : 1; // if set to one, read and write is set
unsigned int user : 1; // another bit we'll never use, for seperating CPL 0-2 and 3+
unsigned int writethrough_cache : 1; // honestly maybe I should look into caching
unsigned int cachable : 1; // hardware chaching. 0 is enabled, whats the worst that could happen?
unsigned int accessed : 1; // we'll never use any of these!
unsigned int zg0 : 1; // needs to be (and will be) zero'd
unsigned int size : 1;
unsigned int zg1 : 1; // needs to be (and will be) zero'd
unsigned int software_marks : 3; // available for our own use, I doubt we'll use it in such a simple thing
uintptr_t base_ptr : 40;
unsigned int sign:11;
} page_entry;
typedef struct __attribute__((packed)) {
page_entry pml4e[512];
page_entry pdpe[512];
page_entry pde[512];
page_entry pte[512];
} page_table;
#define MEM_AVAILABLE 1
#define MEM_RESERVED 2
#define MEM_APCI_RECLAIMABLE 3
#define MEM_APCI_NVS 4
#define MEM_BAD 5
#define PAGE_SIZE_4K 12
#define PAGE_SIZE_2M 21
#define PAGE_SIZE_1G 30
#define MAX_BUDDY_ORDER 8
#define MAX_ZONE_CNT 16 //should cover all cases
struct memory_table {
void *base;
uint64_t length;
uint32_t type;
uint32_t ACPI;
} __attribute__((packed));
/**
* the bsizes are there so we don't have to calculate it every time.
*
* Keep in mind that at if the allocator reaches the last buddy,
* it _will_ have to calculate the bitwidth, even if it's a multiple of 64.
* This scenario hopefully won't happen much during time sensitive areas,
* and (I think) linux calculates the buddy size every single time anyway.
**/
struct phys_map {
struct phys_map *next;
void *zone_paddr;
uint64_t extra_bits;
uint64_t *buddy[MAX_BUDDY_ORDER];
};
//clean up doing extra work some other time
#define LSIZE_FROM_BITLEN(bitlen) (((bitlen) + 63) / 64)
#define BITLEN_FROM_LSIZE(lsize) ((lsize) * 64)
#define GET_BUDDY_BITLEN(zone_len, order) ((zone_len) / (0x1000 << (order)))
#define GET_ORDER_CHUNKSIZE(order) (0x1000 << ((order)))
#include <stddef.h>
extern void* _meminfo_loc;
extern void* _stage2_pagetable;
bool map_page(void *virtual_addr, void *physical_addr, uint8_t PAGE_SIZE);
void unmap_lowmem();
size_t map_complete_physical();
void debug_print_memory();
void create_pagetable_stage2(uint64_t free_mem);
void init_memory(); //TODO removeme
void init_pmap();
void *palloc();
void *pfree();
struct phys_map *init_pmap(size_t pagetable_size);
void *palloc(size_t size);
void pfree(void *addr, size_t size);
void debug_pmap();
void get_mem_capabilities();
void ram_stresser();
#endif

View File

@ -1,27 +1,59 @@
#ifndef PANIC_INCLUDED
#define PANIC_INCLUDED
#define EASTEREGG_BLOATWARE
#include <stdint.h>
void panic(int reason, int type);
#define EASTEREGG_BLOATWARE
//TODO this needs to be moved once we find somehwere better
struct stack_frame {
struct stack_frame *next;
uint64_t function_base;
struct stack_frame *next; //rbp
void *function_base; //rip
} __attribute__((packed));
//kernel panic reasons, likely to grow at an extremely fast rate
#define KERNEL_PANIC_PMAPSIZE 0
#define KERNEL_PANIC_RSDP_UNFOUND 1
#define KERNEL_PANIC_KERNEL_RETURNED 2
struct registers {
uint64_t rax, rbx, rcx, rdx, rsi, rdi;
} __attribute__((packed));
void panic(int reason, void *frame_p, struct registers *regs);
#define PANIC(reason) panic(reason, 0, 0)
//hardware exceptions/faults
//kernel panic reasons 0-31 reserved for these
//later on, we can make an ugly function to find our vector.
//I did a lot of work assuming it was pushed onto the stack,
//so if I choose not to impliment that these defs will be deleted.
/**
#define KERNEL_PANIC_HWE_DE 0
#define KERNEL_PANIC_HWE_BR 5
#define KERNEL_PANIC_HWE_UD 6
#define KERNEL_PANIC_HWE_NM 7
#define KERNEL_PANIC_HWE_DF 8
#define KERNEL_PANIC_HWE_TS 10
#define KERNEL_PANIC_HWE_NP 11
#define KERNEL_PANIC_HWE_SS 12
#define KERNEL_PANIC_HWE_GP 13
#define KERNEL_PANIC_HWE_PF 14
#define KERNEL_PANIC_HWE_MF 16
#define KERNEL_PANIC_HWE_AC 17
#define KERNEL_PANIC_HWE_MC 18
#define KERNEL_PANIC_HWE_XM 19
#define KERNEL_PANIC_HWE_VE 20
#define KERNEL_PANIC_HWE_SE 30
**/
#define KERNEL_PANIC_HW_EXCEPTION 0
#define KERNEL_PANIC_HW_EXCEPTION_ERR 1 //exception with error code
//kernel generated exceptions
#define KERNEL_PANIC_RSDP_UNFOUND 32
#define KERNEL_PANIC_KERNEL_RETURNED 33
#define KERNEL_PANIC_INVALID_PFREE 34
#define KERNEL_PANIC_INVALID_RSDT 35
#define KERNEL_PANIC_INVALID_IOAPIC_VEC 36
#define KERNEL_PANIC_HPET_REQUIRED 37
//kernel panic types, may or may not expand once I get further in development
#define KERNEL_PANIC_INVOKED 0
#define KERNEL_PANIC_ERROR 1 //i'll change this once I see what happends
//TODO move this to some kind of global more accesable header
#define DEV_EMAIL "brett_weiland@bpcspace.com"
#endif

5
src/include/random.h Normal file
View File

@ -0,0 +1,5 @@
#ifndef random_header
#define random_header
void randinit();
unsigned int randint();
#endif

View File

@ -3,14 +3,25 @@
#define _SERIAL_H_
#include <stdint.h>
#include <stdbool.h>
#define COM1 0x3f8
#define COM2 0x2f8
#define COM3 0x3e8
#define COM4 0x2e8
int init_serial(uint16_t port);
bool init_serial(uint16_t port);
void serial_out(uint16_t port, char *string);
void _putchar_serial(uint16_t port, char character);
void move_cursor(unsigned int x, unsigned int y);
//TODO fix shitty header
void _putchar_screen(char character);
void clear_screen();
#endif

4
src/include/smp.h Normal file
View File

@ -0,0 +1,4 @@
#ifndef SMP_INCLUDED
#define SMP_INCLUDED
void smp_boot();
#endif

6
src/include/testmalloc.h Normal file
View File

@ -0,0 +1,6 @@
#ifndef testmalloc_header
#define testmalloc_header
void test_malloc(unsigned int cnt);
#endif

8
src/include/timer.h Normal file
View File

@ -0,0 +1,8 @@
#ifndef timer
#define timer
void init_timer();
void usleep(unsigned int us);
uint64_t timestamp();
#endif

Binary file not shown.

View File

@ -1,58 +1,169 @@
#include <acpi.h>
#include <stdint.h>
#include <printf.h>
#include <kernel.h>
#include <stdbool.h>
#include <kernel.h>
#include <io.h>
#include <int.h>
#include <libc.h>
#include <panic.h>
static int RSDP_verify(void *rsdp_pointer) {
printf("Verifying potential RSDP at address 0x%p... ", rsdp_pointer);
union rsdp_t *rsdp = rsdp_pointer;
uint8_t checksum = 0;
char *rsdp_csm_ptr = rsdp_pointer;
int i;
if(checksum) return 0;
if(rsdp->v1.version) {
printf("APCI revision > 2.\n");
checksum = 0;
printf("len : %i\n", rsdp->v2.len);
for(i = 0; i < rsdp->v2.len; i++) {
checksum += rsdp_csm_ptr[i];
//finding rsdp
typedef struct rsdp {
char sig[8];
uint8_t checksum;
char OEMID[6];
uint8_t revision;
uint32_t rsdt;
//these only exist on REV > 1
uint32_t len;
uint64_t xsdt;
uint8_t extended_checksum;
uint8_t reserved[3];
} __attribute__((packed)) root_sdp;
//root descriptor SDTs
struct rsdt {
sdt_head header;
uint32_t sdt_pointers[];
} __attribute__((packed));
struct xsdt {
sdt_head header;
uint64_t sdt_pointers[];
} __attribute__((packed));
root_sdp *rsdp;
void debug_acpi() {
sdt_head *sdt;
int sdt_i, sdt_cnt;
if(rsdp->revision) {
struct xsdt *root = PHYS_TO_VIRT(rsdp->xsdt);
sdt_cnt = (root->header.length - sizeof(root->header)) / 8;
printf("sdts found in apic headers: %i\n", sdt_cnt);
for(sdt_i = 0; sdt_i < sdt_cnt; sdt_i++) {
sdt = PHYS_TO_VIRT(root->sdt_pointers[sdt_i]);
printf("ACPI debug:%.4s\n", sdt->sig);
}
}
else {
printf("APCI revision 1.\n");
for(i = 0; i <= 20; i++) {
checksum += rsdp_csm_ptr[i];
struct rsdt *root = PHYS_TO_VIRT((uint64_t)rsdp->rsdt);
sdt_cnt = (root->header.length - sizeof(root->header)) / 4;
printf("sdts found in apic headers: %i\n", sdt_cnt);
for(sdt_i = 0; sdt_i < sdt_cnt; sdt_i++) {
sdt = PHYS_TO_VIRT((uint64_t)root->sdt_pointers[sdt_i]);
printf("ACPI debug:%.4s\n", sdt->sig);
}
}
if(checksum) {
return 0;
printf("Invalid, searching on.\n");
}
printf("RSDP Verified!\n");
return 1;
}
rsdp_t *find_RSDP() {
//TODO clean up confusing variable name rsdt
void *find_sdt(int type) {
char sig[4];
sdt_head *sdt;
int sdt_i, sdt_cnt;
//If we never need to expand this, then you can clean it up.
switch(type) {
case SDT_MADT:
strcpy(sig, "APIC");
break;
case SDT_HPET:
strcpy(sig, "HPET");
break;
}
if(rsdp->revision) {
struct xsdt *root = PHYS_TO_VIRT(rsdp->xsdt);
sdt_cnt = (root->header.length - sizeof(root->header)) / 8;
for(sdt_i = 0; sdt_i < sdt_cnt - 1; sdt_i++) {
sdt = PHYS_TO_VIRT((uint64_t)root->sdt_pointers[sdt_i]);
if(!(strncmp(sdt->sig, sig, 4))) {
return sdt;
}
}
}
else {
struct rsdt *root = PHYS_TO_VIRT((uint64_t)rsdp->rsdt);
sdt_cnt = (root->header.length - sizeof(root->header)) / 4;
for(sdt_i = 0; sdt_i < sdt_cnt - 1; sdt_i++) {
sdt = PHYS_TO_VIRT((uint64_t)root->sdt_pointers[sdt_i]);
if(!(strncmp(sdt->sig, sig, 4))) {
return sdt;
}
}
}
return 0;
}
bool verify_sdt(sdt_head *sdt) {
uint8_t checksum = 0;
for(int i = 0; i < sdt->length; i++) checksum += ((uint8_t *)sdt)[i];
return checksum == 0;
}
static root_sdp *verify_sdp(root_sdp *uv_rsdp) {
char *rsdp_char_ptr = (char *)uv_rsdp;
int i;
uint8_t checksum = 0;
if(uv_rsdp->revision) {
for(i = 0; i < uv_rsdp->len; i++) {
checksum += rsdp_char_ptr[i];
}
}
else {
for(i = 0; i <= 20; i++) {
checksum += rsdp_char_ptr[i];
}
}
if(!checksum) {
printf("Found RSDP revision %i at address %x.\n", uv_rsdp->revision, uv_rsdp);
return uv_rsdp;
}
printf("Skipped over invalid RSDP at 0x%p.\n", uv_rsdp);
return 0;
}
void find_root_sdp() {
const char sig[9] = "RSD PTR ";
uintptr_t *p = (void *)0x040e;
uintptr_t *ebda_unshifted = (void *)p;
bool rsdt_verified;
void *ebda = (void *)((uintptr_t)ebda_unshifted << (uintptr_t)4 & (uintptr_t)0xfffff);
void *ebda = PHYS_TO_VIRT((void *)((uintptr_t)ebda_unshifted << (uintptr_t)4 & (uintptr_t)0xfffff));
for(void *i = ebda; i <= ebda + 64000; i += 16) {
if(!(memcmp(sig, i, 8))) {
if(RSDP_verify(i)) {
return(i);
}
}
for(void *i = ebda; i <= ebda + 64000; i += 16)
if(!(memcmp(sig, i, 8)) && ((rsdp = verify_sdp(i)))) break;
for(void *i = PHYS_TO_VIRT(0xe0000); i <= PHYS_TO_VIRT(0xfffff); i += 16)
if(!(memcmp(sig, i, 8)) && ((rsdp = verify_sdp(i)))) break;
if(!(rsdp)) {
printf("Couldn't find the RSDP... not sure what to do now.\n");
PANIC(KERNEL_PANIC_RSDP_UNFOUND);
}
if(rsdp->revision) {
rsdt_verified = verify_sdt(PHYS_TO_VIRT(rsdp->xsdt));
}
else {
rsdt_verified = verify_sdt(PHYS_TO_VIRT((uint64_t)rsdp->rsdt));
}
for(void *i = (void *)0xe0000; i <= (void *)0xfffff; i += 16) {
if(!(memcmp(sig, i, 8))) {
if(RSDP_verify(i)) {
return(i);
}
}
if(!(rsdt_verified)) {
PANIC(KERNEL_PANIC_INVALID_RSDT);
}
}

View File

@ -1,39 +0,0 @@
#include <stdint.h>
#include <serial.h>
static inline void outb(uint16_t port, uint8_t value) {
asm volatile(
"outb %0, %1" :: "a"(value), "Nd"(port)
);
}
static inline uint8_t inb(uint16_t port) {
uint8_t ret;
asm volatile(
"inb %1, %0" : "=a"(ret) : "Nd"(port)
);
return(ret);
}
int init_serial(uint16_t port) {
outb(port + 1, 0x00);
outb(port + 3, 0x80);
outb(port + 0, 0x06);
outb(port + 1, 0x00);
outb(port + 3, 0x03);
outb(port + 2, 0xc7);
outb(port + 4, 0x0b);
outb(port + 4, 0x1e);
outb(port + 0, 0xae); // test char
if(inb(port + 0) != 0xae)
return 1;
outb(port + 4, 0x0f); // dissable interupts
return 0;
}
void _putchar_serial(uint16_t port, char msg) {
while(!(inb(port + 5) & 0x20)); //wait for transmit to be done
outb(port, msg);
}

211
src/kernel/heap.c Normal file
View File

@ -0,0 +1,211 @@
#include <paging.h>
#include <printf.h>
#include <libc.h>
#include <io.h>
/** a disclaimer-
* I've never made an OS, so I don't know what kind of structures I'll have.
* Because of this, I'm not YET using a slab allocator.
* This allocator allocates sizes of 2^n, starting at 32, which can waste lots of space.
* In other places, you'll see me reallocating when it's not nessesary,
* as if my allocator doesn't over-allocate.
*
* Later in development, I will create a slab allocator, making it so that this isn't an issue.
**/
#define MIN_CHUNK_ORDER 5 //equal to 32 bytes for heap_chunk struct. Keeping away from 24 for easy division.
#define MAX_CHUNK_I 6
#define CHUNK_LIST_LEN MAX_CHUNK_I + 1
#define LOG_SUB 63 - MIN_CHUNK_ORDER
#define PAGE_SIZE 0x1000 //TODO [minor] maybe move this to some header somewhere
#define CHUNK_SIZE_FROM_INDEX(i) ((1 << ((i) + MIN_CHUNK_ORDER)))
#define CHUNK_INDEX_FROM_SIZE(s) ((63 - MIN_CHUNK_ORDER - __builtin_clzl((s))))
typedef struct __attribute__((packed)) heap_chunk {
unsigned int free:1;
unsigned int size:4; //will use with flags later if needed
unsigned int lsize:4;
unsigned long reserved:55;
struct heap_chunk *fd;
struct heap_chunk *bk;
} chunk;
struct heap_chunk_roots {
void *fd;
};
//TODO this wastes some memory
chunk arena[CHUNK_LIST_LEN];
void debug_heap() {
chunk *on_chunk, *last_chunk;
size_t chunk_expected_size;
size_t chunk_actual_size;
for(unsigned int l = 0; l < CHUNK_LIST_LEN; l++) {
chunk_expected_size = CHUNK_SIZE_FROM_INDEX(l);
printf("\n%i sized chunks:\n", chunk_expected_size);
for(on_chunk = arena[l].fd; on_chunk != 0; on_chunk = on_chunk->fd) {
chunk_actual_size = CHUNK_SIZE_FROM_INDEX(on_chunk->size);
printf("\t%p", on_chunk);
if(chunk_expected_size != chunk_actual_size) {
printf("\nFound chunk of size %i in %i sized freebin!!!!\n", chunk_actual_size, chunk_expected_size);
hlt();
}
if(!on_chunk->free) {
printf("\nChunk found in freebin isn't free!\n");
hlt();
}
//may take a while
if((uintptr_t)on_chunk & (0x1000 - 1)) {
last_chunk = (void *)on_chunk - CHUNK_SIZE_FROM_INDEX(on_chunk->lsize);
if(last_chunk->size != on_chunk->lsize) {
printf("\nChunk before this one is a different size then lsize!");
hlt();
}
}
}
}
printf("\n\n");
}
void malloc_init() {
bzero(&arena, sizeof(arena));
}
void free(void *addr) {
chunk *on_chunk, *buddy_chunk, *next_chunk, *chunk_iter;
on_chunk = addr - 24;
for(;;) {
if(on_chunk->size == (MAX_CHUNK_I + 1)) {
pfree(on_chunk, 0x1000);
printf("freeing page %p\n", on_chunk);
return;
}
unsigned int chunk_size = CHUNK_SIZE_FROM_INDEX(on_chunk->size);
if((uintptr_t)on_chunk & ((chunk_size * 2) - 1)) {
buddy_chunk = (void *)on_chunk - CHUNK_SIZE_FROM_INDEX(on_chunk->lsize);
if((buddy_chunk->size != on_chunk->size) || !(buddy_chunk->free)) {
next_chunk = (void *)on_chunk + chunk_size;
break;
}
chunk_iter = buddy_chunk;
}
else {
buddy_chunk = (void *)on_chunk + chunk_size;
if((buddy_chunk->size != on_chunk->size) || !(buddy_chunk->free)) {
next_chunk = buddy_chunk;
break;
};
chunk_iter = on_chunk;
}
if(buddy_chunk->fd) (buddy_chunk->fd)->bk = buddy_chunk->bk;
(buddy_chunk->bk)->fd = buddy_chunk->fd;
on_chunk = chunk_iter;
on_chunk->size++;
}
next_chunk->lsize = on_chunk->size;
on_chunk->free = 1;
on_chunk->bk = (chunk *)&arena[on_chunk->size];
on_chunk->fd = arena[on_chunk->size].fd;
if(arena[on_chunk->size].fd) (arena[on_chunk->size].fd)->bk = on_chunk;
arena[on_chunk->size].fd = on_chunk;
}
static void split_chunk(chunk *full_chunk, unsigned int full_size, unsigned int req_size) {
chunk *on_chunk;
full_chunk->size = req_size;
full_chunk->free = 0;
//chunk isn't newly allocated page
//our free funciton won't check lsize if addr is divisible by 0x1000
if(full_chunk->bk) {
if(full_chunk->fd) (full_chunk->fd)->bk = (chunk *)&arena[full_size];
arena[full_size].fd = full_chunk->fd;
on_chunk = (void *)full_chunk + CHUNK_SIZE_FROM_INDEX(full_size);
if((uintptr_t)on_chunk & (0x1000 - 1)) on_chunk->lsize = full_size - 1;
}
on_chunk = (void *)full_chunk + CHUNK_SIZE_FROM_INDEX(req_size);
for(unsigned int sz = req_size; sz < full_size; sz++) {
on_chunk->lsize = (sz == req_size) ? req_size : sz - 1;
on_chunk->fd = arena[sz].fd;
on_chunk->bk = (chunk *)&arena[sz];
on_chunk->size = sz;
on_chunk->free = 1;
arena[sz].fd = on_chunk;
on_chunk = (void *)on_chunk + CHUNK_SIZE_FROM_INDEX(sz);
}
}
void *malloc(size_t size) {
size += sizeof(chunk);
chunk *on_chunk;
unsigned int chunk_sz_i; //desired chunk size index
unsigned int chunk_ck_i; //backup chunk size index in case we need to count up
//rounding size to nearest 2^n to find which size it fits into
if(size < 32) size = 32;
chunk_sz_i = CHUNK_INDEX_FROM_SIZE(size);
if(size & (size - 1)) chunk_sz_i++;
if(chunk_sz_i > MAX_CHUNK_I) return(void *)0;
//search for direct hits
if(arena[chunk_sz_i].fd) {
on_chunk = arena[chunk_sz_i].fd;
if(on_chunk->fd) (on_chunk->fd)->bk = (chunk *)&arena[chunk_sz_i];
arena[chunk_sz_i].fd = on_chunk->fd;
on_chunk->free = 0;
return (void *)on_chunk + sizeof(chunk);
}
//search for chunks above our size
for(chunk_ck_i = chunk_sz_i + 1; chunk_ck_i < CHUNK_LIST_LEN; chunk_ck_i++) {
if(arena[chunk_ck_i].fd) {
on_chunk = arena[chunk_ck_i].fd;
split_chunk(on_chunk, chunk_ck_i, chunk_sz_i);
return (void *)on_chunk + sizeof(chunk);
}
}
on_chunk = (void *)palloc(0x1000);
printf("allocating page %p\n", on_chunk);
split_chunk(on_chunk, 7, chunk_sz_i);
return (void *)on_chunk + sizeof(chunk);
}
void *realloc(void *old_chunk, size_t size) {
void *new_chunk = malloc(size);
memcpy(new_chunk, old_chunk, CHUNK_SIZE_FROM_INDEX(((chunk *)(old_chunk-24))->size));
free(old_chunk);
return new_chunk;
}

253
src/kernel/int.c Normal file
View File

@ -0,0 +1,253 @@
#include <int.h>
#include <acpi.h>
#include <io.h>
#include <paging.h>
#include <printf.h>
#include <isv.h>
#include <kernel.h>
#include <libc.h>
#include <madt.h>
#include <timer.h>
#include <panic.h>
#define SEGMENT_GDT 0
#define SEGMENT_LDT 1
//io commands
#define ICW1_ICW4 0x01
#define ICW1_INIT 0x10
#define ICW4_8086 0x01
// PIC
#define PICM_COMMAND 0x20
#define PICM_DATA 0x21
#define PICS_COMMAND 0xa0
#define PICS_DATA 0xa1
#define KERNEL_IDT_DESC_GATE(addr) \
((struct idt_descriptor) { \
.offset_1 = (uint16_t)((uint64_t)addr & 0xffff), \
.offset_2 = (uint16_t)((uint64_t)addr >> 16), \
.offset_3 = (uint32_t)((uint64_t)addr >> 32), \
.segment = 0x8, \
.ist = 0, \
.type = INT_GATE_64, \
.priv = 0, \
.reserved = 0, \
.reserved_1 = 0, \
.reserved_2 = 0, \
.present = 1 \
})
#define GDT_ADDR (uint64_t)PHYS_TO_VIRT(0x7e22)
#define IOAPICR_VER 1
struct idt_descriptor {
uint16_t offset_1;
uint16_t segment;
unsigned int ist:3; //interupt stack table offset
unsigned int reserved:5;
unsigned int type:4;
unsigned int reserved_1:1;
unsigned int priv:2;
unsigned int present:1;
uint16_t offset_2;
uint32_t offset_3;
uint32_t reserved_2;
} __attribute__((packed));
struct segment_descriptor {
unsigned int priv:2;
unsigned int type:1;
unsigned int index:13;
} __attribute__((packed));
struct idt_reg {
uint16_t size;
uint64_t offset;
} __attribute__((packed));
struct gdt_reg {
uint16_t size;
uint64_t offset;
} __attribute__((packed));
struct idt_descriptor *idt;
static struct idt_reg idtr;
struct gdt_reg gdtr = {
.size = 0x17,
.offset = GDT_ADDR
};
lapic_t lapic; //would make this static but it's needed by timer.c
static struct ioapic_fixedlen ioapic;
static uint64_t isr_bitmap[4]; //first 32 is for exceptions;
void disable_pic() {
//remapping PIC
outb_wait(PICM_COMMAND, ICW1_INIT | ICW1_ICW4); //initilizing
outb_wait(PICS_COMMAND, ICW1_INIT | ICW1_ICW4);
outb_wait(PICM_DATA, 32); //IRQ offsets
outb_wait(PICS_DATA, 40);
outb_wait(PICM_DATA, 4); //notify master of slave
outb_wait(PICS_DATA, 2); //nofity slave of mediocore social ethical standards
outb_wait(PICM_DATA, ICW4_8086);
outb_wait(PICS_DATA, ICW4_8086);
//masking.
outb_wait(PICS_DATA, 0xff);
outb_wait(PICM_DATA, 0xff);
}
void write_ioapic(void *ioapic_addr, uint32_t reg, uint32_t value) {
uint32_t *ioapic_r = ioapic_addr;
ioapic_r[0] = reg & 0xff;
ioapic_r[4] = value;
}
uint32_t read_ioapic(void *ioapic_addr, uint32_t reg) {
uint32_t *ioapic_r = ioapic_addr;
ioapic_r[0] = reg & 0xff;
return ioapic_r[4];
}
void create_fixed_interrupt(unsigned int irq, struct apic_vt *redirect) {
irq = irq_to_gsi(irq); //make sure there's no redirection entries.
struct ioapic_t *ioapic_smi = (void *)0;
for(unsigned int i = 0; i < ioapic.count; i++) {
if((irq >= ioapic.ioapic[i].gsi_base) &&
(irq < (ioapic.ioapic[i].gsi_base + ioapic.ioapic[i].gsi_count))) {
ioapic_smi = ioapic.ioapic[i].address;
break;
}
}
if(!ioapic_smi) PANIC(KERNEL_PANIC_INVALID_IOAPIC_VEC);
write_ioapic(ioapic_smi, (0x10 + (irq * 2)), *(uint32_t *)redirect);
write_ioapic(ioapic_smi, ((0x10 + (irq * 2)) + 1), *(uint64_t *)redirect >> 32);
}
unsigned int alloc_idt(struct idt_entry *entry) {
unsigned int b, i;
for(b = 0; b < (sizeof(isr_bitmap) / 8); b++) {
for(i = 0; i < 64; i++) {
if(!((isr_bitmap[b] >> i) & 1)) {
isr_bitmap[b] |= (uint64_t)1 << i;
idt[(b * 64) + i] = (struct idt_descriptor) {
.offset_1 = (uint16_t)((uint64_t)entry->addr & 0xffff),
.offset_2 = (uint16_t)((uint64_t)entry->addr >> 16),
.offset_3 = (uint32_t)((uint64_t)entry->addr >> 32),
.segment = 0x8,
.ist = entry->ist,
.type = entry->type,
.priv = entry->priv,
.reserved = 0,
.reserved_1 = 0,
.reserved_2 = 0,
.present = 1
};
return i;
}
}
}
return 0;
}
void modify_idt(struct idt_entry *entry, unsigned int vector) {
idt[vector] = (struct idt_descriptor) {
.offset_1 = (uint16_t)((uint64_t)entry->addr & 0xffff),
.offset_2 = (uint16_t)((uint64_t)entry->addr >> 16),
.offset_3 = (uint32_t)((uint64_t)entry->addr >> 32),
.segment = 0x8,
.ist = entry->ist,
.type = entry->type,
.priv = entry->priv,
.reserved = 0,
.reserved_1 = 0,
.reserved_2 = 0,
.present = 1
};
}
void free_idt(unsigned int entry) {
isr_bitmap[entry / 64] ^= ((uint64_t)1 << (entry & 63));
idt[entry].present = 0;
}
inline void clear_int() {
lapic[44] = 0;
}
void init_exceptions() {
//for idt index explainations, see include/panic.h
//TODO maybe utalize add_idt_entry?
unsigned int e;
idt[0] = KERNEL_IDT_DESC_GATE(fatal_hwexception);
idt[8] = KERNEL_IDT_DESC_GATE(fatal_hwexception_errcode);
idt[16] = KERNEL_IDT_DESC_GATE(fatal_hwexception);
idt[17] = KERNEL_IDT_DESC_GATE(fatal_hwexception_errcode);
idt[30] = KERNEL_IDT_DESC_GATE(fatal_hwexception_errcode);
for(e = 5; e < 8; e++) idt[e] = KERNEL_IDT_DESC_GATE(fatal_hwexception);
for(e = 10; e < 15; e++) idt[e] = KERNEL_IDT_DESC_GATE(fatal_hwexception_errcode);
for(e = 18; e < 21; e++) idt[e] = KERNEL_IDT_DESC_GATE(fatal_hwexception);
}
//void new_idt
void init_interrupts() {
init_madt();
debug_madt();
lapic = get_lapic();
get_ioapic(&ioapic);
for(unsigned int i = 0; i < ioapic.count; i++) {
ioapic.ioapic[i].gsi_count = ((read_ioapic(ioapic.ioapic[i].address, IOAPICR_VER) >> 16) & 0xff);
printf("Found ioapic %i at addr %p, gsi %i to %i\n", i, ioapic.ioapic[i].address,
ioapic.ioapic[0].gsi_base, ioapic.ioapic[0].gsi_base + ioapic.ioapic[0].gsi_count);
}
disable_pic();
idt = palloc(0x1000);
idtr.size = 0x1000;
idtr.offset = (uint64_t)idt;
bzero(&isr_bitmap, sizeof(isr_bitmap));
isr_bitmap[0] = (((uint64_t)1 << 32) - 1);
//set idt and new gdt
asm("lgdt [%0]\n"
"lidt [%1]\n":: "m"(gdtr), "m"(idtr));
//Spurrious Interrupt (permanent, required for APIC)
idt[SPURRIOUS_VECTOR] = KERNEL_IDT_DESC_GATE(spurrious_int);
//Spurrious interrupt on IDT
lapic[60] = 0x1ff; //page 395
//initlize exceptions
init_exceptions();
//initilize the timers
init_timer();
asm("sti\n");
}

24
src/kernel/io.c Normal file
View File

@ -0,0 +1,24 @@
#include <io.h>
#include <stdint.h>
//these functions are assumed not to be time-sensitive.
void outb_wait(uint16_t port, uint8_t value) {
outb(port, value);
io_wait();
}
void read_msr(uint32_t addr, uint64_t *value) {
uint32_t low, high;
asm volatile("rdmsr" : "=a"(low), "=d"(high) : "c"(addr));
*value = low | ((uint64_t)high << 32);
}
void write_msr(uint32_t addr, uint64_t value) {
uint32_t low = value & UINT32_MAX;
uint32_t high = value >> 32;
asm volatile("wrmsr"::"a"(low), "d"(high), "c"(addr));
}
void hlt() {
asm("cli\nhlt");
}

12
src/kernel/isv.c Normal file
View File

@ -0,0 +1,12 @@
#include <printf.h>
#include <stdint.h>
#include <isv.h>
#include <panic.h>
__attribute__((interrupt)) void spurrious_int(void *unused) {
printf("Detected spurrious interrupt, doing nothing.\n");
}

65
src/kernel/isv_asm.asm Normal file
View File

@ -0,0 +1,65 @@
global timer_init_pit
global kernel_block
global fatal_hwexception
global fatal_hwexception_errcode
global lapic_timer_racefixer
extern clear_int
extern panic
extern calibrate_lapic_waiting
;I need to get my timing right and consistently boot smp,
;_then_ I'll work on cleaning this up.
timer_init_pit:
cli
mov ecx, DWORD [rbx + 0x390]
mov DWORD [rbx + 0x380], 0
in al, 0x61
and al, 0xfe
out 0x61, al
mov DWORD [rbx + 0xb0], 0
iretq
kernel_block:
call clear_int
iretq
fatal_hwexception:
push qword rdi
push qword rsi
push qword rdx
push qword rcx
push qword rbx
push qword rax
mov rdi, 0
mov rdx, rsp
mov rsi, rdx
add rsi, 48
call panic
; rdi: reason
; rsi: frame
; rdx: regs
; 0xffff80000010af70
fatal_hwexception_errcode:
push qword rdi
push qword rsi
push qword rdx
push qword rcx
push qword rbx
push qword rax
mov rdi, 1
mov rdx, rsp
mov rsi, rdx
add rsi, 48
call panic
lapic_timer_racefixer: ;in case the lapic makes it to zero, impliment me
inc rbx
call clear_int
mov QWORD [rsp], calibrate_lapic_waiting
iretq

View File

@ -1,31 +1,45 @@
#include <stdbool.h>
#include <smp.h>
#include <serial.h>
#include <printf.h>
#include <paging.h>
#include <video.h>
#include <acpi.h>
#include <panic.h>
#include <kernel.h>
#include <stdbool.h>
#include <int.h>
#include <io.h>
#include <cpuid.h>
#include <heap.h>
#include <random.h>
#include <timer.h>
#include <libc.h>
//testing headers
#include <testmalloc.h>
void main() {
if(!(init_serial(COM1))) {
printf("\nKernal started on CPU 1!\n"); // will detect cpu later
}
#ifndef SCREEN_OUTPUT
if(init_serial(COM1)) printf("\nKernal started on CPU 1!\n");
#endif
get_mem_capabilities();
init_pmap(map_complete_physical());
unmap_lowmem();
rsdp_t *rsdp;
rsdp = find_RSDP();
if(!(rsdp)) {
printf("Couldn't find the RSDP... not sure what to do now.\n");
panic(KERNEL_PANIC_RSDP_UNFOUND, KERNEL_PANIC_INVOKED);
}
find_root_sdp();
debug_acpi();
init_interrupts();
randinit();
dump_video();
debug_print_memory();
init_pmap();
debug_pmap();
panic(KERNEL_PANIC_KERNEL_RETURNED, KERNEL_PANIC_INVOKED);
smp_boot();
PANIC(KERNEL_PANIC_KERNEL_RETURNED);
}

100
src/kernel/klog.c Normal file
View File

@ -0,0 +1,100 @@
#include <stdint.h>
#include <serial.h>
#include <io.h>
#include <printf.h>
#include <kernel.h>
#include <stdbool.h>
#include <libc.h>
//README
//this file has some temporary workarounds until I get further into development
//this isn't going to be the serial driver I end up keeping
#define VIDEO_BUFFER PHYS_TO_VIRT(0xb8000)
#define MAX_LOOPBACK_ATTEMPTS 20
//later we'll set it up for interrupting instead of polling
bool init_serial(uint16_t port) {
outb_wait(port + 1, 0x00);
outb_wait(port + 2, 0x00);
outb_wait(port + 3, 0x80);
outb_wait(port + 0, 0x01); //here
outb_wait(port + 1, 0x00);
outb_wait(port + 3, 0x03);
outb_wait(port + 2, 0xc7);
outb_wait(port + 4, 0x0b);
outb_wait(port + 4, 0x1e);
//we shouldn't accept this as a loopback success.
//I'm making an exception because my dell desktop has a faulty serial port
//that requires some extra effort.
uint8_t loopback_byte;
for(int attempts = 0; attempts < MAX_LOOPBACK_ATTEMPTS; attempts++) {
outb_wait(port + 0, 0xae); // test char
loopback_byte = inb(port);
if(loopback_byte == 0xae) break;
}
if(loopback_byte != 0xae) {
//I'm sorry if these next few lines give you a stroke. They gave me one.
//My old Dell test desktop has disfunctional serial hardware that only works after a reboot.
//Seeing we don't have ACPI set up yet, we can't reboot without crashing.
//IM GOING TO REMOVE THIS ONCE I DEVELOP THE "REAL" SERIAL DRIVER!!!!
asm("mov rax, 0\n"
"div rax\n");
return false;
}
outb_wait(port + 4, 0x0f);
return true;
}
void _putchar_serial(uint16_t port, char msg) {
while(!(inb(port + 5) & 0x20)); //wait for transmit to be done
outb_wait(port, msg);
}
static unsigned int on_char = 0;
static unsigned int on_line = 0;
void clear_screen() {
char *screen_buffer = (void *)VIDEO_BUFFER;
for(unsigned int i = 0; i <= (80 * 25); i++) screen_buffer[i * 2] = ' ';
on_char = 0;
on_line = 0;
}
void next_line() {
if(on_line >= 25) {
on_line = 0;
clear_screen();
}
else {
on_line++;
}
on_char = 0;
}
void move_cursor(unsigned int x, unsigned int y) {
//yeah yeah yeah, it doens't test for overflows... I'm gonna delete this once i get serial working
//on my second desktop
on_char = x;
on_line = y;
}
//until we get serial working on hp
void _putchar_screen(char msg) {
if(on_char >= 80) {
next_line();
} else if(msg == '\n') {
next_line();
return;
}
char *screen_buffer = (void *)((uint64_t)VIDEO_BUFFER + ((on_line * 160) + (on_char * 2)));
*screen_buffer = msg;
on_char++;
}

View File

@ -3,7 +3,7 @@
// TODO clean up variable names
int strncmp(const char *s1, const char *s2, unsigned int n) {
int i;
for(i = 0; ((i <= n) && (s1[i] != '\0') && (s2[i] != '\0')); i++) {
for(i = 0; ((i < (n - 1)) && (s1[i] != '\0') && (s2[i] != '\0')); i++) {
if(s1[i] != s2[i]) {
return(s1[i] - s2[i]);
}
@ -11,6 +11,13 @@ int strncmp(const char *s1, const char *s2, unsigned int n) {
return(s1[i] - s2[i]);
}
//this one hasn't been tested
size_t strlen(const char *s) {
size_t len = 0;
while(s[len + 1] != '\0') len++;
return(len);
}
int strcmp(const char *s1, const char *s2) {
int i;
for(i = 0; ((s1[i] != '\0') && (s2[i] != '\0')); i++) {
@ -25,7 +32,7 @@ int memcmp(const void *s1, const void *s2, size_t n) {
const unsigned char *p1 = s1;
const unsigned char *p2 = s2;
int i;
for(i = 0; i < n; i++) {
for(i = 0; i < n - 1; i++) {
if(p1[i] != p2[i]) {
return(p1[i] - p2[i]);
}
@ -41,7 +48,7 @@ void strcpy(char *dest, char *src) {
void *memcpy(void *dest, char *src, size_t n) {
char *p = dest;
for(unsigned int i = 0; i <= n; i++) {
for(unsigned int i = 0; i < n; i++) {
p[i] = src[i];
}
return(dest);
@ -56,12 +63,16 @@ void *bzero(void *dest, size_t size) {
}
//TODO move this function to a seperate math library
unsigned int ceil(double n) {
int ceil(double n) {
int low_n = (int)n;
if(n == (double)low_n) return(low_n);
return(low_n + 1);
}
int round(double n) {
return(int)(n + 0.5);
}
void *memset(void *s, char c, size_t n) {
char *p = s;
for(size_t i = 0; i < n; i++) {

179
src/kernel/madt.c Normal file
View File

@ -0,0 +1,179 @@
#include <stdint.h>
#include <libc.h>
#include <acpi.h>
#include <int.h>
#include <kernel.h>
#include <stddef.h>
#include <heap.h>
#include <printf.h>
#include <smp.h>
#include <madt.h>
#include <cpuid.h>
//madt entry types. there's more, we don't need em yet
#define LOCAL_APIC 0
#define IO_APIC 1
#define IO_INT_OVERRIDE 2
#define IO_APIC_NMI_SOURCE 3
#define IO_APIC_NMI 4
#define MADT_LOCAL_APIC_ADDR_OVERRIDE 5
struct madt_sdt {
sdt_head header;
uint32_t apic_base;
uint32_t flags;
} __attribute__((packed));
struct madt_entry {
uint8_t type;
uint8_t length;
} __attribute__((packed));
struct madt_local_apic {
uint8_t processor_id;
uint8_t apic_id;
uint32_t flags;
} __attribute__((packed));
struct madt_local_apic_override {
uint16_t reserved;
void *apic_base;
} __attribute__((packed));
struct madt_ioapic {
uint8_t apic_id;
uint8_t reserved;
uint32_t addr;
uint32_t gsi_base;
} __attribute__((packed));
struct madt_io_int_override {
uint8_t bus_src; //not sure what this is used for.
uint8_t irq;
uint32_t global_vector;
uint16_t flags;
} __attribute__((packed));
static struct madt_sdt *madt;
void init_madt() {
madt = find_sdt(SDT_MADT);
}
void debug_madt() {
struct madt_entry *entry;
for(
entry = (void *)madt + sizeof(*madt);
(void *)entry < (void *)madt + madt->header.length;
entry = (void *)entry + entry->length) {
printf("MADT debug: ");
switch(entry->type) {
case LOCAL_APIC:
printf("local APIC\n");
break;
case IO_APIC:
printf("IOAPIC\n");
break;
case IO_INT_OVERRIDE:
printf("IOAPIC interupt override\n");
break;
case IO_APIC_NMI_SOURCE:
printf("IOAPIC NMI\n");
break;
case IO_APIC_NMI:
printf("Local APIC NMI\n");
break;
case MADT_LOCAL_APIC_ADDR_OVERRIDE:
printf("APIC addr override\n");
break;
default:
printf("other\n");
break;
}
}
}
unsigned int irq_to_gsi(unsigned int irq) {
struct madt_io_int_override *irq_override;
struct madt_entry *entry;
for(
entry = (void *)madt + sizeof(*madt);
(void *)entry < (void *)madt + madt->header.length;
entry = (void *)entry + entry->length) {
if(entry->type == IO_INT_OVERRIDE) {
irq_override = (void *)entry + 2;
if(irq_override->irq == irq) return irq_override->global_vector;
}
}
return (uint32_t)irq; //no entry found
}
void get_ioapic(struct ioapic_fixedlen *ret) {
struct madt_entry *entry;
struct madt_ioapic *mpio_entry;
ret->count = 0;
for(
entry = (void *)madt + sizeof(*madt);
(void *)entry < (void *)madt + madt->header.length;
entry = (void *)entry + entry->length) {
if(entry->type == IO_APIC) {
mpio_entry = (void *)entry + 2;
if(ret->count) {
ret->ioapic = realloc(ret->ioapic, ++ret->count * sizeof(struct ioapic_t));
}
else {
ret->ioapic = malloc(++ret->count * sizeof(struct ioapic_t));
}
ret->ioapic[ret->count - 1].address = PHYS_TO_VIRT((uintptr_t)mpio_entry->addr);
ret->ioapic[ret->count - 1].gsi_base = mpio_entry->gsi_base;
}
}
}
lapic_t get_lapic() {
struct madt_local_apic_override *ap_override;
struct madt_entry *entry;
for(
entry = (void *)madt + sizeof(*madt);
(void *)entry < (void *)madt + madt->header.length;
entry = (void *)entry + entry->length) {
if(entry->type == MADT_LOCAL_APIC_ADDR_OVERRIDE) {
ap_override = (void *)entry + 2;
return (lapic_t)PHYS_TO_VIRT(ap_override->apic_base);
}
}
return (lapic_t)PHYS_TO_VIRT((uint64_t)madt->apic_base);
}
void get_coreinfo(struct cores_info *cores) {
struct madt_local_apic *ap_info;
struct madt_entry *entry;
uint32_t unused_cpuid, ebx;
__get_cpuid(1, &unused_cpuid, &ebx, &unused_cpuid, &unused_cpuid);
cores->bsp = ebx << 24;
bzero(cores, sizeof(*cores));
for(
entry = (void *)madt + sizeof(*madt);
(void *)entry < (void *)madt + madt->header.length;
entry = (void *)entry + entry->length) {
if(entry->type == LOCAL_APIC) {
ap_info = (void *)entry + 2;
if(ap_info->flags & 1) {
cores->apic_id[cores->corecount] = ap_info->processor_id;
cores->corecount++;
}
}
}
}

View File

@ -1,12 +1,106 @@
#include <printf.h>
#include <paging.h>
#include <stdint.h>
#include <libc.h>
#include <limits.h>
#include <panic.h>
#include <math.h>
#include <kernel.h>
#include <stdbool.h>
#include <cpuid.h>
//just using char because c is a lil bitch and won't let us use void
extern char _kernel_shared_zone_begin;
// PAGE MAPPING
#define PAGEMAP_LOCATION 0x10000
#define MAX_BUDDY_ORDER 8
#define PALLOC_AUTO_BLEVEL MAX_BUDDY_ORDER
typedef struct phys_map {
struct phys_map *next;
unsigned int max_buddy;
uint64_t bsize[MAX_BUDDY_ORDER];
uint64_t *buddy[MAX_BUDDY_ORDER];
} pmap_t;
static pmap_t *first_pmap;
#define MEM_AVAILABLE 1
#define MEM_RESERVED 2
#define MEM_APCI_RECLAIMABLE 3
#define MEM_APCI_NVS 4
#define MEM_BAD 5
// ZONES
#define ZONE_MAP_PLOC 0x7000
#define ZONE_MAP PHYS_TO_VIRT(ZONE_MAP_PLOC)
//reorganized (moved) from header
typedef struct __attribute__((packed)) {
unsigned int present : 1; // present, must be one when accessed.
unsigned int read_write : 1; // if set to one, read and write is set
unsigned int user : 1; // For seperating CPL 0-2 and 3+
unsigned int writethrough_cache : 1; // honestly maybe I should look into caching
unsigned int cachable : 1; // hardware chaching. 0 is enabled, whats the worst that could happen?
unsigned int accessed : 1; // we'll never use any of these!
unsigned int zg0 : 1; // needs to be (and will be) zero'd
unsigned int size : 1; // if set to 1, this entry points to physical memory
unsigned int zg1 : 1; // needs to be (and will be) zero'd
unsigned int software_marks : 3; // available for our own use, I doubt we'll use it in such a simple thing
uintptr_t base_ptr : 40;
unsigned int avail:11;
unsigned int no_exec:1;
} page_table;
struct memory_table {
void *base;
uint64_t length;
uint32_t type;
uint32_t ACPI;
} __attribute__((packed));
static bool NX_capable;
static bool huge_page_capable;
void get_mem_capabilities() {
uint32_t unused, edx;
__get_cpuid(0x80000001, &unused, &unused, &unused, &edx);
huge_page_capable = (edx >> 26) & 1;
NX_capable = (edx >> 20) & 1;
}
void unmap_lowmem() {
struct stack_frame *frame;
asm("addq rsp, %0\n"
"addq rbp, %0\n"
"mov %0, rbp"
:"=r"(frame)
:"r"(PA_OFFSET));
while(frame->next != 0) {
printf("%p\n", frame->function_base);
frame->next = PHYS_TO_VIRT((void *)frame->next);
frame = frame->next;
}
//[future]
//eventually, you should use the function that unmaps pages when you write it
page_table *entry = (page_table *)PAGEMAP_LOCATION;
entry[0].present = 0;
}
void debug_print_memory() {
struct memory_table *memtable = (struct memory_table *)&_meminfo_loc;
struct memory_table *memtable = (void *)ZONE_MAP;
printf(" __________________________________________________________________________\n");
printf("| type\tstart\t\t\tend\t\t\tsize\t\t |\n");
printf("|--------------------------------------------------------------------------|\n");
@ -16,32 +110,39 @@ void debug_print_memory() {
printf("----------------------------------------------------------------------------\n");
}
void ram_stresser() {
struct memory_table *memtable = (void *)ZONE_MAP - PA_OFFSET;
memtable[6].length = 0x10000;
}
void debug_pmap() {
struct phys_map* pmap;
pmap_t *pmap = first_pmap;
int pmap_i = 0, order;
uint64_t buddy_size, blong_i, bbit_i, buddy_chunksize, omit_cnt;
printf("Maximum buddy order: %u (up to %#x sized chunks)\n", MAX_BUDDY_ORDER, (0x1000 << MAX_BUDDY_ORDER - 1));
for(pmap = (struct phys_map*)&_stage2_pagetable; pmap != 0; pmap = pmap->next) {
uint64_t blong_i, bbit_i, buddy_chunksize, omit_cnt;
printf("Maximum buddy order: %u (up to %#x sized chunks)\n", MAX_BUDDY_ORDER, ((0x1000 << MAX_BUDDY_ORDER) - 1));
for(; pmap != 0; pmap = pmap->next) {
printf("Table %u:\n"
"\tPhysical Start:\t%#p\n"
"\tTable location:\t%#p\n", pmap_i, pmap->zone_paddr, pmap);
"\tPhysical/pmap start:\t%#p\n"
"\tTable Size:\t%u\n", pmap_i, pmap,
(uint64_t)(pmap->buddy[MAX_BUDDY_ORDER - 1] + pmap->bsize[MAX_BUDDY_ORDER - 1]) - (uint64_t)pmap);
for(order = 0; order <= MAX_BUDDY_ORDER - 1; order++) {
buddy_chunksize = (0x1000 << order); //TODO just put it in the for loop
buddy_size = (((order == MAX_BUDDY_ORDER - 1)
? (uint64_t *)pmap->next : pmap->buddy[order + 1]) - pmap->buddy[order]);
printf("\tbuddy[%u]:\n"
"\t\tAddress:\t%#x\n"
"\t\tAddress:\t%#p\n"
"\t\tSize:\t\t%u\n"
"\t\tBuddies:\t\t\n", order, pmap->buddy[order], buddy_size);
"\t\tBuddies:\t\t\n", order, pmap->buddy[order], pmap->bsize[order]);
omit_cnt = 0;
for(blong_i = 0; blong_i < buddy_size; blong_i++) {
for(blong_i = 0; blong_i < pmap->bsize[order]; blong_i++) {
for(bbit_i = 0; bbit_i < 64; bbit_i++) {
if(*(pmap->buddy[order] + blong_i) & ((uint64_t)1 << bbit_i)) {
if((omit_cnt < 20) || (blong_i == buddy_size - 1)) {
printf("address %#x\tbit %u: %p\t is free\n",
pmap->buddy[order] + blong_i, bbit_i, (uint64_t)pmap->zone_paddr + (((blong_i * 64) + bbit_i) * buddy_chunksize));
if((pmap->buddy[order][blong_i]) & ((uint64_t)1 << bbit_i)) {
if((omit_cnt < 20) || (blong_i == pmap->bsize[order] - 1)) {
printf("address %#p\tbit %u: %p\t is free\n",
pmap->buddy[order] + blong_i,
bbit_i,
((uint64_t)pmap - PA_OFFSET) + ((((blong_i * 64) + bbit_i) * buddy_chunksize)));
}
omit_cnt++;
if(omit_cnt == 20) {
@ -55,237 +156,291 @@ void debug_pmap() {
}
}
void init_pmap() {
struct memory_table *zones = (struct memory_table *)&_meminfo_loc;
struct phys_map *pmap = (struct phys_map*)&_stage2_pagetable;
unsigned int zone_i, pmap_i = 0;
int budorder;
//we keep this info out of the struct because we won't need it after setting up
uint64_t zone_len[MAX_ZONE_CNT], buddy_bitlen[MAX_ZONE_CNT][MAX_BUDDY_ORDER], *buddy_end;
uint64_t pmap_size, pmap_bbitsize, pmap_blongsize, buddy_size, buddy_bit, pmap_bit;
uint64_t threshold_bitsize, threshold_longsize = 0;
//TODO I know you don't want to, but you need to thoroughly check this.
void pfree(void *addr, size_t size) {
int blevel = 0;
uint64_t *onbyte; //the byte out buddy resides on in the current level
uint64_t page_bitloc; // how many bits we are away from buddy[0]. Helps calculate bitshifts
int bbitlen; //length of free'd area in current level
int lshift; //lshift is how many bits we shift over, rightbit is what it sounds like dumbass
pmap_t *pmap = first_pmap;
void *paged_mem = (void *)&_stage2_pagetable + 0x200000;
map_page(&_stage2_pagetable, &_stage2_pagetable, PAGE_SIZE_2M);
for(zone_i = 0; zones[zone_i].length > 0; zone_i++) {
if((zones[zone_i].type == MEM_AVAILABLE) && (zones[zone_i].ACPI & 1)) {
//hopefully this should never happen...
//I should dig into the docs to check before removing this.
//We also could forget about MAX_ZONE_CNT if we did.
if(zone_i >= MAX_ZONE_CNT) {
printf("Only %u zones can be used! Modify MAX_ZONE_CNT in paging.h to use all memory.\n", MAX_ZONE_CNT);
break;
}
/* note: there's no security check to see if the page is actually allocated,
* or if we are freeing the table itself.
* This should be okay, as only the kernel will be calling it.
* If it gets too messy we can always come back.
*/
if((zones[zone_i].base <= (void*)&_stage2_pagetable) &&
(zones[zone_i].base + zones[zone_i].length >= (void *)&_stage2_pagetable)) {
pmap->zone_paddr = &_stage2_pagetable;
zone_len[pmap_i] = zones[zone_i].length - (pmap->zone_paddr - zones[zone_i].base);
}
else {
pmap->zone_paddr = zones[zone_i].base;
zone_len[pmap_i] = zones[zone_i].length;
}
pmap->buddy[0] = (void *)pmap + sizeof(*pmap);
for(budorder = 1; budorder < MAX_BUDDY_ORDER; budorder++) {
buddy_bitlen[pmap_i][budorder - 1] = GET_BUDDY_BITLEN(zone_len[pmap_i], budorder - 1);
pmap->buddy[budorder] = (uint64_t *)pmap->buddy[budorder - 1] +
LSIZE_FROM_BITLEN(buddy_bitlen[pmap_i][budorder - 1]);
}
buddy_bitlen[pmap_i][MAX_BUDDY_ORDER - 1] = GET_BUDDY_BITLEN(zone_len[pmap_i], MAX_BUDDY_ORDER - 1);
pmap->next = (void *)pmap->buddy[MAX_BUDDY_ORDER - 1] +
(LSIZE_FROM_BITLEN(buddy_bitlen[pmap_i][MAX_BUDDY_ORDER - 1]) * 8);
pmap = pmap->next;
pmap_i++;
//allocates by an extra sizeof(struct phys_map),
//but were about to discard anyway
while((void *)pmap + sizeof(*pmap) >= paged_mem) {
map_page(paged_mem, paged_mem, PAGE_SIZE_2M);
paged_mem += 0x200000;
}
}
if(((uintptr_t)addr & 4095) || (size & 4095)) {
PANIC(KERNEL_PANIC_INVALID_PFREE);
return; //TODO [minor] some more specificity, not a huge deal
}
size /= 0x1000;
for(; pmap != 0; pmap = pmap->next) {
page_bitloc = (addr - (void *)pmap) / 0x1000;
onbyte = pmap->buddy[0] + (page_bitloc / 64);
if((addr >= (void *)pmap) && onbyte < pmap->buddy[1]) break;
}
pmap_size = (void*)(pmap) - (void*)&_stage2_pagetable;
if(pmap_size >= zone_len[0]) panic(); //TODO debugging
pmap_i = 0;
for(pmap = (struct phys_map*)&_stage2_pagetable; pmap->next != 0; pmap = pmap->next) {
for(budorder = MAX_BUDDY_ORDER - 1; budorder >= 0; budorder--) {
pmap_bbitsize = ceil((float)pmap_size / ((uint64_t)0x1000 << budorder));
pmap_blongsize = pmap_bbitsize / 64;
if(budorder == MAX_BUDDY_ORDER - 1) {
buddy_size = (uint64_t *)pmap->next - pmap->buddy[budorder];
buddy_end = (uint64_t *)pmap->next - 1;
threshold_bitsize = ((pmap_blongsize * 64) + pmap_bbitsize) * 2;
}
else {
buddy_size = pmap->buddy[budorder + 1] - pmap->buddy[budorder];
buddy_end = pmap->buddy[budorder + 1] - 1;
threshold_longsize = threshold_bitsize / 64;
}
pmap_bit = pmap_bbitsize & 63;
buddy_bit = buddy_bitlen[pmap_i][budorder] & 63;
while(blevel < MAX_BUDDY_ORDER) {
lshift = (page_bitloc / (1 << blevel)) & 63;
onbyte = pmap->buddy[blevel] + ((page_bitloc / 64) / (1 << blevel));
bbitlen = size / (1 << blevel);
if((pmap_bbitsize >= BITLEN_FROM_LSIZE(buddy_size)) && (pmap == (void *)&_stage2_pagetable)) {
bzero(pmap->buddy[budorder], buddy_size * 8);
}
else {
if(budorder == MAX_BUDDY_ORDER - 1) {
if(pmap == (void*)&_stage2_pagetable) {
if(pmap_blongsize) bzero(pmap->buddy[budorder], (pmap_blongsize - 1) * 8);
if(pmap_bit) {
*(pmap->buddy[budorder] + pmap_blongsize) = ~(((uint64_t)1 << pmap_bit) - 1);
}
else {
*(pmap->buddy[budorder] + pmap_blongsize) = UINT64_MAX;
}
if(pmap_blongsize + 1 == buddy_size) {
*buddy_end &= ((uint64_t)1 << buddy_bit) - 1;
}
else {
memset(pmap->buddy[budorder] + pmap_blongsize + 1, UINT8_MAX, (buddy_size - 1) * 8);
if(buddy_bit) {
*buddy_end = ((uint64_t)1 << buddy_bit) - 1;
}
else {
*buddy_end = UINT64_MAX;
}
}
threshold_bitsize = ((pmap_blongsize * 64) + pmap_bbitsize) * 2;
}
else {
memset(pmap->buddy[budorder], UINT8_MAX, (buddy_size - 1) * 8);
if(buddy_bit) {
*buddy_end = ((uint64_t)1 << buddy_bit) - 1;
}
else {
*buddy_end = UINT64_MAX;
}
}
}
else if(pmap == (void *)&_stage2_pagetable) {
if(threshold_longsize) bzero(pmap->buddy[budorder], (threshold_longsize - 1) * 8);
if(threshold_bitsize > pmap_bbitsize)
*(pmap->buddy[budorder] + threshold_longsize) = ((uint64_t)1 << ((threshold_bitsize - 1) & 63));
if(buddy_size - threshold_longsize)
bzero(pmap->buddy[budorder] + threshold_longsize + 1, buddy_size - threshold_longsize);
if(buddy_bit & 1) {
*buddy_end = ((uint64_t)1 << (buddy_bit - 1));
}
threshold_bitsize = ((pmap_blongsize * 64) + pmap_bbitsize) * 2;
}
else {
bzero(pmap->buddy[budorder], buddy_size);
if(buddy_bit & 1) {
*buddy_end = ((uint64_t)1 << ((buddy_bit) - 1));
}
//TODO clean up this part ------------------------------------------------------------- (below)
if(bbitlen <= 1) {
if(lshift & 1) {
if((*onbyte >> (lshift - 1)) & 1) {
*onbyte &= ~(((uint64_t)1 << (lshift - 1)) | ((uint64_t)1 << lshift));
size += (1 << blevel);
page_bitloc -= (1 << blevel);
bbitlen = size / (1 << blevel);
}
}
else if((*onbyte >> (lshift + 1)) & 1) {
*onbyte &= ~(((uint64_t)1 << (lshift + 1)) | ((uint64_t)1 << lshift));
size += (1 << blevel);
bbitlen = size / (1 << blevel);
}
}
else if(((lshift + bbitlen) & 1) && ((*onbyte >> (lshift + bbitlen)) & 1)) {
*onbyte ^= ((uint64_t)1 << (lshift + bbitlen));
size += (1 << blevel);
bbitlen = size / (1 << blevel);
}
//TODO clean up this part ------------------------------------------------------------- (above)
if((!((size - 1) & size)) && (bbitlen != 1)) {
blevel = 63 - __builtin_clzl(size);
}
else {
if(bbitlen <= 1) {
*onbyte |= ((uint64_t)1 << lshift);
break;
} else if(bbitlen & 1) { //check me
size -= (1 << blevel);
*onbyte |= ((uint64_t)1 << (bbitlen + lshift));
}
blevel++;
}
pmap_i++;
}
}
/**
* BIG TODO:
* Paging turned out to be simpler then I thought. I've temporarily fixed the code, but needs to be rewritten/simplified.
* Let's get rid of those nasty GOTOs if we can.
* Also, once we get physical memory allocator up and running, impliment that in this function.
**/
void *palloc(size_t size) {
bool self_alloc;
int min_blevel, blevel;
uint64_t bbit, unshifted_entry, threshold, bloc; //TODO move when you've confirmed casting stuff
uint64_t buddy_i, *ret, *bentry;
int itercount;
pmap_t *pmap = first_pmap;
bool map_page(void *virtual_addr, void *physical_addr, uint8_t size) {
//printf("map page called\n");
uintptr_t va_ptr = (uintptr_t)virtual_addr;
uintptr_t pa_ptr = (uintptr_t)physical_addr;
if((va_ptr % (1 << size)) || (pa_ptr % (1 << size))) {
return 0;
}
page_table *table = (page_table *)PAGEMAP_LOCATION;
int pte_i = (va_ptr >> 12) & 0x1ff;
int pde_i = (va_ptr >> 21) & 0x1ff;
int pdpe_i = (va_ptr >> 30) & 0x1ff;
int pml4e_i = (va_ptr >> 39) & 0x1ff;
if(table->pml4e[pml4e_i].present) {
if(table->pml4e[pml4e_i].base_ptr != (uintptr_t)&table->pdpe[pdpe_i] >> 12) goto error;
if(table->pdpe[pdpe_i].present) {
if(size == PAGE_SIZE_1G) {
if(table->pdpe[pdpe_i].base_ptr == ((uintptr_t)pa_ptr >> 30 & 0x1ff))
return true;
goto error;
}
if(table->pdpe[pdpe_i].base_ptr != (uintptr_t)&table->pde[pde_i] >> 12) goto error;
if(table->pde[pde_i].present) {
if(size == PAGE_SIZE_2M) {
if(table->pde[pde_i].base_ptr == ((uintptr_t)pa_ptr >> 21 & 0x1ff))
return true;
goto error;
}
if(table->pde[pde_i].base_ptr != (uintptr_t)&table->pte[pte_i] >> 12) goto error;
if(table->pte[pte_i].present) {
if(table->pte[pte_i].base_ptr != ((pa_ptr >> 12) & 0x1ff)) goto error;
return true;
}
else goto mod_page_pte;
}
else goto mod_page_pde;
}
else goto mod_page_pdpe;
if(size == 0) return 0;
if(size & 4095) {
size = DIV_ROUND_UP(size, 0x1000);
}
else {
table->pml4e[pml4e_i].base_ptr = (uintptr_t)&table->pdpe[pdpe_i] >> 12;
table->pdpe[pml4e_i].read_write = 1;
table->pml4e[pml4e_i].present = 1;
mod_page_pdpe:
table->pdpe[pdpe_i].read_write = 1;
//TODO you just found out things are a lot more simple then you thought!
if(size == PAGE_SIZE_1G) {
table->pdpe[pdpe_i].size = 1;
table->pdpe[pdpe_i].base_ptr = pa_ptr >> 12;
table->pdpe[pdpe_i].present = 1;
return true;
}
table->pdpe[pdpe_i].base_ptr = (uintptr_t)&table->pde[pde_i] >> 12;
table->pdpe[pdpe_i].present = 1;
mod_page_pde:
table->pde[pde_i].read_write = 1;
if(size == PAGE_SIZE_2M) {
table->pde[pde_i].size = 1;
table->pde[pde_i].base_ptr = pa_ptr >> 12;
table->pde[pde_i].present = 1;
return true;
}
table->pde[pde_i].base_ptr = (uintptr_t)&table->pte[pte_i] >> 12;
table->pde[pde_i].present = 1;
mod_page_pte:
table->pte[pte_i].base_ptr = pa_ptr >> 12;
table->pte[pte_i].read_write = 1;
table->pte[pte_i].present = 1;
return true;
size = size / 0x1000;
}
error:
printf("Page allocation error!\n");
return false;
//checking if pmap has been initilized; if not we've been called to self allocate
//the first buddy should never be allocated; that's where our pmap lives
if(pmap->buddy[pmap->max_buddy][0] & 1) {
self_alloc = true;
min_blevel = pmap->max_buddy;
}
else {
//log(size, 2)
self_alloc = false;
min_blevel = 63 - __builtin_clzl(size);
if(size & (size - 1)) min_blevel++;
if(min_blevel > MAX_BUDDY_ORDER - 1) return 0;
}
for(blevel = min_blevel; blevel < MAX_BUDDY_ORDER; blevel++) {
for(pmap = first_pmap; pmap != 0; pmap = pmap->next) {
for(buddy_i = 0; buddy_i < pmap->bsize[blevel]; buddy_i++) {
if(pmap->buddy[blevel][buddy_i] > (uint64_t)0) {
bentry = &pmap->buddy[blevel][buddy_i];
bbit = __builtin_ctzl(*bentry);
bloc = bbit;
*bentry ^= (uint64_t)1 << bbit;
ret = (((buddy_i * 64) + bbit) * (0x1000 << blevel)) + (void *)pmap;
threshold = 0b11;
itercount = 1;
for(blevel--; blevel >= 0; blevel--) {
bentry = pmap->buddy[blevel] + ((bentry - pmap->buddy[blevel + 1]) * 2);
itercount++;
if(bloc >= 32) bentry += 1;
bloc = (bloc * 2) & 63; // will be the amount we need to shift
bbit = ceil((float)size / (1 << blevel));
unshifted_entry = ((uint64_t)1 << bbit) & threshold;
if(unshifted_entry) {
threshold = ((uint64_t)1 << (bbit * 2)) - 1;
}
else {
threshold = (threshold << 2) | threshold;
}
*bentry |= (unshifted_entry << bloc);
}
if(!self_alloc) bzero(ret, size * 0x1000);
return ret;
}
}
}
}
return 0;
}
//returns size of pages needed
size_t map_complete_physical() {
uint64_t total_mem;
unsigned int pdpe_cnt, pde_cnt, pde_max_i;
int zone_i, entry_i;
struct memory_table *zones = (void *)ZONE_MAP_PLOC;
page_table *pml4 = (page_table *)PAGEMAP_LOCATION;
page_table *pdpe = (page_table *)&_kernel_shared_zone_begin;
page_table *pde;
for(zone_i = 0; zones[zone_i].length > 0; zone_i++);
total_mem = (uint64_t)zones[zone_i - 1].base + zones[zone_i - 1].length;
pdpe_cnt = (total_mem + (0x40000000 - 1)) / 0x40000000;
entry_i = (PA_OFFSET >> 39) & 0x1ff;
pml4[entry_i].base_ptr = (uintptr_t)&_kernel_shared_zone_begin >> 12;
pml4[entry_i].read_write = 1;
pml4[entry_i].user = 0;
pml4[entry_i].size = 0;
pml4[entry_i].no_exec = 1;
pml4[entry_i].present = 1;
if(huge_page_capable) {
for(int pdpe_i = 0; pdpe_i < pdpe_cnt; pdpe_i++) {
pdpe[pdpe_i].base_ptr = pdpe_i << 18;
pdpe[pdpe_i].read_write = 1;
pdpe[pdpe_i].user = 0;
pdpe[pdpe_i].size = 1;
pdpe[pdpe_i].no_exec = NX_capable;
pdpe[pdpe_i].present = 1;
}
return pdpe_cnt * 0x1000;
}
else {
pde_cnt = (total_mem + 0x100000) / 0x200000;
for(int pdpe_i = 0; pdpe_i < pdpe_cnt; pdpe_i++) {
pde = (page_table *)(&_kernel_shared_zone_begin + (pdpe_cnt * 0x1000) + (pdpe_i * 0x1000));
if((pdpe_i < pdpe_cnt - 1) || (!(pde_cnt & 511))) {
pde_max_i = 512;
}
else {
pde_max_i = pde_cnt & 511;
}
pdpe[pdpe_i].base_ptr = (uintptr_t)pde >> 12;
pdpe[pdpe_i].read_write = 1;
pdpe[pdpe_i].user = 0;
pdpe[pdpe_i].size = 0;
pdpe[pdpe_i].no_exec = NX_capable;
pdpe[pdpe_i].present = 1;
for(int pde_i = 0; pde_i < pde_max_i; pde_i++) {
pde[pde_i].base_ptr = ((pdpe_i << 9) + pde_i) << 9;
pde[pde_i].read_write = 1;
pde[pde_i].user = 0;
pde[pde_i].size = 1;
pde[pde_i].no_exec = NX_capable;
pde[pde_i].present = 1;
}
}
return (pdpe_cnt * 2) * 0x1000;
}
}
pmap_t *init_pmap(size_t pagetable_size) {
pmap_t *pmap, *last_pmap;
struct memory_table *zones = (void *)ZONE_MAP;
int budorder, zone_i;
uint64_t pmap_size, pmap_bbitsize, zone_size;
bool first_pmap_i = true;
for(zone_i = 0; zones[zone_i].length > 0; zone_i++) {
if((zones[zone_i].type == MEM_AVAILABLE) && (zones[zone_i].ACPI & 1) &&
zones[zone_i].length >= (0x2000)) {
printf("found allocatable map at %p\n", zones[zone_i].base);
last_pmap = pmap;
if(zones[zone_i].base == (void *)0x100000) {
zone_size = zones[zone_i].length - (((uint64_t)&_kernel_shared_zone_begin - 0x100000) + pagetable_size);
pmap = PHYS_TO_VIRT((void *)&_kernel_shared_zone_begin + pagetable_size);
}
else {
zone_size = zones[zone_i].length;
pmap = PHYS_TO_VIRT(zones[zone_i].base);
}
if(first_pmap_i) {
pmap->next = NULL;
first_pmap_i = false;
}
else {
pmap->next = last_pmap;
}
for(budorder = 0; budorder < MAX_BUDDY_ORDER; budorder++) {
pmap_bbitsize = zone_size / (0x1000 << budorder);
pmap->bsize[budorder] = DIV_ROUND_UP(pmap_bbitsize , 64);
if(budorder) {
pmap->buddy[budorder] = pmap->buddy[budorder - 1] + pmap->bsize[budorder - 1];
}
else {
pmap->buddy[0] = (void *)pmap + sizeof(*pmap);
}
if(budorder < MAX_BUDDY_ORDER - 1) {
bzero(pmap->buddy[budorder], pmap->bsize[budorder] * 8);
if(pmap_bbitsize & 1) {
pmap->buddy[budorder][pmap->bsize[budorder] - 1] =
((uint64_t)1 << ((pmap_bbitsize - 1) & 63));
}
if(pmap_bbitsize == 1) {
pmap->max_buddy = budorder;
for(budorder++; budorder < MAX_BUDDY_ORDER; budorder++) {
pmap->buddy[budorder] = 0;
pmap->bsize[budorder] = 0;
}
break;
}
}
else {
pmap->max_buddy = MAX_BUDDY_ORDER - 1;
memset(pmap->buddy[budorder], UINT8_MAX, pmap->bsize[budorder] * 8);
if((pmap_bbitsize / 64) != (pmap->bsize[budorder])) {
pmap->buddy[budorder][pmap->bsize[budorder] - 1] =
(((uint64_t)1 << (pmap_bbitsize & 63)) - 1);
}
}
}
pmap_size = (uint64_t)(pmap->buddy[pmap->max_buddy] + pmap->bsize[pmap->max_buddy]) - (uint64_t)pmap;
first_pmap = pmap; //we spoof palloc into allocating from the specific required pmap.
palloc(pmap_size);
}
}
return pmap;
}

View File

@ -1,13 +1,15 @@
#include <panic.h>
#include <stdint.h>
#include <printf.h>
#include <kernel.h>
#include <serial.h>
#include <isv.h>
void panic(int reason, int type) { // will fill with debugging info latter
void panic(int reason, void *eframe_p, struct registers *regs) { // will fill with debugging info latter
struct stack_frame *frame;
#ifdef EASTEREGG_BLOATWARE
printf("Kernel PANIC!!!!!!!\n");
printf("\nKernel PANIC!!!!!!!\n");
printf(
" _.-^^---....,,-- \n"
" _-- --_ \n"
@ -24,31 +26,65 @@ void panic(int reason, int type) { // will fill with debugging info latter
printf("Kernel Panic!\n");
#endif
printf("Reason:\n");
switch(reason) {
case KERNEL_PANIC_PMAPSIZE :
printf("\tThe physical map can't fit in the first memory zone.\n"
"\tNote: If you have this issue, please contact me.\n"
"\tIt's a fixable bug caused by an unlikely scenario.\n");
break;
case KERNEL_PANIC_RSDP_UNFOUND:
printf("\tRSDP unfound!\n");
break;
case KERNEL_PANIC_KERNEL_RETURNED:
printf("\tThe kernel (almost) reached its return!\n");
break;
case KERNEL_PANIC_HW_EXCEPTION:
case KERNEL_PANIC_HW_EXCEPTION_ERR:
printf("\nHardware exception in kernel space!\n");
break;
case KERNEL_PANIC_RSDP_UNFOUND:
printf("\nRSDP unfound!\n");
break;
case KERNEL_PANIC_KERNEL_RETURNED:
printf("\nThe kernel (almost) reached its return!\n");
break;
case KERNEL_PANIC_INVALID_PFREE:
printf("\npfree was called with invalid paramaters.\n");
break;
case KERNEL_PANIC_INVALID_RSDT:
printf("\nInvalid RSDT or XSDT after RSDP verification.\nIf this happens, go buy some lottery tickets.\n");
break;
case KERNEL_PANIC_INVALID_IOAPIC_VEC:
printf("\nThe kernel tried to make a vector that was too high for the IOAPIC to handle.\n");
break;
case KERNEL_PANIC_HPET_REQUIRED:
printf("\nHPET is required. \nIf you get this error, let know;\nif enough people share this issue, I'll impliment PIT usage.\n");
default:
printf("\nUnknown panic code %i\n.", reason);
break;
}
printf("\nStack trace:\n");
struct stack_frame *frame;
asm("mov %%rbp,%0" : "=r"(frame) ::);
char reg_str[156];
if((reason == KERNEL_PANIC_HW_EXCEPTION) || (reason == KERNEL_PANIC_HW_EXCEPTION_ERR)) {
sprintf(reg_str, "rax:\t0x%lx\n"
"rbx:\t0x%lx\n"
"rcx:\t0x%lx\n"
"rdx:\t0x%lx\n"
"rsi:\t0x%lx\n"
"rdi:\t0x%lx\n", regs->rax, regs->rbx, regs->rcx, regs->rdx, regs->rsi, regs->rdi);
}
if(reason == KERNEL_PANIC_HW_EXCEPTION) {
struct int_frame *ex_frame = eframe_p;
printf("Offending instruction:\t0x%p\n", ex_frame->rip);
printf(reg_str);
}
else if (reason == KERNEL_PANIC_HW_EXCEPTION_ERR) {
struct exception_frame *ex_frame = eframe_p;
printf("Offending instruction: 0x%p\nError value: %x\n", ex_frame->frame.rip, ex_frame->err);
printf(reg_str);
}
asm("mov %0, rbp" : "=r"(frame) ::);
printf("\nCall trace:\n");
for(; frame->next != 0; frame = frame->next) {
printf("\t%x\n", frame->function_base);
printf("\t0x%p\n", frame->function_base);
}
//It's not public yet, but it will be
//It's not public yet, but if I ever get somewhere it will be
printf("\nAfter ensuring your computer meets the requirements specified, if you think this is a bug, please open an issue on the git repo or email me at %s.\n", DEV_EMAIL);
for(;;);
asm("cli\nhlt");
}

View File

@ -39,10 +39,12 @@
#include <serial.h>
//and my options
/**
#define PRINTF_DISABLE_SUPPORT_FLOAT
#define PRINTF_DISABLE_SUPPORT_EXPONENTIAL
#define PRINTF_DISABLE_SUPPORT_LONG_LONG
#define PRINTF_DISABLE_SUPPORT_PTRDIFF_T
**/
// define this globally (e.g. gcc -DPRINTF_INCLUDE_CONFIG_H ...) to include the
@ -159,7 +161,11 @@ static inline void _out_char(char character, void* buffer, size_t idx, size_t ma
{
(void)buffer; (void)idx; (void)maxlen;
if (character) {
#ifdef SCREEN_OUTPUT
_putchar_screen(character);
#else
_putchar_serial(COM1, character); // later we should figure out a way to not specifify exclusively com1
#endif
}
}

36
src/kernel/random.c Normal file
View File

@ -0,0 +1,36 @@
#include <stdbool.h>
#include <cpuid.h>
#include <printf.h>
#include <stdint.h>
static bool hw_random = false;
static unsigned long int seed = -1;
void randinit() {
unsigned int unused, eax, ecx;
eax = 0;
ecx = 0;
__get_cpuid(1, &eax, &unused, &ecx, &unused);
hw_random = (ecx >> 30) & 1;
printf("Kernel random source: %s.\n", (hw_random) ? "rdrand" : "pseudo");
}
//As of now, this function isn't nessesarily meant to be fast.
unsigned int randint() {
uint64_t random_long = 0;
if(hw_random) {
asm("rand:\n"
"rdrand %0\n"
"jc finished\n"
"pause\n"
"jmp rand\n"
"finished:" :"=r"(random_long)::"rax" );
}
else {
//TODO FUCKING ENTROPY!!!!!!!111
seed = 1103515245 * seed + 12345;
return(unsigned int)(seed / 65536) % 32768;
}
return 0;
}

80
src/kernel/smp.c Normal file
View File

@ -0,0 +1,80 @@
#include <kernel.h>
#include <madt.h>
#include <timer.h>
#include <cpuid.h>
#include <int.h>
#include <libc.h>
#define LAPIC_ICR_LOW 192
#define LAPIC_ICR_HIGH 196
extern lapic_t lapic;
extern char __load_start_smp_bootloader, __load_stop_smp_bootloader;
extern uint8_t *smp_bootstrap_corecount;
struct icr_reg {
uint8_t vector;
unsigned int deliv_mode:3;
unsigned int dest_mode:1;
unsigned int deliv_status:1;
unsigned int reserved:1;
unsigned int level:1; //0 = de-assert
unsigned int trig_mode:1;
unsigned int reserved_1:2;
unsigned int dest_shorthand:2;
unsigned int reserved_2:12;
}__attribute__((packed));
//1: get bsp number 2
//2: get list of lapics 3
//3: copy code 1
//4:
static inline void write_icr(uint8_t dest, uint32_t message) {
lapic[LAPIC_ICR_HIGH] = (uint32_t)dest << 24;
lapic[LAPIC_ICR_LOW] = message;
}
void smp_boot() {
uint8_t cores_active = 0; //TODO change in asm
struct cores_info cores;
unsigned int ci;
struct icr_reg icr;
bzero(&icr, sizeof(icr));
memcpy(PHYS_TO_VIRT((void *)0x8000), PHYS_TO_VIRT(&__load_start_smp_bootloader),
&__load_stop_smp_bootloader - &__load_start_smp_bootloader);
*(uint8_t **)PHYS_TO_VIRT(&smp_bootstrap_corecount) = &cores_active;
get_coreinfo(&cores);
icr.deliv_mode = 0b101;
icr.dest_shorthand = 0b11;
icr.level = 1;
write_icr(0, *(uint32_t *)&icr);
usleep(10000);
icr.deliv_mode = 0b110;
icr.vector = 8;
write_icr(0, *(uint32_t *)&icr);
usleep(200);
write_icr(0, *(uint32_t *)&icr);
/**
if(cores_active != cores.corecount) {
write_icr(0, *(uint32_t *)&icr);
usleep(200);
}
**/
/**
for(ci = 0; ci < cores.corecount; ci++) {
if(ci == cores.bsp) continue;
//send init sipi
write_icr(ci, *(uint32_t *)&icr);
}
**/
}

View File

@ -0,0 +1,140 @@
global smp_bootstrap_corecount;
[bits 16]
smp_trampoline:
cli
hlt
hlt
xor ax, ax
mov ds, ax
lgdt [.gdt_descriptor_p]
mov eax, cr0
or eax, 0x1
mov cr0, eax ; now in long mode
jmp 0x8:.smp_protected
;________________________________________________________________________________________
;TODO find how to use old gdt
.gdt_start_p: ; we need to start with a null gdt
dd 0
dd 0
.gdt_code_p:
dw 0xffff
dw 0x0000
db 0x00
db 10011010b
db 11001111b
db 0x0000
.gdt_data_p:
dw 0xffff
dw 0x0000
db 0x00
db 10010010b
db 11001111b
db 0x0000
.gdt_end_p:
.gdt_descriptor_p:
dw .gdt_end_p - .gdt_start_p - 1
dq .gdt_start_p
SMP_PROTECTED_CODE_SEGMENT equ .gdt_code_p - .gdt_start_p
SMP_PROTECTED_DATA_SEGMENT equ .gdt_data_p - .gdt_start_p
;________________________________________________________________________________________
.smp_protected
[bits 32]
mov ax, SMP_PROTECTED_DATA_SEGMENT
mov ds, ax
mov ss, ax
mov es, ax
mov fs, ax
mov gs, ax
mov eax, 0x10000 ;TODO clarify _why_ this is a thing
mov cr3, eax
;setting up misc features
mov eax, cr4
; PAE, OSFXSR, OSXMMEXCPT
or eax, 1 << 5 | 1 << 9 | 1 << 10
mov cr4, eax
;set NX and LME
mov eax, 0x80000001
cpuid
and edx, 1 << 20
shr edx, 9
mov ecx, 0xc0000080
rdmsr
or eax, 1 << 8 | 1 << 11
or eax, edx
wrmsr
;enable paging
mov eax, cr0
or eax, 1 << 31 | 1 << 0;
and ax, ~(1 << 2)
mov cr0, eax
jmp SMP_LONG_CODE_SEGMENT:.counter
;________________________________________________________________________________________
[bits 64]
;TODO do we really need all this?
.gdt_start_l:
;and now we set up a temporary GDT creating a 1:1 mapping
dw 0xffff
dw 0
db 0 ;this is the invalid GDT
db 0
db 1
db 0
;now for the code GDT:
.gdt_code_l:
dw 0 ; segment limit 15:00 (I don't think this matters in 64 bit mode!)
dw 0 ; base address 15:00
db 0 ; base address 23:16
db 10011010b
db 00100000b
db 0
.gdt_data_l:
dw 0
dw 0
db 0
db 10010010b
db 00100000b
db 0
.gdt_end_l: ; later calculates offset in defs below
.descriptor_l:
dw .gdt_end_l - .gdt_start_l - 1
dq .gdt_start_l
SMP_LONG_CODE_SEGMENT equ .gdt_code_l - .gdt_start_l
SMP_LONG_DATA_SEGMENT equ .gdt_data_l - .gdt_start_l
.counter:
mov rax, [smp_bootstrap_corecount]
inc QWORD [rax]
cli
hlt
;TODO set up stack; enter kernel
smp_bootstrap_corecount: db 0

35
src/kernel/testmalloc.c Normal file
View File

@ -0,0 +1,35 @@
#define CHUNK_SIZE_FROM_INDEX(i) ((1 << ((i) + 5)))
#include <printf.h>
#include <heap.h>
#include <libc.h>
#include <random.h>
//will delete later
void test_malloc(unsigned int cnt) {
void *testchunks[cnt];
unsigned int rindex[cnt], testchunk_size, i, x;
bzero(rindex, cnt * sizeof(unsigned int));
for(x = 0; x < cnt; x++) {
testchunk_size = (CHUNK_SIZE_FROM_INDEX(randint() % 7) - 24);
testchunks[x] = malloc(testchunk_size);
//printf("ALLOCATING CHUNK %p SIZE %i\n", (void *)testchunks[x] - 24, testchunk_size);
}
for(x = 0; x < cnt;) {
i = randint() % cnt;
if(rindex[i]) continue;
rindex[i] = x;
x++;
}
for(x = 0; x < cnt; x++) {
//printf("FREEING CHUNK %p\n", (void *)testchunks[rindex[x]]);
free(testchunks[rindex[x]]);
}
printf("\nmalloc tester:\n");
printf("THIS NEEDS TO BE EMPTY______________\n");
debug_heap();
printf("____________________________________\n");
}

296
src/kernel/timer.c Normal file
View File

@ -0,0 +1,296 @@
#include <acpi.h>
#include <stdbool.h>
#include <io.h>
#include <printf.h>
#include <libc.h>
#include <kernel.h>
#include <cpuid.h>
#include <panic.h>
#include <int.h>
#include <isv.h>
#define PIT_COMMAND 0x43
#define PIT_FREQ 1193182
#define PIT_CHANNEL_0 0x40
#define PIT_CHANNEL_2 0x42
#define PIT_INPUT_2 0x61
//you can move these if it makes more sense to put them in a lapic.h
#define LAPIC_TIMER_LVT 200
#define LAPIC_TIMER_INIT_COUNT 224
#define LAPIC_TIMER_CURRENT_COUNT 228
#define LAPIC_TIMER_DIVIDE 248
#define LAPIC_TIMER_MODE_ONESHOT 0b00
#define LAPIC_TIMER_MODE_PERIODIC 0b01
#define LAPIC_TIMER_MODE_TSC_DEADLINE 0b10
#define IA32_TSC_DEADLINE 0x6E0
struct acpi_addr_fmt {
uint8_t addr_space; //0 is system memory, 1 is system io
uint8_t reg_bit_width;
uint8_t reg_bit_offset;
uint8_t resrved;
uint64_t addr;
} __attribute__((packed));
struct hpet_sdt {
sdt_head header;
uint8_t hw_rev_id;
unsigned int comparator_cnt:5;
unsigned int counter_size_64bit:1;
unsigned int reserved:1;
unsigned int legacy_capable:1;
uint16_t vendor_id;
struct acpi_addr_fmt acpi_addr;
uint8_t hpet_number;
uint16_t min_tick;
unsigned int protection:4; //0 = no protection, 1 = 4kb, 2 = 64kb
unsigned int oem_attribute:4;
} __attribute__((packed));
struct hpet_info_struct {
uint64_t *hpet_reg;
unsigned int hpet_period;
double hpet_freq_mhz; //improves speed with us quantum
unsigned int timer_cnt;
unsigned int irqs;
bool long_mode;
} hpet_info;
struct lapic_lvt {
uint8_t vector;
unsigned int reserved:4;
unsigned int delivery_status:1;
unsigned int reserved_1:3;
unsigned int mask:1;
unsigned int timer_mode:2;
unsigned int reserved_2:13;
} __attribute__((packed));
extern lapic_t lapic;
#define TSC_TIME_TSC_EVENT 0
#define TSC_TIME_LAPIC_EVENT 1
#define LAPIC_TIME_HPET_EVENT 2
static int clock_mode;
//time quantum is 1us
static double tsc_freq;
static double apic_freq;
static unsigned int early_event_vector;
static struct lapic_lvt lapic_timer_lvt;
uint64_t timestamp() {
if((clock_mode == TSC_TIME_TSC_EVENT) || (clock_mode == TSC_TIME_LAPIC_EVENT)) {
return read_tsc() / tsc_freq;
}
else {
return UINT32_MAX - (lapic[LAPIC_TIMER_CURRENT_COUNT] / apic_freq);
}
return 0;
}
//timer test code
//there is a race condition here.
//I'm leaving it unattended, as I'm only expecting this to be used for SMP booting.
void usleep(unsigned int us) {
if(clock_mode == TSC_TIME_TSC_EVENT) {
write_msr(IA32_TSC_DEADLINE, ((uint64_t)((us * (double)tsc_freq) + 0.5)) + read_tsc());
}
else if(clock_mode == TSC_TIME_LAPIC_EVENT) {
lapic[LAPIC_TIMER_INIT_COUNT] = (uint32_t)((us * (double)apic_freq) + 0.5);
}
else {
hpet_info.hpet_reg[33] = hpet_info.hpet_reg[30] +
(uint32_t)((us * hpet_info.hpet_freq_mhz) + 0.5);
}
asm("hlt\n");
}
/**
we either use:
1: tsc
2: tsc
1: tsc
2: lapic (no tsc deadline)
1: lapic
2: hpet
PIT not implimented, I've yet to find a computer that doesn't have HPET
**/
void calibrate_lapic() {
unsigned int timer_idt = alloc_idt(&KERNEL_IDT_GATE(lapic_timer_racefixer));
uint32_t lapic_ticks;
uint64_t hpet_ticks = (100000000000000 * (1 / (double)hpet_info.hpet_period)) + 0.5;
uint64_t timer_loops = 0;
struct lapic_lvt timer_lvt = {
.vector = timer_idt,
.delivery_status = 0,
.mask = 0,
.timer_mode = LAPIC_TIMER_MODE_PERIODIC
};
printf("Starting LAPIC timer calibration...\n");
lapic[LAPIC_TIMER_LVT] = *(uint32_t *)&timer_lvt;
lapic[LAPIC_TIMER_DIVIDE] = 0b1011;
asm(".global calibrate_lapic_waiting\n"
"xor rbx, rbx\n"
"mov rax, %2\n"
"add rax, %3\n"
"mov [%4], rax\n" //hpet timer
"mov [%5], 0xffffffff\n" //lapic timer
"calibrate_lapic_waiting:\n"
"hlt\n"
"mov %0, %6\n" //save lapic time
"mov [%5], 0\n"
"mov %1, rbx\n"
:"=r"(lapic_ticks), "=r"(timer_loops)
:"r"(hpet_ticks), "m"(hpet_info.hpet_reg[30]), "m"(hpet_info.hpet_reg[33]),
"m"(lapic[LAPIC_TIMER_INIT_COUNT]), "m"(lapic[LAPIC_TIMER_CURRENT_COUNT])
:"rax","rbx" );
apic_freq = ((UINT32_MAX * timer_loops) + (UINT32_MAX - lapic_ticks)) / (double)100000;
printf("LAPIC timer frequency: %f MHz\n", apic_freq);
free_idt(timer_idt);
}
void init_timer() {
struct hpet_sdt *hpet_desc = find_sdt(SDT_HPET);
early_event_vector = alloc_idt(&KERNEL_IDT_GATE(kernel_block));
if(!hpet_desc) PANIC(KERNEL_PANIC_HPET_REQUIRED);
unsigned int first_irq;
struct apic_vt hpet_redirect;
bzero(&hpet_redirect, sizeof(hpet_redirect));
hpet_redirect.vector = early_event_vector;
hpet_redirect.polarity = 1;
hpet_info.hpet_reg = PHYS_TO_VIRT(hpet_desc->acpi_addr.addr);
hpet_info.hpet_period = (hpet_info.hpet_reg[0] >> 32) & UINT32_MAX;
hpet_info.long_mode = (hpet_info.hpet_reg[0] >> 13) & 1;
hpet_info.irqs = (hpet_info.hpet_reg[32] >> 32) & UINT32_MAX;
hpet_info.hpet_freq_mhz = (1 / (double)hpet_info.hpet_period) * 1000000000;
//get first irq available
first_irq = __builtin_ctzl(hpet_info.irqs) + 1;
printf("hpet frequency: %f MHz\n", hpet_info.hpet_freq_mhz);
//do we need to worry about bypassing irq remappings?
create_fixed_interrupt(first_irq, &hpet_redirect);
//set irq, enable triggering of interrupts
hpet_info.hpet_reg[32] = (first_irq << 9) | (1 << 2) | 1;
//enable main counter
hpet_info.hpet_reg[2] = 1;
asm("sti\n");
//detect consistent TSC
uint32_t unused, cpuid_reg;
__get_cpuid(0x80000007, &unused, &unused, &unused, &cpuid_reg);
//goto debug_tsc;
if((cpuid_reg >> 8) & 1) {
printf("Detected invariant TSC\n");
//.1 second to calibrate, TODO do we need to check if the register is big enough?
uint64_t hpet_ticks = (100000000000000 * (1 / (double)hpet_info.hpet_period)) + 0.5;
printf("Starting TSC calibration...\n");
uint64_t volatile start_tsc, end_tsc;
//this may not be the fastest, but I'll improve if needed further into development
asm volatile(
"xor rax, rax\n"
"rdtsc\n"
"shl rdx, 32\n"
"or rax, rdx\n"
"mov %0, rax\n"
"mov rbx, %4\n"
"add rbx, [%3]\n"
"mov [%2], rbx\n"
//if you're worried about a single instruction being a race condition, it's time for an upgrade
"xor rax, rax\n"
"hlt\n"
"rdtsc\n"
"shl rdx, 32\n"
"or rax, rdx\n"
"mov %1, rax\n"
:"=&r"(start_tsc), "=&r"(end_tsc)
:"m"(hpet_info.hpet_reg[33]), "m"(hpet_info.hpet_reg[30]),"r"(hpet_ticks)
:"rax","rbx","rdx");
tsc_freq = (end_tsc - start_tsc) / (double)100000;
printf("TSC: Detected frequency is %f MHz\n", tsc_freq);
__get_cpuid(0x1, &unused, &unused, &cpuid_reg, &unused);
if((cpuid_reg >> 24) & 1) {
clock_mode = TSC_TIME_TSC_EVENT;
lapic_timer_lvt.vector = early_event_vector;
lapic_timer_lvt.timer_mode = LAPIC_TIMER_MODE_TSC_DEADLINE;
lapic_timer_lvt.delivery_status = 0;
lapic_timer_lvt.mask = 0;
}
else {
clock_mode = TSC_TIME_LAPIC_EVENT;
calibrate_lapic();
lapic_timer_lvt.vector = early_event_vector;
lapic_timer_lvt.timer_mode = LAPIC_TIMER_MODE_ONESHOT;
lapic_timer_lvt.delivery_status = 0;
lapic_timer_lvt.mask = 0;
}
lapic[LAPIC_TIMER_LVT] = *(uint32_t *)&lapic_timer_lvt;
}
else {
uint32_t apic_div;
clock_mode = LAPIC_TIME_HPET_EVENT;
calibrate_lapic();
//because lapic is used as scheduler, we want to cause as little interrupts as possible
//while retaining 1 mhz accuracy
apic_div = __builtin_clz((uint32_t)apic_freq);
if(!apic_div) {
apic_div = 0b1011;
}
else if(apic_div >= 7) {
apic_div = 0b1010;
apic_freq /= 128;
}
else {
apic_freq /= (1 << --apic_div);
apic_div = (((apic_div & 0b100) << 1) | (apic_div * 0b1011));
}
lapic_timer_lvt.vector = SPURRIOUS_VECTOR; //TODO CHANGE ME
lapic_timer_lvt.timer_mode = LAPIC_TIMER_MODE_PERIODIC;
lapic_timer_lvt.delivery_status = 0;
lapic_timer_lvt.mask = 1;
lapic[LAPIC_TIMER_DIVIDE] = apic_div;
lapic[LAPIC_TIMER_LVT] = *(uint32_t *)&lapic_timer_lvt;
lapic[LAPIC_TIMER_INIT_COUNT] = UINT32_MAX;
}
switch(clock_mode) {
case TSC_TIME_TSC_EVENT:
printf("Clock source: tsc\nEvent source: tsc-deadline\n");
break;
case TSC_TIME_LAPIC_EVENT:
printf("Clock source: tsc\nEvent source: lapic\n");
break;
case LAPIC_TIME_HPET_EVENT:
printf("Clock source: lapic\nEvent source: hpet\n");
break;
}
}

View File

@ -5,5 +5,6 @@
void dump_video() {
struct mode_info *video = (struct mode_info *)0x600;
printf("Video info:\nx:\t%u\ny:\t%u\nbbp:\t%u\nloc:\t0x%p\n", video->width, video->height, video->bpp, video->framebuffer);
printf("Video info:\nx:\t%u\ny:\t%u\nbbp:\t%u\nloc:\t0x%p\n",
video->width, video->height, video->bpp, video->framebuffer);
}

View File

@ -1,38 +1,34 @@
SEARCH_DIR(objects)
INPUT(
bootloader.o
libc.o
serial.o
video.o
printf.o
page.o
acpi.o
kernel.o
panic.o
)
_kernel_stack_loc = 0x200000 - 8;
_kernel_loc = 0x100000;
_meminfo_loc = 0x7000;
_kernel_s1_loc = 0xffffffff80000000;
_meminfo = 0x7000;
_vbe_infoblock = 0x500;
_stage2_pagetable = 0x200000;
_stage1_pagetable = 0x4000;
SECTIONS
{
. = 0x0;
bootloader 0x7c00 :
bootloader 0x7c00 : AT(0)
{
bootloader.o(.text)
bootloader.o(.text)
}
kernel _kernel_loc :
AT (ADDR(bootloader) + SIZEOF(bootloader))
/*ALIGN(4096)*/
smp_bootloader 0x8000 :
AT(SIZEOF(bootloader))
{
smp_trampoline.o(.text)
}
kernel _kernel_s1_loc :
AT (SIZEOF(bootloader))
{
EXCLUDE_FILE (*bootloader.o) *(.text .data .bss .rodata .comment .eh_frame)
}
}
}
_kernel_size = ((SIZEOF(kernel) / 512) + (SIZEOF(bootloader) / 512)); /* there's a bug here I think!*/
_bootloader_stage1_size = (SIZEOF(bootloader) / 512);
_kernel_sector_size = ((SIZEOF(kernel) + SIZEOF(smp_bootloader) + 511) / 512);
_kernel_page_size = ((SIZEOF(kernel) + SIZEOF(smp_bootloader) + 0xfff) / 0x1000);
_bootloader_stage1_size = (SIZEOF(bootloader) + 511) / 512;
__load_start_smp_bootloader = LOADADDR(smp_bootloader);
__load_stop_smp_bootloader = LOADADDR(smp_bootloader) + SIZEOF(smp_bootloader);
_kernel_stack_loc = (0x100000 + (_kernel_page_size * 0x1000) + 0x2000) - 8;
_kernel_shared_zone_begin = 0x100000 + (_kernel_page_size * 0x1000) + 0x2000;

View File

@ -4,60 +4,98 @@ OBJCPY=../compiler/indigo_gcc/bin/x86_64-elf-objcopy
INC=-I./include
EMU_CORES=4
EMU_RAM=4G
EMU_RAM=5G
XRES=1024
YRES=768
CCFLAGS=-g -ffreestanding -Wall
CFLAGS=-g -ffreestanding -mcmodel=kernel -masm=intel -Wall -O0
make:
nasm -g -felf64 bootloader/bootloader.asm -o objects/bootloader.o
$(CC) $(INC) $(CCFLAGS) -c kernel/kernel.c -o objects/kernel.o
$(CC) $(INC) $(CCFLAGS) -c kernel/panic.c -o objects/panic.o
$(CC) $(INC) $(CCFLAGS) -c kernel/acpi.c -o objects/acpi.o
$(CC) $(INC) $(CCFLAGS) -c kernel/drivers/serial.c -o objects/serial.o
$(CC) $(INC) $(CCFLAGS) -c kernel/drivers/video.c -o objects/video.o
$(CC) $(INC) $(CCFLAGS) -c kernel/printf.c -o objects/printf.o
$(CC) $(INC) $(CCFLAGS) -c kernel/page.c -o objects/page.o
$(CC) $(INC) $(CCFLAGS) -c kernel/libc.c -o objects/libc.o
$(LD) -o indigo_os.elf --oformat=elf64-x86-64 -T link.ld
C_SRCFILES := $(wildcard kernel/*.c)
C_SRCFILES := $(filter-out kernel/isv.c, $(C_SRCFILES))
C_OBJFILES := $(patsubst kernel/%.c,%.o,$(C_SRCFILES))
ASM_SRCFILES := $(wildcard kernel/*.asm)
ASM_SRCFILES := $(filter-out kernel/smp_trampoline.asm, $(ASM_SRCFILES))
ASM_OBJFILES := $(patsubst kernel/%.asm,%.o,$(ASM_SRCFILES))
MISC_OBJFILES := isv.o #this will probably expand over time for files that require special treatment
OBJFILES_PATH := $(patsubst %,objects/%,$(C_OBJFILES) $(ASM_OBJFILES) $(MISC_OBJFILES))
QEMU_OPTS=-smp $(EMU_CORES) -m $(EMU_RAM) -no-reboot -drive format=raw,file=./indigo_os
QEMU_CPU=-cpu host,+invtsc,+tsc-deadline
ifndef disable-kvm
QEMU_OPTS := $(QEMU_OPTS) -enable-kvm $(QEMU_CPU)
endif
QEMU_PRINTDEBUG=-d int,guest_errors
all: $(ASM_OBJFILES) $(C_OBJFILES) $(MISC_OBJFILES) bootloader.o smp_trampoline.o
$(LD) -o indigo_os.elf --no-check-sections --oformat=elf64-x86-64 -T link.ld $(OBJFILES_PATH)
$(OBJCPY) --only-keep-debug indigo_os.elf debug/debug_syms.o
$(OBJCPY) -O binary --strip-all indigo_os.elf indigo_os
ifneq ("$(wildcard $(./debug/serial.in))","")
mkfifo debug/serial.in
endif
ifneq ("$(wildcard $(./debug/serial.out))","")
mkfifo debug/serial.out
endif
# rm -f indigo_os.elf
compiledb make --dry-run > /dev/null
preproc_debug:
nasm -g -felf64 bootloader/bootloader.asm
$(CC) $(INC) $(CCFLAGS) -E -c kernel/kernel.c
$(CC) $(INC) $(CCFLAGS) -E -c kernel/acpi.c
$(CC) $(INC) $(CCFLAGS) -E -c kernel/drivers/serial.c
$(CC) $(INC) $(CCFLAGS) -E -c kernel/drivers/video.c
$(CC) $(INC) $(CCFLAGS) -E -c kernel/printf.c
$(CC) $(INC) $(CCFLAGS) -E -c kernel/page.c
$(CC) $(INC) $(CCFLAGS) -E -c kernel/libc.c
isv.o: kernel/isv.c
$(CC) $(CFLAGS) $(INC) -mgeneral-regs-only -MMD -MP -c kernel/isv.c -o objects/$@
printf.o: kernel/printf.c
ifdef screen
$(CC) $(CFLAGS) $(INC) -MMD -MP -DSCREEN_OUTPUT -c kernel/printf.c -o objects/$@
else
$(CC) $(CFLAGS) $(INC) -MMD -MP -c kernel/printf.c -o objects/$@
endif
kernel.o: kernel/kernel.c
ifdef screen
$(CC) $(CFLAGS) $(INC) -MMD -MP -DSCREEN_OUTPUT -c kernel/kernel.c -o objects/$@
else
$(CC) $(CFLAGS) $(INC) -MMD -MP -c kernel/kernel.c -o objects/$@
endif
%.o: kernel/%.c makefile
$(CC) $(CFLAGS) $(INC) -MMD -MP -c $< -o objects/$@
%.o: kernel/%.asm makefile
nasm -g -felf64 $< -o objects/$@
bootloader.o:
nasm -g -felf64 bootloader/bootloader.asm -o objects/bootloader.o
smp_trampoline.o:
nasm -g -felf64 kernel/smp_trampoline.asm -o objects/smp_trampoline.o
run:
qemu-system-x86_64 -smp $(EMU_CORES) -m $(EMU_RAM) -nographic -no-reboot -drive format=raw,file=./indigo_os
qemu-system-x86_64 $(QEMU_OPTS) $(QEMU_PRINTDEBUG) -nographic
gdb: indigo_os
tmux new-session -s os_gdb "qemu-system-x86_64 -smp $(EMU_CORES) -nographic -no-reboot -S -s -drive format=raw,file=./indigo_os -m $(EMU_RAM)"\;\
tmux new-session -s os_gdb "qemu-system-x86_64 -S -s $(QEMU_OPTS) -nographic"\;\
split-window -h "gdb -x debug/gdbinit.gdb; killall qemu-system-x86_64"
run-graphical:
qemu-system-x86_64 -S -s -smp $(EMU_CORES) -m $(EMU_RAM) -no-reboot -serial pipe:debug/serial -device VGA,edid=on,xres=$(XRES),yres=$(YRES) -drive format=raw,file=./indigo_os &\
qemu-system-x86_64 -device VGA,edid=on,xres=$(XRES),yres=$(YRES) -S -s $(QEMU_OPTS) &\
gdb -x debug/gdbinit.gdb; killall qemu-system-x86_64
install-sandisk: indigo_os
test -h /dev/disk/by-id/usb-SanDisk_Cruzer_Fit_4C531001351010112404-0:0 && \
sudo dd if=indigo_os of=/dev/disk/by-id/usb-SanDisk_Cruzer_Fit_4C531001351010112404-0:0 && \
sync
install-trav: indigo_os
test -h /dev/disk/by-id/usb-Kingston_DataTraveler_3.0_E0D55EA573F1E3A11963008B-0:0 && \
sudo dd if=indigo_os of=/dev/disk/by-id/usb-Kingston_DataTraveler_3.0_E0D55EA573F1E3A11963008B-0:0 && \
sync
clean:
rm -f objects/*
rm -f indigo_os
rm -f indigo_os.elf
rm -f bin/*
rm -f debug/debug_syms.o
rm -f debug/serial.in
rm -f debug/serial.out