Skip to content

Commit

Permalink
Add at_exit function rust-lang#4450
Browse files Browse the repository at this point in the history
  • Loading branch information
brson committed Jan 13, 2013
1 parent 91e3cb2 commit 5a78d0a
Show file tree
Hide file tree
Showing 6 changed files with 157 additions and 0 deletions.
3 changes: 3 additions & 0 deletions src/libcore/private.rs
Original file line number Diff line number Diff line change
Expand Up @@ -28,6 +28,9 @@ use task;
use task::{TaskBuilder, atomically};
use uint;

#[path = "private/at_exit.rs"]
pub mod at_exit;

extern mod rustrt {
#[legacy_exports];
unsafe fn rust_task_weaken(ch: rust_port_id);
Expand Down
86 changes: 86 additions & 0 deletions src/libcore/private/at_exit.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,86 @@
use sys;
use cast;
use ptr;
use task;
use uint;
use vec;
use rand;
use libc::{c_void, size_t};

/**
Register a function to be run during runtime shutdown.
After all non-weak tasks have exited, registered exit functions will
execute, in random order, on the primary scheduler. Each function runs
in its own unsupervised task.
*/
pub fn at_exit(f: ~fn()) unsafe {
let runner: &fn(*ExitFunctions) = exit_runner;
let runner_pair: sys::Closure = cast::transmute(runner);
let runner_ptr = runner_pair.code;
let runner_ptr = cast::transmute(runner_ptr);
rustrt::rust_register_exit_function(runner_ptr, ~f);
}

// NB: The double pointer indirection here is because ~fn() is a fat
// pointer and due to FFI problems I am more comfortable making the
// interface use a normal pointer
extern mod rustrt {
fn rust_register_exit_function(runner: *c_void, f: ~~fn());
}

struct ExitFunctions {
// The number of exit functions
count: size_t,
// The buffer of exit functions
start: *~~fn()
}

fn exit_runner(exit_fns: *ExitFunctions) unsafe {
let exit_fns = &*exit_fns;
let count = (*exit_fns).count;
let start = (*exit_fns).start;

// NB: from_buf memcpys from the source, which will
// give us ownership of the array of functions
let mut exit_fns_vec = vec::from_buf(start, count as uint);
// Let's not make any promises about execution order
rand::Rng().shuffle_mut(exit_fns_vec);

debug!("running %u exit functions", exit_fns_vec.len());

while exit_fns_vec.is_not_empty() {
match exit_fns_vec.pop() {
~f => {
task::task().supervised().spawn(f);
}
}
}
}

#[abi = "rust-intrinsic"]
pub extern mod rusti {
fn move_val_init<T>(dst: &mut T, -src: T);
fn init<T>() -> T;
}

#[test]
fn test_at_exit() {
let i = 10;
do at_exit {
debug!("at_exit1");
assert i == 10;
}
}

#[test]
fn test_at_exit_many() {
let i = 10;
for uint::range(20, 100) |j| {
do at_exit {
debug!("at_exit2");
assert i == 10;
assert j > i;
}
}
}
5 changes: 5 additions & 0 deletions src/rt/rust_builtin.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -1005,6 +1005,11 @@ rust_raw_thread_join_delete(raw_thread *thread) {
delete thread;
}

extern "C" void
rust_register_exit_function(spawn_fn runner, fn_env_pair *f) {
rust_task *task = rust_get_current_task();
task->kernel->register_exit_function(runner, f);
}

//
// Local Variables:
Expand Down
44 changes: 44 additions & 0 deletions src/rt/rust_kernel.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -36,6 +36,8 @@ rust_kernel::rust_kernel(rust_env *env) :
non_weak_tasks(0),
global_loop_chan(0),
global_env_chan(0),
at_exit_runner(NULL),
at_exit_started(false),
env(env)

{
Expand Down Expand Up @@ -427,6 +429,7 @@ rust_kernel::begin_shutdown() {
}
}

run_exit_functions();
allow_scheduler_exit();
end_weak_tasks();
}
Expand All @@ -446,6 +449,47 @@ rust_kernel::send_to_port(rust_port_id chan, void *sptr) {
}
}

void
rust_kernel::register_exit_function(spawn_fn runner, fn_env_pair *f) {
scoped_lock with(at_exit_lock);

assert(!at_exit_started && "registering at_exit function after exit");

if (at_exit_runner) {
assert(runner == at_exit_runner
&& "there can be only one at_exit_runner");
}

at_exit_runner = runner;
at_exit_fns.push_back(f);
}

void
rust_kernel::run_exit_functions() {
rust_task *task;

{
scoped_lock with(at_exit_lock);

assert(!at_exit_started && "running exit functions twice?");

at_exit_started = true;

if (at_exit_runner == NULL) {
return;
}

rust_scheduler *sched = get_scheduler_by_id(main_sched_id());
assert(sched);
task = sched->create_task(NULL, "at_exit");

final_exit_fns.count = at_exit_fns.size();
final_exit_fns.start = at_exit_fns.data();
}

task->start(at_exit_runner, NULL, &final_exit_fns);
}

//
// Local Variables:
// mode: C++
Expand Down
18 changes: 18 additions & 0 deletions src/rt/rust_kernel.h
Original file line number Diff line number Diff line change
Expand Up @@ -50,6 +50,7 @@
#include "memory_region.h"
#include "rust_log.h"
#include "rust_sched_reaper.h"
#include "rust_type.h"
#include "util/hash_map.h"

class rust_scheduler;
Expand All @@ -66,6 +67,13 @@ typedef intptr_t rust_port_id;

typedef std::map<rust_sched_id, rust_scheduler*> sched_map;

// This is defined as a struct only because we need a single pointer to pass
// to the Rust function that runs the at_exit functions
struct exit_functions {
size_t count;
fn_env_pair **start;
};

class rust_kernel {
memory_region _region;
rust_log _log;
Expand Down Expand Up @@ -126,6 +134,14 @@ class rust_kernel {
// Used to serialize access to getenv/setenv
uintptr_t global_env_chan;

lock_and_signal at_exit_lock;
spawn_fn at_exit_runner;
bool at_exit_started;
std::vector<fn_env_pair*> at_exit_fns;
exit_functions final_exit_fns;

void run_exit_functions();

public:
struct rust_env *env;

Expand Down Expand Up @@ -175,6 +191,8 @@ class rust_kernel {

uintptr_t* get_global_loop() { return &global_loop_chan; }
uintptr_t* get_global_env_chan() { return &global_env_chan; }

void register_exit_function(spawn_fn runner, fn_env_pair *f);
};

template <typename T> struct kernel_owned {
Expand Down
1 change: 1 addition & 0 deletions src/rt/rustrt.def.in
Original file line number Diff line number Diff line change
Expand Up @@ -210,3 +210,4 @@ linenoiseHistorySave
linenoiseHistoryLoad
rust_raw_thread_start
rust_raw_thread_join_delete
rust_register_exit_function

0 comments on commit 5a78d0a

Please sign in to comment.