about summary refs log tree commit diff
path: root/src/doc
diff options
context:
space:
mode:
authorLorenz <lorenzb@student.ethz.ch>2015-05-30 14:26:50 +0200
committerLorenz <lorenzb@student.ethz.ch>2015-05-30 14:26:50 +0200
commitbabb2684d80aee86ba73ffb1af04617fb2943a06 (patch)
tree0fc516c6a9cb136ad5300fa52c84ff8585ec9dd6 /src/doc
parenta2b61e16b13226ba6ae3cecce9752eee6a6999aa (diff)
downloadrust-babb2684d80aee86ba73ffb1af04617fb2943a06.tar.gz
rust-babb2684d80aee86ba73ffb1af04617fb2943a06.zip
Extend rust reference with a section about type coercions
Diffstat (limited to 'src/doc')
-rw-r--r--src/doc/reference.md120
1 files changed, 120 insertions, 0 deletions
diff --git a/src/doc/reference.md b/src/doc/reference.md
index 68c17c04079..cb595c22a25 100644
--- a/src/doc/reference.md
+++ b/src/doc/reference.md
@@ -3628,6 +3628,126 @@ fn bar<'a>() {
 Since `'static` "lives longer" than `'a`, `&'static str` is a subtype of
 `&'a str`.
 
+## Type coercions
+
+Coercions are defined in [RFC401]. A coercion is implicit and has no syntax.
+
+[RFC401]: https://github.com/rust-lang/rfcs/blob/master/text/0401-coercions.md
+
+### Coercion sites
+
+A coercion can only occur at certain coercion sites in a program; these are
+typically places where the desired type is explicit or can be dervied by
+propagation from explicit types (without type inference). Possible coercion
+sites are:
+
+* `let` statements where an explicit type is given.
+
+    In `let _: U = e;`, `e` is coerced to have type `U`.
+
+* `static` and `const` statements (similar to `let` statements).
+
+* arguments for function calls.
+
+    The value being coerced is the
+    actual parameter and it is coerced to the type of the formal parameter. For
+    example, let `foo` be defined as `fn foo(x: U) { ... }` and call it as
+    `foo(e);`. Then `e` is coerced to have type `U`;
+
+* instantiations of struct or variant fields.
+
+    Assume we have a `struct
+    Foo { x: U }` and instantiate it as `Foo { x: e }`. Then `e` is coerced to
+    have type `U`.
+
+* function results (either the final line of a block if it is not semicolon
+terminated or any expression in a `return` statement).
+
+    In `fn foo() -> U { e }`, `e` is coerced to to have type `U`.
+
+If the expression in one of these coercion sites is a coercion-propagating
+expression, then the relevant sub-expressions in that expression are also
+coercion sites. Propagation recurses from these new coercion sites.
+Propagating expressions and their relevant sub-expressions are:
+
+* array literals, where the array has type `[U; n]`. Each sub-expression in
+the array literal is a coercion site for coercion to type `U`.
+
+* array literals with repeating syntax, where the array has type `[U; n]`. The
+repeated sub-expression is a coercion site for coercion to type `U`.
+
+* tuples, where a tuple is a coercion site to type `(U_0, U_1, ..., U_n)`.
+Each sub-expression is a coercion site to the respective type, e.g. the
+zeroth sub-expression is a coercion site to type `U_0`.
+
+* parenthesised sub-expressions (`(e)`). If the expression has type `U`, then
+the sub-expression is a coercion site to `U`.
+
+* blocks. If a block has type `U`, then the last expression in the block (if
+it is not semicolon-terminated) is a coercion site to `U`. This includes
+blocks which are part of control flow statements, such as `if`/`else`, if
+the block has a known type.
+
+### Coercion types
+
+Coercion is allowed between the following types:
+
+* `T` to `U` if `T` is a subtype of `U` (*reflexive case*).
+
+* `T_1` to `T_3` where `T_1` coerces to `T_2` and `T_2` coerces to `T_3`
+(*transitive case*).
+
+    Note that this is not fully supported yet
+
+* `&mut T` to `&T`.
+
+* `*mut T` to `*const T`.
+
+* `&T` to `*const T`.
+
+* `&mut T` to `*mut T`.
+
+* `&T` to `&U` if `T` implements `Deref<Target = U>`. For example:
+    ```
+    use std::ops::Deref;
+
+    struct CharContainer {
+        value: char
+    }
+
+    impl Deref for CharContainer {
+        type Target = char;
+
+        fn deref<'a>(&'a self) -> &'a char {
+            &self.value
+        }
+    }
+
+    fn foo(arg: &char) {}
+
+    fn main() {
+        let x = &mut CharContainer { value: 'y' };
+        foo(x); //&mut CharContainer is coerced to &char.
+    }
+    ```
+* `&mut T` to `&mut U` if `T` implements `DerefMut<Target = U>`.
+
+* TyCtor(`T`) to TyCtor(coerce_inner(`T`)), where TyCtor(`T`) is one of
+    - `&T`
+    - `&mut T`
+    - `*const T`
+    - `*mut T`
+    - `Box<T>`
+
+    and where
+    - coerce_inner(`[T, ..n]`) = `[T]`
+    - coerce_inner(`T`) = `U` where `T` is a concrete type which implements the
+    trait `U`.
+
+    In the future, coerce_inner will be recursively extended to tuples and
+    structs. In addition, coercions from sub-traits to super-traits will be
+    added. See [RFC401] for more details.
+
 # Special traits
 
 Several traits define special evaluation behavior.