9788: fix: extract_function does not move locals defined outside of loops r=Veykril a=Veykril

Fixes #8234

Co-authored-by: Lukas Wirth <lukastw97@gmail.com>
This commit is contained in:
bors[bot] 2021-08-04 18:03:41 +00:00 committed by GitHub
commit ec753847bd
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
4 changed files with 149 additions and 21 deletions

View file

@ -138,8 +138,9 @@ pub struct HlRange {
// injected:: Emitted for doc-string injected highlighting like rust source blocks in documentation. // injected:: Emitted for doc-string injected highlighting like rust source blocks in documentation.
// intraDocLink:: Emitted for intra doc links in doc-strings. // intraDocLink:: Emitted for intra doc links in doc-strings.
// library:: Emitted for items that are defined outside of the current crate. // library:: Emitted for items that are defined outside of the current crate.
// mutable:: Emitted for mutable locals and statics as well as functions taking `&mut self`.
// public:: Emitted for items that are from the current crate and are `pub`. // public:: Emitted for items that are from the current crate and are `pub`.
// mutable:: Emitted for mutable locals and statics. // reference: Emitted for locals behind a reference and functions taking `self` by reference.
// static:: Emitted for "static" functions, also known as functions that do not take a `self` param, as well as statics and consts. // static:: Emitted for "static" functions, also known as functions that do not take a `self` param, as well as statics and consts.
// trait:: Emitted for associated trait items. // trait:: Emitted for associated trait items.
// unsafe:: Emitted for unsafe operations, like unsafe function calls, as well as the `unsafe` token. // unsafe:: Emitted for unsafe operations, like unsafe function calls, as well as the `unsafe` token.

View file

@ -45,6 +45,8 @@ pub enum HlTag {
pub enum HlMod { pub enum HlMod {
/// Used for items in traits and impls. /// Used for items in traits and impls.
Associated = 0, Associated = 0,
/// Used with keywords like `async` and `await`.
Async,
/// Used to differentiate individual elements within attributes. /// Used to differentiate individual elements within attributes.
Attribute, Attribute,
/// Callable item or value. /// Callable item or value.
@ -62,20 +64,18 @@ pub enum HlMod {
Injected, Injected,
/// Used for intra doc links in doc injection. /// Used for intra doc links in doc injection.
IntraDocLink, IntraDocLink,
/// Used for items from other crates.
Library,
/// Mutable binding. /// Mutable binding.
Mutable, Mutable,
/// Used for public items.
Public,
/// Immutable reference. /// Immutable reference.
Reference, Reference,
/// Used for associated functions. /// Used for associated functions.
Static, Static,
/// Used for items in traits and trait impls. /// Used for items in traits and trait impls.
Trait, Trait,
/// Used with keywords like `async` and `await`.
Async,
/// Used for items from other crates.
Library,
/// Used for public items.
Public,
// Keep this last! // Keep this last!
/// Used for unsafe functions, unsafe traits, mutable statics, union accesses and unsafe operations. /// Used for unsafe functions, unsafe traits, mutable statics, union accesses and unsafe operations.
Unsafe, Unsafe,

View file

@ -96,7 +96,8 @@ pub(crate) fn extract_function(acc: &mut Assists, ctx: &AssistContext) -> Option
return; return;
} }
let params = body.extracted_function_params(ctx, locals_used.iter().copied()); let params =
body.extracted_function_params(ctx, &container_info, locals_used.iter().copied());
let fun = Function { let fun = Function {
name: "fun_name".to_string(), name: "fun_name".to_string(),
@ -183,8 +184,8 @@ struct Function {
struct Param { struct Param {
var: Local, var: Local,
ty: hir::Type, ty: hir::Type,
has_usages_afterwards: bool, move_local: bool,
has_mut_inside_body: bool, requires_mut: bool,
is_copy: bool, is_copy: bool,
} }
@ -226,6 +227,7 @@ struct ControlFlow {
struct ContainerInfo { struct ContainerInfo {
is_const: bool, is_const: bool,
is_in_tail: bool, is_in_tail: bool,
parent_loop: Option<SyntaxNode>,
/// The function's return type, const's type etc. /// The function's return type, const's type etc.
ret_type: Option<hir::Type>, ret_type: Option<hir::Type>,
} }
@ -335,11 +337,11 @@ impl ParamKind {
impl Param { impl Param {
fn kind(&self) -> ParamKind { fn kind(&self) -> ParamKind {
match (self.has_usages_afterwards, self.has_mut_inside_body, self.is_copy) { match (self.move_local, self.requires_mut, self.is_copy) {
(true, true, _) => ParamKind::MutRef, (false, true, _) => ParamKind::MutRef,
(true, false, false) => ParamKind::SharedRef, (false, false, false) => ParamKind::SharedRef,
(false, true, _) => ParamKind::MutValue, (true, true, _) => ParamKind::MutValue,
(true, false, true) | (false, false, _) => ParamKind::Value, (_, false, _) => ParamKind::Value,
} }
} }
@ -622,6 +624,15 @@ impl FunctionBody {
fn analyze_container(&self, sema: &Semantics<RootDatabase>) -> Option<ContainerInfo> { fn analyze_container(&self, sema: &Semantics<RootDatabase>) -> Option<ContainerInfo> {
let mut ancestors = self.parent()?.ancestors(); let mut ancestors = self.parent()?.ancestors();
let infer_expr_opt = |expr| sema.type_of_expr(&expr?).map(TypeInfo::adjusted); let infer_expr_opt = |expr| sema.type_of_expr(&expr?).map(TypeInfo::adjusted);
let mut parent_loop = None;
let mut set_parent_loop = |loop_: &dyn ast::LoopBodyOwner| {
if loop_
.loop_body()
.map_or(false, |it| it.syntax().text_range().contains_range(self.text_range()))
{
parent_loop.get_or_insert(loop_.syntax().clone());
}
};
let (is_const, expr, ty) = loop { let (is_const, expr, ty) = loop {
let anc = ancestors.next()?; let anc = ancestors.next()?;
break match_ast! { break match_ast! {
@ -658,6 +669,18 @@ impl FunctionBody {
}, },
ast::Variant(__) => return None, ast::Variant(__) => return None,
ast::Meta(__) => return None, ast::Meta(__) => return None,
ast::LoopExpr(it) => {
set_parent_loop(&it);
continue;
},
ast::ForExpr(it) => {
set_parent_loop(&it);
continue;
},
ast::WhileExpr(it) => {
set_parent_loop(&it);
continue;
},
_ => continue, _ => continue,
} }
}; };
@ -670,7 +693,7 @@ impl FunctionBody {
container_tail.zip(self.tail_expr()).map_or(false, |(container_tail, body_tail)| { container_tail.zip(self.tail_expr()).map_or(false, |(container_tail, body_tail)| {
container_tail.syntax().text_range().contains_range(body_tail.syntax().text_range()) container_tail.syntax().text_range().contains_range(body_tail.syntax().text_range())
}); });
Some(ContainerInfo { is_in_tail, is_const, ret_type: ty }) Some(ContainerInfo { is_in_tail, is_const, parent_loop, ret_type: ty })
} }
fn return_ty(&self, ctx: &AssistContext) -> Option<RetType> { fn return_ty(&self, ctx: &AssistContext) -> Option<RetType> {
@ -780,34 +803,38 @@ impl FunctionBody {
Some(ControlFlow { kind, is_async, is_unsafe: _is_unsafe }) Some(ControlFlow { kind, is_async, is_unsafe: _is_unsafe })
} }
/// find variables that should be extracted as params /// find variables that should be extracted as params
/// ///
/// Computes additional info that affects param type and mutability /// Computes additional info that affects param type and mutability
fn extracted_function_params( fn extracted_function_params(
&self, &self,
ctx: &AssistContext, ctx: &AssistContext,
container_info: &ContainerInfo,
locals: impl Iterator<Item = Local>, locals: impl Iterator<Item = Local>,
) -> Vec<Param> { ) -> Vec<Param> {
locals locals
.map(|local| (local, local.source(ctx.db()))) .map(|local| (local, local.source(ctx.db())))
.filter(|(_, src)| is_defined_outside_of_body(ctx, self, src)) .filter(|(_, src)| is_defined_outside_of_body(ctx, self, src))
.filter_map(|(local, src)| { .filter_map(|(local, src)| {
if src.value.is_left() { if let Either::Left(src) = src.value {
Some(local) Some((local, src))
} else { } else {
stdx::never!(false, "Local::is_self returned false, but source is SelfParam"); stdx::never!(false, "Local::is_self returned false, but source is SelfParam");
None None
} }
}) })
.map(|var| { .map(|(var, src)| {
let usages = LocalUsages::find_local_usages(ctx, var); let usages = LocalUsages::find_local_usages(ctx, var);
let ty = var.ty(ctx.db()); let ty = var.ty(ctx.db());
let is_copy = ty.is_copy(ctx.db()); let is_copy = ty.is_copy(ctx.db());
Param { Param {
var, var,
ty, ty,
has_usages_afterwards: self.has_usages_after_body(&usages), move_local: container_info.parent_loop.as_ref().map_or(true, |it| {
has_mut_inside_body: has_exclusive_usages(ctx, &usages, self), it.text_range().contains_range(src.syntax().text_range())
}) && !self.has_usages_after_body(&usages),
requires_mut: has_exclusive_usages(ctx, &usages, self),
is_copy, is_copy,
} }
}) })
@ -4009,6 +4036,83 @@ const FOO: () = {
const fn $0fun_name() { const fn $0fun_name() {
() ()
} }
"#,
);
}
#[test]
fn extract_does_not_move_outer_loop_vars() {
check_assist(
extract_function,
r#"
fn foo() {
let mut x = 5;
for _ in 0..10 {
$0x += 1;$0
}
}
"#,
r#"
fn foo() {
let mut x = 5;
for _ in 0..10 {
fun_name(&mut x);
}
}
fn $0fun_name(x: &mut i32) {
*x += 1;
}
"#,
);
check_assist(
extract_function,
r#"
fn foo() {
for _ in 0..10 {
let mut x = 5;
$0x += 1;$0
}
}
"#,
r#"
fn foo() {
for _ in 0..10 {
let mut x = 5;
fun_name(x);
}
}
fn $0fun_name(mut x: i32) {
x += 1;
}
"#,
);
check_assist(
extract_function,
r#"
fn foo() {
loop {
let mut x = 5;
for _ in 0..10 {
$0x += 1;$0
}
}
}
"#,
r#"
fn foo() {
loop {
let mut x = 5;
for _ in 0..10 {
fun_name(&mut x);
}
}
}
fn $0fun_name(x: &mut i32) {
*x += 1;
}
"#, "#,
); );
} }

View file

@ -17,6 +17,7 @@
//! deref_mut: deref //! deref_mut: deref
//! index: sized //! index: sized
//! fn: //! fn:
//! try:
//! pin: //! pin:
//! future: pin //! future: pin
//! option: //! option:
@ -266,6 +267,28 @@ pub mod ops {
} }
pub use self::function::{Fn, FnMut, FnOnce}; pub use self::function::{Fn, FnMut, FnOnce};
// endregion:fn // endregion:fn
// region:try
mod try_ {
pub enum ControlFlow<B, C = ()> {
Continue(C),
Break(B),
}
pub trait FromResidual<R = Self::Residual> {
#[lang = "from_residual"]
fn from_residual(residual: R) -> Self;
}
#[lang = "try"]
pub trait Try: FromResidual<Self::Residual> {
type Output;
type Residual;
#[lang = "from_output"]
fn from_output(output: Self::Output) -> Self;
#[lang = "branch"]
fn branch(self) -> ControlFlow<Self::Residual, Self::Output>;
}
}
pub use self::try_::{ControlFlow, FromResidual, Try};
// endregion:try
} }
// region:eq // region:eq