2013-07-21 17:20:52 -07:00
|
|
|
// Copyright 2013 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.
|
|
|
|
|
2014-10-27 15:37:07 -07:00
|
|
|
#![allow(missing_docs)]
|
2014-04-30 20:38:31 -07:00
|
|
|
#![experimental]
|
2014-02-16 00:04:33 -08:00
|
|
|
|
2014-03-03 01:01:13 +01:00
|
|
|
//! Contains struct definitions for the layout of compiler built-in types.
|
|
|
|
//!
|
|
|
|
//! They can be used as targets of transmutes in unsafe code for manipulating
|
|
|
|
//! the raw representations directly.
|
|
|
|
//!
|
2014-04-21 00:49:39 -04:00
|
|
|
//! Their definition should always match the ABI defined in `rustc::back::abi`.
|
2014-03-03 01:01:13 +01:00
|
|
|
|
2015-01-07 11:33:42 +13:00
|
|
|
use marker::Copy;
|
core: Remove the cast module
This commit revisits the `cast` module in libcore and libstd, and scrutinizes
all functions inside of it. The result was to remove the `cast` module entirely,
folding all functionality into the `mem` module. Specifically, this is the fate
of each function in the `cast` module.
* transmute - This function was moved to `mem`, but it is now marked as
#[unstable]. This is due to planned changes to the `transmute`
function and how it can be invoked (see the #[unstable] comment).
For more information, see RFC 5 and #12898
* transmute_copy - This function was moved to `mem`, with clarification that is
is not an error to invoke it with T/U that are different
sizes, but rather that it is strongly discouraged. This
function is now #[stable]
* forget - This function was moved to `mem` and marked #[stable]
* bump_box_refcount - This function was removed due to the deprecation of
managed boxes as well as its questionable utility.
* transmute_mut - This function was previously deprecated, and removed as part
of this commit.
* transmute_mut_unsafe - This function doesn't serve much of a purpose when it
can be achieved with an `as` in safe code, so it was
removed.
* transmute_lifetime - This function was removed because it is likely a strong
indication that code is incorrect in the first place.
* transmute_mut_lifetime - This function was removed for the same reasons as
`transmute_lifetime`
* copy_lifetime - This function was moved to `mem`, but it is marked
`#[unstable]` now due to the likelihood of being removed in
the future if it is found to not be very useful.
* copy_mut_lifetime - This function was also moved to `mem`, but had the same
treatment as `copy_lifetime`.
* copy_lifetime_vec - This function was removed because it is not used today,
and its existence is not necessary with DST
(copy_lifetime will suffice).
In summary, the cast module was stripped down to these functions, and then the
functions were moved to the `mem` module.
transmute - #[unstable]
transmute_copy - #[stable]
forget - #[stable]
copy_lifetime - #[unstable]
copy_mut_lifetime - #[unstable]
[breaking-change]
2014-05-09 10:34:51 -07:00
|
|
|
use mem;
|
2013-07-21 17:20:52 -07:00
|
|
|
|
|
|
|
/// The representation of a Rust slice
|
2014-10-25 13:28:17 -04:00
|
|
|
#[repr(C)]
|
2013-07-21 17:20:52 -07:00
|
|
|
pub struct Slice<T> {
|
2014-06-25 12:47:34 -07:00
|
|
|
pub data: *const T,
|
2014-03-27 15:09:47 -07:00
|
|
|
pub len: uint,
|
2013-07-21 17:20:52 -07:00
|
|
|
}
|
|
|
|
|
librustc: Make `Copy` opt-in.
This change makes the compiler no longer infer whether types (structures
and enumerations) implement the `Copy` trait (and thus are implicitly
copyable). Rather, you must implement `Copy` yourself via `impl Copy for
MyType {}`.
A new warning has been added, `missing_copy_implementations`, to warn
you if a non-generic public type has been added that could have
implemented `Copy` but didn't.
For convenience, you may *temporarily* opt out of this behavior by using
`#![feature(opt_out_copy)]`. Note though that this feature gate will never be
accepted and will be removed by the time that 1.0 is released, so you should
transition your code away from using it.
This breaks code like:
#[deriving(Show)]
struct Point2D {
x: int,
y: int,
}
fn main() {
let mypoint = Point2D {
x: 1,
y: 1,
};
let otherpoint = mypoint;
println!("{}{}", mypoint, otherpoint);
}
Change this code to:
#[deriving(Show)]
struct Point2D {
x: int,
y: int,
}
impl Copy for Point2D {}
fn main() {
let mypoint = Point2D {
x: 1,
y: 1,
};
let otherpoint = mypoint;
println!("{}{}", mypoint, otherpoint);
}
This is the backwards-incompatible part of #13231.
Part of RFC #3.
[breaking-change]
2014-12-05 17:01:33 -08:00
|
|
|
impl<T> Copy for Slice<T> {}
|
|
|
|
|
2013-07-21 17:20:52 -07:00
|
|
|
/// The representation of a Rust closure
|
2014-10-25 13:28:17 -04:00
|
|
|
#[repr(C)]
|
2015-01-03 22:54:18 -05:00
|
|
|
#[derive(Copy)]
|
2013-07-21 17:20:52 -07:00
|
|
|
pub struct Closure {
|
2014-06-25 12:47:34 -07:00
|
|
|
pub code: *mut (),
|
|
|
|
pub env: *mut (),
|
2013-07-21 17:20:52 -07:00
|
|
|
}
|
|
|
|
|
2014-03-03 01:01:13 +01:00
|
|
|
/// The representation of a Rust trait object.
|
|
|
|
///
|
|
|
|
/// This struct does not have a `Repr` implementation
|
|
|
|
/// because there is no way to refer to all trait objects generically.
|
2014-10-25 13:28:17 -04:00
|
|
|
#[repr(C)]
|
2015-01-03 22:54:18 -05:00
|
|
|
#[derive(Copy)]
|
DST coercions and DST structs
[breaking-change]
1. The internal layout for traits has changed from (vtable, data) to (data, vtable). If you were relying on this in unsafe transmutes, you might get some very weird and apparently unrelated errors. You should not be doing this! Prefer not to do this at all, but if you must, you should use raw::TraitObject rather than hardcoding rustc's internal representation into your code.
2. The minimal type of reference-to-vec-literals (e.g., `&[1, 2, 3]`) is now a fixed size vec (e.g., `&[int, ..3]`) where it used to be an unsized vec (e.g., `&[int]`). If you want the unszied type, you must explicitly give the type (e.g., `let x: &[_] = &[1, 2, 3]`). Note in particular where multiple blocks must have the same type (e.g., if and else clauses, vec elements), the compiler will not coerce to the unsized type without a hint. E.g., `[&[1], &[1, 2]]` used to be a valid expression of type '[&[int]]'. It no longer type checks since the first element now has type `&[int, ..1]` and the second has type &[int, ..2]` which are incompatible.
3. The type of blocks (including functions) must be coercible to the expected type (used to be a subtype). Mostly this makes things more flexible and not less (in particular, in the case of coercing function bodies to the return type). However, in some rare cases, this is less flexible. TBH, I'm not exactly sure of the exact effects. I think the change causes us to resolve inferred type variables slightly earlier which might make us slightly more restrictive. Possibly it only affects blocks with unreachable code. E.g., `if ... { fail!(); "Hello" }` used to type check, it no longer does. The fix is to add a semicolon after the string.
2014-08-04 14:20:11 +02:00
|
|
|
pub struct TraitObject {
|
2014-08-06 11:59:40 +02:00
|
|
|
pub data: *mut (),
|
|
|
|
pub vtable: *mut (),
|
DST coercions and DST structs
[breaking-change]
1. The internal layout for traits has changed from (vtable, data) to (data, vtable). If you were relying on this in unsafe transmutes, you might get some very weird and apparently unrelated errors. You should not be doing this! Prefer not to do this at all, but if you must, you should use raw::TraitObject rather than hardcoding rustc's internal representation into your code.
2. The minimal type of reference-to-vec-literals (e.g., `&[1, 2, 3]`) is now a fixed size vec (e.g., `&[int, ..3]`) where it used to be an unsized vec (e.g., `&[int]`). If you want the unszied type, you must explicitly give the type (e.g., `let x: &[_] = &[1, 2, 3]`). Note in particular where multiple blocks must have the same type (e.g., if and else clauses, vec elements), the compiler will not coerce to the unsized type without a hint. E.g., `[&[1], &[1, 2]]` used to be a valid expression of type '[&[int]]'. It no longer type checks since the first element now has type `&[int, ..1]` and the second has type &[int, ..2]` which are incompatible.
3. The type of blocks (including functions) must be coercible to the expected type (used to be a subtype). Mostly this makes things more flexible and not less (in particular, in the case of coercing function bodies to the return type). However, in some rare cases, this is less flexible. TBH, I'm not exactly sure of the exact effects. I think the change causes us to resolve inferred type variables slightly earlier which might make us slightly more restrictive. Possibly it only affects blocks with unreachable code. E.g., `if ... { fail!(); "Hello" }` used to type check, it no longer does. The fix is to add a semicolon after the string.
2014-08-04 14:20:11 +02:00
|
|
|
}
|
2014-03-03 01:01:13 +01:00
|
|
|
|
2013-07-21 17:20:52 -07:00
|
|
|
/// This trait is meant to map equivalences between raw structs and their
|
|
|
|
/// corresponding rust values.
|
2015-01-04 21:39:02 -05:00
|
|
|
pub trait Repr<T> {
|
2013-07-21 17:20:52 -07:00
|
|
|
/// This function "unwraps" a rust value (without consuming it) into its raw
|
|
|
|
/// struct representation. This can be used to read/write different values
|
|
|
|
/// for the struct. This is a safe method because by default it does not
|
2014-03-03 01:01:13 +01:00
|
|
|
/// enable write-access to the fields of the return value in safe code.
|
2013-09-06 22:29:29 -07:00
|
|
|
#[inline]
|
2014-10-23 10:43:18 -05:00
|
|
|
fn repr(&self) -> T { unsafe { mem::transmute_copy(&self) } }
|
2013-07-21 17:20:52 -07:00
|
|
|
}
|
|
|
|
|
2014-10-23 10:43:18 -05:00
|
|
|
impl<T> Repr<Slice<T>> for [T] {}
|
|
|
|
impl Repr<Slice<u8>> for str {}
|