From 9bdfd0683f7e36651cf11da1d63e8d7e326ff117 Mon Sep 17 00:00:00 2001 From: Michael Wright Date: Sun, 8 Sep 2019 11:03:45 +0200 Subject: [PATCH] Fix `or_fun_call` bad suggestion Closes #4514 --- clippy_lints/src/methods/mod.rs | 55 ++++++++++++++++++--------------- tests/ui/or_fun_call.fixed | 12 +++++++ tests/ui/or_fun_call.rs | 12 +++++++ 3 files changed, 54 insertions(+), 25 deletions(-) diff --git a/clippy_lints/src/methods/mod.rs b/clippy_lints/src/methods/mod.rs index 5578dd2654e2..cccba0327bb7 100644 --- a/clippy_lints/src/methods/mod.rs +++ b/clippy_lints/src/methods/mod.rs @@ -1377,6 +1377,7 @@ fn lint_or_fun_call<'a, 'tcx>( let mut finder = FunCallFinder { cx: &cx, found: false }; if { finder.visit_expr(&arg); finder.found }; + if !contains_return(&arg); let self_ty = cx.tables.expr_ty(self_expr); @@ -2190,28 +2191,6 @@ fn lint_option_and_then_some(cx: &LateContext<'_, '_>, expr: &hir::Expr, args: & const LINT_MSG: &str = "using `Option.and_then(|x| Some(y))`, which is more succinctly expressed as `map(|x| y)`"; const NO_OP_MSG: &str = "using `Option.and_then(Some)`, which is a no-op"; - // Searches an return expressions in `y` in `_.and_then(|x| Some(y))`, which we don't lint - struct RetCallFinder { - found: bool, - } - - impl<'tcx> intravisit::Visitor<'tcx> for RetCallFinder { - fn visit_expr(&mut self, expr: &'tcx hir::Expr) { - if self.found { - return; - } - if let hir::ExprKind::Ret(..) = &expr.node { - self.found = true; - } else { - intravisit::walk_expr(self, expr); - } - } - - fn nested_visit_map<'this>(&'this mut self) -> intravisit::NestedVisitorMap<'this, 'tcx> { - intravisit::NestedVisitorMap::None - } - } - let ty = cx.tables.expr_ty(&args[0]); if !match_type(cx, ty, &paths::OPTION) { return; @@ -2229,9 +2208,7 @@ fn lint_option_and_then_some(cx: &LateContext<'_, '_>, expr: &hir::Expr, args: & then { let inner_expr = &some_args[0]; - let mut finder = RetCallFinder { found: false }; - finder.visit_expr(inner_expr); - if finder.found { + if contains_return(inner_expr) { return; } @@ -2988,3 +2965,31 @@ fn is_bool(ty: &hir::Ty) -> bool { false } } + +// Returns `true` if `expr` contains a return expression +fn contains_return(expr: &hir::Expr) -> bool { + struct RetCallFinder { + found: bool, + } + + impl<'tcx> intravisit::Visitor<'tcx> for RetCallFinder { + fn visit_expr(&mut self, expr: &'tcx hir::Expr) { + if self.found { + return; + } + if let hir::ExprKind::Ret(..) = &expr.node { + self.found = true; + } else { + intravisit::walk_expr(self, expr); + } + } + + fn nested_visit_map<'this>(&'this mut self) -> intravisit::NestedVisitorMap<'this, 'tcx> { + intravisit::NestedVisitorMap::None + } + } + + let mut visitor = RetCallFinder{ found: false }; + visitor.visit_expr(expr); + visitor.found +} diff --git a/tests/ui/or_fun_call.fixed b/tests/ui/or_fun_call.fixed index c572e0f2f451..6d9ad16989a2 100644 --- a/tests/ui/or_fun_call.fixed +++ b/tests/ui/or_fun_call.fixed @@ -99,4 +99,16 @@ fn test_or_with_ctors() { .or(Some(Bar(b, Duration::from_secs(2)))); } + +// Issue 4514 - early return +fn f() -> Option<()> { + let a = Some(1); + let b = 1i32; + + let _ = a.unwrap_or(b.checked_mul(3)?.min(240)); + + Some(()) +} + + fn main() {} diff --git a/tests/ui/or_fun_call.rs b/tests/ui/or_fun_call.rs index 3c94542774b1..78bcf896ec1d 100644 --- a/tests/ui/or_fun_call.rs +++ b/tests/ui/or_fun_call.rs @@ -99,4 +99,16 @@ fn test_or_with_ctors() { .or(Some(Bar(b, Duration::from_secs(2)))); } + +// Issue 4514 - early return +fn f() -> Option<()> { + let a = Some(1); + let b = 1i32; + + let _ = a.unwrap_or(b.checked_mul(3)?.min(240)); + + Some(()) +} + + fn main() {}