import cast = unsafe::reinterpret_cast; import comm; import option::some; import option::none; import option = option::t; import ptr; native "rust" mod rustrt { fn task_sleep(time_in_us: uint); fn task_yield(); fn task_join(t: task_id) -> int; fn unsupervise(); fn pin_task(); fn unpin_task(); fn get_task_id() -> task_id; type rust_chan; fn set_min_stack(stack_size: uint); fn new_task() -> task_id; fn drop_task(task: *rust_task); fn get_task_pointer(id: task_id) -> *rust_task; fn start_task(id: task_id); fn get_task_trampoline() -> u32; fn migrate_alloc(alloc: *u8, target: task_id); fn leak<@T>(thing: -T); } type rust_task = {id: task, mutable notify_enabled: u8, mutable notify_chan: comm::chan_handle, ctx: task_context, stack_ptr: *u8}; type task_context = {regs: x86_registers, next: *u8}; resource rust_task_ptr(task: *rust_task) { rustrt::drop_task(task); } fn get_task_ptr(id: task) -> rust_task_ptr { ret rust_task_ptr(rustrt::get_task_pointer(id)); } type task = int; type task_id = task; fn get_task_id() -> task_id { rustrt::get_task_id() } /** * Hints the scheduler to yield this task for a specified ammount of time. * * arg: time_in_us maximum number of microseconds to yield control for */ fn sleep(time_in_us: uint) { ret rustrt::task_sleep(time_in_us); } fn yield() { ret rustrt::task_yield(); } tag task_result { tr_success; tr_failure; } tag task_notification { exit(task, task_result); } fn join(task_port: (task_id, comm::port)) -> task_result { let (id, port) = task_port; alt comm::recv::(port) { exit(_id, res) { if _id == id { ret res } else { fail #fmt["join received id %d, expected %d", _id, id] } } } } fn join_id(t: task_id) -> task_result { alt rustrt::task_join(t) { 0 { tr_success } _ { tr_failure } } } fn unsupervise() { ret rustrt::unsupervise(); } fn pin() { rustrt::pin_task(); } fn unpin() { rustrt::unpin_task(); } fn set_min_stack(stack_size: uint) { rustrt::set_min_stack(stack_size); } fn _spawn(thunk: -fn()) -> task { spawn(thunk) } fn spawn(thunk: -fn()) -> task { spawn_inner(thunk, none) } fn spawn_notify(thunk: -fn(), notify: comm::chan) -> task { spawn_inner(thunk, some(notify)) } fn spawn_joinable(thunk: -fn()) -> (task_id, comm::port) { let p = comm::port::(); let id = spawn_notify(thunk, comm::chan::(p)); ret (id, p); } // FIXME: make this a fn~ once those are supported. fn spawn_inner(thunk: -fn(), notify: option>) -> task_id { let id = rustrt::new_task(); // the order of arguments are outptr, taskptr, envptr. // LLVM fastcall puts the first two in ecx, edx, and the rest on the // stack. // set up the task pointer let task_ptr = get_task_ptr(id); let regs = ptr::addr_of((**task_ptr).ctx.regs); (*regs).edx = cast(*task_ptr);; (*regs).esp = cast((**task_ptr).stack_ptr); assert (ptr::null() != (**task_ptr).stack_ptr); let raw_thunk: {code: u32, env: u32} = cast(thunk); (*regs).eip = raw_thunk.code; // set up notifications if they are enabled. alt notify { some(c) { (**task_ptr).notify_enabled = 1u8;; (**task_ptr).notify_chan = *c; } none { } }; // okay, now we align the stack and add the environment pointer and a fake // return address. // -12 for the taskm output location, the env pointer // -4 for the return address. (*regs).esp = align_down((*regs).esp - 12u32) - 4u32; let ra: *mutable u32 = cast((*regs).esp); let env: *mutable u32 = cast((*regs).esp + 4u32); let tptr: *mutable u32 = cast((*regs).esp + 12u32); // put the return pointer in ecx. (*regs).ecx = (*regs).esp + 8u32;; *tptr = cast(*task_ptr);; *env = raw_thunk.env;; *ra = rustrt::get_task_trampoline(); rustrt::migrate_alloc(cast(raw_thunk.env), id); rustrt::start_task(id); rustrt::leak(thunk); ret id; } // Who says we can't write an operating system in Rust? type x86_registers = // This needs to match the structure in context.h {mutable eax: u32, mutable ebx: u32, mutable ecx: u32, mutable edx: u32, mutable ebp: u32, mutable esi: u32, mutable edi: u32, mutable esp: u32, mutable cs: u16, mutable ds: u16, mutable ss: u16, mutable es: u16, mutable fs: u16, mutable gs: u16, mutable eflags: u32, mutable eip: u32}; fn align_down(x: u32) -> u32 { // Aligns x down to 16 bytes x & !15u32 } // Local Variables: // mode: rust; // fill-column: 78; // indent-tabs-mode: nil // c-basic-offset: 4 // buffer-file-coding-system: utf-8-unix // compile-command: "make -k -C $RBUILD 2>&1 | sed -e 's/\\/x\\//x:\\//g'"; // End: