rust/src/libcore/sys.rs

373 lines
9.6 KiB
Rust
Raw Normal View History

// 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.
//! Misc low level stuff
2012-03-10 02:04:09 -06:00
use option::{Some, None};
use cast;
2012-08-27 18:26:35 -05:00
use cmp::{Eq, Ord};
use gc;
use io;
use libc;
use libc::{c_void, c_char, size_t};
use repr;
use str;
2012-08-27 18:26:35 -05:00
pub type FreeGlue<'self> = &'self fn(*TypeDesc, *c_void);
2012-09-05 23:10:22 -05:00
// Corresponds to runtime type_desc type
pub struct TypeDesc {
size: uint,
2012-09-05 23:10:22 -05:00
align: uint,
take_glue: uint,
drop_glue: uint,
free_glue: uint
// Remaining fields not listed
}
2012-08-29 16:04:10 -05:00
/// The representation of a Rust closure
pub struct Closure {
code: *(),
env: *(),
2012-08-29 16:04:10 -05:00
}
pub mod rusti {
#[abi = "rust-intrinsic"]
pub extern "rust-intrinsic" {
fn get_tydesc<T>() -> *();
fn size_of<T>() -> uint;
fn pref_align_of<T>() -> uint;
fn min_align_of<T>() -> 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<T:Eq>(x1: &T, x2: &T) -> bool {
*x1 == *x2
}
pub fn shape_lt<T:Ord>(x1: &T, x2: &T) -> bool {
*x1 < *x2
}
pub fn shape_le<T:Ord>(x1: &T, x2: &T) -> bool {
2012-08-27 18:26:35 -05:00
*x1 <= *x2
}
/**
* 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)]
pub fn get_type_desc<T>() -> *TypeDesc {
unsafe { rusti::get_tydesc::<T>() as *TypeDesc }
}
/// Returns a pointer to a type descriptor.
#[inline(always)]
pub fn get_type_desc_val<T>(_val: &T) -> *TypeDesc {
get_type_desc::<T>()
}
/// Returns the size of a type
2012-06-14 13:38:45 -05:00
#[inline(always)]
pub fn size_of<T>() -> uint {
unsafe { rusti::size_of::<T>() }
}
/// Returns the size of the type that `_val` points to
#[inline(always)]
pub fn size_of_val<T>(_val: &T) -> uint {
size_of::<T>()
}
/**
* 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<T>() -> uint {
let s = size_of::<T>();
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<T>(_val: &T) -> uint {
nonzero_size_of::<T>()
}
/**
* 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<T>() -> uint {
unsafe { rusti::min_align_of::<T>() }
}
/// Returns the ABI-required minimum alignment of the type of the value that
/// `_val` points to
#[inline(always)]
pub fn min_align_of_val<T>(_val: &T) -> uint {
min_align_of::<T>()
}
/// Returns the preferred alignment of a type
#[inline(always)]
pub fn pref_align_of<T>() -> uint {
unsafe { rusti::pref_align_of::<T>() }
}
/// Returns the preferred alignment of the type of the value that
/// `_val` points to
#[inline(always)]
pub fn pref_align_of_val<T>(_val: &T) -> uint {
pref_align_of::<T>()
}
/// Returns the refcount of a shared box (as just before calling this)
2012-09-11 17:57:32 -05:00
#[inline(always)]
pub fn refcount<T>(t: @T) -> uint {
unsafe {
2013-04-26 16:04:39 -05:00
let ref_ptr: *uint = cast::transmute_copy(&t);
*ref_ptr - 1
}
}
pub fn log_str<T>(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)
}
2012-01-17 19:28:21 -06:00
#[cfg(test)]
mod tests {
use cast;
use sys::*;
2012-01-17 19:28:21 -06:00
#[test]
fn size_of_basic() {
2013-03-28 20:39:09 -05:00
assert!(size_of::<u8>() == 1u);
assert!(size_of::<u16>() == 2u);
assert!(size_of::<u32>() == 4u);
assert!(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")]
fn size_of_32() {
2013-03-28 20:39:09 -05:00
assert!(size_of::<uint>() == 4u);
assert!(size_of::<*uint>() == 4u);
2012-01-17 19:28:21 -06:00
}
#[test]
#[cfg(target_arch = "x86_64")]
fn size_of_64() {
2013-03-28 20:39:09 -05:00
assert!(size_of::<uint>() == 8u);
assert!(size_of::<*uint>() == 8u);
2012-01-17 19:28:21 -06: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);
}
#[test]
fn nonzero_size_of_basic() {
type Z = [i8, ..0];
2013-03-28 20:39:09 -05:00
assert!(size_of::<Z>() == 0u);
assert!(nonzero_size_of::<Z>() == 1u);
assert!(nonzero_size_of::<uint>() == size_of::<uint>());
}
#[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]
fn align_of_basic() {
2013-03-28 20:39:09 -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")]
2013-01-29 08:28:08 -06:00
#[cfg(target_arch = "mips")]
fn align_of_32() {
2013-03-28 20:39:09 -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() {
2013-03-28 20:39:09 -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 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]
fn synthesize_closure() {
unsafe {
let x = 10;
let f: &fn(int) -> int = |y| x + y;
2012-08-29 16:04:10 -05:00
2013-03-28 20:39:09 -05:00
assert!(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
let actual_function_pointer = original_closure.code;
let environment = original_closure.env;
2012-08-29 16:04:10 -05:00
let new_closure = Closure {
code: actual_function_pointer,
env: environment
};
2012-08-29 16:04:10 -05:00
let new_f: &fn(int) -> int = cast::transmute(new_closure);
2013-03-28 20:39:09 -05:00
assert!(new_f(20) == 30);
}
2012-08-29 16:04:10 -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
}
// Local Variables:
// mode: rust;
// fill-column: 78;
// indent-tabs-mode: nil
// c-basic-offset: 4
// buffer-file-coding-system: utf-8-unix
// End: