2012-12-03 18:48:01 -06: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 16:53:12 -05:00
|
|
|
//! Misc low level stuff
|
2012-03-10 02:04:09 -06:00
|
|
|
|
2013-05-28 16:35:52 -05:00
|
|
|
#[allow(missing_doc)];
|
|
|
|
|
2013-04-23 17:11:28 -05:00
|
|
|
use option::{Some, None};
|
2012-12-23 16:41:37 -06:00
|
|
|
use cast;
|
|
|
|
use gc;
|
|
|
|
use io;
|
|
|
|
use libc;
|
2012-12-10 19:22:10 -06:00
|
|
|
use libc::{c_void, c_char, size_t};
|
2012-12-23 16:41:37 -06:00
|
|
|
use repr;
|
|
|
|
use str;
|
2013-05-09 14:49:14 -05:00
|
|
|
use unstable::intrinsics;
|
2012-08-27 18:26:35 -05:00
|
|
|
|
2013-03-25 15:21:04 -05:00
|
|
|
pub type FreeGlue<'self> = &'self fn(*TypeDesc, *c_void);
|
2012-09-05 23:10:22 -05:00
|
|
|
|
2012-08-13 18:20:27 -05:00
|
|
|
// Corresponds to runtime type_desc type
|
2013-03-05 19:36:59 -06:00
|
|
|
pub struct 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
|
2013-03-05 19:36:59 -06:00
|
|
|
}
|
2011-12-13 18:25:51 -06:00
|
|
|
|
2012-08-29 16:04:10 -05:00
|
|
|
/// The representation of a Rust closure
|
2012-09-26 18:01:40 -05:00
|
|
|
pub struct Closure {
|
2012-09-07 16:50:47 -05:00
|
|
|
code: *(),
|
|
|
|
env: *(),
|
2012-08-29 16:04:10 -05:00
|
|
|
}
|
|
|
|
|
2013-03-05 13:57:50 -06:00
|
|
|
pub mod rustrt {
|
|
|
|
use libc::{c_char, size_t};
|
|
|
|
|
|
|
|
pub extern {
|
|
|
|
#[rust_stack]
|
|
|
|
unsafe fn rust_upcall_fail(expr: *c_char,
|
|
|
|
file: *c_char,
|
|
|
|
line: size_t);
|
|
|
|
}
|
2012-12-10 19:22:10 -06: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.
|
|
|
|
*/
|
2013-06-18 16:45:18 -05:00
|
|
|
#[inline]
|
2013-03-21 23:20:48 -05:00
|
|
|
pub fn get_type_desc<T>() -> *TypeDesc {
|
2013-05-09 14:49:14 -05:00
|
|
|
unsafe { intrinsics::get_tydesc::<T>() as *TypeDesc }
|
2011-12-13 18:25:51 -06:00
|
|
|
}
|
|
|
|
|
2013-04-19 23:34:18 -05:00
|
|
|
/// Returns a pointer to a type descriptor.
|
2013-06-18 16:45:18 -05:00
|
|
|
#[inline]
|
2013-04-19 23:34:18 -05:00
|
|
|
pub fn get_type_desc_val<T>(_val: &T) -> *TypeDesc {
|
|
|
|
get_type_desc::<T>()
|
|
|
|
}
|
|
|
|
|
2012-07-04 16:53:12 -05:00
|
|
|
/// Returns the size of a type
|
2013-06-18 16:45:18 -05:00
|
|
|
#[inline]
|
2013-03-21 23:20:48 -05:00
|
|
|
pub fn size_of<T>() -> uint {
|
2013-05-09 14:49:14 -05:00
|
|
|
unsafe { intrinsics::size_of::<T>() }
|
2011-12-13 18:25:51 -06:00
|
|
|
}
|
|
|
|
|
2013-04-19 23:34:18 -05:00
|
|
|
/// Returns the size of the type that `_val` points to
|
2013-06-18 16:45:18 -05:00
|
|
|
#[inline]
|
2013-04-19 23:34:18 -05:00
|
|
|
pub fn size_of_val<T>(_val: &T) -> uint {
|
|
|
|
size_of::<T>()
|
|
|
|
}
|
|
|
|
|
2013-01-08 02:24:43 -06:00
|
|
|
/**
|
|
|
|
* Returns the size of a type, or 1 if the actual size is zero.
|
|
|
|
*
|
|
|
|
* Useful for building structures containing variable-length arrays.
|
|
|
|
*/
|
2013-06-18 16:45:18 -05:00
|
|
|
#[inline]
|
2013-03-21 23:20:48 -05:00
|
|
|
pub fn nonzero_size_of<T>() -> uint {
|
2013-01-08 02:24:43 -06:00
|
|
|
let s = size_of::<T>();
|
|
|
|
if s == 0 { 1 } else { s }
|
|
|
|
}
|
|
|
|
|
2013-04-19 23:34:18 -05:00
|
|
|
/// Returns the size of the type of the value that `_val` points to
|
2013-06-18 16:45:18 -05:00
|
|
|
#[inline]
|
2013-04-19 23:34:18 -05:00
|
|
|
pub fn nonzero_size_of_val<T>(_val: &T) -> uint {
|
|
|
|
nonzero_size_of::<T>()
|
|
|
|
}
|
|
|
|
|
|
|
|
|
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.
|
|
|
|
*/
|
2013-06-18 16:45:18 -05:00
|
|
|
#[inline]
|
2013-03-21 23:20:48 -05:00
|
|
|
pub fn min_align_of<T>() -> uint {
|
2013-05-09 14:49:14 -05:00
|
|
|
unsafe { intrinsics::min_align_of::<T>() }
|
2012-04-27 00:28:03 -05:00
|
|
|
}
|
|
|
|
|
2013-04-19 23:34:18 -05:00
|
|
|
/// Returns the ABI-required minimum alignment of the type of the value that
|
|
|
|
/// `_val` points to
|
2013-06-18 16:45:18 -05:00
|
|
|
#[inline]
|
2013-04-19 23:34:18 -05:00
|
|
|
pub fn min_align_of_val<T>(_val: &T) -> uint {
|
|
|
|
min_align_of::<T>()
|
|
|
|
}
|
|
|
|
|
2012-07-04 16:53:12 -05:00
|
|
|
/// Returns the preferred alignment of a type
|
2013-06-18 16:45:18 -05:00
|
|
|
#[inline]
|
2013-03-21 23:20:48 -05:00
|
|
|
pub fn pref_align_of<T>() -> uint {
|
2013-05-09 14:49:14 -05:00
|
|
|
unsafe { intrinsics::pref_align_of::<T>() }
|
2011-12-13 18:25:51 -06:00
|
|
|
}
|
|
|
|
|
2013-04-19 23:34:18 -05:00
|
|
|
/// Returns the preferred alignment of the type of the value that
|
|
|
|
/// `_val` points to
|
2013-06-18 16:45:18 -05:00
|
|
|
#[inline]
|
2013-04-19 23:34:18 -05:00
|
|
|
pub fn pref_align_of_val<T>(_val: &T) -> uint {
|
|
|
|
pref_align_of::<T>()
|
|
|
|
}
|
|
|
|
|
2012-07-04 16:53:12 -05:00
|
|
|
/// Returns the refcount of a shared box (as just before calling this)
|
2013-06-18 16:45:18 -05:00
|
|
|
#[inline]
|
2013-03-21 23:20:48 -05:00
|
|
|
pub fn refcount<T>(t: @T) -> uint {
|
2012-06-01 21:47:04 -05:00
|
|
|
unsafe {
|
2013-04-26 16:04:39 -05:00
|
|
|
let ref_ptr: *uint = cast::transmute_copy(&t);
|
2013-04-22 23:19:58 -05:00
|
|
|
*ref_ptr - 1
|
2012-06-01 21:47:04 -05:00
|
|
|
}
|
2011-12-13 18:25:51 -06:00
|
|
|
}
|
|
|
|
|
2013-03-21 23:20:48 -05:00
|
|
|
pub fn log_str<T>(t: &T) -> ~str {
|
2013-04-09 00:31:23 -05:00
|
|
|
do io::with_str_writer |wr| {
|
|
|
|
repr::write_repr(wr, t)
|
2012-06-01 21:47:04 -05:00
|
|
|
}
|
2012-01-11 11:09:46 -06:00
|
|
|
}
|
|
|
|
|
2013-04-23 15:30:58 -05:00
|
|
|
/// Trait for initiating task failure.
|
|
|
|
pub trait FailWithCause {
|
|
|
|
/// Fail the current task, taking ownership of `cause`
|
|
|
|
fn fail_with(cause: Self, file: &'static str, line: uint) -> !;
|
|
|
|
}
|
|
|
|
|
|
|
|
impl FailWithCause for ~str {
|
|
|
|
fn fail_with(cause: ~str, file: &'static str, line: uint) -> ! {
|
|
|
|
do str::as_buf(cause) |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)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl FailWithCause for &'static str {
|
|
|
|
fn fail_with(cause: &'static str, file: &'static str, line: uint) -> ! {
|
|
|
|
do str::as_buf(cause) |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 17:16:15 -06:00
|
|
|
// FIXME #4427: Temporary until rt::rt_fail_ goes away
|
2013-03-21 23:20:48 -05:00
|
|
|
pub fn begin_unwind_(msg: *c_char, file: *c_char, line: size_t) -> ! {
|
2013-06-19 20:37:50 -05:00
|
|
|
use cell::Cell;
|
2013-05-04 19:30:31 -05:00
|
|
|
use option::Option;
|
2013-06-19 20:37:50 -05:00
|
|
|
use either::Left;
|
2013-05-04 19:30:31 -05:00
|
|
|
use rt::{context, OldTaskContext, TaskContext};
|
2013-05-19 18:50:21 -05:00
|
|
|
use rt::task::{Task, Unwinder};
|
|
|
|
use rt::local::Local;
|
2013-06-19 20:37:50 -05:00
|
|
|
use rt::logging::Logger;
|
2013-04-22 19:15:31 -05:00
|
|
|
|
2013-05-04 19:30:31 -05:00
|
|
|
let context = context();
|
|
|
|
match context {
|
2013-04-22 19:15:31 -05:00
|
|
|
OldTaskContext => {
|
2013-05-02 21:13:56 -05:00
|
|
|
unsafe {
|
|
|
|
gc::cleanup_stack_for_failure();
|
|
|
|
rustrt::rust_upcall_fail(msg, file, line);
|
|
|
|
cast::transmute(())
|
2013-04-22 19:15:31 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
_ => {
|
2012-12-10 19:22:10 -06:00
|
|
|
unsafe {
|
2013-05-04 19:30:31 -05:00
|
|
|
// XXX: Bad re-allocations. fail! needs some refactoring
|
|
|
|
let msg = str::raw::from_c_str(msg);
|
|
|
|
let file = str::raw::from_c_str(file);
|
|
|
|
|
2013-06-19 20:37:50 -05:00
|
|
|
let outmsg = fmt!("task failed: '%s' at line %i of file %s",
|
|
|
|
msg, line as int, file);
|
2013-05-04 19:30:31 -05:00
|
|
|
|
|
|
|
// XXX: Logging doesn't work correctly in non-task context because it
|
|
|
|
// invokes the local heap
|
|
|
|
if context == TaskContext {
|
2013-06-19 20:37:50 -05:00
|
|
|
// XXX: Logging doesn't work here - the check to call the log
|
|
|
|
// function never passes - so calling the log function directly.
|
|
|
|
let outmsg = Cell::new(outmsg);
|
|
|
|
do Local::borrow::<Task, ()> |task| {
|
|
|
|
task.logger.log(Left(outmsg.take()));
|
|
|
|
}
|
2013-05-04 19:30:31 -05:00
|
|
|
} else {
|
|
|
|
rtdebug!("%s", outmsg);
|
|
|
|
}
|
|
|
|
|
|
|
|
gc::cleanup_stack_for_failure();
|
|
|
|
|
2013-05-19 18:50:21 -05:00
|
|
|
let task = Local::unsafe_borrow::<Task>();
|
2013-06-14 01:16:27 -05:00
|
|
|
(*task).unwinder.begin_unwind();
|
2012-12-10 19:22:10 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-01-17 19:28:21 -06:00
|
|
|
#[cfg(test)]
|
2013-04-15 10:08:52 -05:00
|
|
|
mod tests {
|
2012-12-27 19:53:04 -06:00
|
|
|
use cast;
|
2013-04-19 23:34:18 -05:00
|
|
|
use sys::*;
|
2012-01-17 19:28:21 -06:00
|
|
|
|
|
|
|
#[test]
|
2013-04-15 10:08:52 -05:00
|
|
|
fn size_of_basic() {
|
2013-05-18 21:02:45 -05:00
|
|
|
assert_eq!(size_of::<u8>(), 1u);
|
|
|
|
assert_eq!(size_of::<u16>(), 2u);
|
|
|
|
assert_eq!(size_of::<u32>(), 4u);
|
|
|
|
assert_eq!(size_of::<u64>(), 8u);
|
2012-01-17 19:28:21 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
#[cfg(target_arch = "x86")]
|
|
|
|
#[cfg(target_arch = "arm")]
|
2013-01-29 08:28:08 -06:00
|
|
|
#[cfg(target_arch = "mips")]
|
2013-04-15 10:08:52 -05:00
|
|
|
fn size_of_32() {
|
2013-05-18 21:02:45 -05:00
|
|
|
assert_eq!(size_of::<uint>(), 4u);
|
|
|
|
assert_eq!(size_of::<*uint>(), 4u);
|
2012-01-17 19:28:21 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
#[cfg(target_arch = "x86_64")]
|
2013-04-15 10:08:52 -05:00
|
|
|
fn size_of_64() {
|
2013-05-18 21:02:45 -05:00
|
|
|
assert_eq!(size_of::<uint>(), 8u);
|
|
|
|
assert_eq!(size_of::<*uint>(), 8u);
|
2012-01-17 19:28:21 -06:00
|
|
|
}
|
|
|
|
|
2013-04-19 23:34:18 -05:00
|
|
|
#[test]
|
|
|
|
fn size_of_val_basic() {
|
|
|
|
assert_eq!(size_of_val(&1u8), 1);
|
|
|
|
assert_eq!(size_of_val(&1u16), 2);
|
|
|
|
assert_eq!(size_of_val(&1u32), 4);
|
|
|
|
assert_eq!(size_of_val(&1u64), 8);
|
|
|
|
}
|
|
|
|
|
2013-01-08 02:24:43 -06:00
|
|
|
#[test]
|
2013-04-15 10:08:52 -05:00
|
|
|
fn nonzero_size_of_basic() {
|
2013-03-22 20:52:04 -05:00
|
|
|
type Z = [i8, ..0];
|
2013-05-18 21:02:45 -05:00
|
|
|
assert_eq!(size_of::<Z>(), 0u);
|
|
|
|
assert_eq!(nonzero_size_of::<Z>(), 1u);
|
|
|
|
assert_eq!(nonzero_size_of::<uint>(), size_of::<uint>());
|
2013-01-08 02:24:43 -06:00
|
|
|
}
|
|
|
|
|
2013-04-19 23:34:18 -05:00
|
|
|
#[test]
|
|
|
|
fn nonzero_size_of_val_basic() {
|
|
|
|
let z = [0u8, ..0];
|
|
|
|
assert_eq!(size_of_val(&z), 0u);
|
|
|
|
assert_eq!(nonzero_size_of_val(&z), 1u);
|
|
|
|
assert_eq!(nonzero_size_of_val(&1u), size_of_val(&1u));
|
|
|
|
}
|
|
|
|
|
2012-01-17 19:28:21 -06:00
|
|
|
#[test]
|
2013-04-15 10:08:52 -05:00
|
|
|
fn align_of_basic() {
|
2013-05-18 21:02:45 -05:00
|
|
|
assert_eq!(pref_align_of::<u8>(), 1u);
|
|
|
|
assert_eq!(pref_align_of::<u16>(), 2u);
|
|
|
|
assert_eq!(pref_align_of::<u32>(), 4u);
|
2012-01-17 19:28:21 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
#[cfg(target_arch = "x86")]
|
|
|
|
#[cfg(target_arch = "arm")]
|
2013-01-29 08:28:08 -06:00
|
|
|
#[cfg(target_arch = "mips")]
|
2013-04-15 10:08:52 -05:00
|
|
|
fn align_of_32() {
|
2013-05-18 21:02:45 -05:00
|
|
|
assert_eq!(pref_align_of::<uint>(), 4u);
|
|
|
|
assert_eq!(pref_align_of::<*uint>(), 4u);
|
2012-01-17 19:28:21 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
#[cfg(target_arch = "x86_64")]
|
2013-04-15 10:08:52 -05:00
|
|
|
fn align_of_64() {
|
2013-05-18 21:02:45 -05:00
|
|
|
assert_eq!(pref_align_of::<uint>(), 8u);
|
|
|
|
assert_eq!(pref_align_of::<*uint>(), 8u);
|
2012-01-17 19:28:21 -06:00
|
|
|
}
|
2012-08-29 16:04:10 -05:00
|
|
|
|
2013-04-19 23:34:18 -05:00
|
|
|
#[test]
|
|
|
|
fn align_of_val_basic() {
|
|
|
|
assert_eq!(pref_align_of_val(&1u8), 1u);
|
|
|
|
assert_eq!(pref_align_of_val(&1u16), 2u);
|
|
|
|
assert_eq!(pref_align_of_val(&1u32), 4u);
|
|
|
|
}
|
|
|
|
|
2012-08-29 16:04:10 -05:00
|
|
|
#[test]
|
2013-04-15 10:08:52 -05:00
|
|
|
fn synthesize_closure() {
|
2013-01-23 13:43:58 -06:00
|
|
|
unsafe {
|
|
|
|
let x = 10;
|
2013-03-07 16:38:38 -06:00
|
|
|
let f: &fn(int) -> int = |y| x + y;
|
2012-08-29 16:04:10 -05:00
|
|
|
|
2013-05-18 21:02:45 -05:00
|
|
|
assert_eq!(f(20), 30);
|
2012-08-29 16:04:10 -05:00
|
|
|
|
2013-02-15 02:51:28 -06:00
|
|
|
let original_closure: Closure = cast::transmute(f);
|
2012-08-29 16:04:10 -05:00
|
|
|
|
2013-01-23 13:43:58 -06:00
|
|
|
let actual_function_pointer = original_closure.code;
|
|
|
|
let environment = original_closure.env;
|
2012-08-29 16:04:10 -05:00
|
|
|
|
2013-01-23 13:43:58 -06:00
|
|
|
let new_closure = Closure {
|
|
|
|
code: actual_function_pointer,
|
|
|
|
env: environment
|
|
|
|
};
|
2012-08-29 16:04:10 -05:00
|
|
|
|
2013-03-07 16:38:38 -06:00
|
|
|
let new_f: &fn(int) -> int = cast::transmute(new_closure);
|
2013-05-18 21:02:45 -05:00
|
|
|
assert_eq!(new_f(20), 30);
|
2013-01-23 13:43:58 -06:00
|
|
|
}
|
2012-08-29 16:04:10 -05:00
|
|
|
}
|
2013-04-23 15:30:58 -05:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
#[should_fail]
|
|
|
|
fn fail_static() { FailWithCause::fail_with("cause", file!(), line!()) }
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
#[should_fail]
|
|
|
|
fn fail_owned() { FailWithCause::fail_with(~"cause", file!(), line!()) }
|
2012-01-17 19:28:21 -06:00
|
|
|
}
|