-
Notifications
You must be signed in to change notification settings - Fork 13.2k
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
Implement stack maps for unwinding #2361
Comments
Commit b99038c places boxes in I have code working in a branch to root Additionally, I discovered that |
Based on some further conversations with the upstream LLVM maintainers, this approach will not work due to the inability to trace back through |
Ouch. Curious how that relates to moving gc. |
Code generation passes can introduce arbitrary register-to-register copies, which means that we wouldn't be able to find all the copies of each GC root to update for moving GC. |
Steps (1) through (3) are basically implemented in the |
Step (4), GC metadata and strategy infrastructure, is now implemented in the |
Realized I forgot about callee-saved registers; updated the list above. |
Realized I'll want GEPs of allocas to be able to be rooted as well. Updated the list. |
Steps (5) through (7) are now complete in the |
Steps (8) and (9) are implemented in the |
Liveness has now been implemented according to the paper and lightly tested in my Also, I discovered that LLVM already has support for tracing |
Steps (11) and (12) have now been completed in my I'm now going to look into getting |
applause but not done in time, moving to 0.4. |
Not critical for 0.6; removing milestone |
@pcwalton What's the status of this? |
This is not going to happen any time in the near future. |
Unwinding now works fine on Windows, and LLVM does have experimental stack map support. It's not clear if there's anything else needed for a precise Rust garbage collector so I don't see a need to leave this open. |
test that we also find bad uses of mem::uninitialized Also we really don't need to separately test signed and unsigned integers... our test suite is big enough as it is. ;)
This issue tracks implementation of stack maps in LLVM. The goal is to have stack maps working for unwinding for 0.3 or 0.4.
The idea is to use precise stack maps instead of landing pads to run destructors of unique pointers and resources during task failure. This will allow unwinding to work on Windows and should also lead to dramatic code size reduction. Unwinding performance should be slightly improved as well. Eventually this can be extended to support accurate tracing garbage collection for shared boxes (either as a backup for reference counting or in lieu of reference counting).
This requires fairly invasive changes to LLVM. All optimization passes will be turned off at first, and will be reenabled one by one. They should not be hard to reenable.
At the moment, I believe the LLVM steps needed here are:
llvm.gcregroot
intrinsic. Translate it in the fast instruction selector. This needs to be done during call lowering, to make sure that the register roots end up within the call sequence.llvm.gcregroot
intrinsics automatically after call sites based on the types of SSA values. This must be done very late in IR transformation, probably right around theCodeGenPrepare
pass.llvm.gcregroot
intrinsics properly in the fast instruction selector. This requires creating a new, fakeMachineInstr
that tracks GC roots and their address spaces.llvm.gcroot
intrinsics in the fast instruction selector as well.getelementptr
instructions that reference analloca
instruction to be rooted withllvm.gcroot
. This allows structs containing pointers to be rooted without complex type encoding schemes.llvm.gcroot
, and uses the metadata field to track the locations of the pointers therein. (I have a patch for this that needs to be resurrected.)llvm.gcregroot
insertion pass to throw out dead register roots.llvm.gcregroot
insertion pass to root only pointer origins, not any derived pointers. Consider pointer origins and derived pointers a single value for the purposes of liveness.LowerCall
and/orLowerCallTo
. Thus all targets will need to be updated.We will eventually need to have a comprehensive LLVM-level test suite before all of this can be sent upstream.
On the
rustc
side, we will need to tag all shared and unique boxes withaddrspace(1)
. (This currently happens for shared boxes, but not for uniques.) Unique boxes will need to become self-describing; this is not necessary in theory, but in practice it helps the LLVM side of things if all that needs to be tracked for each virtual register is a single address space value. Additionally, we will need to use thellvm.gcroot
intrinsics to root (a) every enum that contains shared pointers, unique pointers, or resources; and (b) every resource.In the Rust runtime, we will need to implement the stack crawler. (I have a prototype of this written already.) We will then need to use it to locate all of the roots during unwinding and then run their destructors. This may require fixes to (or replacement of) the shape code.
I intend to keep this issue up to date with the latest changes to our strategy here.
The text was updated successfully, but these errors were encountered: