Improve scheduler
This commit is contained in:
parent
28870fc375
commit
58b706d40f
3 changed files with 133 additions and 2 deletions
|
@ -1,12 +1,67 @@
|
||||||
|
#include "scheduler.h"
|
||||||
#include "libc/stdio.h"
|
#include "libc/stdio.h"
|
||||||
#include "gdt.h"
|
#include "gdt.h"
|
||||||
#include "mem.h"
|
|
||||||
#include "paging.h"
|
#include "paging.h"
|
||||||
|
|
||||||
char show_tics=0;
|
char show_tics=0;
|
||||||
|
char scheduler_on=0;
|
||||||
|
PROC procs[MAX_PROC];
|
||||||
|
u16 current_id;
|
||||||
|
u16 nproc;
|
||||||
|
|
||||||
void schedule(){
|
void schedule(){
|
||||||
|
// Note that this function is called by clock
|
||||||
|
// clock is called by INT_CLOCK (core/int.S)
|
||||||
|
// which store all the process information on
|
||||||
|
// the stack. Thus, knowing the C calling conventions
|
||||||
|
// and that schedule() is call by two functions with no parameters,
|
||||||
|
// the first process register can be accessed by ebp+2
|
||||||
|
u32 *stack;
|
||||||
|
asm("mov %%ebp, %0":"=r" (stack));
|
||||||
|
|
||||||
|
// No proc to schedule
|
||||||
|
if(nproc<2)
|
||||||
|
return;
|
||||||
|
|
||||||
|
PROC *p=&procs[current_id];
|
||||||
|
p->regs.gs=stack[2]; // ebp+2=gs cf note above
|
||||||
|
p->regs.fs=stack[3];
|
||||||
|
p->regs.es=stack[4];
|
||||||
|
p->regs.ds=stack[5];
|
||||||
|
p->regs.edi=stack[6];
|
||||||
|
p->regs.esi=stack[7];
|
||||||
|
p->regs.ebp=stack[8];
|
||||||
|
// We do not take p->regs.esp=stack[9]
|
||||||
|
// since it corresponds to the kernel stack
|
||||||
|
// (it was push during the interruption)
|
||||||
|
p->regs.edx=stack[10];
|
||||||
|
p->regs.ecx=stack[11];
|
||||||
|
p->regs.ebx=stack[12];
|
||||||
|
p->regs.eax=stack[13];
|
||||||
|
p->regs.eip=stack[14];
|
||||||
|
p->regs.cs=stack[15];
|
||||||
|
p->regs.eflags=stack[16];
|
||||||
|
p->regs.esp=stack[17];
|
||||||
|
p->regs.ss=stack[18];
|
||||||
|
|
||||||
|
current_id++;
|
||||||
|
if(current_id>=nproc)
|
||||||
|
current_id=0;
|
||||||
|
|
||||||
|
// Have a clean stack on next interrupt
|
||||||
|
TSS.esp0=(u32)stack+19;
|
||||||
|
asm("mov %%ss, %0": "=m" (TSS.ss0));
|
||||||
|
|
||||||
|
// Ensure interrupts are activated and NT flag is clear
|
||||||
|
p->regs.eflags|=0x200;
|
||||||
|
p->regs.eflags&=0xffffbfff;
|
||||||
|
|
||||||
|
// Perform task switch
|
||||||
|
asm(
|
||||||
|
"mov %0, %%esi \n\t"
|
||||||
|
"jmp task_switch \n\t"
|
||||||
|
:: "a" (p)
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
void clock(){
|
void clock(){
|
||||||
|
@ -19,6 +74,7 @@ void clock(){
|
||||||
if(show_tics)
|
if(show_tics)
|
||||||
putchar('.');
|
putchar('.');
|
||||||
}
|
}
|
||||||
|
if(scheduler_on==1)
|
||||||
schedule();
|
schedule();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -1,7 +1,36 @@
|
||||||
#ifndef SCHEDULER_H
|
#ifndef SCHEDULER_H
|
||||||
#define SCHEDULER_H
|
#define SCHEDULER_H
|
||||||
|
|
||||||
|
#include "mem.h"
|
||||||
|
|
||||||
|
|
||||||
|
#define MAX_PROC 10
|
||||||
|
|
||||||
|
// If you change the following struct
|
||||||
|
// please consider updating the code in scheduler_asm.S
|
||||||
|
typedef struct REGISTERS {
|
||||||
|
u32 eax, ebx, ecx, edx;
|
||||||
|
u32 cs, eip;
|
||||||
|
u32 ss, esp, ebp;
|
||||||
|
u32 esi, edi;
|
||||||
|
u32 ds, es, fs, gs;
|
||||||
|
u32 eflags;
|
||||||
|
} __attribute__((packed)) REGISTERS;
|
||||||
|
|
||||||
|
// If you change the following struct
|
||||||
|
// please consider updating the code in scheduler_asm.S
|
||||||
|
typedef struct PROC {
|
||||||
|
u16 id;
|
||||||
|
u16 pid;
|
||||||
|
u32* page_dir;
|
||||||
|
REGISTERS regs;
|
||||||
|
} __attribute__((packed)) PROC;
|
||||||
|
|
||||||
extern char show_tics;
|
extern char show_tics;
|
||||||
|
extern char scheduler_on;
|
||||||
|
extern PROC procs[MAX_PROC];
|
||||||
|
extern u16 current_id;
|
||||||
|
extern u16 nproc;
|
||||||
|
|
||||||
void clock();
|
void clock();
|
||||||
void schedule();
|
void schedule();
|
||||||
|
|
46
src/core/scheduler_asm.S
Normal file
46
src/core/scheduler_asm.S
Normal file
|
@ -0,0 +1,46 @@
|
||||||
|
.globl task_switch
|
||||||
|
|
||||||
|
// This function takes a pointer
|
||||||
|
// to a PROC structure and jump
|
||||||
|
// to this process
|
||||||
|
task_switch:
|
||||||
|
push 12(%esi) # eax (cf PROC struct)
|
||||||
|
push 16(%esi) # ebx
|
||||||
|
push 20(%esi) # ecx
|
||||||
|
push 24(%esi) # edx
|
||||||
|
push 44(%esi) # ebp
|
||||||
|
push 48(%esi) # esi
|
||||||
|
push 52(%esi) # edi
|
||||||
|
push 56(%esi) # ds
|
||||||
|
push 60(%esi) # es
|
||||||
|
push 64(%esi) # fs
|
||||||
|
push 68(%esi) # gs
|
||||||
|
|
||||||
|
// Don't forget to clear the interrupt
|
||||||
|
movb $0x20, %al
|
||||||
|
outb %al, $0x20
|
||||||
|
|
||||||
|
// Setup process page directory
|
||||||
|
mov 8(%esi), %eax
|
||||||
|
mov %eax, %cr3
|
||||||
|
|
||||||
|
// Setup registers
|
||||||
|
pop %gs
|
||||||
|
pop %fs
|
||||||
|
pop %es
|
||||||
|
pop %ds
|
||||||
|
pop %edi
|
||||||
|
pop %esi
|
||||||
|
pop %ebp
|
||||||
|
pop %edx
|
||||||
|
pop %ecx
|
||||||
|
pop %ebx
|
||||||
|
pop %eax
|
||||||
|
|
||||||
|
// Perform the task switch
|
||||||
|
push 36(%esi) # ss
|
||||||
|
push 40(%esi) # esp
|
||||||
|
push 72(%esi) # eflags
|
||||||
|
push 28(%esi) # cs
|
||||||
|
push 32(%esi) # eip
|
||||||
|
iret
|
Loading…
Add table
Reference in a new issue