Debug GDT enable multitasking

This commit is contained in:
Loic Guegan 2021-05-02 14:46:18 +02:00
parent 4f08ba2b1d
commit e59104ffb5
14 changed files with 283 additions and 85 deletions

View file

@ -1,5 +1,5 @@
EXEC := boucane EXEC := boucane
CC := g++ -Wno-write-strings -Wno-int-to-pointer-cast -mcmodel=large -nostdlib -nostdinc -no-pie -fno-builtin -fno-stack-protector -I ./ -I include CC := g++ -g -Wno-write-strings -Wno-int-to-pointer-cast -mcmodel=large -nostdlib -nostdinc -no-pie -fno-builtin -fno-stack-protector -I ./ -I include
LD_SCRIPT := linker.ld LD_SCRIPT := linker.ld
# Note that BOOT_OBJ do not match boot.S # Note that BOOT_OBJ do not match boot.S

View file

@ -28,15 +28,18 @@ extern u64 gdt64_tss;
void configure_tss(){ void configure_tss(){
// Get TSS physical address // Get TSS physical address
u64 tss_addr=(u64)PHY(&kvar_tss); u64 tss_addr=(u64)&kvar_tss;
u32 limit=sizeof(TSS); u32 limit=sizeof(TSS);
u32 desc1=(tss_addr&0xFFFF)<<16|(limit&0xFFFF); u32 desc1=(tss_addr&0xFFFF)<<16|(limit&0xFFFF);
tss_addr>>=16;
u32 desc2=(tss_addr&0xFF); u32 desc2=(tss_addr&0xFF);
desc2|=0b1001<<8; // Type desc2|=0b11101001<<8; // Type, Permission, present
desc2|=0b111<<13; // Permission & present tss_addr>>=8;
u32 desc3=((limit>>8)&0xFFFF)|(limit>>24); desc2|=tss_addr<<24;
tss_addr>>=8;
u32 desc3=tss_addr;
// Configure GDT // Configure GDT
u32 *gdt_entry=(u32*)&gdt64_tss; u32 *gdt_entry=(u32*)&gdt64_tss;
gdt_entry[0]=desc1; gdt_entry[0]=desc1;
@ -46,6 +49,7 @@ void configure_tss(){
// Configure segment // Configure segment
kvar_tss.rsp0=(u64)VIRT(kvar_stack_pma); kvar_tss.rsp0=(u64)VIRT(kvar_stack_pma);
kvar_tss.iomap_address=0;
asm( asm(
"mov $0x28, %ax \n\t" "mov $0x28, %ax \n\t"
"ltr %ax" "ltr %ax"
@ -53,9 +57,15 @@ void configure_tss(){
} }
void task1(){ void task1(){
u64 a=0xEEEEEE; while(1){
DUMP(a); asm("mov $1, %rdi;int $0x30");
while(1); }
}
void task2(){
while(1){
asm("mov $2, %rdi;int $0x30");
}
} }
extern "C" void boucane(u64 mb_info){ extern "C" void boucane(u64 mb_info){
@ -77,7 +87,7 @@ extern "C" void boucane(u64 mb_info){
memtext_init(); memtext_init();
idt_enable_interrupt(); idt_enable_interrupt();
apic_enable(); apic_enable();
// Looking for framebuffer // Looking for framebuffer
FRAMEBUFFER fb_info; FRAMEBUFFER fb_info;
if(mb2_find_framebuffer((u32*)mb_info, &fb_info)){ if(mb2_find_framebuffer((u32*)mb_info, &fb_info)){
@ -97,23 +107,25 @@ extern "C" void boucane(u64 mb_info){
__putchar=vgatext_putchar; __putchar=vgatext_putchar;
} }
} }
// Booting! // Booting!
printk("Booting Boucane v%d.%d.%d\n",VERSION_MAJOR,VERSION_MINOR, VERSION_PATH); printk("Booting Boucane v%d.%d.%d\n",VERSION_MAJOR,VERSION_MINOR, VERSION_PATH);
printk("System informations -- ");
char bootloader[20]; char bootloader[20];
if(mb2_find_bootloader_name((u32*)mb_info,bootloader)){ if(mb2_find_bootloader_name((u32*)mb_info,bootloader)){
printk("System informations -- BOOT:%s ", bootloader);
printk("BOOT:%s ", bootloader);
} }
MEM_INFO mem_infos; MEM_INFO mem_infos;
if(mb2_find_mem((u32*)mb_info,&mem_infos)){ if(mb2_find_mem((u32*)mb_info,&mem_infos)){
u64 mem=mem_infos.mem_upper-mem_infos.mem_lower; u64 mem=mem_infos.mem_upper-mem_infos.mem_lower;
mem/=1024; mem/=1024;
printk("RAM:%dMB\n", mem); printk("RAM:%dMB\n", mem);
} }
show_ticks=1;
//create_task((void*)task1, 50); create_task((void*)task1, 50);
//scheduler_start(); create_task((void*)task2, 50);
scheduler_start();
while(1); while(1);
} }

View file

@ -17,23 +17,23 @@
u8 lapic_space[4096] __attribute__((aligned(4096))); u8 lapic_space[4096] __attribute__((aligned(4096)));
u8 ioapic_space[4096] __attribute__((aligned(4096))); u8 ioapic_space[4096] __attribute__((aligned(4096)));
char enable=0;
void apic_enable(){ void apic_enable(){
// Memory Allocation // Memory Allocation
PAGE_MAP(lapic_space,APIC_LAPIC_ADDR, PAGING_OPT_DEFAULTS); PAGE_MAP(lapic_space,APIC_LAPIC_ADDR, PAGING_OPT_PCD|PAGING_OPT_DEFAULTS);
PAGE_MAP(lapic_space,APIC_LAPIC_ADDR,PAGING_OPT_DEFAULTS); PAGE_MAP(ioapic_space,APIC_IOAPIC_ADDR,PAGING_OPT_PCD|PAGING_OPT_DEFAULTS);
// Configure APIC register location and enable it via MSR // Configure APIC register location and enable it via MSR
u64 lapic_addr=(u64)APIC_LAPIC_ADDR; /* u64 lapic_addr=(u64)APIC_LAPIC_ADDR;
u32 high=lapic_addr>>32; u32 high=lapic_addr>>32;
u32 low=((u64)APIC_LAPIC_ADDR&0xFFFFFFFF); u32 low=((u64)APIC_LAPIC_ADDR&0xFFFFFFFF);
low|=0x800; // Enable apic low|=0x800; // Enable apic
WRITE_MSR(0x1B,high,low); WRITE_MSR(0x1B,high,low);*/
// Configure LAPIC device using mmap // Configure LAPIC device using mmap
apic_write(APIC_LAPIC_REG_SPURIOUS, 0x100&apic_read(APIC_LAPIC_REG_SPURIOUS)); apic_write(APIC_LAPIC_REG_SPURIOUS, 0x100&apic_read(APIC_LAPIC_REG_SPURIOUS));
apic_write(APIC_DFR, 0xFFFFFFFF); //apic_write(APIC_DFR, 0xFFFFFFFF);
apic_write(APIC_PRIOR, 0); // apic_write(APIC_PRIOR, 0);
apic_write(APIC_LAPIC_TIMER_DVD, 1); apic_write(APIC_LAPIC_TIMER_DVD, 1);
apic_write(APIC_LAPIC_TIMER_LVT, (1<<17)|61); apic_write(APIC_LAPIC_TIMER_LVT, (1<<17)|61);
apic_write(APIC_LAPIC_TIMER_IC, 100000); apic_write(APIC_LAPIC_TIMER_IC, 100000);
@ -43,6 +43,7 @@ void apic_enable(){
*ioapic_reg=0x12; // Select the 0x12 IRQ *ioapic_reg=0x12; // Select the 0x12 IRQ
ioapic_reg=(u32*)(((u64)ioapic_space)+0x10); // Now use the IOREGWIN to write ioapic_reg=(u32*)(((u64)ioapic_space)+0x10); // Now use the IOREGWIN to write
*ioapic_reg=(0x0<<12)|60; // Enable IRQ 1 (0x12) and assign it to the vector 0x3C (index 60 in the IDT) *ioapic_reg=(0x0<<12)|60; // Enable IRQ 1 (0x12) and assign it to the vector 0x3C (index 60 in the IDT)
enable=1;
} }
void apic_write(u32 reg, u32 value){ void apic_write(u32 reg, u32 value){
@ -55,6 +56,15 @@ u32 apic_read(u32 reg){
return *lapic_reg; return *lapic_reg;
} }
extern "C" void ack(){ extern "C" void apic_ack(){
apic_write(APIC_EOI, 0); if(enable){
/* u8 data;
do {
inb(0x64,data);
}
while((data&0x01) == 0);
inb(0x60,data); */
apic_write(APIC_EOI, 0);
}
} }

View file

@ -4,6 +4,7 @@
#include "boucane.hpp" #include "boucane.hpp"
extern "C" void apic_ack();
void apic_enable(); void apic_enable();
void apic_write(u32 reg, u32 value); void apic_write(u32 reg, u32 value);
u32 apic_read(u32 reg); u32 apic_read(u32 reg);

View file

@ -3,31 +3,35 @@
gdt64: gdt64:
gdt64_null: gdt64_null:
.long 0 .word 0xFFF
.long 0 .word 0x0
.byte 0
.byte 0
.byte 1
.byte 0
gdt64_cs: gdt64_cs:
.long 0 .long 0
.byte 0 .byte 0
.byte 0b10011100 .byte 0b10011010 # Present and non-conforming Readable
.byte 0b00100000 .byte 0b00100000 # Long mode
.byte 0 .byte 0
gdt64_ds: gdt64_ds:
.long 0 .long 0
.byte 0 .byte 0
.byte 0b10010010 .byte 0b10010010 # Present, writable
.word 0 .word 0
gdt64_cs_user: gdt64_cs_user:
.long 0 .long 0
.byte 0 .byte 0
.byte 0b11111100 .byte 0b11111010 # Present, Privilege 3 and non-conforming Readable
.byte 0b00100000 .byte 0b00100000 # Long mode
.byte 0 .byte 0
gdt64_ds_user: gdt64_ds_user:
.long 0 .long 0
.byte 0 .byte 0
.byte 0b11110010 .byte 0b11110010 # Present, Privilege 3 and writable
.word 0 .word 0
gdt64_tss: gdt64_tss: # Will be setup in the boucane() function
.long 0 .long 0
.long 0 .long 0
.long 0 .long 0

View file

@ -6,7 +6,7 @@
u32 idt[IDT_MAX_ENTRIES][4] __attribute__((aligned(4096)));; u32 idt[IDT_MAX_ENTRIES][4] __attribute__((aligned(4096)));;
IDT_REGISTER IDTR; IDT_REGISTER IDTR;
extern u64 INT_DEFAULT,INT_0,INT_14,INT_KBD,INT_CLK; extern u64 INT_DEFAULT,INT_0,INT_10,INT_14,INT_KBD,INT_CLK,INT_SYSCALL;
void idt_enable_interrupt(void){ void idt_enable_interrupt(void){
IDTR.base=((u64)idt); IDTR.base=((u64)idt);
@ -28,6 +28,10 @@ void idt_enable_interrupt(void){
d.offset=(u64)&INT_14; d.offset=(u64)&INT_14;
idt_write_descriptor(d, i); idt_write_descriptor(d, i);
} }
else if(i==10){
d.offset=(u64)&INT_10;
idt_write_descriptor(d, i);
}
else if(i==60){ // Keyboard else if(i==60){ // Keyboard
d.offset=(u64)&INT_KBD; d.offset=(u64)&INT_KBD;
idt_write_descriptor(d, i); idt_write_descriptor(d, i);
@ -36,12 +40,20 @@ void idt_enable_interrupt(void){
d.offset=(u64)&INT_CLK; d.offset=(u64)&INT_CLK;
idt_write_descriptor(d, i); idt_write_descriptor(d, i);
} }
else if(i==0x30){ // Syscall
IDT_DESCRIPTOR d2;
d2.ign=0;
d2.ist=0;
d2.selector=0x08;
d2.options=IDT_OPT_P|IDT_OPT_PRVL_3|IDT_OPT_TYPE_INT;
d2.offset=(u64)&INT_SYSCALL;
idt_write_descriptor(d2, i);
}
else { else {
d.offset=(u64)&INT_DEFAULT; d.offset=(u64)&INT_DEFAULT;
idt_write_descriptor(d, i); idt_write_descriptor(d, i);
} }
} }
// Enable interrupts // Enable interrupts
asm( asm(
"lidt (IDTR) \n\t" "lidt (IDTR) \n\t"

View file

@ -9,6 +9,7 @@
#define IDT_OPT_P (1 << 15) #define IDT_OPT_P (1 << 15)
#define IDT_OPT_TYPE_INT 0xE << 8 #define IDT_OPT_TYPE_INT 0xE << 8
#define IDT_OPT_TYPE_TRAP 0xF << 8
#define IDT_OPT_PRVL_0 0 #define IDT_OPT_PRVL_0 0
#define IDT_OPT_PRVL_1 (1 << 13) #define IDT_OPT_PRVL_1 (1 << 13)
#define IDT_OPT_PRVL_2 (2 << 13) #define IDT_OPT_PRVL_2 (2 << 13)

View file

@ -1,50 +1,112 @@
.code64 .code64
.extern printk .extern printk
.extern ack .extern apic_ack
.macro call_printk msg .macro CALL_PRINTK msg
mov \msg, %rdi mov \msg, %rdi
mov $0, %eax # Required for variadic functions mov $0, %eax # Required for variadic functions
mov $printk,%rcx mov $printk,%rcx
call *(%rcx) call *(%rcx)
.endm .endm
.macro SAVE_REGS
push %r8
push %r9
push %r10
push %r11
push %r12
push %r13
push %r14
push %r15
push %rax
push %rbx
push %rcx
push %rdx
push %rbp
push %rsi
push %rdi
push %rax
mov $0x10, %ax
mov %ax, %ds
pop %rax
.endm
.macro RESTORE_REGS
pop %rdi
pop %rsi
pop %rbp
pop %rdx
pop %rcx
pop %rbx
pop %rax
pop %r15
pop %r14
pop %r13
pop %r12
pop %r11
pop %r10
pop %r9
pop %r8
.endm
.globl INT_DEFAULT .globl INT_DEFAULT
INT_DEFAULT: INT_DEFAULT:
SAVE_REGS
RESTORE_REGS
iretq iretq
.globl INT_0 .globl INT_0
INT_0: INT_0:
call_printk $MSG_INT_0 CALL_PRINTK $MSG_INT_0
INT_0_INFINITE: INT_0_INFINITE:
jmp INT_0_INFINITE jmp INT_0_INFINITE
iretq iretq
.globl INT_10
INT_10:
CALL_PRINTK $MSG_INT_10
INT_10_INFINITE:
jmp INT_10_INFINITE
iretq
.globl INT_14 .globl INT_14
INT_14: INT_14:
call_printk $MSG_INT_14 CALL_PRINTK $MSG_INT_14
mov $0, %eax
call printk
INT_14_INFINITE: INT_14_INFINITE:
jmp INT_14_INFINITE jmp INT_14_INFINITE
iretq iretq
.globl INT_KBD .globl INT_KBD
INT_KBD: INT_KBD:
#call_printk $MSG_INT_KBD SAVE_REGS
call ack CALL_PRINTK $MSG_INT_KBD
call apic_ack
RESTORE_REGS
iretq iretq
.globl INT_CLK .globl INT_CLK
.extern clock .extern clock
INT_CLK: INT_CLK:
SAVE_REGS
call clock call clock
call ack call apic_ack
RESTORE_REGS
iretq iretq
.globl INT_SYSCALL
.extern syscall
INT_SYSCALL:
SAVE_REGS
call syscall
RESTORE_REGS
iretq
MSG_INT_0: MSG_INT_0:
.asciz "Zero Division error!" .asciz "Zero Division error!"
MSG_INT_10:
.asciz "Invalid TSS!"
MSG_INT_14: MSG_INT_14:
.asciz "Page fault!" .asciz "Page fault!"
MSG_INT_KBD: MSG_INT_KBD:

View file

@ -52,7 +52,7 @@ void paging_enable() {
// 4096 bytes stack // 4096 bytes stack
PAGE_MAP(kvar_kernel_vma-4096, kvar_stack_pma,PAGING_OPT_DEFAULTS); PAGE_MAP(kvar_kernel_vma-4096, kvar_stack_pma,PAGING_OPT_DEFAULTS);
// Load new pml4 // Load new pml4
kpml4=(u64*)((u64)kpages[0]-kvar_kernel_vma); kpml4=(u64*)((u64)kpages[0]-kvar_kernel_vma);
lpml4(kpml4); lpml4(kpml4);
@ -190,9 +190,15 @@ void paging_allocate_addr(u64* pml4_table, u64 virt, u64 phy, u16 options, char
u64* paging_create_task(int npages){ u64* paging_create_task(int npages){
u64 *pml4=paging_allocate_utable(); u64 *pml4=paging_allocate_utable();
for(int i=0;i<npages;i++){ int i;
paging_allocate_addr(pml4, i*4096, (u64)PAGE_ALLOCATE(), PAGING_OPT_P|PAGING_OPT_RW|PAGING_OPT_US, 0); for(i=0;i<npages;i++){
paging_allocate_addr(pml4, i*4096, (u64)PAGE_ALLOCATE(), PAGING_OPT_DEFAULTS|PAGING_OPT_US, 0);
} }
// Allocate a page for the user stack
paging_allocate_addr(pml4, i*4096, (u64)PAGE_ALLOCATE(), PAGING_OPT_DEFAULTS|PAGING_OPT_US, 0);
// Allocate a page for the kernel stack
paging_allocate_addr(pml4, (i+1)*4096, (u64)PAGE_ALLOCATE(), PAGING_OPT_DEFAULTS, 0);
// Enable kernel access // Enable kernel access
u16 pml4_entry=kvar_kernel_vma>>39&0x1FF; u16 pml4_entry=kvar_kernel_vma>>39&0x1FF;

View file

@ -1,40 +1,107 @@
#include "scheduler.hpp" #include "scheduler.hpp"
#include "boucane.hpp"
#include "core/apic.hpp"
TASK tasks[MAX_TASK]; PROC procs[MAX_TASK];
u32 ntasks=0; u32 nproc=0;
char show_ticks=0; char show_ticks=0;
char scheduling=0;
u32 active_process=0;
extern "C" void clock(){ extern "C" void clock(){
if(show_ticks) if(show_ticks)
print("."); print(".");
if(scheduling)
schedule();
} }
void schedule(){ void schedule(){
// First get a pointer to the first process saved register.
// Since this is called by clock(), %rbp contains a pointer
// to the clock() %rbp value and then we access to the registers SAVE_REGS in int.S
u64* stack;
asm("mov %%rbp, %%rax;mov (%%rax), %%rbx; add $16, %%rbx; mov %%rbx,%0": "=m"(stack)::"rax","rbx");
// Save current task
PROC *t=&procs[active_process];
t->registers.r8=stack[0];
t->registers.r9=stack[1];
t->registers.r10=stack[2];
t->registers.r11=stack[3];
t->registers.r12=stack[4];
t->registers.r13=stack[5];
t->registers.r14=stack[6];
t->registers.r15=stack[7];
t->registers.rdi=stack[8];
t->registers.rsi=stack[9];
t->registers.rbp=stack[10];
t->registers.rdx=stack[11];
t->registers.rcx=stack[12];
t->registers.rbx=stack[13];
t->registers.rax=stack[14];
t->registers.rip=stack[15];
t->registers.cs=stack[16];
t->registers.eflags=stack[17];
t->registers.rsp=stack[18];
t->registers.ds=stack[19];
// Goto next task
active_process++;
if(active_process>=nproc)
active_process=0;
t=&procs[active_process];
kvar_tss.rsp0=t->registers.rsp0;
// Clock acknownledgement
apic_ack();
asm volatile(
"mov %0, %%rdi \n\t"
"jmp switch \n\t"
:: "a" (t)
);
} }
void create_task(void* task, u32 size){ void create_task(void* task, u32 size){
if(ntasks>=MAX_TASK){ if(nproc>=MAX_TASK){
printk("Could not create more tasks."); printk("Could not create more tasks.");
return; return;
} }
TASK *t=&tasks[ntasks]; PROC *t=&procs[nproc];
t->id=ntasks; t->id=nproc;
t->pid=ntasks; t->pid=nproc;
t->size=size; t->size=size;
t->pml4=paging_create_task(size/4096+1);
u32 npages=size%4096 ? size/4096 + 1 : size/4096;
// Note that paging_create_task() allocate 2 more pages (one for the user stack and
// the other for the kernel stack)
t->pml4=paging_create_task(npages);
t->registers.rsp=TASK_VMA+npages*4096+4096; // User stack
t->registers.rsp0=TASK_VMA+npages*4096+4096*2; // Kernel stack on the last page
t->registers.rip=TASK_VMA;
t->registers.cs=0x1B; // 0x18 and 0x3 privilege
t->registers.ds=0x23; // 0x20 and 0x3 privilege
// Load task using // Load task using
lpml4(t->pml4); lpml4(t->pml4);
memcpy(task, TASK_VMA, size); memcpy(task, TASK_VMA, size);
lpml4(kpml4); lpml4(kpml4);
ntasks++; nproc++;
} }
void scheduler_start(){ void scheduler_start(){
TASK *t=&tasks[0]; scheduling=1;
lpml4(t->pml4); active_process=0;
asm("jmp switch"); PROC *t=&procs[active_process];
kvar_tss.rsp0=t->registers.rsp0;
asm(
"cli \n\t"
"mov %0, %%rdi \n\t"
"jmp switch \n\t"
:: "r" (t)
);
} }

View file

@ -4,24 +4,32 @@
#define MAX_TASK 5 #define MAX_TASK 5
#define TASK_VMA 0x0 #define TASK_VMA 0x0
#if (TASK_VMA % 0x7)
#error TASK_VMA value is not 4096 bits aligned!
#endif
// DO NOT CHANGE THE FOLLOWING STRUCTURE WITHOUT CONCIDERING UPDATING
// THE SWITCH FUNCTION INTO scheduler_asm.S
typedef struct { typedef struct {
u32 rax, rbx, rcx, rdx; u64 rax, rbx, rcx, rdx;
u32 cs, rip; u64 cs, rip;
u32 ss, rsp, rbp; u64 ss, rsp, rbp;
u32 rsi, rdi; u64 rsi, rdi;
u32 ds, es, fs, gs; u64 ds, es, fs, gs;
u32 eflags; u64 eflags;
u32 ss0, rsp0; u64 rsp0;
u64 r8,r9,r10,r11,r12,r13,r14,r15;
} __attribute__((packed)) REGS; } __attribute__((packed)) REGS;
// DO NOT CHANGE THE FOLLOWING STRUCTURE WITHOUT CONCIDERING UPDATING
// THE SWITCH FUNCTION INTO scheduler_asm.S
typedef struct { typedef struct {
u64* pml4;
REGS registers;
u32 id; u32 id;
u32 pid; u32 pid;
u64* pml4;
u32 size; u32 size;
REGS registers; } __attribute__((packed)) PROC;
} __attribute__((packed)) TASK;
extern char show_ticks; extern char show_ticks;

View file

@ -2,26 +2,33 @@
.extern kvar_stack_pma
switch: switch:
# TODO: Check if we come from kernel mode (use kernel stack)
mov $0x23, %ax # TODO: restore all registers
mov 96(%rdi), %ax
mov %ax, %ds mov %ax, %ds
mov %ax, %es mov %ax, %es
mov %ax, %fs mov %ax, %fs
mov %ax, %gs mov %ax, %gs
push $0x23 # mov (%rdi), %rax
push $0x80 mov %rax, %cr3
push 96(%rdi)
push 64(%rdi)
pushf pushf
pop %rax pop %rax
#orl $0x200, %%eax or $0x200, %rax # Enable interrupt
mov $0xffffbfff, %rbx mov $0xffffffffbfff, %rbx # NT flag
and %rbx, %rax and %rbx, %rax
push %rax push %rax
push $0x1B push 40(%rdi)
push $0x0 push 48(%rdi)
# Perform task switching
iretq iretq

8
src/core/syscalls.cc Normal file
View file

@ -0,0 +1,8 @@
#include "boucane.hpp"
extern "C" void syscall(){
u64 call_number;
asm volatile("mov %%rdi, %0":"=r"(call_number));
printk("%d",call_number);
for(int i=0;i<10000000;i++){}
}

View file

@ -15,7 +15,7 @@ void framebuffer_init(FB_CFG config){
// Ensure we start writing at the begining of the page since // Ensure we start writing at the begining of the page since
// start is not necessarly 4096 bytes aligned // start is not necessarly 4096 bytes aligned
start=PAGE(start); start=PAGE(start);
PAGE_RMAP(start,fb_cfg.location, PAGING_OPT_DEFAULTS,fb_cfg.pitch*fb_cfg.height); PAGE_RMAP(start,fb_cfg.location, PAGING_OPT_PCD|PAGING_OPT_DEFAULTS,fb_cfg.pitch*fb_cfg.height);
fb_cfg.location=start; fb_cfg.location=start;
} }
@ -25,7 +25,7 @@ void framebuffer_draw(FB_PIXEL p){
p.y=p.y>(fb_cfg.width)?fb_cfg.width:p.y; p.y=p.y>(fb_cfg.width)?fb_cfg.width:p.y;
p.x=p.x<0?0:p.x; p.x=p.x<0?0:p.x;
p.y=p.y<0?0:p.y; p.y=p.y<0?0:p.y;
u8 *pixel=(u8*)(fb_cfg.location+p.x*(fb_cfg.depth/8)+p.y*fb_cfg.pitch); u8 *pixel=(u8*)(fb_cfg.location+p.x*(fb_cfg.depth/8)+p.y*fb_cfg.pitch);
pixel[0]=p.r; pixel[0]=p.r;
pixel[1]=p.g; pixel[1]=p.g;