about summary refs log tree commit diff
path: root/src/test/ui/async-await/issues/issue-55324.rs
diff options
context:
space:
mode:
authorbors <bors@rust-lang.org>2019-07-02 12:25:00 +0000
committerbors <bors@rust-lang.org>2019-07-02 12:25:00 +0000
commit848e0a23f34aaab3e4a974b031c86ef2a4e4fcc1 (patch)
tree028f767264e3d179b75d747d57662b93c72fa06b /src/test/ui/async-await/issues/issue-55324.rs
parentef064d2f66bf5851e1da1a016627ff1395f6c9ad (diff)
parenta68e2c716113d4f7a10a149cc13b18b851853000 (diff)
downloadrust-848e0a23f34aaab3e4a974b031c86ef2a4e4fcc1.tar.gz
rust-848e0a23f34aaab3e4a974b031c86ef2a4e4fcc1.zip
Auto merge of #61922 - tmandry:moar-generator-optimization, r=matthewjasper
Don't store locals that have been moved from in generators

This avoids reserving storage in generators for locals that are moved
out of (and not re-initialized) prior to yield points. Fixes #59123.

This adds a new dataflow analysis, `RequiresStorage`, to determine whether the storage of a local can be destroyed without being observed by the program. The rules are:

1. StorageLive(x) => mark x live
2. StorageDead(x) => mark x dead
3. If a local is moved from, _and has never had its address taken_, mark it dead
4. If (any part of) a local is initialized, mark it live'

This is used to determine whether to save a local in the generator object at all, as well as which locals can be overlapped in the generator layout.

Here's the size in bytes of all testcases included in the change, before and after the change:

async fn test    |Size before |Size after
-----------------|------------|----------
single           | 1028       | 1028
single_with_noop | 2056       | 1032
joined           | 5132       | 3084
joined_with_noop | 8208       | 3084

generator test              |Size before |Size after
----------------------------|------------|----------
move_before_yield           | 1028       | 1028
move_before_yield_with_noop | 2056       | 1032
overlap_move_points         | 3080       | 2056

## Future work

Note that there is a possible extension to this optimization, which modifies rule 3 to read: "If a local is moved from, _**and either has never had its address taken, or is Freeze and has never been mutably borrowed**_, mark it dead." This was discussed at length in #59123 and then #61849. Because this would cause some behavior to be UB which was not UB before, it's a step that needs to be taken carefully.

A more immediate priority for me is inlining `std::mem::size_of_val(&x)` so it becomes apparent that the address of `x` is not taken. This way, using `size_of_val` to look at the size of your inner futures does not affect the size of your outer future.

cc @cramertj @eddyb @Matthias247 @nikomatsakis @RalfJung @Zoxc
Diffstat (limited to 'src/test/ui/async-await/issues/issue-55324.rs')
0 files changed, 0 insertions, 0 deletions