PUBLIC inline void Thread::ipc_receiver_ready() { vcpu_disable_irqs(); state_change_dirty(~Thread_ipc_mask, Thread_receive_in_progress); }
PRIVATE void Thread::ipc_send_msg(Receiver *recv) { Syscall_frame *regs = _snd_regs; bool success = transfer_msg(regs->tag(), nonull_static_cast<Thread*>(recv), regs, _ipc_send_rights); sender_dequeue(recv->sender_list()); recv->vcpu_update_state(); //printf(" done\n"); regs->tag(L4_msg_tag(regs->tag(), success ? 0 : L4_msg_tag::Error)); Mword state_del = Thread_ipc_mask | Thread_ipc_transfer; Mword state_add = Thread_ready; if (Receiver::prepared()) // same as in Receiver::prepare_receive_dirty_2 state_add |= Thread_receive_wait; if (cpu() == current_cpu()) { state_change_dirty(~state_del, state_add); if (current_sched()->deblock(cpu(), current_sched(), true)) recv->switch_to_locked(this); } else { drq_state_change(~state_del, state_add); current()->schedule_if(current()->handle_drq()); } }
IMPLEMENTATION [mp]: #include <cstdlib> #include <cstdio> #include "config.h" #include "delayloop.h" #include "fpu.h" #include "globals.h" #include "helping_lock.h" #include "kernel_task.h" #include "processor.h" #include "scheduler.h" #include "task.h" #include "thread.h" #include "thread_state.h" #include "timer.h" #include "timer_tick.h" #include "spin_lock.h" PUBLIC static Kernel_thread * App_cpu_thread::may_be_create(Cpu_number cpu, bool cpu_never_seen_before) { if (!cpu_never_seen_before) { kernel_context(cpu)->reset_kernel_sp(); return static_cast<Kernel_thread *>(kernel_context(cpu)); } Kernel_thread *t = new (Ram_quota::root) App_cpu_thread; assert (t); set_cpu_of(t, cpu); check(t->bind(Kernel_task::kernel_task(), User<Utcb>::Ptr(0))); return t; } // the kernel bootstrap routine IMPLEMENT void App_cpu_thread::bootstrap() { extern Spin_lock<Mword> _tramp_mp_spinlock; state_change_dirty(0, Thread_ready); // Set myself ready Fpu::init(cpu(true)); // initialize the current_mem_space function to point to the kernel space Kernel_task::kernel_task()->make_current(); Mem_unit::tlb_flush(); Cpu::cpus.current().set_online(1); _tramp_mp_spinlock.set(1); kernel_context(cpu(true), this); Sched_context::rq.current().set_idle(this->sched()); Rcu::leave_idle(cpu(true)); Timer_tick::setup(cpu(true)); Timer_tick::enable(cpu(true)); enable_tlb(cpu(true)); // Setup initial timeslice Sched_context::rq.current().set_current_sched(sched()); Per_cpu_data::run_late_ctors(cpu(true)); Scheduler::scheduler.trigger_hotplug_event(); cpu_lock.clear(); printf("CPU[%u]: goes to idle loop\n", cxx::int_value<Cpu_number>(cpu(true))); for (;;) idle_op(); }