Skip to content

Commit 822b10d

Browse files
committed
Use ControlFlow in HIR visitors
1 parent e760c44 commit 822b10d

File tree

12 files changed

+168
-201
lines changed

12 files changed

+168
-201
lines changed

compiler/rustc_borrowck/src/diagnostics/mutability_errors.rs

+29-47
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,7 @@
11
#![allow(rustc::diagnostic_outside_of_impl)]
22
#![allow(rustc::untranslatable_diagnostic)]
33

4+
use core::ops::ControlFlow;
45
use hir::ExprKind;
56
use rustc_errors::{Applicability, Diag};
67
use rustc_hir as hir;
@@ -727,30 +728,12 @@ impl<'a, 'tcx> MirBorrowckCtxt<'a, 'tcx> {
727728
_ => local_decl.source_info.span,
728729
};
729730

730-
struct BindingFinder {
731-
span: Span,
732-
hir_id: Option<hir::HirId>,
733-
}
734-
735-
impl<'tcx> Visitor<'tcx> for BindingFinder {
736-
fn visit_stmt(&mut self, s: &'tcx hir::Stmt<'tcx>) {
737-
if let hir::StmtKind::Local(local) = s.kind {
738-
if local.pat.span == self.span {
739-
self.hir_id = Some(local.hir_id);
740-
}
741-
}
742-
hir::intravisit::walk_stmt(self, s);
743-
}
744-
}
745-
746731
let def_id = self.body.source.def_id();
747732
let hir_id = if let Some(local_def_id) = def_id.as_local()
748733
&& let Some(body_id) = self.infcx.tcx.hir().maybe_body_owned_by(local_def_id)
749734
{
750735
let body = self.infcx.tcx.hir().body(body_id);
751-
let mut v = BindingFinder { span: pat_span, hir_id: None };
752-
v.visit_body(body);
753-
v.hir_id
736+
BindingFinder { span: pat_span }.visit_body(body).break_value()
754737
} else {
755738
None
756739
};
@@ -859,17 +842,18 @@ impl<'a, 'tcx> MirBorrowckCtxt<'a, 'tcx> {
859842
};
860843

861844
let hir_map = self.infcx.tcx.hir();
862-
struct Finder<'tcx> {
845+
struct Finder {
863846
span: Span,
864-
expr: Option<&'tcx Expr<'tcx>>,
865847
}
866848

867-
impl<'tcx> Visitor<'tcx> for Finder<'tcx> {
868-
fn visit_expr(&mut self, e: &'tcx hir::Expr<'tcx>) {
869-
if e.span == self.span && self.expr.is_none() {
870-
self.expr = Some(e);
849+
impl<'tcx> Visitor<'tcx> for Finder {
850+
type Result = ControlFlow<&'tcx Expr<'tcx>>;
851+
fn visit_expr(&mut self, e: &'tcx hir::Expr<'tcx>) -> Self::Result {
852+
if e.span == self.span {
853+
ControlFlow::Break(e)
854+
} else {
855+
hir::intravisit::walk_expr(self, e)
871856
}
872-
hir::intravisit::walk_expr(self, e);
873857
}
874858
}
875859
if let Some(body_id) = hir_map.maybe_body_owned_by(self.mir_def_id())
@@ -878,9 +862,7 @@ impl<'a, 'tcx> MirBorrowckCtxt<'a, 'tcx> {
878862
// `span` corresponds to the expression being iterated, find the `for`-loop desugared
879863
// expression with that span in order to identify potential fixes when encountering a
880864
// read-only iterator that should be mutable.
881-
let mut v = Finder { span, expr: None };
882-
v.visit_block(block);
883-
if let Some(expr) = v.expr
865+
if let ControlFlow::Break(expr) = (Finder { span }).visit_block(block)
884866
&& let Call(_, [expr]) = expr.kind
885867
{
886868
match expr.kind {
@@ -1179,29 +1161,12 @@ impl<'a, 'tcx> MirBorrowckCtxt<'a, 'tcx> {
11791161
);
11801162
}
11811163
Some((false, err_label_span, message)) => {
1182-
struct BindingFinder {
1183-
span: Span,
1184-
hir_id: Option<hir::HirId>,
1185-
}
1186-
1187-
impl<'tcx> Visitor<'tcx> for BindingFinder {
1188-
fn visit_stmt(&mut self, s: &'tcx hir::Stmt<'tcx>) {
1189-
if let hir::StmtKind::Local(local) = s.kind {
1190-
if local.pat.span == self.span {
1191-
self.hir_id = Some(local.hir_id);
1192-
}
1193-
}
1194-
hir::intravisit::walk_stmt(self, s);
1195-
}
1196-
}
11971164
let def_id = self.body.source.def_id();
11981165
let hir_id = if let Some(local_def_id) = def_id.as_local()
11991166
&& let Some(body_id) = self.infcx.tcx.hir().maybe_body_owned_by(local_def_id)
12001167
{
12011168
let body = self.infcx.tcx.hir().body(body_id);
1202-
let mut v = BindingFinder { span: err_label_span, hir_id: None };
1203-
v.visit_body(body);
1204-
v.hir_id
1169+
BindingFinder { span: err_label_span }.visit_body(body).break_value()
12051170
} else {
12061171
None
12071172
};
@@ -1333,6 +1298,23 @@ impl<'a, 'tcx> MirBorrowckCtxt<'a, 'tcx> {
13331298
}
13341299
}
13351300

1301+
struct BindingFinder {
1302+
span: Span,
1303+
}
1304+
1305+
impl<'tcx> Visitor<'tcx> for BindingFinder {
1306+
type Result = ControlFlow<hir::HirId>;
1307+
fn visit_stmt(&mut self, s: &'tcx hir::Stmt<'tcx>) -> Self::Result {
1308+
if let hir::StmtKind::Local(local) = s.kind
1309+
&& local.pat.span == self.span
1310+
{
1311+
ControlFlow::Break(local.hir_id)
1312+
} else {
1313+
hir::intravisit::walk_stmt(self, s)
1314+
}
1315+
}
1316+
}
1317+
13361318
pub fn mut_borrow_of_mutable_ref(local_decl: &LocalDecl<'_>, local_name: Option<Symbol>) -> bool {
13371319
debug!("local_info: {:?}, ty.kind(): {:?}", local_decl.local_info, local_decl.ty.kind());
13381320

compiler/rustc_borrowck/src/lib.rs

+1
Original file line numberDiff line numberDiff line change
@@ -6,6 +6,7 @@
66
#![feature(assert_matches)]
77
#![feature(associated_type_bounds)]
88
#![feature(box_patterns)]
9+
#![feature(control_flow_enum)]
910
#![feature(let_chains)]
1011
#![feature(min_specialization)]
1112
#![feature(never_type)]

compiler/rustc_hir_analysis/src/check/compare_impl_item.rs

+11-10
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,6 @@
11
use super::potentially_plural_count;
22
use crate::errors::{LifetimesOrBoundsMismatchOnTrait, MethodShouldReturnFuture};
3+
use core::ops::ControlFlow;
34
use hir::def_id::{DefId, DefIdMap, LocalDefId};
45
use rustc_data_structures::fx::{FxHashSet, FxIndexMap, FxIndexSet};
56
use rustc_errors::{codes::*, pluralize, struct_span_code_err, Applicability, ErrorGuaranteed};
@@ -1565,24 +1566,24 @@ fn compare_synthetic_generics<'tcx>(
15651566
let (sig, _) = impl_m.expect_fn();
15661567
let input_tys = sig.decl.inputs;
15671568

1568-
struct Visitor(Option<Span>, hir::def_id::LocalDefId);
1569+
struct Visitor(hir::def_id::LocalDefId);
15691570
impl<'v> intravisit::Visitor<'v> for Visitor {
1570-
fn visit_ty(&mut self, ty: &'v hir::Ty<'v>) {
1571-
intravisit::walk_ty(self, ty);
1571+
type Result = ControlFlow<Span>;
1572+
fn visit_ty(&mut self, ty: &'v hir::Ty<'v>) -> Self::Result {
15721573
if let hir::TyKind::Path(hir::QPath::Resolved(None, path)) = ty.kind
15731574
&& let Res::Def(DefKind::TyParam, def_id) = path.res
1574-
&& def_id == self.1.to_def_id()
1575+
&& def_id == self.0.to_def_id()
15751576
{
1576-
self.0 = Some(ty.span);
1577+
ControlFlow::Break(ty.span)
1578+
} else {
1579+
intravisit::walk_ty(self, ty)
15771580
}
15781581
}
15791582
}
15801583

1581-
let mut visitor = Visitor(None, impl_def_id);
1582-
for ty in input_tys {
1583-
intravisit::Visitor::visit_ty(&mut visitor, ty);
1584-
}
1585-
let span = visitor.0?;
1584+
let span = input_tys.iter().find_map(|ty| {
1585+
intravisit::Visitor::visit_ty(&mut Visitor(impl_def_id), ty).break_value()
1586+
})?;
15861587

15871588
let bounds = impl_m.generics.bounds_for_param(impl_def_id).next()?.bounds;
15881589
let bounds = bounds.first()?.span().to(bounds.last()?.span());

compiler/rustc_hir_analysis/src/collect/resolve_bound_vars.rs

+9-13
Original file line numberDiff line numberDiff line change
@@ -6,6 +6,7 @@
66
//! the types in HIR to identify late-bound lifetimes and assign their Debruijn indices. This file
77
//! is also responsible for assigning their semantics to implicit lifetimes in trait objects.
88
9+
use core::ops::ControlFlow;
910
use rustc_ast::visit::walk_list;
1011
use rustc_data_structures::fx::{FxHashSet, FxIndexMap, FxIndexSet};
1112
use rustc_errors::{codes::*, struct_span_code_err};
@@ -417,23 +418,18 @@ impl<'a, 'tcx> Visitor<'tcx> for BoundVarContext<'a, 'tcx> {
417418
{
418419
if let &hir::ClosureBinder::For { span: for_sp, .. } = binder {
419420
fn span_of_infer(ty: &hir::Ty<'_>) -> Option<Span> {
420-
struct V(Option<Span>);
421-
421+
struct V;
422422
impl<'v> Visitor<'v> for V {
423-
fn visit_ty(&mut self, t: &'v hir::Ty<'v>) {
424-
match t.kind {
425-
_ if self.0.is_some() => (),
426-
hir::TyKind::Infer => {
427-
self.0 = Some(t.span);
428-
}
429-
_ => intravisit::walk_ty(self, t),
423+
type Result = ControlFlow<Span>;
424+
fn visit_ty(&mut self, t: &'v hir::Ty<'v>) -> Self::Result {
425+
if matches!(t.kind, hir::TyKind::Infer) {
426+
ControlFlow::Break(t.span)
427+
} else {
428+
intravisit::walk_ty(self, t)
430429
}
431430
}
432431
}
433-
434-
let mut v = V(None);
435-
v.visit_ty(ty);
436-
v.0
432+
V.visit_ty(ty).break_value()
437433
}
438434

439435
let infer_in_rt_sp = match fn_decl.output {

compiler/rustc_hir_analysis/src/collect/type_of.rs

+7-9
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,4 @@
1+
use core::ops::ControlFlow;
12
use rustc_errors::{Applicability, StashKey};
23
use rustc_hir as hir;
34
use rustc_hir::def_id::{DefId, LocalDefId};
@@ -666,19 +667,16 @@ pub fn type_alias_is_lazy<'tcx>(tcx: TyCtxt<'tcx>, def_id: LocalDefId) -> bool {
666667
if tcx.features().lazy_type_alias {
667668
return true;
668669
}
669-
struct HasTait {
670-
has_type_alias_impl_trait: bool,
671-
}
670+
struct HasTait;
672671
impl<'tcx> Visitor<'tcx> for HasTait {
673-
fn visit_ty(&mut self, t: &'tcx hir::Ty<'tcx>) {
672+
type Result = ControlFlow<()>;
673+
fn visit_ty(&mut self, t: &'tcx hir::Ty<'tcx>) -> Self::Result {
674674
if let hir::TyKind::OpaqueDef(..) = t.kind {
675-
self.has_type_alias_impl_trait = true;
675+
ControlFlow::Break(())
676676
} else {
677-
hir::intravisit::walk_ty(self, t);
677+
hir::intravisit::walk_ty(self, t)
678678
}
679679
}
680680
}
681-
let mut has_tait = HasTait { has_type_alias_impl_trait: false };
682-
has_tait.visit_ty(tcx.hir().expect_item(def_id).expect_ty_alias().0);
683-
has_tait.has_type_alias_impl_trait
681+
HasTait.visit_ty(tcx.hir().expect_item(def_id).expect_ty_alias().0).is_break()
684682
}

compiler/rustc_hir_typeck/src/method/suggest.rs

+10-11
Original file line numberDiff line numberDiff line change
@@ -6,6 +6,7 @@
66
use crate::errors::{self, CandidateTraitNote, NoAssociatedItem};
77
use crate::Expectation;
88
use crate::FnCtxt;
9+
use core::ops::ControlFlow;
910
use rustc_ast::ast::Mutability;
1011
use rustc_attr::parse_confusables;
1112
use rustc_data_structures::fx::{FxIndexMap, FxIndexSet};
@@ -2212,30 +2213,28 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
22122213
let map = self.infcx.tcx.hir();
22132214
let body_id = self.tcx.hir().body_owned_by(self.body_id);
22142215
let body = map.body(body_id);
2215-
struct LetVisitor<'a> {
2216-
result: Option<&'a hir::Expr<'a>>,
2216+
struct LetVisitor {
22172217
ident_name: Symbol,
22182218
}
22192219

22202220
// FIXME: This really should be taking scoping, etc into account.
2221-
impl<'v> Visitor<'v> for LetVisitor<'v> {
2222-
fn visit_stmt(&mut self, ex: &'v hir::Stmt<'v>) {
2223-
if let hir::StmtKind::Local(hir::Local { pat, init, .. }) = &ex.kind
2221+
impl<'v> Visitor<'v> for LetVisitor {
2222+
type Result = ControlFlow<Option<&'v hir::Expr<'v>>>;
2223+
fn visit_stmt(&mut self, ex: &'v hir::Stmt<'v>) -> Self::Result {
2224+
if let hir::StmtKind::Local(&hir::Local { pat, init, .. }) = ex.kind
22242225
&& let Binding(_, _, ident, ..) = pat.kind
22252226
&& ident.name == self.ident_name
22262227
{
2227-
self.result = *init;
2228+
ControlFlow::Break(init)
22282229
} else {
2229-
hir::intravisit::walk_stmt(self, ex);
2230+
hir::intravisit::walk_stmt(self, ex)
22302231
}
22312232
}
22322233
}
22332234

2234-
let mut visitor = LetVisitor { result: None, ident_name: seg1.ident.name };
2235-
visitor.visit_body(body);
2236-
22372235
if let Node::Expr(call_expr) = self.tcx.parent_hir_node(seg1.hir_id)
2238-
&& let Some(expr) = visitor.result
2236+
&& let ControlFlow::Break(Some(expr)) =
2237+
(LetVisitor { ident_name: seg1.ident.name }).visit_body(body)
22392238
&& let Some(self_ty) = self.node_ty_opt(expr.hir_id)
22402239
{
22412240
let probe = self.lookup_probe_for_diagnostic(

compiler/rustc_infer/src/infer/error_reporting/mod.rs

+9-12
Original file line numberDiff line numberDiff line change
@@ -79,7 +79,7 @@ use rustc_middle::ty::{
7979
use rustc_span::{sym, symbol::kw, BytePos, DesugaringKind, Pos, Span};
8080
use rustc_target::spec::abi;
8181
use std::borrow::Cow;
82-
use std::ops::Deref;
82+
use std::ops::{ControlFlow, Deref};
8383
use std::path::PathBuf;
8484
use std::{cmp, fmt, iter};
8585

@@ -2129,15 +2129,12 @@ impl<'tcx> TypeErrCtxt<'_, 'tcx> {
21292129
let tykind = match self.tcx.opt_hir_node_by_def_id(trace.cause.body_id) {
21302130
Some(hir::Node::Item(hir::Item { kind: hir::ItemKind::Fn(_, _, body_id), .. })) => {
21312131
let body = hir.body(*body_id);
2132-
struct LetVisitor<'v> {
2132+
struct LetVisitor {
21332133
span: Span,
2134-
result: Option<&'v hir::Ty<'v>>,
21352134
}
2136-
impl<'v> Visitor<'v> for LetVisitor<'v> {
2137-
fn visit_stmt(&mut self, s: &'v hir::Stmt<'v>) {
2138-
if self.result.is_some() {
2139-
return;
2140-
}
2135+
impl<'v> Visitor<'v> for LetVisitor {
2136+
type Result = ControlFlow<&'v hir::TyKind<'v>>;
2137+
fn visit_stmt(&mut self, s: &'v hir::Stmt<'v>) -> Self::Result {
21412138
// Find a local statement where the initializer has
21422139
// the same span as the error and the type is specified.
21432140
if let hir::Stmt {
@@ -2151,13 +2148,13 @@ impl<'tcx> TypeErrCtxt<'_, 'tcx> {
21512148
} = s
21522149
&& init_span == &self.span
21532150
{
2154-
self.result = Some(*array_ty);
2151+
ControlFlow::Break(&array_ty.peel_refs().kind)
2152+
} else {
2153+
ControlFlow::Continue(())
21552154
}
21562155
}
21572156
}
2158-
let mut visitor = LetVisitor { span, result: None };
2159-
visitor.visit_body(body);
2160-
visitor.result.map(|r| &r.peel_refs().kind)
2157+
LetVisitor { span }.visit_body(body).break_value()
21612158
}
21622159
Some(hir::Node::Item(hir::Item { kind: hir::ItemKind::Const(ty, _, _), .. })) => {
21632160
Some(&ty.peel_refs().kind)

0 commit comments

Comments
 (0)