Process gc preserve#58
Conversation
|
|
||
| namespace { | ||
|
|
||
| static void removeGCPreserve(CallInst *call, Instruction *val) |
There was a problem hiding this comment.
Why is this function removed? The compiler removes the GC preserve calls if the values to be preserved are removed, or moved to the stack allocation. I don't know why we want to remove this optimization. Furthermore, I am concerned that in those cases where GC preserve is kept, the preserved value is replaced with a null pointer -- preserving a null pointer sounds meaningless.
There was a problem hiding this comment.
I think I misunderstood the code twice.
(1) I thought I was deleting the gc preserve after it had been lowered, ie., after the necessary objects arguments to preserve_begin have been pushed to the shadow stack, but I don't think that is the case.
(2) Since the if case only covered pass.gc_preserve_begin_func, I thought I'd end up with dangling hooks for the preserve_end and that would cause problems with the push and pop hook functions, but the matching pop functions are also removed.
Can you explain what you meant by "moved to stack allocation"? I'm just trying to understand whether removing those cases could cause any correctness issue.
There was a problem hiding this comment.
Can you explain what you meant by "moved to stack allocation"?
If I understand right, this optimization tries to change heap allocation to stack allocation if possible.
Line 554 in 88fea47
If a value
x that was heap allocated and is replaced with stack allocation by the optimization, all of its uses need to be fixed with replace_inst.Line 636 in 88fea47
If we have
gc_preserve(x), replace_inst either removes the gc_preserve (removeGCPreserve()), or replace x with the alloca buff on the stack.Line 656 in 88fea47
There was a problem hiding this comment.
If you remove the code that checks the gc_preserve_begin_func case, it will fall into the default case.
Line 678 in 88fea47
It either replaces x with the alloca buff (which is fine), or replaces x with null. Then you may see null pointers in the preserve begin hook, and you will need to deal with null pointers in the hook.
I don't think it causes correctness issues, but doing a gc preserve call for null pointers is meaningless.
There was a problem hiding this comment.
Got it! In that case, it might actually be problematic if we remove that particular gc preserve. The reason being the fact that we need to transitively pin everything in that object's transitive closure, independently of that object itself being allocated in the heap or in the stack.
If we have gc_preserve(x), replace_inst either removes the gc_preserve (removeGCPreserve()), or replace x with the alloca buff on the stack.
Okay, then it should be fine, since the gc preserve is removed only if it doesn't have anything referring to it (has_ref == 0, which I believe would also be considering any possible c call).
| #define jl_p_tpin_gcstack (jl_current_task->tpin_gcstack) | ||
|
|
||
| #define JL_GC_PUSHARGS_TPIN_ROOT_OBJS(rts_var,n) \ | ||
| rts_var = ((jl_value_t**)malloc(((n)+2)*sizeof(jl_value_t*)))+2; \ |
There was a problem hiding this comment.
These need to be clarified.
- Why do you need to use
mallocinstead ofallocainLine 982 in d907a06
- Why do you need a separate
tpin_gcstackin the task? Obviously you can put tpinned roots in the normalgcstack? - If
JL_GC_PUSHARGS_TPIN_ROOT_OBJSandtpin_gcstackis only used by gc preserve, probably just call them gc preserve frames or something. Calling them tpin is more confusing, as you can clearly push tpin roots to the normal stack and use existingJL_GC_PUSH. - Add some comments so we know why it is implemented like this.
| args.insert(args.begin(), ConstantInt::get(T_size, nargs)); | ||
|
|
||
| ArrayRef<Value*> args_llvm = ArrayRef<Value*>(args); | ||
| builder.CreateCall(getOrDeclare(jl_well_known::GCPreserveBeginHook), args_llvm ); |
There was a problem hiding this comment.
GCPreserveBeginHook is only compiled when MMTK_GC is set, but this code here is executed for all the builds. The stock build would fail in this case. Same for GCPreserveEndHook below.
There was a problem hiding this comment.
Fixed. It turns out that the stock build was already broken because of a file not being compiled in Makefile, but I've fixed that too and now both builds (stock and MMTk) should work fine.
This PR processes objects from "gc_preserve" regions separately, ensuring they're transitively pinned while the remaining objects rooted by application code are only pinned.
Note that the initial focus is on correctness and that the implementation should change to lower the code inside LLVM rather than calling into C using the hook functions (
jl_gc_preserve_begin_hookandjl_gc_preserve_end_hook) for each preserve region.