rust/src/libcore/sys.rs

209 lines
4.9 KiB
Rust
Raw Normal View History

2012-03-10 02:04:09 -06:00
#[doc = "Misc low level stuff"];
export type_desc;
export get_type_desc;
export size_of;
export min_align_of;
export pref_align_of;
2012-03-10 02:04:09 -06:00
export refcount;
export log_str;
export create_lock, lock_and_signal, condition, methods;
2012-01-19 17:56:54 -06:00
enum type_desc = {
first_param: **libc::c_int,
size: libc::size_t,
align: libc::size_t
// Remaining fields not listed
};
type rust_cond_lock = *libc::c_void;
#[abi = "cdecl"]
native mod rustrt {
fn unsupervise();
2012-06-02 22:49:39 -05:00
pure fn shape_log_str(t: *sys::type_desc, data: *()) -> str;
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;
}
#[abi = "rust-intrinsic"]
native mod rusti {
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;
}
2012-03-06 21:09:32 -06:00
#[doc = "
Returns a pointer to a type descriptor.
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 }
}
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)]
pure fn size_of<T>() -> uint {
2012-06-02 22:49:39 -05:00
unchecked { rusti::size_of::<T>() }
}
#[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.
"]
pure fn min_align_of<T>() -> uint {
2012-06-02 22:49:39 -05:00
unchecked { rusti::min_align_of::<T>() }
}
#[doc = "Returns the preferred alignment of a type"]
pure fn pref_align_of<T>() -> uint {
2012-06-02 22:49:39 -05:00
unchecked { rusti::pref_align_of::<T>() }
}
#[doc = "Returns the refcount of a shared box (as just before calling this)"]
pure fn refcount<T>(+t: @T) -> uint {
unsafe {
let ref_ptr: *uint = unsafe::reinterpret_cast(t);
*ref_ptr - 1
}
}
2012-06-02 22:49:39 -05:00
pure fn log_str<T>(t: T) -> str {
unsafe {
let data_ptr: *() = unsafe::reinterpret_cast(ptr::addr_of(t));
rustrt::shape_log_str(get_type_desc::<T>(), data_ptr)
}
}
2012-06-21 23:30:16 -05:00
class lock_and_signal {
let lock: rust_cond_lock;
new(lock: rust_cond_lock) { self.lock = lock; }
drop { rustrt::rust_destroy_cond_lock(self.lock); }
}
enum condition {
condition_(rust_cond_lock)
}
2012-06-21 23:30:16 -05:00
class unlock {
let lock: rust_cond_lock;
new(lock: rust_cond_lock) { self.lock = lock; }
drop { rustrt::rust_unlock_cond_lock(self.lock); }
}
fn create_lock() -> lock_and_signal {
lock_and_signal(rustrt::rust_create_cond_lock())
}
impl methods for lock_and_signal {
unsafe fn lock<T>(f: fn() -> T) -> T {
2012-06-21 23:30:16 -05:00
rustrt::rust_lock_cond_lock(self.lock);
let _r = unlock(self.lock);
f()
}
unsafe fn lock_cond<T>(f: fn(condition) -> T) -> T {
2012-06-21 23:30:16 -05:00
rustrt::rust_lock_cond_lock(self.lock);
let _r = unlock(self.lock);
f(condition_(self.lock))
}
}
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() {
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() {
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() {
assert pref_align_of::<uint>() == 8u;
assert pref_align_of::<*uint>() == 8u;
2012-01-17 19:28:21 -06:00
}
#[test]
#[ignore] // this can go into infinite loops
fn condition_variable() {
let lock = arc::arc(create_lock());
let lock2 = arc::clone(&lock);
2012-06-30 18:19:07 -05:00
do task::spawn |move lock2| {
let lock = arc::get(&lock2);
2012-06-30 18:19:07 -05:00
do (*lock).lock_cond |c| {
c.wait();
}
}
let mut signaled = false;
while !signaled {
2012-06-30 18:19:07 -05:00
do (*arc::get(&lock)).lock_cond |c| {
signaled = c.signal()
}
}
}
2012-01-17 19:28:21 -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: