Check hidden types in dead code

This commit is contained in:
Oli Scherer 2022-10-01 11:08:58 +00:00
parent 70d39abbc2
commit f85d3a7e33
5 changed files with 55 additions and 23 deletions

View File

@ -560,13 +560,11 @@ impl<'cx, 'tcx> WritebackCx<'cx, 'tcx> {
continue; continue;
} }
let hidden_type = hidden_type let hidden_type = hidden_type.remap_generic_params_to_declaration_params(
.remap_generic_params_to_declaration_params( opaque_type_key,
opaque_type_key, self.fcx.infcx.tcx,
self.fcx.infcx.tcx, true,
true, );
)
.ty;
self.typeck_results.concrete_opaque_types.insert(opaque_type_key.def_id, hidden_type); self.typeck_results.concrete_opaque_types.insert(opaque_type_key.def_id, hidden_type);
} }

View File

@ -564,6 +564,11 @@ fn find_opaque_ty_constraints_for_tait(tcx: TyCtxt<'_>, def_id: LocalDefId) -> T
/// checked against it (we also carry the span of that first /// checked against it (we also carry the span of that first
/// type). /// type).
found: Option<ty::OpaqueHiddenType<'tcx>>, found: Option<ty::OpaqueHiddenType<'tcx>>,
/// In the presence of dead code, typeck may figure out a hidden type
/// while borrowck will now. We collect these cases here and check at
/// the end that we actually found a type that matches (modulo regions).
typeck_types: Vec<ty::OpaqueHiddenType<'tcx>>,
} }
impl ConstraintLocator<'_> { impl ConstraintLocator<'_> {
@ -590,18 +595,23 @@ fn find_opaque_ty_constraints_for_tait(tcx: TyCtxt<'_>, def_id: LocalDefId) -> T
self.found = Some(ty::OpaqueHiddenType { span: DUMMY_SP, ty: self.tcx.ty_error() }); self.found = Some(ty::OpaqueHiddenType { span: DUMMY_SP, ty: self.tcx.ty_error() });
return; return;
} }
if !tables.concrete_opaque_types.contains_key(&self.def_id) { let Some(&typeck_hidden_ty) = tables.concrete_opaque_types.get(&self.def_id) else {
debug!("no constraints in typeck results"); debug!("no constraints in typeck results");
return; return;
};
if self.typeck_types.iter().all(|prev| prev.ty != typeck_hidden_ty.ty) {
self.typeck_types.push(typeck_hidden_ty);
} }
// Use borrowck to get the type with unerased regions. // Use borrowck to get the type with unerased regions.
let concrete_opaque_types = &self.tcx.mir_borrowck(item_def_id).concrete_opaque_types; let concrete_opaque_types = &self.tcx.mir_borrowck(item_def_id).concrete_opaque_types;
debug!(?concrete_opaque_types); debug!(?concrete_opaque_types);
if let Some(&concrete_type) = concrete_opaque_types.get(&self.def_id) { if let Some(&concrete_type) = concrete_opaque_types.get(&self.def_id) {
debug!(?concrete_type, "found constraint"); debug!(?concrete_type, "found constraint");
if let Some(prev) = self.found { if let Some(prev) = &mut self.found {
if concrete_type.ty != prev.ty && !(concrete_type, prev).references_error() { if concrete_type.ty != prev.ty && !(concrete_type, prev.ty).references_error() {
prev.report_mismatch(&concrete_type, self.tcx); prev.report_mismatch(&concrete_type, self.tcx);
prev.ty = self.tcx.ty_error();
} }
} else { } else {
self.found = Some(concrete_type); self.found = Some(concrete_type);
@ -648,7 +658,7 @@ fn find_opaque_ty_constraints_for_tait(tcx: TyCtxt<'_>, def_id: LocalDefId) -> T
let hir_id = tcx.hir().local_def_id_to_hir_id(def_id); let hir_id = tcx.hir().local_def_id_to_hir_id(def_id);
let scope = tcx.hir().get_defining_scope(hir_id); let scope = tcx.hir().get_defining_scope(hir_id);
let mut locator = ConstraintLocator { def_id: def_id, tcx, found: None }; let mut locator = ConstraintLocator { def_id: def_id, tcx, found: None, typeck_types: vec![] };
debug!(?scope); debug!(?scope);
@ -678,16 +688,26 @@ fn find_opaque_ty_constraints_for_tait(tcx: TyCtxt<'_>, def_id: LocalDefId) -> T
} }
} }
match locator.found { let Some(hidden) = locator.found else {
Some(hidden) => hidden.ty, tcx.sess.emit_err(UnconstrainedOpaqueType {
None => { span: tcx.def_span(def_id),
tcx.sess.emit_err(UnconstrainedOpaqueType { name: tcx.item_name(tcx.local_parent(def_id).to_def_id()),
span: tcx.def_span(def_id), });
name: tcx.item_name(tcx.local_parent(def_id).to_def_id()), return tcx.ty_error();
}); };
tcx.ty_error()
// Only check against typeck if we didn't already error
if !hidden.ty.references_error() {
for concrete_type in locator.typeck_types {
if tcx.erase_regions(concrete_type.ty) != tcx.erase_regions(hidden.ty)
&& !(concrete_type, hidden).references_error()
{
hidden.report_mismatch(&concrete_type, tcx);
}
} }
} }
hidden.ty
} }
fn find_opaque_ty_constraints_for_rpit( fn find_opaque_ty_constraints_for_rpit(
@ -788,7 +808,7 @@ fn find_opaque_ty_constraints_for_rpit(
// the `concrete_opaque_types` table. // the `concrete_opaque_types` table.
tcx.ty_error() tcx.ty_error()
} else { } else {
table.concrete_opaque_types.get(&def_id).copied().unwrap_or_else(|| { table.concrete_opaque_types.get(&def_id).map(|ty| ty.ty).unwrap_or_else(|| {
// We failed to resolve the opaque type or it // We failed to resolve the opaque type or it
// resolves to itself. We interpret this as the // resolves to itself. We interpret this as the
// no values of the hidden type ever being constructed, // no values of the hidden type ever being constructed,

View File

@ -542,7 +542,7 @@ pub struct TypeckResults<'tcx> {
/// by this function. We also store the /// by this function. We also store the
/// type here, so that mir-borrowck can use it as a hint for figuring out hidden types, /// type here, so that mir-borrowck can use it as a hint for figuring out hidden types,
/// even if they are only set in dead code (which doesn't show up in MIR). /// even if they are only set in dead code (which doesn't show up in MIR).
pub concrete_opaque_types: VecMap<LocalDefId, Ty<'tcx>>, pub concrete_opaque_types: VecMap<LocalDefId, ty::OpaqueHiddenType<'tcx>>,
/// Tracks the minimum captures required for a closure; /// Tracks the minimum captures required for a closure;
/// see `MinCaptureInformationMap` for more details. /// see `MinCaptureInformationMap` for more details.

View File

@ -1,5 +1,5 @@
#![feature(type_alias_impl_trait)] #![feature(type_alias_impl_trait)]
// check-pass
fn main() {} fn main() {}
// two definitions with different types // two definitions with different types
@ -9,7 +9,7 @@ fn foo() -> Foo {
"" ""
} }
fn bar() -> Foo { fn bar() -> Foo { //~ ERROR: concrete type differs from previous defining opaque type use
panic!() panic!()
} }

View File

@ -0,0 +1,14 @@
error: concrete type differs from previous defining opaque type use
--> $DIR/different_defining_uses_never_type.rs:12:13
|
LL | fn bar() -> Foo {
| ^^^ expected `&'static str`, got `()`
|
note: previous use here
--> $DIR/different_defining_uses_never_type.rs:9:5
|
LL | ""
| ^^
error: aborting due to previous error