2012-12-03 16:48:01 -08:00
|
|
|
// Copyright 2012 The Rust Project Developers. See the COPYRIGHT
|
|
|
|
// file at the top-level directory of this distribution and at
|
|
|
|
// http://rust-lang.org/COPYRIGHT.
|
|
|
|
//
|
|
|
|
// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
|
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
|
|
|
|
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
|
|
|
|
// option. This file may not be copied, modified, or distributed
|
|
|
|
// except according to those terms.
|
|
|
|
|
2012-07-04 22:53:12 +01:00
|
|
|
//! Misc low level stuff
|
2012-03-10 00:04:09 -08:00
|
|
|
|
2012-12-23 17:41:37 -05:00
|
|
|
use cast;
|
2012-08-27 16:26:35 -07:00
|
|
|
use cmp::{Eq, Ord};
|
2012-12-23 17:41:37 -05:00
|
|
|
use gc;
|
|
|
|
use io;
|
|
|
|
use libc;
|
2012-12-10 17:22:10 -08:00
|
|
|
use libc::{c_void, c_char, size_t};
|
2012-12-23 17:41:37 -05:00
|
|
|
use ptr;
|
|
|
|
use repr;
|
|
|
|
use str;
|
|
|
|
use vec;
|
2012-08-27 16:26:35 -07:00
|
|
|
|
2012-09-26 16:01:40 -07:00
|
|
|
pub type FreeGlue = fn(*TypeDesc, *c_void);
|
2012-09-05 21:10:22 -07:00
|
|
|
|
2012-08-13 16:20:27 -07:00
|
|
|
// Corresponds to runtime type_desc type
|
2012-09-26 16:01:40 -07:00
|
|
|
pub enum TypeDesc = {
|
2012-07-09 17:23:13 -07:00
|
|
|
size: uint,
|
2012-09-05 21:10:22 -07:00
|
|
|
align: uint,
|
|
|
|
take_glue: uint,
|
|
|
|
drop_glue: uint,
|
|
|
|
free_glue: uint
|
2011-12-13 16:25:51 -08:00
|
|
|
// Remaining fields not listed
|
|
|
|
};
|
|
|
|
|
2012-08-29 14:04:10 -07:00
|
|
|
/// The representation of a Rust closure
|
2012-09-26 16:01:40 -07:00
|
|
|
pub struct Closure {
|
2012-09-07 14:50:47 -07:00
|
|
|
code: *(),
|
|
|
|
env: *(),
|
2012-08-29 14:04:10 -07:00
|
|
|
}
|
|
|
|
|
2012-03-23 15:05:16 +01:00
|
|
|
#[abi = "rust-intrinsic"]
|
2012-07-03 16:11:00 -07:00
|
|
|
extern mod rusti {
|
2012-03-22 12:30:10 +01:00
|
|
|
fn get_tydesc<T>() -> *();
|
|
|
|
fn size_of<T>() -> uint;
|
2012-04-27 15:40:40 -07:00
|
|
|
fn pref_align_of<T>() -> uint;
|
|
|
|
fn min_align_of<T>() -> uint;
|
2011-12-13 16:25:51 -08:00
|
|
|
}
|
|
|
|
|
2012-12-10 17:22:10 -08:00
|
|
|
extern mod rustrt {
|
|
|
|
#[rust_stack]
|
2013-01-10 21:23:07 -08:00
|
|
|
unsafe fn rust_upcall_fail(expr: *c_char, file: *c_char, line: size_t);
|
2012-12-10 17:22:10 -08:00
|
|
|
}
|
|
|
|
|
2012-08-02 15:42:56 -07:00
|
|
|
/// Compares contents of two pointers using the default method.
|
|
|
|
/// Equivalent to `*x1 == *x2`. Useful for hashtables.
|
2012-09-26 16:01:40 -07:00
|
|
|
pub pure fn shape_eq<T:Eq>(x1: &T, x2: &T) -> bool {
|
2012-08-02 15:42:56 -07:00
|
|
|
*x1 == *x2
|
|
|
|
}
|
|
|
|
|
2012-09-26 16:01:40 -07:00
|
|
|
pub pure fn shape_lt<T:Ord>(x1: &T, x2: &T) -> bool {
|
2012-08-02 15:42:56 -07:00
|
|
|
*x1 < *x2
|
|
|
|
}
|
|
|
|
|
2012-09-26 16:01:40 -07:00
|
|
|
pub pure fn shape_le<T:Ord>(x1: &T, x2: &T) -> bool {
|
2012-08-27 16:26:35 -07:00
|
|
|
*x1 <= *x2
|
2012-08-02 15:42:56 -07:00
|
|
|
}
|
|
|
|
|
2012-07-04 22:53:12 +01: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 15:57:32 -07:00
|
|
|
#[inline(always)]
|
2012-09-26 16:01:40 -07:00
|
|
|
pub pure fn get_type_desc<T>() -> *TypeDesc {
|
2012-09-18 11:17:40 -07:00
|
|
|
unsafe { rusti::get_tydesc::<T>() as *TypeDesc }
|
2011-12-13 16:25:51 -08:00
|
|
|
}
|
|
|
|
|
2012-07-04 22:53:12 +01:00
|
|
|
/// Returns the size of a type
|
2012-06-14 11:38:45 -07:00
|
|
|
#[inline(always)]
|
2012-09-26 16:01:40 -07:00
|
|
|
pub pure fn size_of<T>() -> uint {
|
2012-09-18 11:17:40 -07:00
|
|
|
unsafe { rusti::size_of::<T>() }
|
2011-12-13 16:25:51 -08:00
|
|
|
}
|
|
|
|
|
2013-01-08 00:24:43 -08:00
|
|
|
/**
|
|
|
|
* Returns the size of a type, or 1 if the actual size is zero.
|
|
|
|
*
|
|
|
|
* Useful for building structures containing variable-length arrays.
|
|
|
|
*/
|
|
|
|
#[inline(always)]
|
|
|
|
pub pure fn nonzero_size_of<T>() -> uint {
|
|
|
|
let s = size_of::<T>();
|
|
|
|
if s == 0 { 1 } else { s }
|
|
|
|
}
|
|
|
|
|
2012-07-04 22:53:12 +01: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 15:32:30 -07:00
|
|
|
#[inline(always)]
|
2012-09-26 16:01:40 -07:00
|
|
|
pub pure fn min_align_of<T>() -> uint {
|
2012-09-18 11:17:40 -07:00
|
|
|
unsafe { rusti::min_align_of::<T>() }
|
2012-04-26 22:28:03 -07:00
|
|
|
}
|
|
|
|
|
2012-07-04 22:53:12 +01:00
|
|
|
/// Returns the preferred alignment of a type
|
2012-08-21 15:32:30 -07:00
|
|
|
#[inline(always)]
|
2012-09-26 16:01:40 -07:00
|
|
|
pub pure fn pref_align_of<T>() -> uint {
|
2012-09-18 11:17:40 -07:00
|
|
|
unsafe { rusti::pref_align_of::<T>() }
|
2011-12-13 16:25:51 -08:00
|
|
|
}
|
|
|
|
|
2012-07-04 22:53:12 +01:00
|
|
|
/// Returns the refcount of a shared box (as just before calling this)
|
2012-09-11 15:57:32 -07:00
|
|
|
#[inline(always)]
|
2012-10-02 11:37:37 -07:00
|
|
|
pub pure fn refcount<T>(t: @T) -> uint {
|
2012-06-01 19:47:04 -07:00
|
|
|
unsafe {
|
2012-09-18 17:34:08 -07:00
|
|
|
let ref_ptr: *uint = cast::reinterpret_cast(&t);
|
2012-06-28 14:40:31 -04:00
|
|
|
*ref_ptr - 1
|
2012-06-01 19:47:04 -07:00
|
|
|
}
|
2011-12-13 16:25:51 -08:00
|
|
|
}
|
|
|
|
|
2012-09-26 16:01:40 -07:00
|
|
|
pub pure fn log_str<T>(t: &T) -> ~str {
|
2012-06-01 19:47:04 -07:00
|
|
|
unsafe {
|
2012-09-14 09:40:28 -07:00
|
|
|
do io::with_str_writer |wr| {
|
2012-09-20 15:37:43 -07:00
|
|
|
repr::write_repr(wr, t)
|
2012-09-14 09:40:28 -07:00
|
|
|
}
|
2012-06-01 19:47:04 -07:00
|
|
|
}
|
2012-01-11 09:09:46 -08:00
|
|
|
}
|
|
|
|
|
2012-12-10 17:22:10 -08:00
|
|
|
/** Initiate task failure */
|
|
|
|
pub pure fn begin_unwind(msg: ~str, file: ~str, line: uint) -> ! {
|
|
|
|
do str::as_buf(msg) |msg_buf, _msg_len| {
|
|
|
|
do str::as_buf(file) |file_buf, _file_len| {
|
|
|
|
unsafe {
|
|
|
|
let msg_buf = cast::transmute(msg_buf);
|
|
|
|
let file_buf = cast::transmute(file_buf);
|
|
|
|
begin_unwind_(msg_buf, file_buf, line as libc::size_t)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-01-10 15:16:15 -08:00
|
|
|
// FIXME #4427: Temporary until rt::rt_fail_ goes away
|
2012-12-10 17:22:10 -08:00
|
|
|
pub pure fn begin_unwind_(msg: *c_char, file: *c_char, line: size_t) -> ! {
|
|
|
|
unsafe {
|
|
|
|
gc::cleanup_stack_for_failure();
|
|
|
|
rustrt::rust_upcall_fail(msg, file, line);
|
|
|
|
cast::transmute(())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-01-17 17:28:21 -08:00
|
|
|
#[cfg(test)]
|
2012-09-26 16:01:40 -07:00
|
|
|
pub mod tests {
|
2012-12-27 17:53:04 -08:00
|
|
|
use cast;
|
2013-01-08 00:24:43 -08:00
|
|
|
use sys::{Closure, pref_align_of, size_of, nonzero_size_of};
|
2012-01-17 17:28:21 -08:00
|
|
|
|
|
|
|
#[test]
|
2012-09-26 16:01:40 -07:00
|
|
|
pub fn size_of_basic() {
|
2012-01-17 17:28:21 -08:00
|
|
|
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")]
|
2012-09-26 16:01:40 -07:00
|
|
|
pub fn size_of_32() {
|
2012-01-17 17:28:21 -08:00
|
|
|
assert size_of::<uint>() == 4u;
|
|
|
|
assert size_of::<*uint>() == 4u;
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
#[cfg(target_arch = "x86_64")]
|
2012-09-26 16:01:40 -07:00
|
|
|
pub fn size_of_64() {
|
2012-01-17 17:28:21 -08:00
|
|
|
assert size_of::<uint>() == 8u;
|
|
|
|
assert size_of::<*uint>() == 8u;
|
|
|
|
}
|
|
|
|
|
2013-01-08 00:24:43 -08:00
|
|
|
#[test]
|
|
|
|
pub fn nonzero_size_of_basic() {
|
|
|
|
type Z = [i8 * 0];
|
|
|
|
assert size_of::<Z>() == 0u;
|
|
|
|
assert nonzero_size_of::<Z>() == 1u;
|
|
|
|
assert nonzero_size_of::<uint>() == size_of::<uint>();
|
|
|
|
}
|
|
|
|
|
2012-01-17 17:28:21 -08:00
|
|
|
#[test]
|
2012-09-26 16:01:40 -07:00
|
|
|
pub fn align_of_basic() {
|
2012-04-26 22:28:03 -07:00
|
|
|
assert pref_align_of::<u8>() == 1u;
|
|
|
|
assert pref_align_of::<u16>() == 2u;
|
|
|
|
assert pref_align_of::<u32>() == 4u;
|
2012-01-17 17:28:21 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
#[cfg(target_arch = "x86")]
|
|
|
|
#[cfg(target_arch = "arm")]
|
2012-09-26 16:01:40 -07:00
|
|
|
pub fn align_of_32() {
|
2012-04-26 22:28:03 -07:00
|
|
|
assert pref_align_of::<uint>() == 4u;
|
|
|
|
assert pref_align_of::<*uint>() == 4u;
|
2012-01-17 17:28:21 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
#[cfg(target_arch = "x86_64")]
|
2012-09-26 16:01:40 -07:00
|
|
|
pub fn align_of_64() {
|
2012-04-26 22:28:03 -07:00
|
|
|
assert pref_align_of::<uint>() == 8u;
|
|
|
|
assert pref_align_of::<*uint>() == 8u;
|
2012-01-17 17:28:21 -08:00
|
|
|
}
|
2012-08-29 14:04:10 -07:00
|
|
|
|
|
|
|
#[test]
|
2013-01-23 11:43:58 -08:00
|
|
|
pub fn synthesize_closure() {
|
|
|
|
unsafe {
|
|
|
|
let x = 10;
|
|
|
|
let f: fn(int) -> int = |y| x + y;
|
2012-08-29 14:04:10 -07:00
|
|
|
|
2013-01-23 11:43:58 -08:00
|
|
|
assert f(20) == 30;
|
2012-08-29 14:04:10 -07:00
|
|
|
|
2013-02-15 03:51:28 -05:00
|
|
|
let original_closure: Closure = cast::transmute(f);
|
2012-08-29 14:04:10 -07:00
|
|
|
|
2013-01-23 11:43:58 -08:00
|
|
|
let actual_function_pointer = original_closure.code;
|
|
|
|
let environment = original_closure.env;
|
2012-08-29 14:04:10 -07:00
|
|
|
|
2013-01-23 11:43:58 -08:00
|
|
|
let new_closure = Closure {
|
|
|
|
code: actual_function_pointer,
|
|
|
|
env: environment
|
|
|
|
};
|
2012-08-29 14:04:10 -07:00
|
|
|
|
2013-02-15 03:51:28 -05:00
|
|
|
let new_f: fn(int) -> int = cast::transmute(new_closure);
|
2013-01-23 11:43:58 -08:00
|
|
|
assert new_f(20) == 30;
|
|
|
|
}
|
2012-08-29 14:04:10 -07:00
|
|
|
}
|
2012-01-17 17:28:21 -08:00
|
|
|
}
|
|
|
|
|
2011-12-13 16:25:51 -08:00
|
|
|
// Local Variables:
|
|
|
|
// mode: rust;
|
|
|
|
// fill-column: 78;
|
|
|
|
// indent-tabs-mode: nil
|
|
|
|
// c-basic-offset: 4
|
|
|
|
// buffer-file-coding-system: utf-8-unix
|
|
|
|
// End:
|