Skip to content

Commit 5044be4

Browse files
redact coerced types away
1 parent 2162e9d commit 5044be4

File tree

6 files changed

+350
-29
lines changed

6 files changed

+350
-29
lines changed

compiler/rustc_hir_analysis/messages.ftl

+13
Original file line numberDiff line numberDiff line change
@@ -85,6 +85,19 @@ hir_analysis_cmse_output_stack_spill =
8585
.note1 = functions with the `{$abi}` ABI must pass their result via the available return registers
8686
.note2 = the result must either be a (transparently wrapped) i64, u64 or f64, or be at most 4 bytes in size
8787
88+
hir_analysis_coerce_pointee_cannot_coerce_unsize = `{$ty}` cannot be coerced to an unsized type
89+
.note = `derive(CoercePointee)` demands that `{$ty}` can be coerced to an unsized type
90+
.help = the standard pointers such as `Arc`, `Rc`, `Box`, and other types with `derive(CoercePointee)` can be coerced to their corresponding unsized types
91+
92+
hir_analysis_coerce_pointee_cannot_unsize = `{$ty}` cannot be coerced to any unsized value
93+
.note = `derive(CoercePointee)` demands that `{$ty}` can be coerced to an unsized type
94+
.help = `derive(CoercePointee)` requires exactly one copy of `#[pointee]` type at the end of the `struct` definition, without any further pointer or reference indirection
95+
96+
hir_analysis_coerce_pointee_multiple_targets = `derive(CoercePointee)` only admits exactly one data field, {$diag_trait ->
97+
[DispatchFromDyn] to which `dyn` methods shall be dispatched
98+
*[CoerceUnsized] on which unsize coercion shall be performed
99+
}
100+
88101
hir_analysis_coerce_pointee_no_field = `CoercePointee` can only be derived on `struct`s with at least one field
89102
90103
hir_analysis_coerce_pointee_no_user_validity_assertion = asserting applicability of `derive(CoercePointee)` on a target data is forbidden

compiler/rustc_hir_analysis/src/coherence/builtin.rs

+110-27
Original file line numberDiff line numberDiff line change
@@ -1,9 +1,12 @@
11
//! Check properties that are required by built-in traits and set
22
//! up data structures required by type-checking/codegen.
33
4+
mod diagnostics;
5+
46
use std::assert_matches::assert_matches;
57
use std::collections::BTreeMap;
68

9+
use diagnostics::{extract_coerce_pointee_data, redact_fulfillment_err_for_coerce_pointee};
710
use rustc_data_structures::fx::FxHashSet;
811
use rustc_errors::{ErrorGuaranteed, MultiSpan};
912
use rustc_hir as hir;
@@ -12,6 +15,7 @@ use rustc_hir::def_id::{DefId, LocalDefId};
1215
use rustc_hir::lang_items::LangItem;
1316
use rustc_infer::infer::{self, RegionResolutionError, TyCtxtInferExt};
1417
use rustc_infer::traits::Obligation;
18+
use rustc_middle::bug;
1519
use rustc_middle::ty::adjustment::CoerceUnsizedInfo;
1620
use rustc_middle::ty::print::PrintTraitRefExt as _;
1721
use rustc_middle::ty::{
@@ -307,29 +311,45 @@ fn visit_implementation_of_dispatch_from_dyn(checker: &Checker<'_>) -> Result<()
307311
.collect::<Vec<_>>();
308312

309313
if coerced_fields.is_empty() {
310-
res = Err(tcx.dcx().emit_err(errors::DispatchFromDynSingle {
311-
span,
312-
trait_name: "DispatchFromDyn",
313-
note: true,
314-
}));
314+
if extract_coerce_pointee_data(tcx, def_a.did()).is_some() {
315+
res = Err(tcx.dcx().span_delayed_bug(
316+
span,
317+
"a specialised message for CoercePointee is expected",
318+
));
319+
} else {
320+
res = Err(tcx.dcx().emit_err(errors::DispatchFromDynSingle {
321+
span,
322+
trait_name: "DispatchFromDyn",
323+
note: true,
324+
}));
325+
}
315326
} else if coerced_fields.len() > 1 {
316-
res = Err(tcx.dcx().emit_err(errors::DispatchFromDynMulti {
317-
span,
318-
coercions_note: true,
319-
number: coerced_fields.len(),
320-
coercions: coerced_fields
321-
.iter()
322-
.map(|field| {
323-
format!(
324-
"`{}` (`{}` to `{}`)",
325-
field.name,
326-
field.ty(tcx, args_a),
327-
field.ty(tcx, args_b),
328-
)
329-
})
330-
.collect::<Vec<_>>()
331-
.join(", "),
332-
}));
327+
if extract_coerce_pointee_data(tcx, def_a.did()).is_some() {
328+
let spans =
329+
coerced_fields.iter().map(|field| tcx.def_span(field.did)).collect();
330+
res = Err(tcx.dcx().emit_err(errors::CoercePointeeMultipleTargets {
331+
spans,
332+
diag_trait: "DispatchFromDyn",
333+
}));
334+
} else {
335+
res = Err(tcx.dcx().emit_err(errors::DispatchFromDynMulti {
336+
span,
337+
coercions_note: true,
338+
number: coerced_fields.len(),
339+
coercions: coerced_fields
340+
.iter()
341+
.map(|field| {
342+
format!(
343+
"`{}` (`{}` to `{}`)",
344+
field.name,
345+
field.ty(tcx, args_a),
346+
field.ty(tcx, args_b),
347+
)
348+
})
349+
.collect::<Vec<_>>()
350+
.join(", "),
351+
}));
352+
}
333353
} else {
334354
let ocx = ObligationCtxt::new_with_diagnostics(&infcx);
335355
for field in coerced_fields {
@@ -344,9 +364,31 @@ fn visit_implementation_of_dispatch_from_dyn(checker: &Checker<'_>) -> Result<()
344364
),
345365
));
346366
}
347-
let errors = ocx.select_all_or_error();
367+
let mut errors = ocx.select_all_or_error();
348368
if !errors.is_empty() {
349-
res = Err(infcx.err_ctxt().report_fulfillment_errors(errors));
369+
if let Some((pointee_idx, _)) = extract_coerce_pointee_data(tcx, def_a.did()) {
370+
let target_pointee = args_b.type_at(pointee_idx);
371+
372+
errors = errors
373+
.into_iter()
374+
.filter_map(|err| {
375+
redact_fulfillment_err_for_coerce_pointee(
376+
tcx,
377+
err,
378+
target_pointee,
379+
span,
380+
)
381+
})
382+
.collect();
383+
}
384+
if errors.is_empty() {
385+
res = Err(tcx.dcx().span_delayed_bug(
386+
span,
387+
"a specialised CoercePointee error is expected",
388+
));
389+
} else {
390+
res = Err(infcx.err_ctxt().report_fulfillment_errors(errors));
391+
}
350392
}
351393

352394
// Finally, resolve all regions.
@@ -372,9 +414,11 @@ pub(crate) fn coerce_unsized_info<'tcx>(
372414
let unsize_trait = tcx.require_lang_item(LangItem::Unsize, Some(span));
373415

374416
let source = tcx.type_of(impl_did).instantiate_identity();
417+
let self_ty = source;
375418
let trait_ref = tcx.impl_trait_ref(impl_did).unwrap().instantiate_identity();
376419
assert_eq!(trait_ref.def_id, coerce_unsized_trait);
377420
let target = trait_ref.args.type_at(1);
421+
let coerced_ty = target;
378422
debug!("visit_implementation_of_coerce_unsized: {:?} -> {:?} (bound)", source, target);
379423

380424
let param_env = tcx.param_env(impl_did);
@@ -509,12 +553,28 @@ pub(crate) fn coerce_unsized_info<'tcx>(
509553
.collect::<Vec<_>>();
510554

511555
if diff_fields.is_empty() {
556+
if extract_coerce_pointee_data(tcx, def_a.did()).is_some() {
557+
return Err(tcx.dcx().span_delayed_bug(
558+
span,
559+
"a specialised message for CoercePointee is expected",
560+
));
561+
}
512562
return Err(tcx.dcx().emit_err(errors::CoerceUnsizedOneField {
513563
span,
514564
trait_name: "CoerceUnsized",
515565
note: true,
516566
}));
517567
} else if diff_fields.len() > 1 {
568+
if extract_coerce_pointee_data(tcx, def_a.did()).is_some() {
569+
let spans = diff_fields
570+
.iter()
571+
.map(|&(idx, _, _)| tcx.def_span(fields[idx].did))
572+
.collect();
573+
return Err(tcx.dcx().emit_err(errors::CoercePointeeMultipleTargets {
574+
spans,
575+
diag_trait: "CoerceUnsized",
576+
}));
577+
}
518578
let item = tcx.hir().expect_item(impl_did);
519579
let span = if let ItemKind::Impl(hir::Impl { of_trait: Some(t), .. }) = &item.kind {
520580
t.path.span
@@ -547,18 +607,41 @@ pub(crate) fn coerce_unsized_info<'tcx>(
547607
};
548608

549609
// Register an obligation for `A: Trait<B>`.
610+
let coerce_pointee_data = if let ty::Adt(def, _) = self_ty.kind() {
611+
extract_coerce_pointee_data(tcx, def.did())
612+
} else {
613+
None
614+
};
550615
let ocx = ObligationCtxt::new_with_diagnostics(&infcx);
551-
let cause = traits::ObligationCause::misc(span, impl_did);
616+
let cause = traits::ObligationCause::misc(
617+
span,
618+
coerce_pointee_data.map_or(impl_did, |(_, impl_did)| impl_did.expect_local()),
619+
);
552620
let obligation = Obligation::new(
553621
tcx,
554622
cause,
555623
param_env,
556624
ty::TraitRef::new(tcx, trait_def_id, [source, target]),
557625
);
558626
ocx.register_obligation(obligation);
559-
let errors = ocx.select_all_or_error();
627+
let mut errors = ocx.select_all_or_error();
560628
if !errors.is_empty() {
561-
infcx.err_ctxt().report_fulfillment_errors(errors);
629+
if let Some((pointee, _)) = coerce_pointee_data {
630+
let ty::Adt(_def, args) = coerced_ty.kind() else { bug!() };
631+
let target_pointee = args.type_at(pointee);
632+
let ty::Adt(_def, _) = self_ty.kind() else { bug!() };
633+
errors = errors
634+
.into_iter()
635+
.filter_map(|err| {
636+
redact_fulfillment_err_for_coerce_pointee(tcx, err, target_pointee, span)
637+
})
638+
.collect();
639+
}
640+
if errors.is_empty() {
641+
tcx.dcx().span_delayed_bug(span, "a specialised CoercePointee error is expected");
642+
} else {
643+
infcx.err_ctxt().report_fulfillment_errors(errors);
644+
}
562645
}
563646

564647
// Finally, resolve all regions.
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,136 @@
1+
use rustc_hir::LangItem;
2+
use rustc_hir::def_id::DefId;
3+
use rustc_middle::ty::fast_reject::SimplifiedType;
4+
use rustc_middle::ty::{
5+
self, GenericParamDefKind, Ty, TyCtxt, TypeSuperVisitable, TypeVisitable, TypeVisitor,
6+
};
7+
use rustc_span::{Span, sym};
8+
use rustc_trait_selection::traits::FulfillmentError;
9+
use tracing::instrument;
10+
11+
use crate::errors;
12+
13+
#[instrument(level = "debug", skip(tcx), ret)]
14+
pub(super) fn extract_coerce_pointee_data<'tcx>(
15+
tcx: TyCtxt<'tcx>,
16+
adt_did: DefId,
17+
) -> Option<(usize, DefId)> {
18+
// It is decided that a query to cache these results is not necessary
19+
// for error reporting.
20+
// We can afford to recompute it on-demand.
21+
if let Some(impls) = tcx.lang_items().get(LangItem::CoercePointeeValidated).and_then(|did| {
22+
tcx.trait_impls_of(did).non_blanket_impls().get(&SimplifiedType::Adt(adt_did))
23+
}) && let [impl_did, ..] = impls[..]
24+
{
25+
// Search for the `#[pointee]`
26+
let mut first_type = None;
27+
for (idx, param) in tcx.generics_of(adt_did).own_params.iter().enumerate() {
28+
if let GenericParamDefKind::Type { .. } = param.kind {
29+
first_type = if first_type.is_some() { None } else { Some(idx) };
30+
}
31+
if tcx.has_attr(param.def_id, sym::pointee) {
32+
return Some((idx, impl_did));
33+
}
34+
}
35+
if let Some(idx) = first_type {
36+
return Some((idx, impl_did));
37+
}
38+
}
39+
None
40+
}
41+
42+
fn contains_coerce_pointee_target_pointee<'tcx>(ty: Ty<'tcx>, target_pointee_ty: Ty<'tcx>) -> bool {
43+
struct Search<'tcx> {
44+
pointee: Ty<'tcx>,
45+
found: bool,
46+
}
47+
impl<'tcx> TypeVisitor<TyCtxt<'tcx>> for Search<'tcx> {
48+
fn visit_binder<T: ty::TypeVisitable<TyCtxt<'tcx>>>(
49+
&mut self,
50+
t: &rustc_type_ir::Binder<TyCtxt<'tcx>, T>,
51+
) {
52+
if self.found {
53+
return;
54+
}
55+
t.super_visit_with(self)
56+
}
57+
58+
fn visit_ty(&mut self, t: Ty<'tcx>) {
59+
if self.found {
60+
return;
61+
}
62+
if t == self.pointee {
63+
self.found = true;
64+
} else {
65+
t.super_visit_with(self)
66+
}
67+
}
68+
69+
fn visit_region(&mut self, r: <TyCtxt<'tcx> as ty::Interner>::Region) {
70+
if self.found {
71+
return;
72+
}
73+
if let rustc_type_ir::ReError(guar) = r.kind() {
74+
self.visit_error(guar)
75+
}
76+
}
77+
78+
fn visit_const(&mut self, c: <TyCtxt<'tcx> as ty::Interner>::Const) {
79+
if self.found {
80+
return;
81+
}
82+
c.super_visit_with(self)
83+
}
84+
85+
fn visit_predicate(&mut self, p: <TyCtxt<'tcx> as ty::Interner>::Predicate) {
86+
if self.found {
87+
return;
88+
}
89+
p.super_visit_with(self)
90+
}
91+
92+
fn visit_clauses(&mut self, p: <TyCtxt<'tcx> as ty::Interner>::Clauses) {
93+
if self.found {
94+
return;
95+
}
96+
p.super_visit_with(self)
97+
}
98+
}
99+
let mut search = Search { pointee: target_pointee_ty, found: false };
100+
ty.visit_with(&mut search);
101+
search.found
102+
}
103+
104+
#[instrument(level = "debug", skip(tcx))]
105+
pub(super) fn redact_fulfillment_err_for_coerce_pointee<'tcx>(
106+
tcx: TyCtxt<'tcx>,
107+
err: FulfillmentError<'tcx>,
108+
target_pointee_ty: Ty<'tcx>,
109+
span: Span,
110+
) -> Option<FulfillmentError<'tcx>> {
111+
if let ty::PredicateKind::Clause(ty::ClauseKind::Trait(pred)) =
112+
err.obligation.predicate.kind().skip_binder()
113+
{
114+
let mentions_pointee = || {
115+
contains_coerce_pointee_target_pointee(
116+
pred.trait_ref.args.type_at(1),
117+
target_pointee_ty,
118+
)
119+
};
120+
let source = pred.trait_ref.self_ty();
121+
if tcx.is_lang_item(pred.def_id(), LangItem::Unsize) {
122+
if mentions_pointee() {
123+
// We should redact it
124+
tcx.dcx()
125+
.emit_err(errors::CoercePointeeCannotUnsize { ty: source.to_string(), span });
126+
return None;
127+
}
128+
} else if tcx.is_lang_item(pred.def_id(), LangItem::CoerceUnsized) && mentions_pointee() {
129+
// We should redact it
130+
tcx.dcx()
131+
.emit_err(errors::CoercePointeeCannotCoerceUnsize { ty: source.to_string(), span });
132+
return None;
133+
}
134+
}
135+
Some(err)
136+
}

compiler/rustc_hir_analysis/src/errors.rs

+24
Original file line numberDiff line numberDiff line change
@@ -1219,6 +1219,30 @@ pub(crate) struct CoercePointeeNoField {
12191219
pub span: Span,
12201220
}
12211221

1222+
#[derive(Diagnostic)]
1223+
#[diag(hir_analysis_coerce_pointee_multiple_targets, code = E0802)]
1224+
pub(crate) struct CoercePointeeMultipleTargets {
1225+
#[primary_span]
1226+
pub spans: Vec<Span>,
1227+
pub diag_trait: &'static str,
1228+
}
1229+
1230+
#[derive(Diagnostic)]
1231+
#[diag(hir_analysis_coerce_pointee_cannot_unsize, code = E0802)]
1232+
pub(crate) struct CoercePointeeCannotUnsize {
1233+
#[primary_span]
1234+
pub span: Span,
1235+
pub ty: String,
1236+
}
1237+
1238+
#[derive(Diagnostic)]
1239+
#[diag(hir_analysis_coerce_pointee_cannot_coerce_unsize, code = E0802)]
1240+
pub(crate) struct CoercePointeeCannotCoerceUnsize {
1241+
#[primary_span]
1242+
pub span: Span,
1243+
pub ty: String,
1244+
}
1245+
12221246
#[derive(Diagnostic)]
12231247
#[diag(hir_analysis_inherent_ty_outside_relevant, code = E0390)]
12241248
#[help]

0 commit comments

Comments
 (0)