mirror of
https://github.com/rust-lang/rust-analyzer
synced 2025-01-28 04:45:05 +00:00
Merge #9788
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:
commit
ec753847bd
4 changed files with 149 additions and 21 deletions
|
@ -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.
|
||||||
|
|
|
@ -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,
|
||||||
|
|
|
@ -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;
|
||||||
|
}
|
||||||
"#,
|
"#,
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
|
@ -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
|
||||||
|
|
Loading…
Reference in a new issue