Commit 9f7d114d authored by Kevin Modzelewski's avatar Kevin Modzelewski

Fix refcounting around invokes

parent 7e195c56
...@@ -249,6 +249,7 @@ void IRGenState::setupFrameInfoVar(llvm::Value* passed_closure, llvm::Value* pas ...@@ -249,6 +249,7 @@ void IRGenState::setupFrameInfoVar(llvm::Value* passed_closure, llvm::Value* pas
// frame_info.exc.type = NULL // frame_info.exc.type = NULL
llvm::Constant* null_value = getNullPtr(g.llvm_value_type_ptr); llvm::Constant* null_value = getNullPtr(g.llvm_value_type_ptr);
getRefcounts()->setType(null_value, RefType::BORROWED);
llvm::Value* exc_info = getExcinfoGep(builder, al); llvm::Value* exc_info = getExcinfoGep(builder, al);
builder.CreateStore(null_value, builder.CreateStore(null_value,
builder.CreateConstInBoundsGEP2_32(exc_info, 0, offsetof(ExcInfo, type) / sizeof(Box*))); builder.CreateConstInBoundsGEP2_32(exc_info, 0, offsetof(ExcInfo, type) / sizeof(Box*)));
...@@ -267,7 +268,9 @@ void IRGenState::setupFrameInfoVar(llvm::Value* passed_closure, llvm::Value* pas ...@@ -267,7 +268,9 @@ void IRGenState::setupFrameInfoVar(llvm::Value* passed_closure, llvm::Value* pas
// frame_info.frame_obj = NULL // frame_info.frame_obj = NULL
static llvm::Type* llvm_frame_obj_type_ptr static llvm::Type* llvm_frame_obj_type_ptr
= llvm::cast<llvm::StructType>(g.llvm_frame_info_type)->getElementType(2); = llvm::cast<llvm::StructType>(g.llvm_frame_info_type)->getElementType(2);
builder.CreateStore(getNullPtr(llvm_frame_obj_type_ptr), getFrameObjGep(builder, al)); auto null_frame = getNullPtr(llvm_frame_obj_type_ptr);
getRefcounts()->setType(null_frame, RefType::BORROWED);
builder.CreateStore(null_frame, getFrameObjGep(builder, al));
// set frame_info.passed_closure // set frame_info.passed_closure
builder.CreateStore(passed_closure, getPassedClosureGep(builder, al)); builder.CreateStore(passed_closure, getPassedClosureGep(builder, al));
...@@ -324,7 +327,6 @@ ScopeInfo* IRGenState::getScopeInfoForNode(AST* node) { ...@@ -324,7 +327,6 @@ ScopeInfo* IRGenState::getScopeInfoForNode(AST* node) {
} }
llvm::Value* IRGenState::getGlobals() { llvm::Value* IRGenState::getGlobals() {
assert(0 && "check refcounting (set to BORROWED?)");
assert(globals); assert(globals);
return globals; return globals;
} }
......
...@@ -69,7 +69,7 @@ void RefcountTracker::refConsumed(llvm::Value* v, llvm::Instruction* inst) { ...@@ -69,7 +69,7 @@ void RefcountTracker::refConsumed(llvm::Value* v, llvm::Instruction* inst) {
//var.ref_consumers.push_back(inst); //var.ref_consumers.push_back(inst);
} }
llvm::Instruction* findIncrefPt(llvm::BasicBlock* BB) { llvm::Instruction* findInsertionPoint(llvm::BasicBlock* BB) {
ASSERT(pred_begin(BB) == pred_end(BB) || pred_end(BB) == ++pred_begin(BB), ASSERT(pred_begin(BB) == pred_end(BB) || pred_end(BB) == ++pred_begin(BB),
"We shouldn't be inserting anything at the beginning of blocks with multiple predecessors"); "We shouldn't be inserting anything at the beginning of blocks with multiple predecessors");
...@@ -402,6 +402,7 @@ void RefcountTracker::addRefcounts(IRGenState* irstate) { ...@@ -402,6 +402,7 @@ void RefcountTracker::addRefcounts(IRGenState* irstate) {
|| (s->getName().startswith("Py") && s->getName().endswith("Object")) || (s->getName().startswith("Py") && s->getName().endswith("Object"))
|| s->getName().startswith("class.pyston::Box")) { || s->getName().startswith("class.pyston::Box")) {
v->dump(); v->dump();
s->dump();
if (s && s->elements().size() >= 2) { if (s && s->elements().size() >= 2) {
s->elements()[0]->dump(); s->elements()[0]->dump();
s->elements()[1]->dump(); s->elements()[1]->dump();
...@@ -453,6 +454,9 @@ void RefcountTracker::addRefcounts(IRGenState* irstate) { ...@@ -453,6 +454,9 @@ void RefcountTracker::addRefcounts(IRGenState* irstate) {
}; };
struct RefState { struct RefState {
// We do a backwards scan and starting/ending here refers to the scan, not the instruction sequence.
// So "starting_refs" are the refs that are inherited, ie the refstate at the end of the basic block.
// "ending_refs" are the refs we calculated, which corresponds to the refstate at the beginning of the block.
llvm::DenseMap<llvm::Value*, int> starting_refs; llvm::DenseMap<llvm::Value*, int> starting_refs;
llvm::DenseMap<llvm::Value*, int> ending_refs; llvm::DenseMap<llvm::Value*, int> ending_refs;
...@@ -532,10 +536,10 @@ void RefcountTracker::addRefcounts(IRGenState* irstate) { ...@@ -532,10 +536,10 @@ void RefcountTracker::addRefcounts(IRGenState* irstate) {
this_refs = it->second; this_refs = it->second;
} }
if (this_refs > min_refs) { if (this_refs > min_refs) {
state.increfs.push_back(RefOp({v, refstate.nullable, this_refs - min_refs, findIncrefPt(SBB)})); state.increfs.push_back(RefOp({v, refstate.nullable, this_refs - min_refs, findInsertionPoint(SBB)}));
} else if (this_refs < min_refs) { } else if (this_refs < min_refs) {
assert(refstate.reftype == RefType::OWNED); assert(refstate.reftype == RefType::OWNED);
state.decrefs.push_back(RefOp({v, refstate.nullable, min_refs - this_refs, findIncrefPt(SBB)})); state.decrefs.push_back(RefOp({v, refstate.nullable, min_refs - this_refs, findInsertionPoint(SBB)}));
} }
} }
...@@ -589,8 +593,8 @@ void RefcountTracker::addRefcounts(IRGenState* irstate) { ...@@ -589,8 +593,8 @@ void RefcountTracker::addRefcounts(IRGenState* irstate) {
// Don't do any updates now since we are iterating over the bb // Don't do any updates now since we are iterating over the bb
if (llvm::InvokeInst* invoke = llvm::dyn_cast<llvm::InvokeInst>(&I)) { if (llvm::InvokeInst* invoke = llvm::dyn_cast<llvm::InvokeInst>(&I)) {
state.decrefs.push_back(RefOp({op, rt->vars[op].nullable, 1, findIncrefPt(invoke->getNormalDest())})); state.decrefs.push_back(RefOp({op, rt->vars[op].nullable, 1, findInsertionPoint(invoke->getNormalDest())}));
state.decrefs.push_back(RefOp({op, rt->vars[op].nullable, 1, findIncrefPt(invoke->getUnwindDest())})); state.decrefs.push_back(RefOp({op, rt->vars[op].nullable, 1, findInsertionPoint(invoke->getUnwindDest())}));
} else { } else {
assert(&I != I.getParent()->getTerminator()); assert(&I != I.getParent()->getTerminator());
auto next = I.getNextNode(); auto next = I.getNextNode();
...@@ -616,14 +620,22 @@ void RefcountTracker::addRefcounts(IRGenState* irstate) { ...@@ -616,14 +620,22 @@ void RefcountTracker::addRefcounts(IRGenState* irstate) {
// Handle variables that were defined in this BB: // Handle variables that were defined in this BB:
for (auto&& p : rt->vars) { for (auto&& p : rt->vars) {
llvm::Instruction* inst = llvm::dyn_cast<llvm::Instruction>(p.first); llvm::Instruction* inst = llvm::dyn_cast<llvm::Instruction>(p.first);
if (inst && inst->getParent() == &BB) { if (!inst)
continue;
// Invokes are special. Handle them here by treating them as if they happened in their normal-dest block.
llvm::InvokeInst* ii = llvm::dyn_cast<llvm::InvokeInst>(inst);
if ((!ii && inst->getParent() == &BB) || (ii && ii->getNormalDest() == &BB)) {
int starting_refs = (p.second.reftype == RefType::OWNED ? 1 : 0); int starting_refs = (p.second.reftype == RefType::OWNED ? 1 : 0);
if (state.ending_refs[inst] != starting_refs) { if (state.ending_refs[inst] != starting_refs) {
llvm::Instruction* insertion_pt = inst->getNextNode(); llvm::Instruction* insertion_pt;
assert(insertion_pt); if (ii) {
while (llvm::isa<llvm::PHINode>(insertion_pt)) { insertion_pt = findInsertionPoint(&BB);
insertion_pt = insertion_pt->getNextNode(); } else {
assert(insertion_pt); insertion_pt = inst->getNextNode();
while (llvm::isa<llvm::PHINode>(insertion_pt)) {
insertion_pt = insertion_pt->getNextNode();
}
} }
if (state.ending_refs[inst] < starting_refs) { if (state.ending_refs[inst] < starting_refs) {
...@@ -661,7 +673,7 @@ void RefcountTracker::addRefcounts(IRGenState* irstate) { ...@@ -661,7 +673,7 @@ void RefcountTracker::addRefcounts(IRGenState* irstate) {
#endif #endif
assert(rt->vars[p.first].reftype == RefType::BORROWED); assert(rt->vars[p.first].reftype == RefType::BORROWED);
state.increfs.push_back(RefOp({p.first, rt->vars[p.first].nullable, p.second, findIncrefPt(&BB)})); state.increfs.push_back(RefOp({p.first, rt->vars[p.first].nullable, p.second, findInsertionPoint(&BB)}));
} }
state.ending_refs.clear(); state.ending_refs.clear();
} }
......
...@@ -263,7 +263,7 @@ void dumpPrettyIR(llvm::Function* f) { ...@@ -263,7 +263,7 @@ void dumpPrettyIR(llvm::Function* f) {
remapPatchpoint(ii); remapPatchpoint(ii);
} }
} else if (llvm::InvokeInst* ii = llvm::dyn_cast<llvm::InvokeInst>(&*it)) { } else if (llvm::InvokeInst* ii = llvm::dyn_cast<llvm::InvokeInst>(&*it)) {
if (ii->getCalledFunction()->isIntrinsic()) { if (ii->getCalledFunction() && ii->getCalledFunction()->isIntrinsic()) {
remapPatchpoint(ii); remapPatchpoint(ii);
} }
} }
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment