// 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 or the MIT license // , at your // option. This file may not be copied, modified, or distributed // except according to those terms. //! Misc low level stuff use option::{Some, None}; use cast; use cmp::{Eq, Ord}; use gc; use io; use libc; use libc::{c_void, c_char, size_t}; use repr; use str; pub type FreeGlue<'self> = &'self fn(*TypeDesc, *c_void); // Corresponds to runtime type_desc type pub struct TypeDesc { size: uint, align: uint, take_glue: uint, drop_glue: uint, free_glue: uint // Remaining fields not listed } /// The representation of a Rust closure pub struct Closure { code: *(), env: *(), } pub mod rusti { #[abi = "rust-intrinsic"] pub extern "rust-intrinsic" { fn get_tydesc() -> *(); fn size_of() -> uint; fn pref_align_of() -> uint; fn min_align_of() -> uint; } } 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); } } /// Compares contents of two pointers using the default method. /// Equivalent to `*x1 == *x2`. Useful for hashtables. pub fn shape_eq(x1: &T, x2: &T) -> bool { *x1 == *x2 } pub fn shape_lt(x1: &T, x2: &T) -> bool { *x1 < *x2 } pub fn shape_le(x1: &T, x2: &T) -> bool { *x1 <= *x2 } /** * Returns a pointer to a type descriptor. * * Useful for calling certain function in the Rust runtime or otherwise * performing dark magick. */ #[inline(always)] pub fn get_type_desc() -> *TypeDesc { unsafe { rusti::get_tydesc::() as *TypeDesc } } /// Returns a pointer to a type descriptor. #[inline(always)] pub fn get_type_desc_val(_val: &T) -> *TypeDesc { get_type_desc::() } /// Returns the size of a type #[inline(always)] pub fn size_of() -> uint { unsafe { rusti::size_of::() } } /// Returns the size of the type that `_val` points to #[inline(always)] pub fn size_of_val(_val: &T) -> uint { size_of::() } /** * 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 fn nonzero_size_of() -> uint { let s = size_of::(); if s == 0 { 1 } else { s } } /// Returns the size of the type of the value that `_val` points to #[inline(always)] pub fn nonzero_size_of_val(_val: &T) -> uint { nonzero_size_of::() } /** * 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. */ #[inline(always)] pub fn min_align_of() -> uint { unsafe { rusti::min_align_of::() } } /// Returns the ABI-required minimum alignment of the type of the value that /// `_val` points to #[inline(always)] pub fn min_align_of_val(_val: &T) -> uint { min_align_of::() } /// Returns the preferred alignment of a type #[inline(always)] pub fn pref_align_of() -> uint { unsafe { rusti::pref_align_of::() } } /// Returns the preferred alignment of the type of the value that /// `_val` points to #[inline(always)] pub fn pref_align_of_val(_val: &T) -> uint { pref_align_of::() } /// Returns the refcount of a shared box (as just before calling this) #[inline(always)] pub fn refcount(t: @T) -> uint { unsafe { let ref_ptr: *uint = cast::transmute_copy(&t); *ref_ptr - 1 } } pub fn log_str(t: &T) -> ~str { do io::with_str_writer |wr| { repr::write_repr(wr, t) } } /// 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) } } } } } // NOTE: remove function after snapshot #[cfg(stage0)] pub fn begin_unwind(msg: ~str, file: ~str, line: uint) -> ! { use rt::{context, OldTaskContext}; use rt::local_services::unsafe_borrow_local_services; match context() { OldTaskContext => { 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) } } } } _ => { gc::cleanup_stack_for_failure(); unsafe { let local_services = unsafe_borrow_local_services(); match local_services.unwinder { Some(ref mut unwinder) => unwinder.begin_unwind(), None => abort!("failure without unwinder. aborting process") } } } } } // FIXME #4427: Temporary until rt::rt_fail_ goes away pub 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(()) } } // NOTE: remove function after snapshot #[cfg(stage0)] pub fn fail_assert(msg: &str, file: &str, line: uint) -> ! { let (msg, file) = (msg.to_owned(), file.to_owned()); begin_unwind(~"assertion failed: " + msg, file, line) } #[cfg(test)] mod tests { use cast; use sys::*; #[test] fn size_of_basic() { assert!(size_of::() == 1u); assert!(size_of::() == 2u); assert!(size_of::() == 4u); assert!(size_of::() == 8u); } #[test] #[cfg(target_arch = "x86")] #[cfg(target_arch = "arm")] #[cfg(target_arch = "mips")] fn size_of_32() { assert!(size_of::() == 4u); assert!(size_of::<*uint>() == 4u); } #[test] #[cfg(target_arch = "x86_64")] fn size_of_64() { assert!(size_of::() == 8u); assert!(size_of::<*uint>() == 8u); } #[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); } #[test] fn nonzero_size_of_basic() { type Z = [i8, ..0]; assert!(size_of::() == 0u); assert!(nonzero_size_of::() == 1u); assert!(nonzero_size_of::() == size_of::()); } #[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)); } #[test] fn align_of_basic() { assert!(pref_align_of::() == 1u); assert!(pref_align_of::() == 2u); assert!(pref_align_of::() == 4u); } #[test] #[cfg(target_arch = "x86")] #[cfg(target_arch = "arm")] #[cfg(target_arch = "mips")] fn align_of_32() { assert!(pref_align_of::() == 4u); assert!(pref_align_of::<*uint>() == 4u); } #[test] #[cfg(target_arch = "x86_64")] fn align_of_64() { assert!(pref_align_of::() == 8u); assert!(pref_align_of::<*uint>() == 8u); } #[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); } #[test] fn synthesize_closure() { unsafe { let x = 10; let f: &fn(int) -> int = |y| x + y; assert!(f(20) == 30); let original_closure: Closure = cast::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 = cast::transmute(new_closure); assert!(new_f(20) == 30); } } #[test] #[should_fail] fn fail_static() { FailWithCause::fail_with("cause", file!(), line!()) } #[test] #[should_fail] fn fail_owned() { FailWithCause::fail_with(~"cause", file!(), line!()) } } // Local Variables: // mode: rust; // fill-column: 78; // indent-tabs-mode: nil // c-basic-offset: 4 // buffer-file-coding-system: utf-8-unix // End: