89 lines
2.5 KiB
Rust
Raw Normal View History

2019-05-14 10:06:21 +02:00
use crate::utils::{get_pat_name, match_var, snippet};
2020-01-09 16:13:22 +09:00
use rustc::hir::map::Map;
2020-03-01 12:23:33 +09:00
use rustc_ast::ast::Name;
2020-01-09 16:13:22 +09:00
use rustc_hir::intravisit::{walk_expr, NestedVisitorMap, Visitor};
2020-02-21 09:39:38 +01:00
use rustc_hir::{Body, BodyId, Expr, ExprKind, Param};
2020-01-12 15:08:41 +09:00
use rustc_lint::LateContext;
use rustc_span::source_map::Span;
2018-11-27 21:14:15 +01:00
use std::borrow::Cow;
2017-10-08 17:51:44 +09:00
pub fn get_spans(
2018-07-23 13:01:12 +02:00
cx: &LateContext<'_, '_>,
2017-10-08 17:51:44 +09:00
opt_body_id: Option<BodyId>,
idx: usize,
2019-05-17 23:53:54 +02:00
replacements: &[(&'static str, &'static str)],
2017-10-08 17:51:44 +09:00
) -> Option<Vec<(Span, Cow<'static, str>)>> {
if let Some(body) = opt_body_id.map(|id| cx.tcx.hir().body(id)) {
2019-08-28 18:27:06 +09:00
get_binding_name(&body.params[idx]).map_or_else(
2018-11-27 21:14:15 +01:00
|| Some(vec![]),
|name| extract_clone_suggestions(cx, name, replacements, body),
)
2017-10-08 17:51:44 +09:00
} else {
Some(vec![])
}
}
fn extract_clone_suggestions<'a, 'tcx>(
2017-10-08 17:51:44 +09:00
cx: &LateContext<'a, 'tcx>,
name: Name,
2019-05-17 23:53:54 +02:00
replace: &[(&'static str, &'static str)],
2019-12-22 15:42:41 +01:00
body: &'tcx Body<'_>,
2017-10-08 17:51:44 +09:00
) -> Option<Vec<(Span, Cow<'static, str>)>> {
let mut visitor = PtrCloneVisitor {
cx,
name,
replace,
spans: vec![],
abort: false,
};
visitor.visit_body(body);
if visitor.abort {
None
} else {
Some(visitor.spans)
}
}
struct PtrCloneVisitor<'a, 'tcx> {
2017-10-08 17:51:44 +09:00
cx: &'a LateContext<'a, 'tcx>,
name: Name,
2019-05-17 23:53:54 +02:00
replace: &'a [(&'static str, &'static str)],
2017-10-08 17:51:44 +09:00
spans: Vec<(Span, Cow<'static, str>)>,
abort: bool,
}
impl<'a, 'tcx> Visitor<'tcx> for PtrCloneVisitor<'a, 'tcx> {
2020-01-09 16:13:22 +09:00
type Map = Map<'tcx>;
2019-12-27 16:12:26 +09:00
fn visit_expr(&mut self, expr: &'tcx Expr<'_>) {
2017-10-08 17:51:44 +09:00
if self.abort {
return;
}
2019-09-27 17:16:06 +02:00
if let ExprKind::MethodCall(ref seg, _, ref args) = expr.kind {
2017-10-08 17:51:44 +09:00
if args.len() == 1 && match_var(&args[0], self.name) {
2019-05-17 23:53:54 +02:00
if seg.ident.name.as_str() == "capacity" {
2017-10-08 17:51:44 +09:00
self.abort = true;
return;
}
for &(fn_name, suffix) in self.replace {
2019-05-17 23:53:54 +02:00
if seg.ident.name.as_str() == fn_name {
2017-10-08 17:51:44 +09:00
self.spans
.push((expr.span, snippet(self.cx, args[0].span, "_") + suffix));
return;
}
}
}
return;
}
walk_expr(self, expr);
}
2020-01-09 16:13:22 +09:00
fn nested_visit_map(&mut self) -> NestedVisitorMap<'_, Self::Map> {
2017-10-08 17:51:44 +09:00
NestedVisitorMap::None
}
}
2019-12-27 16:12:26 +09:00
fn get_binding_name(arg: &Param<'_>) -> Option<Name> {
2017-10-08 17:51:44 +09:00
get_pat_name(&arg.pat)
}