Skip to content

Commit 22ac838

Browse files
authoredApr 19, 2020
Merge pull request #514 from matthewjasper/drop-scopes
Document drop scopes
2 parents 3ce94ca + c5648e6 commit 22ac838

File tree

7 files changed

+375
-131
lines changed

7 files changed

+375
-131
lines changed
 

‎src/destructors.md

+358-58
Large diffs are not rendered by default.

‎src/expressions.md

+10-66
Original file line numberDiff line numberDiff line change
@@ -125,7 +125,7 @@ the remaining situations if that type is [`Sized`], then it may be possible to
125125
move the value. Only the following place expressions may be moved out of:
126126

127127
* [Variables] which are not currently borrowed.
128-
* [Temporary values](#temporary-lifetimes).
128+
* [Temporary values](#temporaries).
129129
* [Fields][field] of a place expression which can be moved out of and
130130
doesn't implement [`Drop`].
131131
* The result of [dereferencing][deref] an expression with type [`Box<T>`] and
@@ -159,69 +159,12 @@ The following expressions can be mutable place expression contexts:
159159
then evaluates the value being indexed, but not the index, in mutable place
160160
expression context.
161161

162-
### Temporary lifetimes
162+
### Temporaries
163163

164164
When using a value expression in most place expression contexts, a temporary
165165
unnamed memory location is created initialized to that value and the expression
166-
evaluates to that location instead, except if promoted to `'static`. Promotion
167-
of a value expression to a `'static` slot occurs when the expression could be
168-
written in a constant, borrowed, and dereferencing that borrow where the
169-
expression was originally written, without changing the runtime behavior. That
170-
is, the promoted expression can be evaluated at compile-time and the resulting
171-
value does not contain [interior mutability] or [destructors] (these properties
172-
are determined based on the value where possible, e.g. `&None` always has the
173-
type `&'static Option<_>`, as it contains nothing disallowed). Otherwise, the
174-
lifetime of temporary values is typically
175-
176-
- the innermost enclosing statement; the tail expression of a block is
177-
considered part of the statement that encloses the block, or
178-
- the condition expression or the loop conditional expression if the
179-
temporary is created in the condition expression of an `if` or in the loop
180-
conditional expression of a `while` expression.
181-
182-
When a temporary value expression is being created that is assigned into a
183-
[`let` declaration][let], however, the temporary is created with the lifetime of
184-
the enclosing block instead, as using the enclosing [`let` declaration][let]
185-
would be a guaranteed error (since a pointer to the temporary
186-
would be stored into a variable, but the temporary would be freed before the
187-
variable could be used). The compiler uses simple syntactic rules to decide
188-
which values are being assigned into a `let` binding, and therefore deserve a
189-
longer temporary lifetime.
190-
191-
Here are some examples:
192-
193-
- `let x = foo(&temp())`. The expression `temp()` is a value expression. As it
194-
is being borrowed, a temporary is created which will be freed after
195-
the innermost enclosing statement; in this case, the `let` declaration.
196-
- `let x = temp().foo()`. This is the same as the previous example,
197-
except that the value of `temp()` is being borrowed via autoref on a
198-
method-call. Here we are assuming that `foo()` is an `&self` method
199-
defined in some trait, say `Foo`. In other words, the expression
200-
`temp().foo()` is equivalent to `Foo::foo(&temp())`.
201-
- `let x = if foo(&temp()) {bar()} else {baz()};`. The expression `temp()` is
202-
a value expression. As the temporary is created in the condition expression
203-
of an `if`, it will be freed at the end of the condition expression;
204-
in this example before the call to `bar` or `baz` is made.
205-
- `let x = if temp().must_run_bar {bar()} else {baz()};`.
206-
Here we assume the type of `temp()` is a struct with a boolean field
207-
`must_run_bar`. As the previous example, the temporary corresponding to
208-
`temp()` will be freed at the end of the condition expression.
209-
- `while foo(&temp()) {bar();}`. The temporary containing the return value from
210-
the call to `temp()` is created in the loop conditional expression. Hence it
211-
will be freed at the end of the loop conditional expression; in this example
212-
before the call to `bar` if the loop body is executed.
213-
- `let x = &temp()`. Here, the same temporary is being assigned into
214-
`x`, rather than being passed as a parameter, and hence the
215-
temporary's lifetime is considered to be the enclosing block.
216-
- `let x = SomeStruct { foo: &temp() }`. As in the previous case, the
217-
temporary is assigned into a struct which is then assigned into a
218-
binding, and hence it is given the lifetime of the enclosing block.
219-
- `let x = [ &temp() ]`. As in the previous case, the
220-
temporary is assigned into an array which is then assigned into a
221-
binding, and hence it is given the lifetime of the enclosing block.
222-
- `let ref x = temp()`. In this case, the temporary is created using a ref
223-
binding, but the result is the same: the lifetime is extended to the enclosing
224-
block.
166+
evaluates to that location instead, except if [promoted] to a `static`. The
167+
[drop scope] of the temporary is usually the end of the enclosing statement.
225168

226169
### Implicit Borrows
227170

@@ -304,24 +247,25 @@ They are never allowed before:
304247
[deref]: expressions/operator-expr.md#the-dereference-operator
305248

306249
[destructors]: destructors.md
307-
[interior mutability]: interior-mutability.md
308-
[`Box<T>`]: ../std/boxed/struct.Box.md
250+
[drop scope]: destructors.md#drop-scopes
251+
252+
[`Box<T>`]: ../std/boxed/struct.Box.html
309253
[`Copy`]: special-types-and-traits.md#copy
310254
[`Drop`]: special-types-and-traits.md#drop
311255
[`Sized`]: special-types-and-traits.md#sized
312256
[implicit borrow]: #implicit-borrows
313257
[implicitly mutably borrowed]: #implicit-borrows
314-
[let]: statements.md#let-statements
258+
[interior mutability]: interior-mutability.md
315259
[let statement]: statements.md#let-statements
316260
[Mutable `static` items]: items/static-items.md#mutable-statics
317261
[scrutinee]: glossary.md#scrutinee
262+
[promoted]: destructors.md#constant-promotion
318263
[slice]: types/slice.md
319264
[statement]: statements.md
320265
[static variables]: items/static-items.md
321-
[Temporary values]: #temporary-lifetimes
266+
[Temporary values]: #temporaries
322267
[Variables]: variables.md
323268

324-
325269
[_ArithmeticOrLogicalExpression_]: expressions/operator-expr.md#arithmetic-and-logical-binary-operators
326270
[_ArrayExpression_]: expressions/array-expr.md
327271
[_AsyncBlockExpression_]: expressions/block-expr.md#async-blocks

‎src/expressions/loop-expr.md

+1-1
Original file line numberDiff line numberDiff line change
@@ -295,6 +295,6 @@ expression `()`.
295295
[_Pattern_]: ../patterns.md
296296
[`match` expression]: match-expr.md
297297
[scrutinee]: ../glossary.md#scrutinee
298-
[temporary values]: ../expressions.md#temporary-lifetimes
298+
[temporary values]: ../expressions.md#temporaries
299299
[_LazyBooleanOperatorExpression_]: operator-expr.md#lazy-boolean-operators
300300
[`if let` expressions]: if-expr.md#if-let-expressions

‎src/expressions/operator-expr.md

+1-1
Original file line numberDiff line numberDiff line change
@@ -455,7 +455,7 @@ assert_eq!(x, 14);
455455

456456
[place expression]: ../expressions.md#place-expressions-and-value-expressions
457457
[value expression]: ../expressions.md#place-expressions-and-value-expressions
458-
[temporary value]: ../expressions.md#temporary-lifetimes
458+
[temporary value]: ../expressions.md#temporaries
459459
[float-int]: https://github.com/rust-lang/rust/issues/10184
460460
[float-float]: https://github.com/rust-lang/rust/issues/15536
461461
[`unit` type]: ../types/tuple.md

‎src/introduction.md

+3-3
Original file line numberDiff line numberDiff line change
@@ -139,9 +139,9 @@ attention to making those sections the best that they can be.
139139
[_Expression_]: expressions.md
140140
[cargo book]: ../cargo/index.html
141141
[cargo reference]: ../cargo/reference/index.html
142-
[expressions chapter]: expressions.md
143-
[lifetime of temporaries]: expressions.md#temporary-lifetimes
144-
[linkage]: linkage.md
142+
[expressions chapter]: expressions.html
143+
[lifetime of temporaries]: expressions.html#temporaries
144+
[linkage]: linkage.html
145145
[rustc book]: ../rustc/index.html
146146
[Notation]: notation.md
147147
[Discord]: https://discord.gg/rust-lang

‎src/types/pointer.md

+1-1
Original file line numberDiff line numberDiff line change
@@ -56,4 +56,4 @@ and raw pointers.
5656
[_TypeNoBounds_]: ../types.md#type-expressions
5757
[`unsafe` operation]: ../unsafety.md
5858
[dynamically sized types]: ../dynamically-sized-types.md
59-
[temporary value]: ../expressions.md#temporary-lifetimes
59+
[temporary value]: ../expressions.md#temporaries

‎src/variables.md

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,7 @@
11
# Variables
22

33
A _variable_ is a component of a stack frame, either a named function parameter,
4-
an anonymous [temporary](expressions.md#temporary-lifetimes), or a named local
4+
an anonymous [temporary](expressions.md#temporaries), or a named local
55
variable.
66

77
A _local variable_ (or *stack-local* allocation) holds a value directly,

0 commit comments

Comments
 (0)
Please sign in to comment.