-
Notifications
You must be signed in to change notification settings - Fork 12.9k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Pattern errors are too imprecise and should be removed in favor of MIR borrowck #45600
Comments
I'm marking this as deferred. I believe that indeed MIR-based borrowck can generalize many of these patterns, particularly once @pnkfelix's work is done. |
Re-triaging for #56754. NLL-fixed-by-NLL(via PRs #49870 and #54034). Normally that would mean I'd leave the issue open, but I think the tracking issue for #15287 already serves in terms of tracking the feature that is being requested here. (Note that when you have NLL enabled, on nightly, you get a suggestion to enable |
Ah there is the second example that was given in the issue: struct Foo(String, String);
fn x(f: Foo) {
match f {
Foo(a, ref b) => {}
}
} which still errors under NLL. The argument given above is that we already allow partial moves (when you use I'll reopen this issue to allow discussion of that case (which is indeed something that we might fix in the future.) |
in terms of the re-triaging effort, I'll call this NLL-complete, P-medium. |
oh also I think this is probably related to #54987 |
which means it is probably related to Centril/rfcs#16 |
ran into a similar issue recently - the following code is rejected: fn bind(a: &Option<String>, b: Option<String>) {
match (a, b) {
(&Some(ref x), Some(y)) => println!("{} {}", x, y),
_ => (),
}
}
but |
I made a test on top of #60125 (which continues evaluating after "misc checking 2" errors) disabling |
I think the main problem is that the 2015 edition doesn't use NLL yet, so we'd suddenly accept unsound code if we removed the match checks. |
We could remove the checks only in 2018, right? |
Yes |
I want to briefly bring this ticket up in tomorrow's meeting. |
Changed the title to make it clearer that this refers to the errors that are not part of #15287. The next steps for this are:
|
Discussed in compiler team meeting, removing nomination as it seems like we sort of know what steps we need now. |
@rust-lang/lang can you confirm wether relying on the MIR borrowck instead of checking binding modes in patterns ("cannot bind by-move and by-ref in the same pattern") without an RFC? |
So this is just for https://doc.rust-lang.org/error-index.html#E0009 as discussed in @matthewjasper's report? in #15287 (comment)? |
I think it is indeed. |
I would say we should deal with #15287 first and then revisit this once that's done. |
@estebank I personally like the answer in #15287 (comment) of thinking of these restrictions as "AST borrowck details", and thus removing them if MIR borrowck is sufficient in in the spirit of NLL's approval. |
Is there a plan for this now that #15287 is done? I just ran into the confusing "cannot bind by-move and by-ref in the same pattern" message today. |
@lily-commure I don't believe we have any current plans. We have however relaxed |
We discussed this on yesterday's language team meeting. Our conclusion was that there's no good (e.g. soundness related) reason to keep this an error and that it would be an improvement to allow this code. We noted that a good test suite would be needed before stabilizing. Moreover, we thought that it should remain an error to move out of a field where the data type implements I will implement this change along with adding the tests, with a new fresh tracking issue. |
Update by @pnkfelix :
NLL (aka MIR-borrowck) addresses the first case given in the issue, but not this one:
Original bug report follows
For this code:
rustc produces:
This error comes from
check_legality_of_move_bindings
inlibrustc_const_eval/check_match.rs
.The message is at least arguably incorrect: since
x
is not used in the pattern guard, it doesn't make much sense to claim that the code "moves value into pattern guard".I suppose you could argue that all bindings in a pattern are always moved/copied to the pattern guard, and the fact that this one is unused doesn't matter. Under that interpretation, it's a matter of improving the error message, and this issue is a dupe of rust-lang/rfcs#684 which seeks to improve that error message in general.
However, I think the code should be allowed, as it's not unsound. I'm not too familiar with compiler internals, but it seems like once MIR borrow checking is finished, it would catch any actually-unsound moves into pattern guards, and this error could be removed altogether. Is that correct?
The other two errors generated by
check_legality_of_move_bindings
have similar issues:For one thing, this error seems to be a subset of a different one from elsewhere in
check_match.rs
, "pattern bindings are not allowed after an@
" - at least, I can't think of any code that would trigger this and not that. (Feel free to let me know what I'm missing.) But in any case, this is overly conservative. If we havethen it should be legal to match
x @ Foo(y)
(becausey
isCopy
). Again, this seems like something MIR borrowck might be able to handle better, although the frontend might have to make sure the copy comes from the right place (pre- or post-move).I don't even know what this error is trying to prevent. The following works fine even with the existing borrowck (
f
properly becomes a partially moved value):So what is wrong with putting b into the pattern?
The text was updated successfully, but these errors were encountered: