Skip to content

Commit 45859b7

Browse files
committed
Auto merge of #63118 - Centril:stabilize-bind-by-move, r=matthewjasper
Stabilize `bind_by_move_pattern_guards` in Rust 1.39.0 Closes #15287. After stabilizing `#![feature(bind_by_move_pattern_guards)]`, you can now use bind-by-move bindings in patterns and take references to those bindings in `if` guards of `match` expressions. For example, the following now becomes legal: ```rust fn main() { let array: Box<[u8; 4]> = Box::new([1, 2, 3, 4]); match array { nums // ---- `nums` is bound by move. if nums.iter().sum::<u8>() == 10 // ^------ `.iter()` implicitly takes a reference to `nums`. => { drop(nums); // --------- Legal as `nums` was bound by move and so we have ownership. } _ => unreachable!(), } } ``` r? @matthewjasper
2 parents 824383d + aaa9762 commit 45859b7

Some content is hidden

Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.

42 files changed

+51
-427
lines changed

src/libcore/lib.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -87,7 +87,7 @@
8787
#![feature(link_llvm_intrinsics)]
8888
#![feature(never_type)]
8989
#![feature(nll)]
90-
#![feature(bind_by_move_pattern_guards)]
90+
#![cfg_attr(boostrap_stdarch_ignore_this, feature(bind_by_move_pattern_guards))]
9191
#![feature(exhaustive_patterns)]
9292
#![feature(no_core)]
9393
#![feature(on_unimplemented)]

src/librustc_mir/build/matches/mod.rs

+3-7
Original file line numberDiff line numberDiff line change
@@ -1345,13 +1345,9 @@ impl<'a, 'tcx> Builder<'a, 'tcx> {
13451345
/// any, and then branches to the arm. Returns the block for the case where
13461346
/// the guard fails.
13471347
///
1348-
/// Note: we check earlier that if there is a guard, there cannot be move
1349-
/// bindings (unless feature(bind_by_move_pattern_guards) is used). This
1350-
/// isn't really important for the self-consistency of this fn, but the
1351-
/// reason for it should be clear: after we've done the assignments, if
1352-
/// there were move bindings, further tests would be a use-after-move.
1353-
/// bind_by_move_pattern_guards avoids this by only moving the binding once
1354-
/// the guard has evaluated to true (see below).
1348+
/// Note: we do not check earlier that if there is a guard,
1349+
/// there cannot be move bindings. We avoid a use-after-move by only
1350+
/// moving the binding once the guard has evaluated to true (see below).
13551351
fn bind_and_guard_matched_candidate<'pat>(
13561352
&mut self,
13571353
candidate: Candidate<'pat, 'tcx>,

src/librustc_mir/error_codes.rs

+7-78
Original file line numberDiff line numberDiff line change
@@ -157,81 +157,6 @@ match x {
157157
See also the error E0303.
158158
"##,
159159

160-
E0008: r##"
161-
Names bound in match arms retain their type in pattern guards. As such, if a
162-
name is bound by move in a pattern, it should also be moved to wherever it is
163-
referenced in the pattern guard code. Doing so however would prevent the name
164-
from being available in the body of the match arm. Consider the following:
165-
166-
```compile_fail,E0008
167-
match Some("hi".to_string()) {
168-
Some(s) if s.len() == 0 => {}, // use s.
169-
_ => {},
170-
}
171-
```
172-
173-
The variable `s` has type `String`, and its use in the guard is as a variable of
174-
type `String`. The guard code effectively executes in a separate scope to the
175-
body of the arm, so the value would be moved into this anonymous scope and
176-
therefore becomes unavailable in the body of the arm.
177-
178-
The problem above can be solved by using the `ref` keyword.
179-
180-
```
181-
match Some("hi".to_string()) {
182-
Some(ref s) if s.len() == 0 => {},
183-
_ => {},
184-
}
185-
```
186-
187-
Though this example seems innocuous and easy to solve, the problem becomes clear
188-
when it encounters functions which consume the value:
189-
190-
```compile_fail,E0008
191-
struct A{}
192-
193-
impl A {
194-
fn consume(self) -> usize {
195-
0
196-
}
197-
}
198-
199-
fn main() {
200-
let a = Some(A{});
201-
match a {
202-
Some(y) if y.consume() > 0 => {}
203-
_ => {}
204-
}
205-
}
206-
```
207-
208-
In this situation, even the `ref` keyword cannot solve it, since borrowed
209-
content cannot be moved. This problem cannot be solved generally. If the value
210-
can be cloned, here is a not-so-specific solution:
211-
212-
```
213-
#[derive(Clone)]
214-
struct A{}
215-
216-
impl A {
217-
fn consume(self) -> usize {
218-
0
219-
}
220-
}
221-
222-
fn main() {
223-
let a = Some(A{});
224-
match a{
225-
Some(ref y) if y.clone().consume() > 0 => {}
226-
_ => {}
227-
}
228-
}
229-
```
230-
231-
If the value will be consumed in the pattern guard, using its clone will not
232-
move its ownership, so the code works.
233-
"##,
234-
235160
E0009: r##"
236161
In a pattern, all values that don't implement the `Copy` trait have to be bound
237162
the same way. The goal here is to avoid binding simultaneously by-move and
@@ -475,13 +400,15 @@ for item in xs {
475400
"##,
476401

477402
E0301: r##"
403+
#### Note: this error code is no longer emitted by the compiler.
404+
478405
Mutable borrows are not allowed in pattern guards, because matching cannot have
479406
side effects. Side effects could alter the matched object or the environment
480407
on which the match depends in such a way, that the match would not be
481408
exhaustive. For instance, the following would not match any arm if mutable
482409
borrows were allowed:
483410
484-
```compile_fail,E0301
411+
```compile_fail,E0596
485412
match Some(()) {
486413
None => { },
487414
option if option.take().is_none() => {
@@ -493,13 +420,15 @@ match Some(()) {
493420
"##,
494421

495422
E0302: r##"
423+
#### Note: this error code is no longer emitted by the compiler.
424+
496425
Assignments are not allowed in pattern guards, because matching cannot have
497426
side effects. Side effects could alter the matched object or the environment
498427
on which the match depends in such a way, that the match would not be
499428
exhaustive. For instance, the following would not match any arm if assignments
500429
were allowed:
501430
502-
```compile_fail,E0302
431+
```compile_fail,E0594
503432
match Some(()) {
504433
None => { },
505434
option if { option = None; false } => { },
@@ -1989,7 +1918,6 @@ When matching on a variable it cannot be mutated in the match guards, as this
19891918
could cause the match to be non-exhaustive:
19901919
19911920
```compile_fail,E0510
1992-
#![feature(bind_by_move_pattern_guards)]
19931921
let mut x = Some(0);
19941922
match x {
19951923
None => (),
@@ -2451,6 +2379,7 @@ There are some known bugs that trigger this message.
24512379

24522380
;
24532381

2382+
// E0008, // cannot bind by-move into a pattern guard
24542383
// E0298, // cannot compare constants
24552384
// E0299, // mismatched types between arms
24562385
// E0471, // constant evaluation error (in pattern)

src/librustc_mir/hair/pattern/check_match.rs

+4-86
Original file line numberDiff line numberDiff line change
@@ -5,11 +5,6 @@ use super::_match::WitnessPreference::*;
55
use super::{Pattern, PatternContext, PatternError, PatternKind};
66

77
use rustc::middle::borrowck::SignalledError;
8-
use rustc::middle::expr_use_visitor::{ConsumeMode, Delegate, ExprUseVisitor};
9-
use rustc::middle::expr_use_visitor::{LoanCause, MutateMode};
10-
use rustc::middle::expr_use_visitor as euv;
11-
use rustc::middle::mem_categorization::cmt_;
12-
use rustc::middle::region;
138
use rustc::session::Session;
149
use rustc::ty::{self, Ty, TyCtxt};
1510
use rustc::ty::subst::{InternalSubsts, SubstsRef};
@@ -36,9 +31,7 @@ crate fn check_match(tcx: TyCtxt<'_>, def_id: DefId) -> SignalledError {
3631

3732
let mut visitor = MatchVisitor {
3833
tcx,
39-
body_owner: def_id,
4034
tables: tcx.body_tables(body_id),
41-
region_scope_tree: &tcx.region_scope_tree(def_id),
4235
param_env: tcx.param_env(def_id),
4336
identity_substs: InternalSubsts::identity_for_item(tcx, def_id),
4437
signalled_error: SignalledError::NoErrorsSeen,
@@ -53,11 +46,9 @@ fn create_e0004(sess: &Session, sp: Span, error_message: String) -> DiagnosticBu
5346

5447
struct MatchVisitor<'a, 'tcx> {
5548
tcx: TyCtxt<'tcx>,
56-
body_owner: DefId,
5749
tables: &'a ty::TypeckTables<'tcx>,
5850
param_env: ty::ParamEnv<'tcx>,
5951
identity_substs: SubstsRef<'tcx>,
60-
region_scope_tree: &'a region::ScopeTree,
6152
signalled_error: SignalledError,
6253
}
6354

@@ -151,11 +142,8 @@ impl<'a, 'tcx> MatchVisitor<'a, 'tcx> {
151142

152143
// Second, if there is a guard on each arm, make sure it isn't
153144
// assigning or borrowing anything mutably.
154-
if let Some(ref guard) = arm.guard {
145+
if arm.guard.is_some() {
155146
self.signalled_error = SignalledError::SawSomeError;
156-
if !self.tcx.features().bind_by_move_pattern_guards {
157-
check_for_mutation_in_guard(self, &guard);
158-
}
159147
}
160148

161149
// Third, perform some lints.
@@ -582,19 +570,10 @@ fn check_legality_of_move_bindings(
582570
"cannot bind by-move with sub-bindings")
583571
.span_label(p.span, "binds an already bound by-move value by moving it")
584572
.emit();
585-
} else if has_guard {
586-
if !cx.tcx.features().bind_by_move_pattern_guards {
587-
let mut err = struct_span_err!(cx.tcx.sess, p.span, E0008,
588-
"cannot bind by-move into a pattern guard");
589-
err.span_label(p.span, "moves value into pattern guard");
590-
if cx.tcx.sess.opts.unstable_features.is_nightly_build() {
591-
err.help("add `#![feature(bind_by_move_pattern_guards)]` to the \
592-
crate attributes to enable");
593-
}
594-
err.emit();
573+
} else if !has_guard {
574+
if let Some(_by_ref_span) = by_ref_span {
575+
span_vec.push(p.span);
595576
}
596-
} else if let Some(_by_ref_span) = by_ref_span {
597-
span_vec.push(p.span);
598577
}
599578
};
600579

@@ -636,67 +615,6 @@ fn check_legality_of_move_bindings(
636615
}
637616
}
638617

639-
/// Ensures that a pattern guard doesn't borrow by mutable reference or assign.
640-
//
641-
// FIXME: this should be done by borrowck.
642-
fn check_for_mutation_in_guard(cx: &MatchVisitor<'_, '_>, guard: &hir::Guard) {
643-
let mut checker = MutationChecker {
644-
cx,
645-
};
646-
match guard {
647-
hir::Guard::If(expr) =>
648-
ExprUseVisitor::new(&mut checker,
649-
cx.tcx,
650-
cx.body_owner,
651-
cx.param_env,
652-
cx.region_scope_tree,
653-
cx.tables,
654-
None).walk_expr(expr),
655-
};
656-
}
657-
658-
struct MutationChecker<'a, 'tcx> {
659-
cx: &'a MatchVisitor<'a, 'tcx>,
660-
}
661-
662-
impl<'a, 'tcx> Delegate<'tcx> for MutationChecker<'a, 'tcx> {
663-
fn matched_pat(&mut self, _: &Pat, _: &cmt_<'_>, _: euv::MatchMode) {}
664-
fn consume(&mut self, _: hir::HirId, _: Span, _: &cmt_<'_>, _: ConsumeMode) {}
665-
fn consume_pat(&mut self, _: &Pat, _: &cmt_<'_>, _: ConsumeMode) {}
666-
fn borrow(&mut self,
667-
_: hir::HirId,
668-
span: Span,
669-
_: &cmt_<'_>,
670-
_: ty::Region<'tcx>,
671-
kind:ty:: BorrowKind,
672-
_: LoanCause) {
673-
match kind {
674-
ty::MutBorrow => {
675-
let mut err = struct_span_err!(self.cx.tcx.sess, span, E0301,
676-
"cannot mutably borrow in a pattern guard");
677-
err.span_label(span, "borrowed mutably in pattern guard");
678-
if self.cx.tcx.sess.opts.unstable_features.is_nightly_build() {
679-
err.help("add `#![feature(bind_by_move_pattern_guards)]` to the \
680-
crate attributes to enable");
681-
}
682-
err.emit();
683-
}
684-
ty::ImmBorrow | ty::UniqueImmBorrow => {}
685-
}
686-
}
687-
fn decl_without_init(&mut self, _: hir::HirId, _: Span) {}
688-
fn mutate(&mut self, _: hir::HirId, span: Span, _: &cmt_<'_>, mode: MutateMode) {
689-
match mode {
690-
MutateMode::JustWrite | MutateMode::WriteAndRead => {
691-
struct_span_err!(self.cx.tcx.sess, span, E0302, "cannot assign in a pattern guard")
692-
.span_label(span, "assignment in pattern guard")
693-
.emit();
694-
}
695-
MutateMode::Init => {}
696-
}
697-
}
698-
}
699-
700618
/// Forbids bindings in `@` patterns. This is necessary for memory safety,
701619
/// because of the way rvalues are handled in the borrow check. (See issue
702620
/// #14587.)

src/librustc_passes/lib.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -8,7 +8,7 @@
88

99
#![feature(in_band_lifetimes)]
1010
#![feature(nll)]
11-
#![feature(bind_by_move_pattern_guards)]
11+
#![cfg_attr(bootstrap, feature(bind_by_move_pattern_guards))]
1212

1313
#![recursion_limit="256"]
1414

src/librustdoc/lib.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,7 @@
11
#![doc(html_root_url = "https://doc.rust-lang.org/nightly/",
22
html_playground_url = "https://play.rust-lang.org/")]
33

4-
#![feature(bind_by_move_pattern_guards)]
4+
#![cfg_attr(bootstrap, feature(bind_by_move_pattern_guards))]
55
#![feature(rustc_private)]
66
#![feature(arbitrary_self_types)]
77
#![feature(box_patterns)]

src/libstd/lib.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -238,7 +238,7 @@
238238
#![feature(array_error_internals)]
239239
#![feature(asm)]
240240
#![feature(associated_type_bounds)]
241-
#![feature(bind_by_move_pattern_guards)]
241+
#![cfg_attr(bootstrap, feature(bind_by_move_pattern_guards))]
242242
#![feature(box_syntax)]
243243
#![feature(c_variadic)]
244244
#![feature(cfg_target_has_atomic)]

src/libsyntax/feature_gate/accepted.rs

+2
Original file line numberDiff line numberDiff line change
@@ -241,6 +241,8 @@ declare_features! (
241241
(accepted, underscore_const_names, "1.37.0", Some(54912), None),
242242
/// Allows free and inherent `async fn`s, `async` blocks, and `<expr>.await` expressions.
243243
(accepted, async_await, "1.39.0", Some(50547), None),
244+
/// Allows mixing bind-by-move in patterns and references to those identifiers in guards.
245+
(accepted, bind_by_move_pattern_guards, "1.39.0", Some(15287), None),
244246

245247
// -------------------------------------------------------------------------
246248
// feature-group-end: accepted features

src/libsyntax/feature_gate/active.rs

-3
Original file line numberDiff line numberDiff line change
@@ -461,9 +461,6 @@ declare_features! (
461461
/// Allows non-builtin attributes in inner attribute position.
462462
(active, custom_inner_attributes, "1.30.0", Some(54726), None),
463463

464-
/// Allows mixing bind-by-move in patterns and references to those identifiers in guards.
465-
(active, bind_by_move_pattern_guards, "1.30.0", Some(15287), None),
466-
467464
/// Allows `impl Trait` in bindings (`let`, `const`, `static`).
468465
(active, impl_trait_in_bindings, "1.30.0", Some(63065), None),
469466

src/libsyntax/lib.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -7,7 +7,7 @@
77
#![doc(html_root_url = "https://doc.rust-lang.org/nightly/",
88
test(attr(deny(warnings))))]
99

10-
#![feature(bind_by_move_pattern_guards)]
10+
#![cfg_attr(bootstrap, feature(bind_by_move_pattern_guards))]
1111
#![feature(box_syntax)]
1212
#![feature(const_fn)]
1313
#![feature(const_transmute)]

src/test/mir-opt/match-arm-scopes.rs

-2
Original file line numberDiff line numberDiff line change
@@ -8,8 +8,6 @@
88
// all of the bindings for that scope.
99
// * No drop flags are used.
1010

11-
#![feature(nll, bind_by_move_pattern_guards)]
12-
1311
fn complicated_match(cond: bool, items: (bool, bool, String)) -> i32 {
1412
match items {
1513
(false, a, s) | (a, false, s) if if cond { return 3 } else { a } => 1,

src/test/ui/bind-by-move/bind-by-move-no-guards.rs

-13
This file was deleted.

src/test/ui/bind-by-move/bind-by-move-no-guards.stderr

-11
This file was deleted.

0 commit comments

Comments
 (0)