|  | @@ -3,21 +3,24 @@
 | 
	
		
			
				|  |  |  #include <kernel/mm.hpp>
 | 
	
		
			
				|  |  |  #include <kernel/process.hpp>
 | 
	
		
			
				|  |  |  #include <kernel/stdio.h>
 | 
	
		
			
				|  |  | +#include <kernel/tty.h>
 | 
	
		
			
				|  |  |  #include <kernel_main.h>
 | 
	
		
			
				|  |  |  #include <types/types.h>
 | 
	
		
			
				|  |  | +#include <types/lock.h>
 | 
	
		
			
				|  |  |  #include <hello-world.res>
 | 
	
		
			
				|  |  |  #include <interrupt-test.res>
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |  extern "C" void NORETURN go_user_space(void* eip);
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -static inline void* align_down_to_16byte(void* addr)
 | 
	
		
			
				|  |  | -{
 | 
	
		
			
				|  |  | -    return (void*)((uint32_t)addr & 0xfffffff0);
 | 
	
		
			
				|  |  | -}
 | 
	
		
			
				|  |  | +extern "C" void NORETURN to_kernel(interrupt_stack* ret_stack);
 | 
	
		
			
				|  |  | +extern "C" void NORETURN to_user(interrupt_stack* ret_stack);
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |  static bool is_scheduler_ready;
 | 
	
		
			
				|  |  |  static types::list<process>* processes;
 | 
	
		
			
				|  |  |  static types::list<thread*>* ready_thds;
 | 
	
		
			
				|  |  | +static pid_t max_pid = 1;
 | 
	
		
			
				|  |  | +void (* volatile kthreadd_new_thd_func)(void*);
 | 
	
		
			
				|  |  | +void  * volatile kthreadd_new_thd_data;
 | 
	
		
			
				|  |  | +uint32_t kthreadd_lock = 0;
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |  thread* current_thread;
 | 
	
		
			
				|  |  |  process* current_process;
 | 
	
	
		
			
				|  | @@ -25,6 +28,7 @@ process* current_process;
 | 
	
		
			
				|  |  |  process::process(process&& val)
 | 
	
		
			
				|  |  |      : mms(types::move(val.mms))
 | 
	
		
			
				|  |  |      , thds(types::move(val.thds))
 | 
	
		
			
				|  |  | +    , pid(val.pid)
 | 
	
		
			
				|  |  |  {
 | 
	
		
			
				|  |  |      if (current_process == &val)
 | 
	
		
			
				|  |  |          current_process = this;
 | 
	
	
		
			
				|  | @@ -39,49 +43,155 @@ process::process(process&& val)
 | 
	
		
			
				|  |  |      val.attr.system = 0;
 | 
	
		
			
				|  |  |  }
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | +process::process(const process& val, const thread& main_thd)
 | 
	
		
			
				|  |  | +    : mms(*kernel_mms)
 | 
	
		
			
				|  |  | +    , attr { .system = val.attr.system }
 | 
	
		
			
				|  |  | +    , pid { max_pid++ }
 | 
	
		
			
				|  |  | +{
 | 
	
		
			
				|  |  | +    auto iter_thd = thds.emplace_back(main_thd);
 | 
	
		
			
				|  |  | +    iter_thd->owner = this;
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +    if (!val.attr.system) {
 | 
	
		
			
				|  |  | +        // TODO: allocate low mem
 | 
	
		
			
				|  |  | +        k_esp = (void*)page_to_phys_addr(alloc_n_raw_pages(2));
 | 
	
		
			
				|  |  | +        memset((char*)k_esp, 0x00, THREAD_KERNEL_STACK_SIZE);
 | 
	
		
			
				|  |  | +        k_esp = (char*)k_esp + THREAD_KERNEL_STACK_SIZE;
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +        page_directory_entry* pd = alloc_pd();
 | 
	
		
			
				|  |  | +        memcpy(pd, mms_get_pd(kernel_mms), PAGE_SIZE);
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +        mms.begin()->pd = pd;
 | 
	
		
			
				|  |  | +        // skip kernel heap
 | 
	
		
			
				|  |  | +        for (auto iter_src = ++val.mms.cbegin(); iter_src != val.mms.cend(); ++iter_src) {
 | 
	
		
			
				|  |  | +            auto iter_dst = mms.emplace_back(iter_src->start, pd, iter_src->attr.write, iter_src->attr.system);
 | 
	
		
			
				|  |  | +            iter_dst->pd = pd;
 | 
	
		
			
				|  |  | +            for (auto pg = iter_src->pgs->begin(); pg != iter_src->pgs->end(); ++pg)
 | 
	
		
			
				|  |  | +                k_map(iter_dst.ptr(),
 | 
	
		
			
				|  |  | +                        &*pg,
 | 
	
		
			
				|  |  | +                        iter_src->attr.read,
 | 
	
		
			
				|  |  | +                        iter_src->attr.write,
 | 
	
		
			
				|  |  | +                        iter_src->attr.system,
 | 
	
		
			
				|  |  | +                        1);
 | 
	
		
			
				|  |  | +        }
 | 
	
		
			
				|  |  | +    } else {
 | 
	
		
			
				|  |  | +        // TODO: allocate low mem
 | 
	
		
			
				|  |  | +        k_esp = (void*)page_to_phys_addr(alloc_n_raw_pages(2));
 | 
	
		
			
				|  |  | +        memcpy(k_esp, main_thd.owner->k_esp, THREAD_KERNEL_STACK_SIZE);
 | 
	
		
			
				|  |  | +        k_esp = (char*)k_esp + THREAD_KERNEL_STACK_SIZE;
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +        auto orig_k_esp = (uint32_t)main_thd.owner->k_esp;
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +        iter_thd->regs.ebp -= orig_k_esp;
 | 
	
		
			
				|  |  | +        iter_thd->regs.ebp += (uint32_t)k_esp;
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +        iter_thd->regs.esp -= orig_k_esp;
 | 
	
		
			
				|  |  | +        iter_thd->regs.esp += (uint32_t)k_esp;
 | 
	
		
			
				|  |  | +    }
 | 
	
		
			
				|  |  | +}
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  |  process::process(void* start_eip, uint8_t* image, size_t image_size, bool system)
 | 
	
		
			
				|  |  |      : mms(*kernel_mms)
 | 
	
		
			
				|  |  |      , thds {}
 | 
	
		
			
				|  |  |      , attr { .system = system }
 | 
	
		
			
				|  |  | +    , pid { max_pid++ }
 | 
	
		
			
				|  |  |  {
 | 
	
		
			
				|  |  | -    k_esp = align_down_to_16byte((char*)k_malloc(THREAD_KERNEL_STACK_SIZE) + THREAD_KERNEL_STACK_SIZE);
 | 
	
		
			
				|  |  | -    memset((char*)k_esp - THREAD_KERNEL_STACK_SIZE, 0x00, THREAD_KERNEL_STACK_SIZE);
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -    page_directory_entry* pd = alloc_pd();
 | 
	
		
			
				|  |  | -    memcpy(pd, mms_get_pd(kernel_mms), PAGE_SIZE);
 | 
	
		
			
				|  |  | -    for (auto& item : mms)
 | 
	
		
			
				|  |  | -        item.pd = pd;
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -    auto user_mm = mms.emplace_back(0x40000000U, pd, 1, system);
 | 
	
		
			
				|  |  | +    // TODO: allocate low mem
 | 
	
		
			
				|  |  | +    k_esp = (void*)page_to_phys_addr(alloc_n_raw_pages(2));
 | 
	
		
			
				|  |  | +    memset((char*)k_esp, 0x00, THREAD_KERNEL_STACK_SIZE);
 | 
	
		
			
				|  |  | +    k_esp = (char*)k_esp + THREAD_KERNEL_STACK_SIZE;
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |      auto thd = thds.emplace_back(thread {
 | 
	
		
			
				|  |  |          .eip = start_eip,
 | 
	
		
			
				|  |  |          .owner = this,
 | 
	
		
			
				|  |  | -        .regs {},
 | 
	
		
			
				|  |  | -        .eflags {},
 | 
	
		
			
				|  |  |          // TODO: change this
 | 
	
		
			
				|  |  | -        .esp = 0x40100000U,
 | 
	
		
			
				|  |  | +        .regs {
 | 
	
		
			
				|  |  | +            .edi {},
 | 
	
		
			
				|  |  | +            .esi {},
 | 
	
		
			
				|  |  | +            .ebp = system ? (uint32_t)k_esp : 0x40100000U,
 | 
	
		
			
				|  |  | +            .esp = system ? (uint32_t)k_esp : 0x40100000U,
 | 
	
		
			
				|  |  | +            .ebx {},
 | 
	
		
			
				|  |  | +            .edx {},
 | 
	
		
			
				|  |  | +            .ecx {},
 | 
	
		
			
				|  |  | +            .eax {},
 | 
	
		
			
				|  |  | +        },
 | 
	
		
			
				|  |  | +        .eflags {},
 | 
	
		
			
				|  |  | +        .attr {
 | 
	
		
			
				|  |  | +            .system = system,
 | 
	
		
			
				|  |  | +            .ready = 1,
 | 
	
		
			
				|  |  | +            .wait = 0,
 | 
	
		
			
				|  |  | +        },
 | 
	
		
			
				|  |  |      });
 | 
	
		
			
				|  |  |      ready_thds->push_back(thd.ptr());
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | -    // TODO: change this
 | 
	
		
			
				|  |  | -    for (int i = 0; i < 1 * 1024 * 1024 / PAGE_SIZE; ++i)
 | 
	
		
			
				|  |  | -        k_map(user_mm.ptr(), &empty_page, 1, 1, 0, 1);
 | 
	
		
			
				|  |  | +    if (!system) {
 | 
	
		
			
				|  |  | +        page_directory_entry* pd = alloc_pd();
 | 
	
		
			
				|  |  | +        memcpy(pd, mms_get_pd(kernel_mms), PAGE_SIZE);
 | 
	
		
			
				|  |  | +        for (auto& item : mms)
 | 
	
		
			
				|  |  | +            item.pd = pd;
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +        auto user_mm = mms.emplace_back(0x40000000U, pd, 1, system);
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +        // TODO: change this
 | 
	
		
			
				|  |  | +        for (int i = 0; i < 1 * 1024 * 1024 / PAGE_SIZE; ++i)
 | 
	
		
			
				|  |  | +            k_map(user_mm.ptr(), &empty_page, 1, 1, 0, 1);
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +        auto* old_pd = reinterpret_cast<page_directory_entry*>(p_ptr_to_v_ptr(current_pd()));
 | 
	
		
			
				|  |  | +        auto* old_proc = current_process;
 | 
	
		
			
				|  |  | +        auto* old_thd = current_thread;
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +        current_process = this;
 | 
	
		
			
				|  |  | +        current_thread = thd.ptr();
 | 
	
		
			
				|  |  | +        asm_switch_pd(pd);
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | -    auto* old_pd = reinterpret_cast<page_directory_entry*>(p_ptr_to_v_ptr(current_pd()));
 | 
	
		
			
				|  |  | -    auto* old_proc = current_process;
 | 
	
		
			
				|  |  | -    auto* old_thd = current_thread;
 | 
	
		
			
				|  |  | +        // TODO: change this
 | 
	
		
			
				|  |  | +        memcpy((void*)0x40000000U, image, image_size);
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | -    current_process = this;
 | 
	
		
			
				|  |  | -    current_thread = thd.ptr();
 | 
	
		
			
				|  |  | -    asm_switch_pd(pd);
 | 
	
		
			
				|  |  | +        current_process = old_proc;
 | 
	
		
			
				|  |  | +        current_thread = old_thd;
 | 
	
		
			
				|  |  | +        asm_switch_pd(old_pd);
 | 
	
		
			
				|  |  | +    }
 | 
	
		
			
				|  |  | +}
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | -    // TODO: change this
 | 
	
		
			
				|  |  | -    memcpy((void*)0x40000000U, image, image_size);
 | 
	
		
			
				|  |  | +void _example_io_thread(void* _d)
 | 
	
		
			
				|  |  | +{
 | 
	
		
			
				|  |  | +    const char* data = reinterpret_cast<const char*>(_d);
 | 
	
		
			
				|  |  | +    tty_print(console, data);
 | 
	
		
			
				|  |  | +    // syscall_sleep
 | 
	
		
			
				|  |  | +    asm volatile("movl $0x02, %%eax\nint $0x80":::"eax");
 | 
	
		
			
				|  |  | +}
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | -    current_process = old_proc;
 | 
	
		
			
				|  |  | -    current_thread = old_thd;
 | 
	
		
			
				|  |  | -    asm_switch_pd(old_pd);
 | 
	
		
			
				|  |  | +void kernel_threadd_main(void)
 | 
	
		
			
				|  |  | +{
 | 
	
		
			
				|  |  | +    tty_print(console, "kernel thread daemon started\n");
 | 
	
		
			
				|  |  | +    kthreadd_new_thd_func = _example_io_thread;
 | 
	
		
			
				|  |  | +    kthreadd_new_thd_data = (void*)"data in io thread\n";
 | 
	
		
			
				|  |  | +    for (;;) {
 | 
	
		
			
				|  |  | +        spin_lock(&kthreadd_lock);
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +        if (kthreadd_new_thd_func) {
 | 
	
		
			
				|  |  | +            int return_value = 0;
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +            void (*func)(void*) = kthreadd_new_thd_func;
 | 
	
		
			
				|  |  | +            void* data = kthreadd_new_thd_data;
 | 
	
		
			
				|  |  | +            kthreadd_new_thd_func = nullptr;
 | 
	
		
			
				|  |  | +            kthreadd_new_thd_data = nullptr;
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +            spin_unlock(&kthreadd_lock);
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +            // syscall_fork
 | 
	
		
			
				|  |  | +            asm volatile("movl $0x00, %%eax\nint $0x80\nmovl %%eax, %0": "=a" (return_value): :);
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +            if (return_value != 0) {
 | 
	
		
			
				|  |  | +                // child
 | 
	
		
			
				|  |  | +                func(data);
 | 
	
		
			
				|  |  | +                for (;;) asm_hlt();
 | 
	
		
			
				|  |  | +                // TODO: syscall_exit()
 | 
	
		
			
				|  |  | +            }
 | 
	
		
			
				|  |  | +        } else {
 | 
	
		
			
				|  |  | +            spin_unlock(&kthreadd_lock);
 | 
	
		
			
				|  |  | +        }
 | 
	
		
			
				|  |  | +        asm_hlt();
 | 
	
		
			
				|  |  | +    }
 | 
	
		
			
				|  |  |  }
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |  void NORETURN init_scheduler()
 | 
	
	
		
			
				|  | @@ -93,6 +203,7 @@ void NORETURN init_scheduler()
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |      processes->emplace_back(user_space_start, hello_world_bin, hello_world_bin_len, false);
 | 
	
		
			
				|  |  |      processes->emplace_back(user_space_start, interrupt_test_bin, interrupt_test_bin_len, false);
 | 
	
		
			
				|  |  | +    processes->emplace_back((void*)kernel_threadd_main, nullptr, 0, true);
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |      // we need interrupts enabled for cow mapping
 | 
	
		
			
				|  |  |      asm_cli();
 | 
	
	
		
			
				|  | @@ -108,55 +219,82 @@ void NORETURN init_scheduler()
 | 
	
		
			
				|  |  |      go_user_space(user_space_start);
 | 
	
		
			
				|  |  |  }
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | -void context_switch(irq0_data* intrpt_data)
 | 
	
		
			
				|  |  | +void thread_context_save(interrupt_stack* int_stack, thread* thd)
 | 
	
		
			
				|  |  | +{
 | 
	
		
			
				|  |  | +    thd->eflags = int_stack->eflags;
 | 
	
		
			
				|  |  | +    thd->eip = int_stack->v_eip;
 | 
	
		
			
				|  |  | +    memcpy(&thd->regs, &int_stack->s_regs, sizeof(regs_32));
 | 
	
		
			
				|  |  | +    if (thd->attr.system)
 | 
	
		
			
				|  |  | +        thd->regs.esp = int_stack->s_regs.esp + 0x0c;
 | 
	
		
			
				|  |  | +    else
 | 
	
		
			
				|  |  | +        thd->regs.esp = int_stack->esp;
 | 
	
		
			
				|  |  | +}
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +void thread_context_load(interrupt_stack* int_stack, thread* thd)
 | 
	
		
			
				|  |  | +{
 | 
	
		
			
				|  |  | +    int_stack->eflags = (thd->eflags | 0x200); // OR $STI
 | 
	
		
			
				|  |  | +    int_stack->v_eip = thd->eip;
 | 
	
		
			
				|  |  | +    memcpy(&int_stack->s_regs, &thd->regs, sizeof(regs_32));
 | 
	
		
			
				|  |  | +    current_thread = thd;
 | 
	
		
			
				|  |  | +}
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +void process_context_save(interrupt_stack*, process*)
 | 
	
		
			
				|  |  | +{
 | 
	
		
			
				|  |  | +}
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +void process_context_load(interrupt_stack*, process* proc)
 | 
	
		
			
				|  |  | +{
 | 
	
		
			
				|  |  | +    if (!proc->attr.system)
 | 
	
		
			
				|  |  | +        tss.esp0 = (uint32_t)proc->k_esp;
 | 
	
		
			
				|  |  | +    asm_switch_pd(mms_get_pd(&proc->mms));
 | 
	
		
			
				|  |  | +    current_process = proc;
 | 
	
		
			
				|  |  | +}
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +void add_to_process_list(process&& proc)
 | 
	
		
			
				|  |  | +{
 | 
	
		
			
				|  |  | +    processes->push_back(types::move(proc));
 | 
	
		
			
				|  |  | +}
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +void add_to_ready_list(thread* thd)
 | 
	
		
			
				|  |  | +{
 | 
	
		
			
				|  |  | +    ready_thds->push_back(thd);
 | 
	
		
			
				|  |  | +}
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +static inline void next_task(const types::list<thread*>::iterator_type& iter_to_remove, thread* cur_thd)
 | 
	
		
			
				|  |  | +{
 | 
	
		
			
				|  |  | +    ready_thds->erase(iter_to_remove);
 | 
	
		
			
				|  |  | +    if (cur_thd->attr.ready)
 | 
	
		
			
				|  |  | +        ready_thds->push_back(cur_thd);
 | 
	
		
			
				|  |  | +}
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +void do_scheduling(interrupt_stack* intrpt_data)
 | 
	
		
			
				|  |  |  {
 | 
	
		
			
				|  |  |      if (!is_scheduler_ready)
 | 
	
		
			
				|  |  |          return;
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | -    thread* thd = *ready_thds->begin();
 | 
	
		
			
				|  |  | +    auto iter_thd = ready_thds->begin();
 | 
	
		
			
				|  |  | +    while (!((*iter_thd)->attr.ready))
 | 
	
		
			
				|  |  | +        iter_thd = ready_thds->erase(iter_thd);
 | 
	
		
			
				|  |  | +    auto thd = *iter_thd;
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  |      if (current_thread == thd) {
 | 
	
		
			
				|  |  | -        ready_thds->erase(ready_thds->begin());
 | 
	
		
			
				|  |  | -        // check if the thread is ready
 | 
	
		
			
				|  |  | -        ready_thds->push_back(thd);
 | 
	
		
			
				|  |  | +        next_task(iter_thd, thd);
 | 
	
		
			
				|  |  |          return;
 | 
	
		
			
				|  |  |      }
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | -    process* pro = thd->owner;
 | 
	
		
			
				|  |  | -    if (current_process != pro) {
 | 
	
		
			
				|  |  | -        if (!pro->attr.system) {
 | 
	
		
			
				|  |  | -            tss.esp0 = (uint32_t)pro->k_esp;
 | 
	
		
			
				|  |  | -        }
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -        current_process = pro;
 | 
	
		
			
				|  |  | -        asm_switch_pd(pro->mms.begin()->pd);
 | 
	
		
			
				|  |  | +    process* proc = thd->owner;
 | 
	
		
			
				|  |  | +    if (current_process != proc) {
 | 
	
		
			
				|  |  | +        process_context_save(intrpt_data, current_process);
 | 
	
		
			
				|  |  | +        process_context_load(intrpt_data, proc);
 | 
	
		
			
				|  |  |      }
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | -    // save current thread info
 | 
	
		
			
				|  |  | -    current_thread->eflags = intrpt_data->eflags;
 | 
	
		
			
				|  |  | -    current_thread->eip = intrpt_data->v_eip;
 | 
	
		
			
				|  |  | -    memcpy(¤t_thread->regs, &intrpt_data->s_regs, sizeof(regs_32));
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -    // load ready thread info
 | 
	
		
			
				|  |  | -    intrpt_data->eflags = thd->eflags;
 | 
	
		
			
				|  |  | -    intrpt_data->eflags |= 0x200; // sti
 | 
	
		
			
				|  |  | -    intrpt_data->v_eip = thd->eip;
 | 
	
		
			
				|  |  | -    memcpy(&intrpt_data->s_regs, &thd->regs, sizeof(regs_32));
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -    if (!pro->attr.system) {
 | 
	
		
			
				|  |  | -        // user mode
 | 
	
		
			
				|  |  | -        current_thread->esp = intrpt_data->esp;
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -        intrpt_data->cs = USER_CODE_SELECTOR;
 | 
	
		
			
				|  |  | -        intrpt_data->ss = USER_DATA_SELECTOR;
 | 
	
		
			
				|  |  | -        intrpt_data->esp = thd->esp;
 | 
	
		
			
				|  |  | -    } else {
 | 
	
		
			
				|  |  | -        // supervisor mode
 | 
	
		
			
				|  |  | -        intrpt_data->cs = KERNEL_CODE_SEGMENT;
 | 
	
		
			
				|  |  | -    }
 | 
	
		
			
				|  |  | +    thread_context_save(intrpt_data, current_thread);
 | 
	
		
			
				|  |  | +    thread_context_load(intrpt_data, thd);
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | -    ready_thds->erase(ready_thds->begin());
 | 
	
		
			
				|  |  | -    // check if the thread is ready
 | 
	
		
			
				|  |  | -    ready_thds->push_back(thd);
 | 
	
		
			
				|  |  | +    next_task(iter_thd, thd);
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | -    current_thread = thd;
 | 
	
		
			
				|  |  | +    if (thd->attr.system)
 | 
	
		
			
				|  |  | +        to_kernel(intrpt_data);
 | 
	
		
			
				|  |  | +    else
 | 
	
		
			
				|  |  | +        to_user(intrpt_data);
 | 
	
		
			
				|  |  |  }
 |