rfcs/text/1228-placement-left-arrow.md

164 lines
6.2 KiB
Markdown
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

- Feature Name: place_left_arrow_syntax
- Start Date: 2015-07-28
- RFC PR: https://github.com/rust-lang/rfcs/pull/1228
- Rust Issue: https://github.com/rust-lang/rust/issues/27779
# Summary
Rather than trying to find a clever syntax for placement-new that leverages
the `in` keyword, instead use the syntax `PLACE_EXPR <- VALUE_EXPR`.
This takes advantage of the fact that `<-` was reserved as a token via
historical accident (that for once worked out in our favor).
# Motivation
One sentence: the syntax `a <- b` is short, can be parsed without
ambiguity, and is strongly connotated already with assignment.
Further text (essentially historical background):
There is much debate about what syntax to use for placement-new.
We started with `box (PLACE_EXPR) VALUE_EXPR`, then migrated towards
leveraging the `in` keyword instead of `box`, yielding `in (PLACE_EXPR) VALUE_EXPR`.
A lot of people disliked the `in (PLACE_EXPR) VALUE_EXPR` syntax
(see discussion from [RFC 809]).
[RFC 809]: https://github.com/rust-lang/rfcs/pull/809
In response to that discussion (and also due to personal preference)
I suggested the alternative syntax `in PLACE_EXPR { BLOCK_EXPR }`,
which is what landed when [RFC 809] was merged.
However, it is worth noting that this alternative syntax actually
failed to address a number of objections (some of which also
applied to the original `in (PLACE_EXPR) VALUE_EXPR` syntax):
* [kennytm](https://github.com/rust-lang/rfcs/pull/809#issuecomment-73071324)
> While in (place) value is syntactically unambiguous, it looks
> completely unnatural as a statement alone, mainly because there
> are no verbs in the correct place, and also using in alone is
> usually associated with iteration (for x in y) and member
> testing (elem in set).
* [petrochenkov](https://github.com/rust-lang/rfcs/pull/809#issuecomment-73142168)
> As C++11 experience has shown, when it's available, it will
> become the default method of inserting elements in containers,
> since it's never performing worse than "normal insertion" and
> is often better. So it should really have as short and
> convenient syntax as possible.
* [p1start](https://github.com/rust-lang/rfcs/pull/809#issuecomment-73837430)
> Im not a fan of in <place> { <stmts> }, simply because the
> requirement of a block suggests that its some kind of control
> flow structure, or that all the statements inside will be
> somehow run in the given <place> (or perhaps, as @m13253
> seems to have interpreted it, for all box expressions to go
> into the given place). It would be our first syntactical
> construct which is basically just an operator that has to
> have a block operand.
I believe the `PLACE_EXPR <- VALUE_EXPR` syntax addresses all of the
above concerns.
Thus cases like allocating into an arena (which needs to take as input the arena itself
and a value-expression, and returns a reference or handle for the allocated entry in the arena -- i.e. *cannot* return unit)
would look like:
```rust
let ref_1 = arena <- value_expression;
let ref_2 = arena <- value_expression;
```
compare the above against the way this would look under [RFC 809]:
```rust
let ref_1 = in arena { value_expression };
let ref_2 = in arena { value_expression };
```
# Detailed design
Extend the parser to parse `EXPR <- EXPR`.
`EXPR <- EXPR` is parsed into an AST form that is desugared in much
the same way that `in EXPR { BLOCK }` or `box (EXPR) EXPR` are
desugared (see [PR 27215]).
Thus the static and dynamic semantics of `PLACE_EXPR <- VALUE_EXPR`
are *equivalent* to `box (PLACE_EXPR) VALUE_EXPR`. Namely, it is
still an expression form that operates by:
1. Evaluate the `PLACE_EXPR` to a place
2. Evaluate `VALUE_EXPR` directly into the constructed place
3. Return the finalized place value.
(See protocol as documented in [RFC 809] for more details here.)
[PR 27215]: https://github.com/rust-lang/rust/pull/27215
This parsing form can be separately feature-gated (this RFC was
written assuming that would be the procedure). However, since
placement-`in` landed very recently ([PR 27215]) and is still
feature-gated, we can also just fold this change in with
the pre-existing `placement_in_syntax` feature gate
(though that may be non-intuitive since the keyword `in` is
no longer part of the syntactic form).
This feature has already been prototyped, see [place-left-syntax branch].
[place-left-syntax branch]: https://github.com/rust-lang/rust/compare/rust-lang:master...pnkfelix:place-left-syntax
Then, (after sufficient snapshot and/or time passes) remove the following syntaxes:
* `box (PLACE_EXPR) VALUE_EXPR`
* `in PLACE_EXPR { VALUE_BLOCK }`
That is, `PLACE_EXPR <- VALUE_EXPR` will be the "one true way" to
express placement-new.
(Note that support for `box VALUE_EXPR` will remain, and in fact, the
expression `(box ())` expression will become unambiguous and thus we
could make it legal. Because, you know, those boxes of unit have a
syntax that is really important to optimize.)
Finally, it would may be good, as part of this process, to actually
amend the text [RFC 809] itself to use the `a <- b` syntax.
At least, it seems like many people use the RFC's as a reference source
even when they are later outdated.
(An easier option though may be to just add a forward reference to this
RFC from [RFC 809], if this RFC is accepted.)
# Drawbacks
The only drawback I am aware of is this [comment from nikomataskis](https://github.com/rust-lang/rfcs/pull/809#issuecomment-73903777)
> the intent is less clear than with a devoted keyword.
Note however that this was stated with regards to a hypothetical
overloading of the `=` operator (at least that is my understanding).
I think the use of the `<-` operator can be considered sufficiently
"devoted" (i.e. separate) syntax to placate the above concern.
# Alternatives
See [different surface syntax] from the alternatives from [RFC 809].
[different surface syntax]: https://github.com/pnkfelix/rfcs/blob/fsk-placement-box-rfc/text/0000-placement-box.md#same-semantics-but-different-surface-syntax
Also, if we want to try to make it clear that this is not *just*
an assignment, we could combine `in` and `<-`, yielding e.g.:
```rust
let ref_1 = in arena <- value_expression;
let ref_2 = in arena <- value_expression;
```
# Unresolved questions
None