Multitasking works!

This commit is contained in:
Loic Guegan 2021-04-16 14:39:24 +02:00
parent 58b706d40f
commit 98d524bc8a
7 changed files with 121 additions and 72 deletions

View file

@ -9,14 +9,26 @@ extern void interrupt_enable();
void utask(){ void utask(){
char *msg=(char*)4206592+10; char *msg=(char*)4206592+10;
msg[0]='T'; msg[0]='A';
msg[1]='a'; msg[1]='\0';
msg[2]='s'; while(1){
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); for(int i=0;i<10000;i++){
}
}
}
void utask2(){
char *msg=(char*)4206592+10;
msg[0]='B';
msg[1]='\0';
while(1){
asm("mov $0x1, %%eax;int $0x30"::"b"(msg));
for(int i=0;i<10000;i++){
}
}
} }
void bringelle(){ void bringelle(){
@ -34,10 +46,15 @@ void bringelle(){
show_tics=1; show_tics=1;
// Utask // Utask
print("Launch user task \n"); print("Launch user tasks \n");
int* page_dir=paging_allocate(2);
run_task(page_dir, utask,100);
int* page_dir=paging_allocate(2);
task_create(page_dir, utask,100);
int* page_dir2=paging_allocate(2);
task_create(page_dir2, utask2,100);
scheduler_start();
while(1); while(1);
} }

View file

@ -25,7 +25,7 @@ void idt_init(){
if(i==33) if(i==33)
idt_write_entry((IDT_ENTRY){0x08,(u32)&INT_KEYPRESS,IDT_INT_GATE},i); idt_write_entry((IDT_ENTRY){0x08,(u32)&INT_KEYPRESS,IDT_INT_GATE},i);
if(i==48) if(i==48)
idt_write_entry((IDT_ENTRY){0x08,(u32)&INT_SYSCALL,IDT_TRAP_GATE},i); idt_write_entry((IDT_ENTRY){0x08,(u32)&INT_SYSCALL,IDT_INT_GATE|0x6000},i);
} }
// Load IDT // Load IDT
asm("lidtl (IDTR)"); asm("lidtl (IDTR)");

View file

@ -9,7 +9,13 @@
#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) (((u32)entry)&0xFFFFF000) #define PAGING_PADDR(entry) ((int*)(((u32)entry)&0xFFFFF000))
#define PAGING_ENTRY_POINT_VIRT (1024*PAGING_PAGE_SIZE+3*PAGING_PAGE_SIZE)
#define PAGING_ENTRY_POINT_PHY(page_dir) ((int*)(((int*)((((int*)page_dir)[1])&0xFFFFF000))[3]&0xFFFFF000))
/** /**
* Configure and enable paging * Configure and enable paging

View file

@ -9,15 +9,13 @@ PROC procs[MAX_PROC];
u16 current_id; u16 current_id;
u16 nproc; u16 nproc;
void schedule(){ void schedule(u32 *stack){
// Note that this function is called by clock // Note that this function is called by clock
// clock is called by INT_CLOCK (core/int.S) // clock is called by INT_CLOCK (core/int.S)
// which store all the process information on // which store all the process information on
// the stack. Thus, knowing the C calling conventions // the stack. Thus, knowing the C calling conventions
// and that schedule() is call by two functions with no parameters, // and that schedule() is call by two functions with no parameters,
// the first process register can be accessed by ebp+2 // the first process register can be accessed by ebp+2
u32 *stack;
asm("mov %%ebp, %0":"=r" (stack));
// No proc to schedule // No proc to schedule
if(nproc<2) if(nproc<2)
@ -47,6 +45,7 @@ void schedule(){
current_id++; current_id++;
if(current_id>=nproc) if(current_id>=nproc)
current_id=0; current_id=0;
p=&procs[current_id];
// Have a clean stack on next interrupt // Have a clean stack on next interrupt
TSS.esp0=(u32)stack+19; TSS.esp0=(u32)stack+19;
@ -58,13 +57,16 @@ void schedule(){
// Perform task switch // Perform task switch
asm( asm(
"mov %0, %%esi \n\t" "push %0 \n\t"
"jmp task_switch \n\t" "jmp task_switch \n\t"
:: "a" (p) :: "a" (p)
); );
} }
void clock(){ void clock(){
u32* stack;
asm("mov %%ebp, %0":"=r" (stack));
static int tic=0; static int tic=0;
static int sec=0; static int sec=0;
tic++; tic++;
@ -75,46 +77,69 @@ void clock(){
putchar('.'); putchar('.');
} }
if(scheduler_on==1) if(scheduler_on==1)
schedule(); schedule(stack);
} }
void run_task(int *page_dir, void *task, int task_size){ void task_create(int *page_dir, void *task, int task_size){
if(nproc<=MAX_PROC){
// Compute various addresses // Compute various addresses
int*pt_addr=(int*)PADDR(page_dir[1]); void *entry_point=PAGING_ENTRY_POINT_VIRT;
void *entry_point=(void*)(PADDR(pt_addr[3])); void *ustack=(void*)(PAGING_ENTRY_POINT_VIRT+0xFF);
void *ustack=(void*)((int)entry_point+0xFF);
// Load the task into memory // Load the task into memory
memcpy(task,entry_point, task_size); memcpy(task,PAGING_ENTRY_POINT_PHY(page_dir), task_size);
// Load page directory // Setup process registers
asm( PROC *p=&procs[nproc];
"mov %0, %%eax \n\t" p->regs.cs=0x23; // Task cs which is 0x30 along with prlv which is 0x3
"mov %%eax,%%cr3 \n\t" p->regs.eip=entry_point;
:: "b"(page_dir) p->regs.ss=0x33; // Task ss which is 0x20 along with prlv which is 0x3
); p->regs.esp=ustack; // Stack is here for now...
p->regs.ds=0x2B; // GDT entry 0x28 along with prlv which is 0x3
p->regs.eax=0;
p->regs.ebx=0;
p->regs.ecx=0;
p->regs.edx=0;
// Setup users adresses // Manage eflags
u32 eflags;
// Switch to user task
asm ( 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 "pushfl \n\t" // Retrieve flags
"popl %%eax \n\t" "popl %%eax \n\t"
"orl $0x200, %%eax \n\t" // Enable interrupt for the user task "orl $0x200, %%eax \n\t" // Enable interrupt for the user task
"and $0xffffbfff, %%eax \n\t" // Clear the NT flags "and $0xffffbfff, %%eax \n\t" // Clear the NT flags
"push %%eax \n\t" // Push task flags "mov %%eax, %0 \n\t" // Get flags into eflag
"push $0x23 \n\t" // Push task cs which is 0x20 along with prlv which is 0x3 : "=m" (eflags)
"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)
); );
p->regs.eflags=eflags;
// Setup other attributes
p->id=nproc;
p->pid=nproc;
p->page_dir=page_dir;
nproc++;
}
}
void scheduler_start(){
if(nproc>0){
asm("cli");
scheduler_on=1; // Enable scheduling
u32 *stack;
asm("mov %%ebp, %0":"=r" (stack));
TSS.esp0=(u32)stack+1; // Remove ebp (c call convention)
asm("mov %%ss, %0": "=m" (TSS.ss0));
current_id=1;
PROC *p=&procs[current_id];
// Ensure interrupts are activated and NT flag is clear
p->regs.eflags|=0x200;
p->regs.eflags&=0xffffbfff;
asm(
"push %0 \n\t"
"jmp task_switch"
:: "a" (p)
);
}
} }

View file

@ -34,6 +34,6 @@ extern u16 nproc;
void clock(); void clock();
void schedule(); void schedule();
void run_task(int *page_dir, void *task, int task_size); void task_create(int *page_dir, void *task, int task_size);
void scheduler_start();
#endif #endif

View file

@ -4,24 +4,24 @@
// to a PROC structure and jump // to a PROC structure and jump
// to this process // to this process
task_switch: task_switch:
push 12(%esi) # eax (cf PROC struct) pop %esi
push 16(%esi) # ebx push 8(%esi) # eax (cf PROC struct)
push 20(%esi) # ecx push 12(%esi) # ebx
push 24(%esi) # edx push 16(%esi) # ecx
push 44(%esi) # ebp push 20(%esi) # edx
push 48(%esi) # esi push 40(%esi) # ebp
push 52(%esi) # edi push 48(%esi) # edi
push 56(%esi) # ds push 52(%esi) # ds
push 60(%esi) # es push 56(%esi) # es
push 64(%esi) # fs push 60(%esi) # fs
push 68(%esi) # gs push 64(%esi) # gs
// Don't forget to clear the interrupt // Don't forget to clear the interrupt
movb $0x20, %al movb $0x20, %al
outb %al, $0x20 outb %al, $0x20
// Setup process page directory // Setup process page directory
mov 8(%esi), %eax mov 4(%esi), %eax
mov %eax, %cr3 mov %eax, %cr3
// Setup registers // Setup registers
@ -30,7 +30,6 @@ task_switch:
pop %es pop %es
pop %ds pop %ds
pop %edi pop %edi
pop %esi
pop %ebp pop %ebp
pop %edx pop %edx
pop %ecx pop %ecx
@ -38,9 +37,11 @@ task_switch:
pop %eax pop %eax
// Perform the task switch // Perform the task switch
push 36(%esi) # ss push 32(%esi) # ss
push 40(%esi) # esp push 36(%esi) # esp
push 72(%esi) # eflags push 68(%esi) # eflags
push 28(%esi) # cs push 24(%esi) # cs
push 32(%esi) # eip push 28(%esi) # eip
iret mov 52(%esi), %ds # Choose the right data segment
mov 44(%esi), %esi # Now restore task esi
iret # Launch task