rust/src/libcore/priv.rs

265 lines
7.0 KiB
Rust
Raw Normal View History

// NB: transitionary, de-mode-ing.
#[forbid(deprecated_mode)];
#[forbid(deprecated_pattern)];
#[doc(hidden)];
export chan_from_global_ptr, weaken_task;
2012-09-04 13:12:17 -05:00
use compare_and_swap = rustrt::rust_compare_and_swap_ptr;
use task::TaskBuilder;
#[allow(non_camel_case_types)] // runtime type
2012-04-06 19:03:00 -05:00
type rust_port_id = uint;
extern mod rustrt {
fn rust_compare_and_swap_ptr(address: *libc::uintptr_t,
oldval: libc::uintptr_t,
newval: libc::uintptr_t) -> bool;
2012-04-06 19:03:00 -05:00
fn rust_task_weaken(ch: rust_port_id);
fn rust_task_unweaken(ch: rust_port_id);
}
type GlobalPtr = *libc::uintptr_t;
/**
* Atomically gets a channel from a pointer to a pointer-sized memory location
* or, if no channel exists creates and installs a new channel and sets up a
* new task to receive from it.
*/
unsafe fn chan_from_global_ptr<T: send>(
global: GlobalPtr,
2012-08-15 16:10:46 -05:00
task_fn: fn() -> task::TaskBuilder,
+f: fn~(comm::Port<T>)
) -> comm::Chan<T> {
enum Msg {
Proceed,
Abort
}
log(debug,~"ENTERING chan_from_global_ptr, before is_prob_zero check");
let is_probably_zero = *global == 0u;
log(debug,~"after is_prob_zero check");
if is_probably_zero {
log(debug,~"is probably zero...");
// There's no global channel. We must make it
2012-08-08 15:38:26 -05:00
let (setup_po, setup_ch) = do task_fn().spawn_conversation
|setup_po, setup_ch| {
2012-08-27 16:22:25 -05:00
let po = comm::Port::<T>();
let ch = comm::Chan(po);
comm::send(setup_ch, ch);
// Wait to hear if we are the official instance of
// this global task
match comm::recv::<Msg>(setup_po) {
Proceed => f(po),
Abort => ()
}
};
log(debug,~"before setup recv..");
// This is the proposed global channel
let ch = comm::recv(setup_po);
// 0 is our sentinal value. It is not a valid channel
2012-08-29 18:00:36 -05:00
assert unsafe::reinterpret_cast(&ch) != 0u;
// Install the channel
log(debug,~"BEFORE COMPARE AND SWAP");
let swapped = compare_and_swap(
2012-08-29 18:00:36 -05:00
global, 0u, unsafe::reinterpret_cast(&ch));
2012-08-22 19:24:52 -05:00
log(debug,fmt!("AFTER .. swapped? %?", swapped));
if swapped {
// Success!
comm::send(setup_ch, Proceed);
ch
} else {
// Somebody else got in before we did
comm::send(setup_ch, Abort);
2012-08-29 18:00:36 -05:00
unsafe::reinterpret_cast(&*global)
}
} else {
log(debug, ~"global != 0");
2012-08-29 18:00:36 -05:00
unsafe::reinterpret_cast(&*global)
}
}
#[test]
fn test_from_global_chan1() {
// This is unreadable, right?
// The global channel
let globchan = 0u;
let globchanp = ptr::addr_of(globchan);
// Create the global channel, attached to a new task
let ch = unsafe {
do chan_from_global_ptr(globchanp, task::task) |po| {
let ch = comm::recv(po);
comm::send(ch, true);
let ch = comm::recv(po);
comm::send(ch, true);
}
};
// Talk to it
2012-08-27 16:22:25 -05:00
let po = comm::Port();
comm::send(ch, comm::Chan(po));
assert comm::recv(po) == true;
// This one just reuses the previous channel
let ch = unsafe {
do chan_from_global_ptr(globchanp, task::task) |po| {
let ch = comm::recv(po);
comm::send(ch, false);
}
};
// Talk to the original global task
2012-08-27 16:22:25 -05:00
let po = comm::Port();
comm::send(ch, comm::Chan(po));
assert comm::recv(po) == true;
}
#[test]
fn test_from_global_chan2() {
for iter::repeat(100u) {
// The global channel
let globchan = 0u;
let globchanp = ptr::addr_of(globchan);
2012-08-27 16:22:25 -05:00
let resultpo = comm::Port();
let resultch = comm::Chan(resultpo);
// Spawn a bunch of tasks that all want to compete to
// create the global channel
2012-06-30 18:19:07 -05:00
for uint::range(0u, 10u) |i| {
do task::spawn {
let ch = unsafe {
do chan_from_global_ptr(
globchanp, task::task) |po| {
2012-06-30 18:19:07 -05:00
for uint::range(0u, 10u) |_j| {
let ch = comm::recv(po);
comm::send(ch, {i});
}
}
};
2012-08-27 16:22:25 -05:00
let po = comm::Port();
comm::send(ch, comm::Chan(po));
// We are The winner if our version of the
// task was installed
let winner = comm::recv(po);
comm::send(resultch, winner == i);
}
}
// There should be only one winner
let mut winners = 0u;
2012-06-30 18:19:07 -05:00
for uint::range(0u, 10u) |_i| {
let res = comm::recv(resultpo);
if res { winners += 1u };
}
assert winners == 1u;
}
}
2012-04-06 19:03:00 -05:00
/**
* Convert the current task to a 'weak' task temporarily
*
* As a weak task it will not be counted towards the runtime's set
* of live tasks. When there are no more outstanding live (non-weak) tasks
* the runtime will send an exit message on the provided channel.
*
* This function is super-unsafe. Do not use.
*
* # Safety notes
*
* * Weak tasks must either die on their own or exit upon receipt of
* the exit message. Failure to do so will cause the runtime to never
* exit
* * Tasks must not call `weaken_task` multiple times. This will
* break the kernel's accounting of live tasks.
* * Weak tasks must not be supervised. A supervised task keeps
* a reference to its parent, so the parent will not die.
*/
2012-08-15 16:10:46 -05:00
unsafe fn weaken_task(f: fn(comm::Port<()>)) {
2012-08-27 16:22:25 -05:00
let po = comm::Port();
let ch = comm::Chan(po);
unsafe {
2012-08-29 18:00:36 -05:00
rustrt::rust_task_weaken(unsafe::reinterpret_cast(&ch));
}
let _unweaken = Unweaken(ch);
2012-04-06 19:03:00 -05:00
f(po);
2012-08-15 20:46:55 -05:00
struct Unweaken {
2012-08-15 16:10:46 -05:00
let ch: comm::Chan<()>;
drop unsafe {
2012-08-29 18:00:36 -05:00
rustrt::rust_task_unweaken(unsafe::reinterpret_cast(&self.ch));
}
2012-04-06 19:03:00 -05:00
}
2012-09-04 17:23:28 -05:00
fn Unweaken(ch: comm::Chan<()>) -> Unweaken {
Unweaken {
ch: ch
}
}
2012-04-06 19:03:00 -05:00
}
#[test]
fn test_weaken_task_then_unweaken() {
do task::try {
unsafe {
2012-06-30 18:19:07 -05:00
do weaken_task |_po| {
}
2012-04-06 19:03:00 -05:00
}
};
}
#[test]
fn test_weaken_task_wait() {
do task::spawn_unlinked {
unsafe {
2012-06-30 18:19:07 -05:00
do weaken_task |po| {
comm::recv(po);
}
2012-04-06 19:03:00 -05:00
}
}
}
#[test]
fn test_weaken_task_stress() {
2012-04-06 19:03:00 -05:00
// Create a bunch of weak tasks
for iter::repeat(100u) {
do task::spawn {
unsafe {
2012-06-30 18:19:07 -05:00
do weaken_task |_po| {
}
2012-04-06 19:03:00 -05:00
}
}
do task::spawn_unlinked {
unsafe {
2012-06-30 18:19:07 -05:00
do weaken_task |po| {
// Wait for it to tell us to die
comm::recv(po);
}
2012-04-06 19:03:00 -05:00
}
}
}
}
#[test]
#[ignore(cfg(windows))]
fn test_weaken_task_fail() {
let res = do task::try {
unsafe {
2012-06-30 18:19:07 -05:00
do weaken_task |_po| {
fail;
}
2012-04-06 19:03:00 -05:00
}
};
assert result::is_err(res);
}