2012-03-10 02:04:09 -06:00
|
|
|
#[doc = "Misc low level stuff"];
|
|
|
|
|
|
|
|
export type_desc;
|
|
|
|
export get_type_desc;
|
|
|
|
export size_of;
|
2012-04-27 00:28:03 -05:00
|
|
|
export min_align_of;
|
|
|
|
export pref_align_of;
|
2012-03-10 02:04:09 -06:00
|
|
|
export refcount;
|
|
|
|
export log_str;
|
2012-06-06 19:58:15 -05:00
|
|
|
export create_lock, lock_and_signal, condition, methods;
|
2011-12-13 18:25:51 -06:00
|
|
|
|
2012-01-19 17:56:54 -06:00
|
|
|
enum type_desc = {
|
2012-03-12 22:04:27 -05:00
|
|
|
first_param: **libc::c_int,
|
|
|
|
size: libc::size_t,
|
|
|
|
align: libc::size_t
|
2011-12-13 18:25:51 -06:00
|
|
|
// Remaining fields not listed
|
|
|
|
};
|
|
|
|
|
2012-06-06 17:06:24 -05:00
|
|
|
type rust_cond_lock = *libc::c_void;
|
|
|
|
|
2011-12-13 18:25:51 -06:00
|
|
|
#[abi = "cdecl"]
|
|
|
|
native mod rustrt {
|
2012-06-02 22:49:39 -05:00
|
|
|
pure fn refcount(t: *()) -> libc::intptr_t;
|
2011-12-13 18:25:51 -06:00
|
|
|
fn unsupervise();
|
2012-06-02 22:49:39 -05:00
|
|
|
pure fn shape_log_str(t: *sys::type_desc, data: *()) -> str;
|
2012-06-06 17:06:24 -05:00
|
|
|
|
|
|
|
fn rust_create_cond_lock() -> rust_cond_lock;
|
|
|
|
fn rust_destroy_cond_lock(lock: rust_cond_lock);
|
|
|
|
fn rust_lock_cond_lock(lock: rust_cond_lock);
|
|
|
|
fn rust_unlock_cond_lock(lock: rust_cond_lock);
|
|
|
|
fn rust_wait_cond_lock(lock: rust_cond_lock);
|
|
|
|
fn rust_signal_cond_lock(lock: rust_cond_lock) -> bool;
|
2011-12-13 18:25:51 -06:00
|
|
|
}
|
|
|
|
|
2012-03-23 09:05:16 -05:00
|
|
|
#[abi = "rust-intrinsic"]
|
2011-12-13 18:25:51 -06:00
|
|
|
native mod rusti {
|
2012-03-22 06:30:10 -05:00
|
|
|
fn get_tydesc<T>() -> *();
|
|
|
|
fn size_of<T>() -> uint;
|
2012-04-27 17:40:40 -05:00
|
|
|
fn pref_align_of<T>() -> uint;
|
|
|
|
fn min_align_of<T>() -> uint;
|
2011-12-13 18:25:51 -06:00
|
|
|
}
|
|
|
|
|
2012-03-06 21:09:32 -06:00
|
|
|
#[doc = "
|
|
|
|
Returns a pointer to a type descriptor.
|
2011-12-13 18:25:51 -06:00
|
|
|
|
2012-03-06 21:09:32 -06:00
|
|
|
Useful for calling certain function in the Rust runtime or otherwise
|
|
|
|
performing dark magick.
|
|
|
|
"]
|
2012-06-02 22:49:39 -05:00
|
|
|
pure fn get_type_desc<T>() -> *type_desc {
|
|
|
|
unchecked { rusti::get_tydesc::<T>() as *type_desc }
|
2011-12-13 18:25:51 -06:00
|
|
|
}
|
|
|
|
|
2012-03-06 21:09:32 -06:00
|
|
|
#[doc = "Returns the size of a type"]
|
2012-06-14 13:38:45 -05:00
|
|
|
#[inline(always)]
|
2012-06-02 22:49:39 -05:00
|
|
|
pure fn size_of<T>() -> uint unsafe {
|
|
|
|
unchecked { rusti::size_of::<T>() }
|
2011-12-13 18:25:51 -06:00
|
|
|
}
|
|
|
|
|
2012-04-27 00:28:03 -05:00
|
|
|
#[doc = "
|
|
|
|
Returns the ABI-required minimum alignment of a type
|
|
|
|
|
|
|
|
This is the alignment used for struct fields. It may be smaller
|
|
|
|
than the preferred alignment.
|
|
|
|
"]
|
2012-06-02 22:49:39 -05:00
|
|
|
pure fn min_align_of<T>() -> uint unsafe {
|
|
|
|
unchecked { rusti::min_align_of::<T>() }
|
2012-04-27 00:28:03 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
#[doc = "Returns the preferred alignment of a type"]
|
2012-06-02 22:49:39 -05:00
|
|
|
pure fn pref_align_of<T>() -> uint unsafe {
|
|
|
|
unchecked { rusti::pref_align_of::<T>() }
|
2011-12-13 18:25:51 -06:00
|
|
|
}
|
|
|
|
|
2012-03-06 21:09:32 -06:00
|
|
|
#[doc = "Returns the refcount of a shared box"]
|
2012-06-02 22:49:39 -05:00
|
|
|
pure fn refcount<T>(t: @T) -> uint {
|
2012-06-01 21:47:04 -05:00
|
|
|
unsafe {
|
|
|
|
ret rustrt::refcount(unsafe::reinterpret_cast(t)) as uint;
|
|
|
|
}
|
2011-12-13 18:25:51 -06:00
|
|
|
}
|
|
|
|
|
2012-06-02 22:49:39 -05:00
|
|
|
pure fn log_str<T>(t: T) -> str {
|
2012-06-01 21:47:04 -05:00
|
|
|
unsafe {
|
|
|
|
let data_ptr: *() = unsafe::reinterpret_cast(ptr::addr_of(t));
|
|
|
|
rustrt::shape_log_str(get_type_desc::<T>(), data_ptr)
|
|
|
|
}
|
2012-01-11 11:09:46 -06:00
|
|
|
}
|
|
|
|
|
2012-06-06 17:06:24 -05:00
|
|
|
resource lock_and_signal(lock: rust_cond_lock) {
|
|
|
|
rustrt::rust_destroy_cond_lock(lock);
|
|
|
|
}
|
|
|
|
|
|
|
|
enum condition {
|
|
|
|
condition_(rust_cond_lock)
|
|
|
|
}
|
|
|
|
|
|
|
|
resource unlock(lock: rust_cond_lock) {
|
|
|
|
rustrt::rust_unlock_cond_lock(lock);
|
|
|
|
}
|
|
|
|
|
|
|
|
fn create_lock() -> lock_and_signal {
|
|
|
|
lock_and_signal(rustrt::rust_create_cond_lock())
|
|
|
|
}
|
|
|
|
|
|
|
|
impl methods for lock_and_signal {
|
|
|
|
fn lock<T>(f: fn() -> T) -> T {
|
|
|
|
rustrt::rust_lock_cond_lock(*self);
|
|
|
|
let _r = unlock(*self);
|
|
|
|
f()
|
|
|
|
}
|
|
|
|
|
|
|
|
fn lock_cond<T>(f: fn(condition) -> T) -> T {
|
|
|
|
rustrt::rust_lock_cond_lock(*self);
|
|
|
|
let _r = unlock(*self);
|
|
|
|
f(condition_(*self))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl methods for condition {
|
|
|
|
fn wait() {
|
|
|
|
rustrt::rust_wait_cond_lock(*self);
|
|
|
|
}
|
|
|
|
|
|
|
|
fn signal() -> bool {
|
|
|
|
rustrt::rust_signal_cond_lock(*self)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-01-17 19:28:21 -06:00
|
|
|
#[cfg(test)]
|
|
|
|
mod tests {
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn size_of_basic() {
|
|
|
|
assert size_of::<u8>() == 1u;
|
|
|
|
assert size_of::<u16>() == 2u;
|
|
|
|
assert size_of::<u32>() == 4u;
|
|
|
|
assert size_of::<u64>() == 8u;
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
#[cfg(target_arch = "x86")]
|
|
|
|
#[cfg(target_arch = "arm")]
|
|
|
|
fn size_of_32() {
|
|
|
|
assert size_of::<uint>() == 4u;
|
|
|
|
assert size_of::<*uint>() == 4u;
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
#[cfg(target_arch = "x86_64")]
|
|
|
|
fn size_of_64() {
|
|
|
|
assert size_of::<uint>() == 8u;
|
|
|
|
assert size_of::<*uint>() == 8u;
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn align_of_basic() {
|
2012-04-27 00:28:03 -05:00
|
|
|
assert pref_align_of::<u8>() == 1u;
|
|
|
|
assert pref_align_of::<u16>() == 2u;
|
|
|
|
assert pref_align_of::<u32>() == 4u;
|
2012-01-17 19:28:21 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
#[cfg(target_arch = "x86")]
|
|
|
|
#[cfg(target_arch = "arm")]
|
|
|
|
fn align_of_32() {
|
2012-04-27 00:28:03 -05:00
|
|
|
assert pref_align_of::<uint>() == 4u;
|
|
|
|
assert pref_align_of::<*uint>() == 4u;
|
2012-01-17 19:28:21 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
#[cfg(target_arch = "x86_64")]
|
|
|
|
fn align_of_64() {
|
2012-04-27 00:28:03 -05:00
|
|
|
assert pref_align_of::<uint>() == 8u;
|
|
|
|
assert pref_align_of::<*uint>() == 8u;
|
2012-01-17 19:28:21 -06:00
|
|
|
}
|
2012-06-06 17:06:24 -05:00
|
|
|
|
|
|
|
#[test]
|
2012-06-19 20:03:28 -05:00
|
|
|
#[ignore] // this can go into infinite loops
|
2012-06-06 17:06:24 -05:00
|
|
|
fn condition_variable() {
|
|
|
|
let lock = arc::arc(create_lock());
|
|
|
|
let lock2 = arc::clone(&lock);
|
|
|
|
|
|
|
|
task::spawn {|move lock2|
|
|
|
|
let lock = arc::get(&lock2);
|
|
|
|
(*lock).lock_cond {|c|
|
|
|
|
c.wait();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
let mut signaled = false;
|
|
|
|
while !signaled {
|
|
|
|
(*arc::get(&lock)).lock_cond {|c|
|
|
|
|
signaled = c.signal()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2012-01-17 19:28:21 -06:00
|
|
|
}
|
|
|
|
|
2011-12-13 18:25:51 -06:00
|
|
|
// Local Variables:
|
|
|
|
// mode: rust;
|
|
|
|
// fill-column: 78;
|
|
|
|
// indent-tabs-mode: nil
|
|
|
|
// c-basic-offset: 4
|
|
|
|
// buffer-file-coding-system: utf-8-unix
|
|
|
|
// End:
|