//! The AST pointer. //! //! Provides `P`, a frozen owned smart pointer. //! //! # Motivations and benefits //! //! * **Identity**: sharing AST nodes is problematic for the various analysis //! passes (e.g., one may be able to bypass the borrow checker with a shared //! `ExprKind::AddrOf` node taking a mutable borrow). //! //! * **Immutability**: `P` disallows mutating its inner `T`, unlike `Box` //! (unless it contains an `Unsafe` interior, but that may be denied later). //! This mainly prevents mistakes, but also enforces a kind of "purity". //! //! * **Efficiency**: folding can reuse allocation space for `P` and `Vec`, //! the latter even when the input and output types differ (as it would be the //! case with arenas or a GADT AST using type parameters to toggle features). //! //! * **Maintainability**: `P` provides a fixed interface - `Deref`, //! `and_then` and `map` - which can remain fully functional even if the //! implementation changes (using a special thread-local heap, for example). //! Moreover, a switch to, e.g., `P<'a, T>` would be easy and mostly automated. use std::fmt::{self, Debug, Display}; use std::iter::FromIterator; use std::ops::{Deref, DerefMut}; use std::{slice, vec}; use rustc_serialize::{Decodable, Decoder, Encodable, Encoder}; use rustc_data_structures::stable_hasher::{HashStable, StableHasher}; /// An owned smart pointer. pub struct P { ptr: Box, } /// Construct a `P` from a `T` value. #[allow(non_snake_case)] pub fn P(value: T) -> P { P { ptr: Box::new(value) } } impl P { /// Move out of the pointer. /// Intended for chaining transformations not covered by `map`. pub fn and_then(self, f: F) -> U where F: FnOnce(T) -> U, { f(*self.ptr) } /// Equivalent to `and_then(|x| x)`. pub fn into_inner(self) -> T { *self.ptr } /// Produce a new `P` from `self` without reallocating. pub fn map(mut self, f: F) -> P where F: FnOnce(T) -> T, { let x = f(*self.ptr); *self.ptr = x; self } /// Optionally produce a new `P` from `self` without reallocating. pub fn filter_map(mut self, f: F) -> Option> where F: FnOnce(T) -> Option, { *self.ptr = f(*self.ptr)?; Some(self) } } impl Deref for P { type Target = T; fn deref(&self) -> &T { &self.ptr } } impl DerefMut for P { fn deref_mut(&mut self) -> &mut T { &mut self.ptr } } impl Clone for P { fn clone(&self) -> P { P((**self).clone()) } } impl Debug for P { fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { Debug::fmt(&self.ptr, f) } } impl Display for P { fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { Display::fmt(&**self, f) } } impl fmt::Pointer for P { fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { fmt::Pointer::fmt(&self.ptr, f) } } impl> Decodable for P { fn decode(d: &mut D) -> P { P(Decodable::decode(d)) } } impl> Encodable for P { fn encode(&self, s: &mut S) { (**self).encode(s); } } impl P<[T]> { pub const fn new() -> P<[T]> { P { ptr: Box::default() } } #[inline(never)] pub fn from_vec(v: Vec) -> P<[T]> { P { ptr: v.into_boxed_slice() } } #[inline(never)] pub fn into_vec(self) -> Vec { self.ptr.into_vec() } } impl Default for P<[T]> { /// Creates an empty `P<[T]>`. fn default() -> P<[T]> { P::new() } } impl Clone for P<[T]> { fn clone(&self) -> P<[T]> { P::from_vec(self.to_vec()) } } impl From> for P<[T]> { fn from(v: Vec) -> Self { P::from_vec(v) } } impl Into> for P<[T]> { fn into(self) -> Vec { self.into_vec() } } impl FromIterator for P<[T]> { fn from_iter>(iter: I) -> P<[T]> { P::from_vec(iter.into_iter().collect()) } } impl IntoIterator for P<[T]> { type Item = T; type IntoIter = vec::IntoIter; fn into_iter(self) -> Self::IntoIter { self.into_vec().into_iter() } } impl<'a, T> IntoIterator for &'a P<[T]> { type Item = &'a T; type IntoIter = slice::Iter<'a, T>; fn into_iter(self) -> Self::IntoIter { self.ptr.into_iter() } } impl> Encodable for P<[T]> { fn encode(&self, s: &mut S) { Encodable::encode(&**self, s); } } impl> Decodable for P<[T]> { fn decode(d: &mut D) -> P<[T]> { P::from_vec(Decodable::decode(d)) } } impl HashStable for P where T: ?Sized + HashStable, { fn hash_stable(&self, hcx: &mut CTX, hasher: &mut StableHasher) { (**self).hash_stable(hcx, hasher); } }