Move unsafety out of the subtyping relation and into coercion.
This commit is contained in:
parent
1b0f0ad280
commit
0947f4076d
19 changed files with 173 additions and 59 deletions
|
|
@ -993,8 +993,14 @@ impl<'a, 'tcx> rbml_writer_helpers<'tcx> for Encoder<'a> {
|
|||
})
|
||||
}
|
||||
|
||||
ty::AdjustUnsafeFnPointer => {
|
||||
this.emit_enum_variant("AdjustUnsafeFnPointer", 2, 0, |_| {
|
||||
Ok(())
|
||||
})
|
||||
}
|
||||
|
||||
ty::AdjustDerefRef(ref auto_deref_ref) => {
|
||||
this.emit_enum_variant("AdjustDerefRef", 2, 2, |this| {
|
||||
this.emit_enum_variant("AdjustDerefRef", 3, 2, |this| {
|
||||
this.emit_enum_variant_arg(0,
|
||||
|this| Ok(this.emit_auto_deref_ref(ecx, auto_deref_ref)))
|
||||
})
|
||||
|
|
@ -1619,6 +1625,9 @@ impl<'a, 'tcx> rbml_decoder_decoder_helpers<'tcx> for reader::Decoder<'a> {
|
|||
ty::AdjustReifyFnPointer(def_id)
|
||||
}
|
||||
2 => {
|
||||
ty::AdjustUnsafeFnPointer
|
||||
}
|
||||
3 => {
|
||||
let auto_deref_ref: ty::AutoDerefRef =
|
||||
this.read_enum_variant_arg(0,
|
||||
|this| Ok(this.read_auto_deref_ref(dcx))).unwrap();
|
||||
|
|
|
|||
|
|
@ -790,7 +790,8 @@ impl<'d,'t,'tcx,TYPER:mc::Typer<'tcx>> ExprUseVisitor<'d,'t,'tcx,TYPER> {
|
|||
None => { }
|
||||
Some(adjustment) => {
|
||||
match *adjustment {
|
||||
ty::AdjustReifyFnPointer(..) => {
|
||||
ty::AdjustReifyFnPointer(..) |
|
||||
ty::AdjustUnsafeFnPointer(..) => {
|
||||
// Creating a closure/fn-pointer consumes the
|
||||
// input and stores it into the resulting
|
||||
// rvalue.
|
||||
|
|
|
|||
|
|
@ -33,8 +33,6 @@ use middle::infer::{cres};
|
|||
use middle::infer::type_variable::{BiTo};
|
||||
use util::ppaux::{Repr};
|
||||
|
||||
use syntax::ast::{Unsafety};
|
||||
|
||||
pub struct Bivariate<'f, 'tcx: 'f> {
|
||||
fields: CombineFields<'f, 'tcx>
|
||||
}
|
||||
|
|
@ -74,14 +72,6 @@ impl<'f, 'tcx> Combine<'tcx> for Bivariate<'f, 'tcx> {
|
|||
Ok(a)
|
||||
}
|
||||
|
||||
fn unsafeties(&self, a: Unsafety, b: Unsafety) -> cres<'tcx, Unsafety> {
|
||||
if a != b {
|
||||
Err(ty::terr_unsafety_mismatch(expected_found(self, a, b)))
|
||||
} else {
|
||||
Ok(a)
|
||||
}
|
||||
}
|
||||
|
||||
fn builtin_bounds(&self,
|
||||
a: BuiltinBounds,
|
||||
b: BuiltinBounds)
|
||||
|
|
|
|||
|
|
@ -263,7 +263,13 @@ pub trait Combine<'tcx> : Sized {
|
|||
self.tys_with_variance(ty::Contravariant, a, b).and_then(|t| Ok(t))
|
||||
}
|
||||
|
||||
fn unsafeties(&self, a: Unsafety, b: Unsafety) -> cres<'tcx, Unsafety>;
|
||||
fn unsafeties(&self, a: Unsafety, b: Unsafety) -> cres<'tcx, Unsafety> {
|
||||
if a != b {
|
||||
Err(ty::terr_unsafety_mismatch(expected_found(self, a, b)))
|
||||
} else {
|
||||
Ok(a)
|
||||
}
|
||||
}
|
||||
|
||||
fn abi(&self, a: abi::Abi, b: abi::Abi) -> cres<'tcx, abi::Abi> {
|
||||
if a == b {
|
||||
|
|
|
|||
|
|
@ -16,8 +16,6 @@ use middle::infer::{Subtype};
|
|||
use middle::infer::type_variable::{EqTo};
|
||||
use util::ppaux::{Repr};
|
||||
|
||||
use syntax::ast::Unsafety;
|
||||
|
||||
pub struct Equate<'f, 'tcx: 'f> {
|
||||
fields: CombineFields<'f, 'tcx>
|
||||
}
|
||||
|
|
@ -54,15 +52,6 @@ impl<'f, 'tcx> Combine<'tcx> for Equate<'f, 'tcx> {
|
|||
Ok(a)
|
||||
}
|
||||
|
||||
|
||||
fn unsafeties(&self, a: Unsafety, b: Unsafety) -> cres<'tcx, Unsafety> {
|
||||
if a != b {
|
||||
Err(ty::terr_unsafety_mismatch(expected_found(self, a, b)))
|
||||
} else {
|
||||
Ok(a)
|
||||
}
|
||||
}
|
||||
|
||||
fn tys(&self, a: Ty<'tcx>, b: Ty<'tcx>) -> cres<'tcx, Ty<'tcx>> {
|
||||
debug!("{}.tys({}, {})", self.tag(),
|
||||
a.repr(self.fields.infcx.tcx), b.repr(self.fields.infcx.tcx));
|
||||
|
|
|
|||
|
|
@ -15,8 +15,6 @@ use super::{cres};
|
|||
use super::Subtype;
|
||||
|
||||
use middle::ty::{self, Ty};
|
||||
use syntax::ast::{MutImmutable, MutMutable, Unsafety};
|
||||
use util::ppaux::mt_to_string;
|
||||
use util::ppaux::Repr;
|
||||
|
||||
/// "Greatest lower bound" (common subtype)
|
||||
|
|
@ -55,14 +53,6 @@ impl<'f, 'tcx> Combine<'tcx> for Glb<'f, 'tcx> {
|
|||
}
|
||||
}
|
||||
|
||||
|
||||
fn unsafeties(&self, a: Unsafety, b: Unsafety) -> cres<'tcx, Unsafety> {
|
||||
match (a, b) {
|
||||
(Unsafety::Normal, _) | (_, Unsafety::Normal) => Ok(Unsafety::Normal),
|
||||
(Unsafety::Unsafe, Unsafety::Unsafe) => Ok(Unsafety::Unsafe)
|
||||
}
|
||||
}
|
||||
|
||||
fn regions(&self, a: ty::Region, b: ty::Region) -> cres<'tcx, ty::Region> {
|
||||
debug!("{}.regions({}, {})",
|
||||
self.tag(),
|
||||
|
|
|
|||
|
|
@ -15,8 +15,6 @@ use super::{cres};
|
|||
use super::{Subtype};
|
||||
|
||||
use middle::ty::{self, Ty};
|
||||
use syntax::ast::{MutMutable, MutImmutable, Unsafety};
|
||||
use util::ppaux::mt_to_string;
|
||||
use util::ppaux::Repr;
|
||||
|
||||
/// "Least upper bound" (common supertype)
|
||||
|
|
@ -55,13 +53,6 @@ impl<'f, 'tcx> Combine<'tcx> for Lub<'f, 'tcx> {
|
|||
}
|
||||
}
|
||||
|
||||
fn unsafeties(&self, a: Unsafety, b: Unsafety) -> cres<'tcx, Unsafety> {
|
||||
match (a, b) {
|
||||
(Unsafety::Unsafe, _) | (_, Unsafety::Unsafe) => Ok(Unsafety::Unsafe),
|
||||
(Unsafety::Normal, Unsafety::Normal) => Ok(Unsafety::Normal),
|
||||
}
|
||||
}
|
||||
|
||||
fn regions(&self, a: ty::Region, b: ty::Region) -> cres<'tcx, ty::Region> {
|
||||
debug!("{}.regions({}, {})",
|
||||
self.tag(),
|
||||
|
|
|
|||
|
|
@ -9,7 +9,7 @@
|
|||
// except according to those terms.
|
||||
|
||||
use super::combine::*;
|
||||
use super::{cres, CresCompare};
|
||||
use super::{cres};
|
||||
use super::higher_ranked::HigherRankedRelations;
|
||||
use super::{Subtype};
|
||||
use super::type_variable::{SubtypeOf, SupertypeOf};
|
||||
|
|
@ -18,9 +18,6 @@ use middle::ty::{self, Ty};
|
|||
use middle::ty::TyVar;
|
||||
use util::ppaux::{Repr};
|
||||
|
||||
use syntax::ast::{MutImmutable, MutMutable, Unsafety};
|
||||
|
||||
|
||||
/// "Greatest lower bound" (common subtype)
|
||||
pub struct Sub<'f, 'tcx: 'f> {
|
||||
fields: CombineFields<'f, 'tcx>
|
||||
|
|
@ -66,12 +63,6 @@ impl<'f, 'tcx> Combine<'tcx> for Sub<'f, 'tcx> {
|
|||
Ok(a)
|
||||
}
|
||||
|
||||
fn unsafeties(&self, a: Unsafety, b: Unsafety) -> cres<'tcx, Unsafety> {
|
||||
self.lub().unsafeties(a, b).compare(b, || {
|
||||
ty::terr_unsafety_mismatch(expected_found(self, a, b))
|
||||
})
|
||||
}
|
||||
|
||||
fn tys(&self, a: Ty<'tcx>, b: Ty<'tcx>) -> cres<'tcx, Ty<'tcx>> {
|
||||
debug!("{}.tys({}, {})", self.tag(),
|
||||
a.repr(self.tcx()), b.repr(self.tcx()));
|
||||
|
|
|
|||
|
|
@ -428,7 +428,8 @@ impl<'t,'tcx,TYPER:Typer<'tcx>> MemCategorizationContext<'t,TYPER> {
|
|||
|
||||
Some(adjustment) => {
|
||||
match *adjustment {
|
||||
ty::AdjustReifyFnPointer(..) => {
|
||||
ty::AdjustReifyFnPointer(..) |
|
||||
ty::AdjustUnsafeFnPointer(..) => {
|
||||
debug!("cat_expr(AdjustReifyFnPointer): {}",
|
||||
expr.repr(self.tcx()));
|
||||
// Convert a bare fn to a closure by adding NULL env.
|
||||
|
|
|
|||
|
|
@ -282,6 +282,7 @@ pub enum Variance {
|
|||
#[derive(Clone, Debug)]
|
||||
pub enum AutoAdjustment<'tcx> {
|
||||
AdjustReifyFnPointer(ast::DefId), // go from a fn-item type to a fn-pointer type
|
||||
AdjustUnsafeFnPointer, // go from a safe fn pointer to an unsafe fn pointer
|
||||
AdjustDerefRef(AutoDerefRef<'tcx>)
|
||||
}
|
||||
|
||||
|
|
@ -2637,6 +2638,17 @@ impl<'tcx> ctxt<'tcx> {
|
|||
substs
|
||||
}
|
||||
|
||||
/// Create an unsafe fn ty based on a safe fn ty.
|
||||
pub fn safe_to_unsafe_fn_ty(&self, bare_fn: &BareFnTy<'tcx>) -> Ty<'tcx> {
|
||||
assert_eq!(bare_fn.unsafety, ast::Unsafety::Normal);
|
||||
let unsafe_fn_ty_a = self.mk_bare_fn(ty::BareFnTy {
|
||||
unsafety: ast::Unsafety::Unsafe,
|
||||
abi: bare_fn.abi,
|
||||
sig: bare_fn.sig.clone()
|
||||
});
|
||||
ty::mk_bare_fn(self, None, unsafe_fn_ty_a)
|
||||
}
|
||||
|
||||
pub fn mk_bare_fn(&self, bare_fn: BareFnTy<'tcx>) -> &'tcx BareFnTy<'tcx> {
|
||||
if let Some(bare_fn) = self.bare_fn_interner.borrow().get(&bare_fn) {
|
||||
return *bare_fn;
|
||||
|
|
@ -4526,6 +4538,18 @@ pub fn adjust_ty<'tcx, F>(cx: &ctxt<'tcx>,
|
|||
}
|
||||
}
|
||||
|
||||
AdjustUnsafeFnPointer => {
|
||||
match unadjusted_ty.sty {
|
||||
ty::ty_bare_fn(None, b) => cx.safe_to_unsafe_fn_ty(b),
|
||||
ref b => {
|
||||
cx.sess.bug(
|
||||
&format!("AdjustReifyFnPointer adjustment on non-fn-item: \
|
||||
{:?}",
|
||||
b));
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
AdjustDerefRef(ref adj) => {
|
||||
let mut adjusted_ty = unadjusted_ty;
|
||||
|
||||
|
|
@ -6695,6 +6719,7 @@ impl<'tcx> AutoAdjustment<'tcx> {
|
|||
pub fn is_identity(&self) -> bool {
|
||||
match *self {
|
||||
AdjustReifyFnPointer(..) => false,
|
||||
AdjustUnsafeFnPointer(..) => false,
|
||||
AdjustDerefRef(ref r) => r.is_identity(),
|
||||
}
|
||||
}
|
||||
|
|
@ -6844,6 +6869,9 @@ impl<'tcx> Repr<'tcx> for AutoAdjustment<'tcx> {
|
|||
AdjustReifyFnPointer(def_id) => {
|
||||
format!("AdjustReifyFnPointer({})", def_id.repr(tcx))
|
||||
}
|
||||
AdjustUnsafeFnPointer => {
|
||||
format!("AdjustUnsafeFnPointer")
|
||||
}
|
||||
AdjustDerefRef(ref data) => {
|
||||
data.repr(tcx)
|
||||
}
|
||||
|
|
|
|||
Loading…
Add table
Add a link
Reference in a new issue