2012-07-04 16:53:12 -05:00
|
|
|
//! Misc low level stuff
|
2012-03-10 02:04:09 -06:00
|
|
|
|
2012-08-27 18:26:35 -05:00
|
|
|
use cmp::{Eq, Ord};
|
2012-09-05 23:10:22 -05:00
|
|
|
use libc::c_void;
|
2012-08-27 18:26:35 -05:00
|
|
|
|
2012-09-05 23:10:22 -05:00
|
|
|
export FreeGlue;
|
2012-08-13 18:20:27 -05:00
|
|
|
export TypeDesc;
|
2012-08-29 16:05:27 -05:00
|
|
|
export Closure;
|
2012-03-10 02:04:09 -06:00
|
|
|
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-08-02 17:42:56 -05:00
|
|
|
export shape_eq, shape_lt, shape_le;
|
2011-12-13 18:25:51 -06:00
|
|
|
|
2012-09-05 23:10:22 -05:00
|
|
|
type FreeGlue = fn(*TypeDesc, *c_void);
|
|
|
|
|
2012-08-13 18:20:27 -05:00
|
|
|
// Corresponds to runtime type_desc type
|
|
|
|
enum TypeDesc = {
|
2012-07-09 19:23:13 -05:00
|
|
|
size: uint,
|
2012-09-05 23:10:22 -05:00
|
|
|
align: uint,
|
|
|
|
take_glue: uint,
|
|
|
|
drop_glue: uint,
|
|
|
|
free_glue: uint
|
2011-12-13 18:25:51 -06:00
|
|
|
// Remaining fields not listed
|
|
|
|
};
|
|
|
|
|
2012-08-29 16:04:10 -05:00
|
|
|
/// The representation of a Rust closure
|
|
|
|
struct Closure {
|
2012-09-07 16:50:47 -05:00
|
|
|
code: *(),
|
|
|
|
env: *(),
|
2012-08-29 16:04:10 -05:00
|
|
|
}
|
|
|
|
|
2011-12-13 18:25:51 -06:00
|
|
|
#[abi = "cdecl"]
|
2012-07-03 18:11:00 -05:00
|
|
|
extern mod rustrt {
|
2012-08-13 18:20:27 -05:00
|
|
|
pure fn shape_log_str(t: *sys::TypeDesc, data: *()) -> ~str;
|
2011-12-13 18:25:51 -06:00
|
|
|
}
|
|
|
|
|
2012-03-23 09:05:16 -05:00
|
|
|
#[abi = "rust-intrinsic"]
|
2012-07-03 18:11:00 -05:00
|
|
|
extern 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-08-02 17:42:56 -05:00
|
|
|
/// Compares contents of two pointers using the default method.
|
|
|
|
/// Equivalent to `*x1 == *x2`. Useful for hashtables.
|
2012-08-27 18:26:35 -05:00
|
|
|
pure fn shape_eq<T:Eq>(x1: &T, x2: &T) -> bool {
|
2012-08-02 17:42:56 -05:00
|
|
|
*x1 == *x2
|
|
|
|
}
|
|
|
|
|
2012-08-27 18:26:35 -05:00
|
|
|
pure fn shape_lt<T:Ord>(x1: &T, x2: &T) -> bool {
|
2012-08-02 17:42:56 -05:00
|
|
|
*x1 < *x2
|
|
|
|
}
|
|
|
|
|
2012-08-27 18:26:35 -05:00
|
|
|
pure fn shape_le<T:Ord>(x1: &T, x2: &T) -> bool {
|
|
|
|
*x1 <= *x2
|
2012-08-02 17:42:56 -05:00
|
|
|
}
|
|
|
|
|
2012-07-04 16:53:12 -05:00
|
|
|
/**
|
|
|
|
* Returns a pointer to a type descriptor.
|
|
|
|
*
|
|
|
|
* Useful for calling certain function in the Rust runtime or otherwise
|
|
|
|
* performing dark magick.
|
|
|
|
*/
|
2012-09-11 17:57:32 -05:00
|
|
|
#[inline(always)]
|
2012-08-13 18:20:27 -05:00
|
|
|
pure fn get_type_desc<T>() -> *TypeDesc {
|
|
|
|
unchecked { rusti::get_tydesc::<T>() as *TypeDesc }
|
2011-12-13 18:25:51 -06:00
|
|
|
}
|
|
|
|
|
2012-07-04 16:53:12 -05:00
|
|
|
/// Returns the size of a type
|
2012-06-14 13:38:45 -05:00
|
|
|
#[inline(always)]
|
2012-06-24 22:18:18 -05:00
|
|
|
pure fn size_of<T>() -> uint {
|
2012-06-02 22:49:39 -05:00
|
|
|
unchecked { rusti::size_of::<T>() }
|
2011-12-13 18:25:51 -06:00
|
|
|
}
|
|
|
|
|
2012-07-04 16:53:12 -05:00
|
|
|
/**
|
|
|
|
* 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-08-21 17:32:30 -05:00
|
|
|
#[inline(always)]
|
2012-06-24 22:18:18 -05:00
|
|
|
pure fn min_align_of<T>() -> uint {
|
2012-06-02 22:49:39 -05:00
|
|
|
unchecked { rusti::min_align_of::<T>() }
|
2012-04-27 00:28:03 -05:00
|
|
|
}
|
|
|
|
|
2012-07-04 16:53:12 -05:00
|
|
|
/// Returns the preferred alignment of a type
|
2012-08-21 17:32:30 -05:00
|
|
|
#[inline(always)]
|
2012-06-24 22:18:18 -05:00
|
|
|
pure fn pref_align_of<T>() -> uint {
|
2012-06-02 22:49:39 -05:00
|
|
|
unchecked { rusti::pref_align_of::<T>() }
|
2011-12-13 18:25:51 -06:00
|
|
|
}
|
|
|
|
|
2012-07-04 16:53:12 -05:00
|
|
|
/// Returns the refcount of a shared box (as just before calling this)
|
2012-09-11 17:57:32 -05:00
|
|
|
#[inline(always)]
|
2012-06-28 13:40:31 -05:00
|
|
|
pure fn refcount<T>(+t: @T) -> uint {
|
2012-06-01 21:47:04 -05:00
|
|
|
unsafe {
|
2012-08-29 18:00:36 -05:00
|
|
|
let ref_ptr: *uint = unsafe::reinterpret_cast(&t);
|
2012-06-28 13:40:31 -05:00
|
|
|
*ref_ptr - 1
|
2012-06-01 21:47:04 -05:00
|
|
|
}
|
2011-12-13 18:25:51 -06:00
|
|
|
}
|
|
|
|
|
2012-07-14 00:57:48 -05:00
|
|
|
pure fn log_str<T>(t: T) -> ~str {
|
2012-06-01 21:47:04 -05:00
|
|
|
unsafe {
|
2012-08-29 18:00:36 -05:00
|
|
|
let data_ptr: *() = unsafe::reinterpret_cast(&ptr::addr_of(t));
|
2012-06-01 21:47:04 -05:00
|
|
|
rustrt::shape_log_str(get_type_desc::<T>(), data_ptr)
|
|
|
|
}
|
2012-01-11 11:09:46 -06:00
|
|
|
}
|
|
|
|
|
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-08-29 16:04:10 -05:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn synthesize_closure() unsafe {
|
|
|
|
let x = 10;
|
|
|
|
let f: fn(int) -> int = |y| x + y;
|
|
|
|
|
|
|
|
assert f(20) == 30;
|
|
|
|
|
|
|
|
let original_closure: Closure = unsafe::transmute(f);
|
|
|
|
|
|
|
|
let actual_function_pointer = original_closure.code;
|
|
|
|
let environment = original_closure.env;
|
|
|
|
|
|
|
|
let new_closure = Closure {
|
|
|
|
code: actual_function_pointer,
|
|
|
|
env: environment
|
|
|
|
};
|
|
|
|
|
|
|
|
let new_f: fn(int) -> int = unsafe::transmute(new_closure);
|
|
|
|
assert new_f(20) == 30;
|
|
|
|
}
|
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:
|