mirror of
https://github.com/rust-lang/rust-analyzer
synced 2025-01-01 07:48:45 +00:00
Auto merge of #14961 - HKalbasi:mir-fix, r=HKalbasi
Fix drop scopes problems in mir Fix false positives of `need-mut` emerged from #14955 There are still 5 `need-mut` false positives on self, all related to `izip!` macro hygenic issue. I will try to do something about that before monday release.
This commit is contained in:
commit
4fb1df6b7a
5 changed files with 237 additions and 24 deletions
|
@ -1896,7 +1896,14 @@ impl Evaluator<'_> {
|
||||||
let mir_body = self
|
let mir_body = self
|
||||||
.db
|
.db
|
||||||
.monomorphized_mir_body(def, generic_args, self.trait_env.clone())
|
.monomorphized_mir_body(def, generic_args, self.trait_env.clone())
|
||||||
.map_err(|e| MirEvalError::MirLowerError(imp, e))?;
|
.map_err(|e| {
|
||||||
|
MirEvalError::InFunction(
|
||||||
|
Either::Left(imp),
|
||||||
|
Box::new(MirEvalError::MirLowerError(imp, e)),
|
||||||
|
span,
|
||||||
|
locals.body.owner,
|
||||||
|
)
|
||||||
|
})?;
|
||||||
let result = self.interpret_mir(&mir_body, arg_bytes.iter().cloned()).map_err(|e| {
|
let result = self.interpret_mir(&mir_body, arg_bytes.iter().cloned()).map_err(|e| {
|
||||||
MirEvalError::InFunction(Either::Left(imp), Box::new(e), span, locals.body.owner)
|
MirEvalError::InFunction(Either::Left(imp), Box::new(e), span, locals.body.owner)
|
||||||
})?;
|
})?;
|
||||||
|
|
|
@ -1,5 +1,6 @@
|
||||||
use base_db::{fixture::WithFixture, FileId};
|
use base_db::{fixture::WithFixture, FileId};
|
||||||
use hir_def::db::DefDatabase;
|
use hir_def::db::DefDatabase;
|
||||||
|
use syntax::{TextRange, TextSize};
|
||||||
|
|
||||||
use crate::{db::HirDatabase, test_db::TestDB, Interner, Substitution};
|
use crate::{db::HirDatabase, test_db::TestDB, Interner, Substitution};
|
||||||
|
|
||||||
|
@ -45,7 +46,21 @@ fn check_pass_and_stdio(ra_fixture: &str, expected_stdout: &str, expected_stderr
|
||||||
match x {
|
match x {
|
||||||
Err(e) => {
|
Err(e) => {
|
||||||
let mut err = String::new();
|
let mut err = String::new();
|
||||||
let span_formatter = |file, range| format!("{:?} {:?}", file, range);
|
let line_index = |size: TextSize| {
|
||||||
|
let mut size = u32::from(size) as usize;
|
||||||
|
let mut lines = ra_fixture.lines().enumerate();
|
||||||
|
while let Some((i, l)) = lines.next() {
|
||||||
|
if let Some(x) = size.checked_sub(l.len()) {
|
||||||
|
size = x;
|
||||||
|
} else {
|
||||||
|
return (i, size);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
(usize::MAX, size)
|
||||||
|
};
|
||||||
|
let span_formatter = |file, range: TextRange| {
|
||||||
|
format!("{:?} {:?}..{:?}", file, line_index(range.start()), line_index(range.end()))
|
||||||
|
};
|
||||||
e.pretty_print(&mut err, &db, span_formatter).unwrap();
|
e.pretty_print(&mut err, &db, span_formatter).unwrap();
|
||||||
panic!("Error in interpreting: {err}");
|
panic!("Error in interpreting: {err}");
|
||||||
}
|
}
|
||||||
|
@ -115,6 +130,58 @@ fn main() {
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn drop_if_let() {
|
||||||
|
check_pass(
|
||||||
|
r#"
|
||||||
|
//- minicore: drop, add, option, cell, builtin_impls
|
||||||
|
|
||||||
|
use core::cell::Cell;
|
||||||
|
|
||||||
|
struct X<'a>(&'a Cell<i32>);
|
||||||
|
impl<'a> Drop for X<'a> {
|
||||||
|
fn drop(&mut self) {
|
||||||
|
self.0.set(self.0.get() + 1)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn should_not_reach() {
|
||||||
|
_ // FIXME: replace this function with panic when that works
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn main() {
|
||||||
|
let s = Cell::new(0);
|
||||||
|
let x = Some(X(&s));
|
||||||
|
if let Some(y) = x {
|
||||||
|
if s.get() != 0 {
|
||||||
|
should_not_reach();
|
||||||
|
}
|
||||||
|
if s.get() != 0 {
|
||||||
|
should_not_reach();
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
should_not_reach();
|
||||||
|
}
|
||||||
|
if s.get() != 1 {
|
||||||
|
should_not_reach();
|
||||||
|
}
|
||||||
|
let x = Some(X(&s));
|
||||||
|
if let None = x {
|
||||||
|
should_not_reach();
|
||||||
|
} else {
|
||||||
|
if s.get() != 1 {
|
||||||
|
should_not_reach();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if s.get() != 1 {
|
||||||
|
should_not_reach();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
"#,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn drop_in_place() {
|
fn drop_in_place() {
|
||||||
check_pass(
|
check_pass(
|
||||||
|
|
|
@ -48,6 +48,7 @@ struct LoopBlocks {
|
||||||
/// `None` for loops that are not terminating
|
/// `None` for loops that are not terminating
|
||||||
end: Option<BasicBlockId>,
|
end: Option<BasicBlockId>,
|
||||||
place: Place,
|
place: Place,
|
||||||
|
drop_scope_index: usize,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug, Clone, Default)]
|
#[derive(Debug, Clone, Default)]
|
||||||
|
@ -101,6 +102,35 @@ pub enum MirLowerError {
|
||||||
GenericArgNotProvided(TypeOrConstParamId, Substitution),
|
GenericArgNotProvided(TypeOrConstParamId, Substitution),
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// A token to ensuring that each drop scope is popped at most once, thanks to the compiler that checks moves.
|
||||||
|
struct DropScopeToken;
|
||||||
|
impl DropScopeToken {
|
||||||
|
fn pop_and_drop(self, ctx: &mut MirLowerCtx<'_>, current: BasicBlockId) -> BasicBlockId {
|
||||||
|
std::mem::forget(self);
|
||||||
|
ctx.pop_drop_scope_internal(current)
|
||||||
|
}
|
||||||
|
|
||||||
|
/// It is useful when we want a drop scope is syntaxically closed, but we don't want to execute any drop
|
||||||
|
/// code. Either when the control flow is diverging (so drop code doesn't reached) or when drop is handled
|
||||||
|
/// for us (for example a block that ended with a return statement. Return will drop everything, so the block shouldn't
|
||||||
|
/// do anything)
|
||||||
|
fn pop_assume_dropped(self, ctx: &mut MirLowerCtx<'_>) {
|
||||||
|
std::mem::forget(self);
|
||||||
|
ctx.pop_drop_scope_assume_dropped_internal();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Uncomment this to make `DropScopeToken` a drop bomb. Unfortunately we can't do this in release, since
|
||||||
|
// in cases that mir lowering fails, we don't handle (and don't need to handle) drop scopes so it will be
|
||||||
|
// actually reached. `pop_drop_scope_assert_finished` will also detect this case, but doesn't show useful
|
||||||
|
// stack trace.
|
||||||
|
//
|
||||||
|
// impl Drop for DropScopeToken {
|
||||||
|
// fn drop(&mut self) {
|
||||||
|
// never!("Drop scope doesn't popped");
|
||||||
|
// }
|
||||||
|
// }
|
||||||
|
|
||||||
impl MirLowerError {
|
impl MirLowerError {
|
||||||
pub fn pretty_print(
|
pub fn pretty_print(
|
||||||
&self,
|
&self,
|
||||||
|
@ -506,7 +536,6 @@ impl<'ctx> MirLowerCtx<'ctx> {
|
||||||
self.lower_loop(current, place.clone(), Some(*label), expr_id.into(), |this, begin| {
|
self.lower_loop(current, place.clone(), Some(*label), expr_id.into(), |this, begin| {
|
||||||
if let Some(current) = this.lower_block_to_place(statements, begin, *tail, place, expr_id.into())? {
|
if let Some(current) = this.lower_block_to_place(statements, begin, *tail, place, expr_id.into())? {
|
||||||
let end = this.current_loop_end()?;
|
let end = this.current_loop_end()?;
|
||||||
let current = this.pop_drop_scope(current);
|
|
||||||
this.set_goto(current, end, expr_id.into());
|
this.set_goto(current, end, expr_id.into());
|
||||||
}
|
}
|
||||||
Ok(())
|
Ok(())
|
||||||
|
@ -516,30 +545,39 @@ impl<'ctx> MirLowerCtx<'ctx> {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
Expr::Loop { body, label } => self.lower_loop(current, place, *label, expr_id.into(), |this, begin| {
|
Expr::Loop { body, label } => self.lower_loop(current, place, *label, expr_id.into(), |this, begin| {
|
||||||
if let Some((_, current)) = this.lower_expr_as_place(begin, *body, true)? {
|
let scope = this.push_drop_scope();
|
||||||
let current = this.pop_drop_scope(current);
|
if let Some((_, mut current)) = this.lower_expr_as_place(begin, *body, true)? {
|
||||||
|
current = scope.pop_and_drop(this, current);
|
||||||
this.set_goto(current, begin, expr_id.into());
|
this.set_goto(current, begin, expr_id.into());
|
||||||
|
} else {
|
||||||
|
scope.pop_assume_dropped(this);
|
||||||
}
|
}
|
||||||
Ok(())
|
Ok(())
|
||||||
}),
|
}),
|
||||||
Expr::While { condition, body, label } => {
|
Expr::While { condition, body, label } => {
|
||||||
self.lower_loop(current, place, *label, expr_id.into(),|this, begin| {
|
self.lower_loop(current, place, *label, expr_id.into(),|this, begin| {
|
||||||
|
let scope = this.push_drop_scope();
|
||||||
let Some((discr, to_switch)) = this.lower_expr_to_some_operand(*condition, begin)? else {
|
let Some((discr, to_switch)) = this.lower_expr_to_some_operand(*condition, begin)? else {
|
||||||
return Ok(());
|
return Ok(());
|
||||||
};
|
};
|
||||||
let end = this.current_loop_end()?;
|
let fail_cond = this.new_basic_block();
|
||||||
let after_cond = this.new_basic_block();
|
let after_cond = this.new_basic_block();
|
||||||
this.set_terminator(
|
this.set_terminator(
|
||||||
to_switch,
|
to_switch,
|
||||||
TerminatorKind::SwitchInt {
|
TerminatorKind::SwitchInt {
|
||||||
discr,
|
discr,
|
||||||
targets: SwitchTargets::static_if(1, after_cond, end),
|
targets: SwitchTargets::static_if(1, after_cond, fail_cond),
|
||||||
},
|
},
|
||||||
expr_id.into(),
|
expr_id.into(),
|
||||||
);
|
);
|
||||||
|
let fail_cond = this.drop_until_scope(this.drop_scopes.len() - 1, fail_cond);
|
||||||
|
let end = this.current_loop_end()?;
|
||||||
|
this.set_goto(fail_cond, end, expr_id.into());
|
||||||
if let Some((_, block)) = this.lower_expr_as_place(after_cond, *body, true)? {
|
if let Some((_, block)) = this.lower_expr_as_place(after_cond, *body, true)? {
|
||||||
let block = this.pop_drop_scope(block);
|
let block = scope.pop_and_drop(this, block);
|
||||||
this.set_goto(block, begin, expr_id.into());
|
this.set_goto(block, begin, expr_id.into());
|
||||||
|
} else {
|
||||||
|
scope.pop_assume_dropped(this);
|
||||||
}
|
}
|
||||||
Ok(())
|
Ok(())
|
||||||
})
|
})
|
||||||
|
@ -637,7 +675,9 @@ impl<'ctx> MirLowerCtx<'ctx> {
|
||||||
Some(l) => self.labeled_loop_blocks.get(l).ok_or(MirLowerError::UnresolvedLabel)?,
|
Some(l) => self.labeled_loop_blocks.get(l).ok_or(MirLowerError::UnresolvedLabel)?,
|
||||||
None => self.current_loop_blocks.as_ref().ok_or(MirLowerError::ContinueWithoutLoop)?,
|
None => self.current_loop_blocks.as_ref().ok_or(MirLowerError::ContinueWithoutLoop)?,
|
||||||
};
|
};
|
||||||
self.set_goto(current, loop_data.begin, expr_id.into());
|
let begin = loop_data.begin;
|
||||||
|
current = self.drop_until_scope(loop_data.drop_scope_index, current);
|
||||||
|
self.set_goto(current, begin, expr_id.into());
|
||||||
Ok(None)
|
Ok(None)
|
||||||
},
|
},
|
||||||
&Expr::Break { expr, label } => {
|
&Expr::Break { expr, label } => {
|
||||||
|
@ -651,10 +691,16 @@ impl<'ctx> MirLowerCtx<'ctx> {
|
||||||
};
|
};
|
||||||
current = c;
|
current = c;
|
||||||
}
|
}
|
||||||
let end = match label {
|
let (end, drop_scope) = match label {
|
||||||
Some(l) => self.labeled_loop_blocks.get(&l).ok_or(MirLowerError::UnresolvedLabel)?.end.expect("We always generate end for labeled loops"),
|
Some(l) => {
|
||||||
None => self.current_loop_end()?,
|
let loop_blocks = self.labeled_loop_blocks.get(&l).ok_or(MirLowerError::UnresolvedLabel)?;
|
||||||
|
(loop_blocks.end.expect("We always generate end for labeled loops"), loop_blocks.drop_scope_index)
|
||||||
|
},
|
||||||
|
None => {
|
||||||
|
(self.current_loop_end()?, self.current_loop_blocks.as_ref().unwrap().drop_scope_index)
|
||||||
|
},
|
||||||
};
|
};
|
||||||
|
current = self.drop_until_scope(drop_scope, current);
|
||||||
self.set_goto(current, end, expr_id.into());
|
self.set_goto(current, end, expr_id.into());
|
||||||
Ok(None)
|
Ok(None)
|
||||||
}
|
}
|
||||||
|
@ -1378,7 +1424,7 @@ impl<'ctx> MirLowerCtx<'ctx> {
|
||||||
let begin = self.new_basic_block();
|
let begin = self.new_basic_block();
|
||||||
let prev = mem::replace(
|
let prev = mem::replace(
|
||||||
&mut self.current_loop_blocks,
|
&mut self.current_loop_blocks,
|
||||||
Some(LoopBlocks { begin, end: None, place }),
|
Some(LoopBlocks { begin, end: None, place, drop_scope_index: self.drop_scopes.len() }),
|
||||||
);
|
);
|
||||||
let prev_label = if let Some(label) = label {
|
let prev_label = if let Some(label) = label {
|
||||||
// We should generate the end now, to make sure that it wouldn't change later. It is
|
// We should generate the end now, to make sure that it wouldn't change later. It is
|
||||||
|
@ -1391,7 +1437,6 @@ impl<'ctx> MirLowerCtx<'ctx> {
|
||||||
None
|
None
|
||||||
};
|
};
|
||||||
self.set_goto(prev_block, begin, span);
|
self.set_goto(prev_block, begin, span);
|
||||||
self.push_drop_scope();
|
|
||||||
f(self, begin)?;
|
f(self, begin)?;
|
||||||
let my = mem::replace(&mut self.current_loop_blocks, prev).ok_or(
|
let my = mem::replace(&mut self.current_loop_blocks, prev).ok_or(
|
||||||
MirLowerError::ImplementationError("current_loop_blocks is corrupt".to_string()),
|
MirLowerError::ImplementationError("current_loop_blocks is corrupt".to_string()),
|
||||||
|
@ -1489,6 +1534,7 @@ impl<'ctx> MirLowerCtx<'ctx> {
|
||||||
place: Place,
|
place: Place,
|
||||||
span: MirSpan,
|
span: MirSpan,
|
||||||
) -> Result<Option<Idx<BasicBlock>>> {
|
) -> Result<Option<Idx<BasicBlock>>> {
|
||||||
|
let scope = self.push_drop_scope();
|
||||||
for statement in statements.iter() {
|
for statement in statements.iter() {
|
||||||
match statement {
|
match statement {
|
||||||
hir_def::hir::Statement::Let { pat, initializer, else_branch, type_ref: _ } => {
|
hir_def::hir::Statement::Let { pat, initializer, else_branch, type_ref: _ } => {
|
||||||
|
@ -1497,6 +1543,7 @@ impl<'ctx> MirLowerCtx<'ctx> {
|
||||||
let Some((init_place, c)) =
|
let Some((init_place, c)) =
|
||||||
self.lower_expr_as_place(current, *expr_id, true)?
|
self.lower_expr_as_place(current, *expr_id, true)?
|
||||||
else {
|
else {
|
||||||
|
scope.pop_assume_dropped(self);
|
||||||
return Ok(None);
|
return Ok(None);
|
||||||
};
|
};
|
||||||
current = c;
|
current = c;
|
||||||
|
@ -1528,18 +1575,25 @@ impl<'ctx> MirLowerCtx<'ctx> {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
hir_def::hir::Statement::Expr { expr, has_semi: _ } => {
|
hir_def::hir::Statement::Expr { expr, has_semi: _ } => {
|
||||||
self.push_drop_scope();
|
let scope2 = self.push_drop_scope();
|
||||||
let Some((_, c)) = self.lower_expr_as_place(current, *expr, true)? else {
|
let Some((_, c)) = self.lower_expr_as_place(current, *expr, true)? else {
|
||||||
|
scope2.pop_assume_dropped(self);
|
||||||
|
scope.pop_assume_dropped(self);
|
||||||
return Ok(None);
|
return Ok(None);
|
||||||
};
|
};
|
||||||
current = self.pop_drop_scope(c);
|
current = scope2.pop_and_drop(self, c);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
match tail {
|
if let Some(tail) = tail {
|
||||||
Some(tail) => self.lower_expr_to_place(tail, place, current),
|
let Some(c) = self.lower_expr_to_place(tail, place, current)? else {
|
||||||
None => Ok(Some(current)),
|
scope.pop_assume_dropped(self);
|
||||||
|
return Ok(None);
|
||||||
|
};
|
||||||
|
current = c;
|
||||||
}
|
}
|
||||||
|
current = scope.pop_and_drop(self, current);
|
||||||
|
Ok(Some(current))
|
||||||
}
|
}
|
||||||
|
|
||||||
fn lower_params_and_bindings(
|
fn lower_params_and_bindings(
|
||||||
|
@ -1625,16 +1679,34 @@ impl<'ctx> MirLowerCtx<'ctx> {
|
||||||
current
|
current
|
||||||
}
|
}
|
||||||
|
|
||||||
fn push_drop_scope(&mut self) {
|
fn push_drop_scope(&mut self) -> DropScopeToken {
|
||||||
self.drop_scopes.push(DropScope::default());
|
self.drop_scopes.push(DropScope::default());
|
||||||
|
DropScopeToken
|
||||||
}
|
}
|
||||||
|
|
||||||
fn pop_drop_scope(&mut self, mut current: BasicBlockId) -> BasicBlockId {
|
/// Don't call directly
|
||||||
|
fn pop_drop_scope_assume_dropped_internal(&mut self) {
|
||||||
|
self.drop_scopes.pop();
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Don't call directly
|
||||||
|
fn pop_drop_scope_internal(&mut self, mut current: BasicBlockId) -> BasicBlockId {
|
||||||
let scope = self.drop_scopes.pop().unwrap();
|
let scope = self.drop_scopes.pop().unwrap();
|
||||||
self.emit_drop_and_storage_dead_for_scope(&scope, &mut current);
|
self.emit_drop_and_storage_dead_for_scope(&scope, &mut current);
|
||||||
current
|
current
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn pop_drop_scope_assert_finished(
|
||||||
|
&mut self,
|
||||||
|
mut current: BasicBlockId,
|
||||||
|
) -> Result<BasicBlockId> {
|
||||||
|
current = self.pop_drop_scope_internal(current);
|
||||||
|
if !self.drop_scopes.is_empty() {
|
||||||
|
implementation_error!("Mismatched count between drop scope push and pops");
|
||||||
|
}
|
||||||
|
Ok(current)
|
||||||
|
}
|
||||||
|
|
||||||
fn emit_drop_and_storage_dead_for_scope(
|
fn emit_drop_and_storage_dead_for_scope(
|
||||||
&mut self,
|
&mut self,
|
||||||
scope: &DropScope,
|
scope: &DropScope,
|
||||||
|
@ -1728,7 +1800,7 @@ pub fn mir_body_for_closure_query(
|
||||||
|_| true,
|
|_| true,
|
||||||
)?;
|
)?;
|
||||||
if let Some(current) = ctx.lower_expr_to_place(*root, return_slot().into(), current)? {
|
if let Some(current) = ctx.lower_expr_to_place(*root, return_slot().into(), current)? {
|
||||||
let current = ctx.pop_drop_scope(current);
|
let current = ctx.pop_drop_scope_assert_finished(current)?;
|
||||||
ctx.set_terminator(current, TerminatorKind::Return, (*root).into());
|
ctx.set_terminator(current, TerminatorKind::Return, (*root).into());
|
||||||
}
|
}
|
||||||
let mut upvar_map: FxHashMap<LocalId, Vec<(&CapturedItem, usize)>> = FxHashMap::default();
|
let mut upvar_map: FxHashMap<LocalId, Vec<(&CapturedItem, usize)>> = FxHashMap::default();
|
||||||
|
@ -1863,7 +1935,7 @@ pub fn lower_to_mir(
|
||||||
ctx.lower_params_and_bindings([].into_iter(), binding_picker)?
|
ctx.lower_params_and_bindings([].into_iter(), binding_picker)?
|
||||||
};
|
};
|
||||||
if let Some(current) = ctx.lower_expr_to_place(root_expr, return_slot().into(), current)? {
|
if let Some(current) = ctx.lower_expr_to_place(root_expr, return_slot().into(), current)? {
|
||||||
let current = ctx.pop_drop_scope(current);
|
let current = ctx.pop_drop_scope_assert_finished(current)?;
|
||||||
ctx.set_terminator(current, TerminatorKind::Return, root_expr.into());
|
ctx.set_terminator(current, TerminatorKind::Return, root_expr.into());
|
||||||
}
|
}
|
||||||
Ok(ctx.result)
|
Ok(ctx.result)
|
||||||
|
|
|
@ -512,6 +512,38 @@ fn main() {
|
||||||
f(x);
|
f(x);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
"#,
|
||||||
|
);
|
||||||
|
check_diagnostics(
|
||||||
|
r#"
|
||||||
|
fn check(_: i32) -> bool {
|
||||||
|
false
|
||||||
|
}
|
||||||
|
fn main() {
|
||||||
|
loop {
|
||||||
|
let x = 1;
|
||||||
|
if check(x) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
let y = (1, 2);
|
||||||
|
if check(y.1) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
let z = (1, 2);
|
||||||
|
match z {
|
||||||
|
(k @ 5, ref mut t) if { continue; } => {
|
||||||
|
//^^^^^^^^^ 💡 error: cannot mutate immutable variable `z`
|
||||||
|
*t = 5;
|
||||||
|
}
|
||||||
|
_ => {
|
||||||
|
let y = (1, 2);
|
||||||
|
if check(y.1) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
"#,
|
"#,
|
||||||
);
|
);
|
||||||
check_diagnostics(
|
check_diagnostics(
|
||||||
|
@ -592,7 +624,7 @@ fn f((x, y): (i32, i32)) {
|
||||||
fn for_loop() {
|
fn for_loop() {
|
||||||
check_diagnostics(
|
check_diagnostics(
|
||||||
r#"
|
r#"
|
||||||
//- minicore: iterators
|
//- minicore: iterators, copy
|
||||||
fn f(x: [(i32, u8); 10]) {
|
fn f(x: [(i32, u8); 10]) {
|
||||||
for (a, mut b) in x {
|
for (a, mut b) in x {
|
||||||
//^^^^^ 💡 weak: variable does not need to be mutable
|
//^^^^^ 💡 weak: variable does not need to be mutable
|
||||||
|
@ -604,6 +636,28 @@ fn f(x: [(i32, u8); 10]) {
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn while_let() {
|
||||||
|
check_diagnostics(
|
||||||
|
r#"
|
||||||
|
//- minicore: iterators, copy
|
||||||
|
fn f(x: [(i32, u8); 10]) {
|
||||||
|
let mut it = x.into_iter();
|
||||||
|
while let Some((a, mut b)) = it.next() {
|
||||||
|
//^^^^^ 💡 weak: variable does not need to be mutable
|
||||||
|
while let Some((c, mut d)) = it.next() {
|
||||||
|
//^^^^^ 💡 weak: variable does not need to be mutable
|
||||||
|
a = 2;
|
||||||
|
//^^^^^ 💡 error: cannot mutate immutable variable `a`
|
||||||
|
c = 2;
|
||||||
|
//^^^^^ 💡 error: cannot mutate immutable variable `c`
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
"#,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn index() {
|
fn index() {
|
||||||
check_diagnostics(
|
check_diagnostics(
|
||||||
|
|
|
@ -726,6 +726,19 @@ pub mod ops {
|
||||||
pub trait AddAssign<Rhs = Self> {
|
pub trait AddAssign<Rhs = Self> {
|
||||||
fn add_assign(&mut self, rhs: Rhs);
|
fn add_assign(&mut self, rhs: Rhs);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// region:builtin_impls
|
||||||
|
macro_rules! add_impl {
|
||||||
|
($($t:ty)*) => ($(
|
||||||
|
impl const Add for $t {
|
||||||
|
type Output = $t;
|
||||||
|
fn add(self, other: $t) -> $t { self + other }
|
||||||
|
}
|
||||||
|
)*)
|
||||||
|
}
|
||||||
|
|
||||||
|
add_impl! { usize u8 u16 u32 u64 u128 isize i8 i16 i32 i64 i128 f32 f64 }
|
||||||
|
// endregion:builtin_impls
|
||||||
// endregion:add
|
// endregion:add
|
||||||
|
|
||||||
// region:generator
|
// region:generator
|
||||||
|
|
Loading…
Reference in a new issue