From 32c360b993d9706d48e1b216b1a35f3bdf422574 Mon Sep 17 00:00:00 2001 From: Aaron Hill Date: Thu, 27 Feb 2020 17:22:32 -0500 Subject: [PATCH 1/2] [perf] Skip attempting to run coerce_unsized on an inference variable See the included comment for a detailed explanation of why this is sound. --- src/librustc_typeck/check/coercion.rs | 30 +++++++++++++++++++++++++++ 1 file changed, 30 insertions(+) diff --git a/src/librustc_typeck/check/coercion.rs b/src/librustc_typeck/check/coercion.rs index 3720b74d92e07..e99470202a0aa 100644 --- a/src/librustc_typeck/check/coercion.rs +++ b/src/librustc_typeck/check/coercion.rs @@ -455,6 +455,36 @@ impl<'f, 'tcx> Coerce<'f, 'tcx> { fn coerce_unsized(&self, source: Ty<'tcx>, target: Ty<'tcx>) -> CoerceResult<'tcx> { debug!("coerce_unsized(source={:?}, target={:?})", source, target); + // These 'if' statements require some explanation. + // The `CoerceUnsized` trait is special - it is only + // possible to write `impl CoerceUnsized for A` where + // A and B have 'matching' fields. This rules out the following + // two types of blanket impls: + // + // `impl CoerceUnsized for SomeType` + // `impl CoerceUnsized for T` + // + // Both of these trigger a special `CoerceUnsized`-related error (E0376) + // + // We can take advantage of this fact to avoid performing unecessary work. + // If either `source` or `target` is a type variable, then any applicable impl + // would need to be generic over the self-type (`impl CoerceUnsized for T`) + // or generic over the `CoerceUnsized` type parameter (`impl CoerceUnsized for + // SomeType`). + // + // However, these are exactly the kinds of impls which are forbidden by + // the compiler! Therefore, we can be sure that coercion will always fail + // when either the source or target type is a type variable. This allows us + // to skip performing any trait selection, and immediately bail out. + if self.shallow_resolve(source).is_ty_var() { + debug!("coerce_unsized: source is a TyVar, bailing out"); + return Err(TypeError::Mismatch); + } + if self.shallow_resolve(target).is_ty_var() { + debug!("coerce_unsized: target is a TyVar, bailing out"); + return Err(TypeError::Mismatch); + } + let traits = (self.tcx.lang_items().unsize_trait(), self.tcx.lang_items().coerce_unsized_trait()); let (unsize_did, coerce_unsized_did) = if let (Some(u), Some(cu)) = traits { From ff6e4ee7ad9b61d1ac1ae85fff14c4dacb66034f Mon Sep 17 00:00:00 2001 From: Aaron Hill Date: Fri, 28 Feb 2020 14:07:51 -0500 Subject: [PATCH 2/2] Use resolved types in body of coerce_unsized --- src/librustc_typeck/check/coercion.rs | 10 +++++++--- 1 file changed, 7 insertions(+), 3 deletions(-) diff --git a/src/librustc_typeck/check/coercion.rs b/src/librustc_typeck/check/coercion.rs index e99470202a0aa..8d02102afc6d8 100644 --- a/src/librustc_typeck/check/coercion.rs +++ b/src/librustc_typeck/check/coercion.rs @@ -452,9 +452,13 @@ impl<'f, 'tcx> Coerce<'f, 'tcx> { // &[T; n] or &mut [T; n] -> &[T] // or &mut [T; n] -> &mut [T] // or &Concrete -> &Trait, etc. - fn coerce_unsized(&self, source: Ty<'tcx>, target: Ty<'tcx>) -> CoerceResult<'tcx> { + fn coerce_unsized(&self, mut source: Ty<'tcx>, mut target: Ty<'tcx>) -> CoerceResult<'tcx> { debug!("coerce_unsized(source={:?}, target={:?})", source, target); + source = self.shallow_resolve(source); + target = self.shallow_resolve(target); + debug!("coerce_unsized: resolved source={:?} target={:?}", source, target); + // These 'if' statements require some explanation. // The `CoerceUnsized` trait is special - it is only // possible to write `impl CoerceUnsized for A` where @@ -476,11 +480,11 @@ impl<'f, 'tcx> Coerce<'f, 'tcx> { // the compiler! Therefore, we can be sure that coercion will always fail // when either the source or target type is a type variable. This allows us // to skip performing any trait selection, and immediately bail out. - if self.shallow_resolve(source).is_ty_var() { + if source.is_ty_var() { debug!("coerce_unsized: source is a TyVar, bailing out"); return Err(TypeError::Mismatch); } - if self.shallow_resolve(target).is_ty_var() { + if target.is_ty_var() { debug!("coerce_unsized: target is a TyVar, bailing out"); return Err(TypeError::Mismatch); }