Improve task loading
This commit is contained in:
parent
e1e75aa387
commit
2fc03e597e
5 changed files with 84 additions and 47 deletions
|
@ -8,7 +8,13 @@
|
||||||
extern void interrupt_enable();
|
extern void interrupt_enable();
|
||||||
|
|
||||||
void utask(){
|
void utask(){
|
||||||
char msg[]="Message from the task :D";
|
char *msg=(char*)4206592+10;
|
||||||
|
msg[0]='T';
|
||||||
|
msg[1]='a';
|
||||||
|
msg[2]='s';
|
||||||
|
msg[3]='k';
|
||||||
|
msg[4]='1';
|
||||||
|
msg[5]='\0';
|
||||||
asm("mov $0x1, %%eax;int $0x30"::"b"(msg));
|
asm("mov $0x1, %%eax;int $0x30"::"b"(msg));
|
||||||
while(1);
|
while(1);
|
||||||
}
|
}
|
||||||
|
@ -22,37 +28,16 @@ void bringelle(){
|
||||||
print("Interrupts enabled\n");
|
print("Interrupts enabled\n");
|
||||||
|
|
||||||
paging_enable();
|
paging_enable();
|
||||||
print("Paging enable!\n");
|
print("Paging enable\n");
|
||||||
print("Kernel started !");
|
print("Kernel started!\n");
|
||||||
|
print("-----------------------\n");
|
||||||
paging_allocate(5);
|
|
||||||
print("\n");
|
|
||||||
paging_dump(1024,-1);
|
|
||||||
print("\n");
|
|
||||||
|
|
||||||
show_tics=1;
|
show_tics=1;
|
||||||
|
|
||||||
// Utask
|
// Utask
|
||||||
print("Launch user task ");
|
print("Launch user task \n");
|
||||||
memcpy((void*)utask,(void*)0x300000, 100); // 100 bytes seems reasonable to load utask
|
int* page_dir=paging_allocate(2);
|
||||||
asm (
|
run_task(page_dir, utask,100);
|
||||||
"cli \n\t" // Ensure we do not get interrupted
|
|
||||||
"movl %%ss, %%eax \n\t"
|
|
||||||
"movl %%eax, %0 \n\t" // Save kernel ss segment into the TSS
|
|
||||||
"movl %%esp, %1 \n\t" // Save kernel esp into the TSS BEFORE setting up the stack
|
|
||||||
"pushl $0x33 \n\t" // Push task ss which is 0x30 along with prlv which is 0x3
|
|
||||||
"pushl $0x30FFFF \n\t" // Push task esp
|
|
||||||
"pushfl \n\t" // Retrieve flags
|
|
||||||
"popl %%eax \n\t"
|
|
||||||
"orl $0x200, %%eax \n\t" // Enable interrupt for the user task
|
|
||||||
"and $0xffffbfff, %%eax \n\t" // Clear the NT flags
|
|
||||||
"push %%eax \n\t" // Push task flags
|
|
||||||
"push $0x23 \n\t" // Push task cs which is 0x20 along with prlv which is 0x3
|
|
||||||
"push $0x300000 \n\t" // Push task entry point
|
|
||||||
"mov $0x2B, %%eax \n\t" // GDT entry 0x28 along with prlv which is 0x3
|
|
||||||
"mov %%eax, %%ds \n\t" // Setting up user data segment
|
|
||||||
"iret \n\t" // Launch user task
|
|
||||||
: "=m" (TSS.ss0), "=m" (TSS.esp0)
|
|
||||||
);
|
|
||||||
|
|
||||||
while(1);
|
while(1);
|
||||||
}
|
}
|
||||||
|
|
|
@ -88,24 +88,30 @@ char* paging_allocate_next_page(){
|
||||||
asm("hlt");
|
asm("hlt");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// TODO: Take p into account
|
||||||
char *paging_allocate(int p){
|
int *paging_allocate(int p){
|
||||||
|
// ----- Allow kernel access during interruption
|
||||||
int *page_dir=(int*)paging_allocate_next_page();
|
int *page_dir=(int*)paging_allocate_next_page();
|
||||||
int *page_table;
|
int *k_page_table=(int*)paging_allocate_next_page();
|
||||||
int current_page_entry=0;
|
// Init page directory
|
||||||
int current_dir_entry=0;
|
page_dir[0]=(int)k_page_table|3;
|
||||||
while(current_page_entry<p){
|
// Init page table 0
|
||||||
if(current_page_entry%1024 == 0){
|
int addr_offset=0;
|
||||||
current_dir_entry++;
|
for(int i=0;i<1024;i++){
|
||||||
page_table=(int*)paging_allocate_next_page();
|
k_page_table[i]=addr_offset;
|
||||||
page_dir[current_dir_entry]=(int)page_table;
|
k_page_table[i]|=3; // Permission
|
||||||
}
|
addr_offset+=PAGING_PAGE_SIZE; // 4Ko pages
|
||||||
int entry=current_page_entry%1024;
|
|
||||||
page_table[current_page_entry]=(int)paging_allocate_next_page();
|
|
||||||
current_page_entry++;
|
|
||||||
print("Jean\n");
|
|
||||||
}
|
}
|
||||||
print("end");
|
|
||||||
|
// ----- Task table
|
||||||
|
int *u_page_table=(int*)paging_allocate_next_page();
|
||||||
|
page_dir[1]=(int)u_page_table|7; // 1024*1024*4096/4
|
||||||
|
u_page_table[0]=(int)page_dir|7;
|
||||||
|
u_page_table[1]=(int)k_page_table|7;
|
||||||
|
u_page_table[2]=(int)u_page_table|7;
|
||||||
|
u_page_table[3]=(int)paging_allocate_next_page()|7;
|
||||||
|
|
||||||
|
return page_dir;
|
||||||
}
|
}
|
||||||
|
|
||||||
void paging_page_fault(){
|
void paging_page_fault(){
|
||||||
|
|
|
@ -9,6 +9,7 @@
|
||||||
#else
|
#else
|
||||||
#define PAGING_MAX_DIR_ENTRY PAGING_MAX_PAGES/1024
|
#define PAGING_MAX_DIR_ENTRY PAGING_MAX_PAGES/1024
|
||||||
#endif
|
#endif
|
||||||
|
#define PADDR(entry) (entry&0xFFFFF000)
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Configure and enable paging
|
* Configure and enable paging
|
||||||
|
@ -26,7 +27,7 @@ void paging_set_usage(int addr,char state);
|
||||||
* Create a new page directory containing
|
* Create a new page directory containing
|
||||||
* p pages.
|
* p pages.
|
||||||
*/
|
*/
|
||||||
char *paging_allocate(int p);
|
int *paging_allocate(int p);
|
||||||
void paging_dump(int min,int max);
|
void paging_dump(int min,int max);
|
||||||
/**
|
/**
|
||||||
* Handler of page fault
|
* Handler of page fault
|
||||||
|
|
|
@ -1,4 +1,7 @@
|
||||||
#include "libc/stdio.h"
|
#include "libc/stdio.h"
|
||||||
|
#include "gdt.h"
|
||||||
|
#include "mem.h"
|
||||||
|
#include "paging.h"
|
||||||
|
|
||||||
char show_tics=0;
|
char show_tics=0;
|
||||||
|
|
||||||
|
@ -17,4 +20,45 @@ void clock(){
|
||||||
putchar('.');
|
putchar('.');
|
||||||
}
|
}
|
||||||
schedule();
|
schedule();
|
||||||
|
}
|
||||||
|
|
||||||
|
void run_task(int *page_dir, void *task, int task_size){
|
||||||
|
// Compute various addresses
|
||||||
|
int*pt_addr=PADDR(page_dir[1]);
|
||||||
|
void *entry_point=(void*)(PADDR(pt_addr[3]));
|
||||||
|
void *ustack=(void*)((int)entry_point+0xFF);
|
||||||
|
|
||||||
|
// Load the task into memory
|
||||||
|
memcpy(task,entry_point, task_size);
|
||||||
|
|
||||||
|
// Load page directory
|
||||||
|
asm(
|
||||||
|
"mov %0, %%eax \n\t"
|
||||||
|
"mov %%eax,%%cr3 \n\t"
|
||||||
|
:: "b"(page_dir)
|
||||||
|
);
|
||||||
|
|
||||||
|
// Setup users adresses
|
||||||
|
|
||||||
|
// Switch to user task
|
||||||
|
asm (
|
||||||
|
"cli \n\t" // Ensure we do not get interrupted
|
||||||
|
"movl %%ss, %%eax \n\t"
|
||||||
|
"movl %%eax, %0 \n\t" // Save kernel ss segment into the TSS
|
||||||
|
"movl %%esp, %1 \n\t" // Save kernel esp into the TSS BEFORE setting up the stack
|
||||||
|
"pushl $0x33 \n\t" // Push task ss which is 0x30 along with prlv which is 0x3
|
||||||
|
"pushl %2 \n\t" // Push task esp
|
||||||
|
"pushfl \n\t" // Retrieve flags
|
||||||
|
"popl %%eax \n\t"
|
||||||
|
"orl $0x200, %%eax \n\t" // Enable interrupt for the user task
|
||||||
|
"and $0xffffbfff, %%eax \n\t" // Clear the NT flags
|
||||||
|
"push %%eax \n\t" // Push task flags
|
||||||
|
"push $0x23 \n\t" // Push task cs which is 0x20 along with prlv which is 0x3
|
||||||
|
"push %3 \n\t" // Push task entry point
|
||||||
|
"mov $0x2B, %%eax \n\t" // GDT entry 0x28 along with prlv which is 0x3
|
||||||
|
"mov %%eax, %%ds \n\t" // Setting up user data segment
|
||||||
|
"iret \n\t" // Launch user task
|
||||||
|
: "=m" (TSS.ss0), "=m" (TSS.esp0)
|
||||||
|
: "b" (ustack), "c" (entry_point)
|
||||||
|
);
|
||||||
}
|
}
|
|
@ -5,5 +5,6 @@ extern char show_tics;
|
||||||
|
|
||||||
void clock();
|
void clock();
|
||||||
void schedule();
|
void schedule();
|
||||||
|
void run_task(int *page_dir, void *task, int task_size);
|
||||||
|
|
||||||
#endif
|
#endif
|
Loading…
Add table
Reference in a new issue