Commit b92d42d8 authored by Kevin Modzelewski's avatar Kevin Modzelewski

Some very simple refcounting in the LLVM jit

parent 962471b7
...@@ -201,7 +201,7 @@ private: ...@@ -201,7 +201,7 @@ private:
llvm::ValueMap<llvm::Value*, RefcountState> vars; llvm::ValueMap<llvm::Value*, RefcountState> vars;
public: public:
void setType(llvm::Value* v, RefType reftype); llvm::Value* setType(llvm::Value* v, RefType reftype);
void refConsumed(llvm::Value* v, llvm::Instruction*); void refConsumed(llvm::Value* v, llvm::Instruction*);
static void addRefcounts(IRGenState* state); static void addRefcounts(IRGenState* state);
}; };
......
...@@ -204,13 +204,15 @@ llvm::Value* IRGenState::getFrameInfoVar() { ...@@ -204,13 +204,15 @@ llvm::Value* IRGenState::getFrameInfoVar() {
// 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( builder.CreateStore(
null_value, builder.CreateConstInBoundsGEP2_32(exc_info, 0, offsetof(ExcInfo, type) / sizeof(Box*))); null_value, builder.CreateConstInBoundsGEP2_32(exc_info, 0, offsetof(ExcInfo, type) / sizeof(Box*)));
// frame_info.boxedLocals = NULL // frame_info.boxedLocals = NULL
llvm::Value* boxed_locals_gep = getBoxedLocalsGep(builder, al); llvm::Value* boxed_locals_gep = getBoxedLocalsGep(builder, al);
builder.CreateStore(getNullPtr(g.llvm_value_type_ptr), boxed_locals_gep); builder.CreateStore(getRefcounts()->setType(getNullPtr(g.llvm_value_type_ptr), RefType::BORROWED),
boxed_locals_gep);
if (getScopeInfo()->usesNameLookup()) { if (getScopeInfo()->usesNameLookup()) {
// frame_info.boxedLocals = createDict() // frame_info.boxedLocals = createDict()
...@@ -222,7 +224,8 @@ llvm::Value* IRGenState::getFrameInfoVar() { ...@@ -222,7 +224,8 @@ llvm::Value* IRGenState::getFrameInfoVar() {
// 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)); builder.CreateStore(getRefcounts()->setType(getNullPtr(llvm_frame_obj_type_ptr), RefType::BORROWED),
getFrameObjGep(builder, al));
this->frame_info = al; this->frame_info = al;
} }
...@@ -500,11 +503,11 @@ public: ...@@ -500,11 +503,11 @@ public:
} }
Box* getIntConstant(int64_t n) override { Box* getIntConstant(int64_t n) override {
return autoDecref(irstate->getSourceInfo()->parent_module->getIntConstant(n)); return irstate->getSourceInfo()->parent_module->getIntConstant(n);
} }
Box* getFloatConstant(double d) override { Box* getFloatConstant(double d) override {
return autoDecref(irstate->getSourceInfo()->parent_module->getFloatConstant(d)); return irstate->getSourceInfo()->parent_module->getFloatConstant(d);
} }
llvm::Value* setType(llvm::Value* v, RefType reftype) { llvm::Value* setType(llvm::Value* v, RefType reftype) {
......
...@@ -15,8 +15,11 @@ ...@@ -15,8 +15,11 @@
#include "codegen/compvars.h" #include "codegen/compvars.h"
#include <cstdio> #include <cstdio>
#include <deque>
#include <sstream> #include <sstream>
#include "llvm/ADT/DenseSet.h"
#include "llvm/IR/CFG.h"
#include "llvm/IR/IntrinsicInst.h" #include "llvm/IR/IntrinsicInst.h"
#include "llvm/IR/Module.h" #include "llvm/IR/Module.h"
#include "llvm/Support/raw_ostream.h" #include "llvm/Support/raw_ostream.h"
...@@ -37,11 +40,12 @@ ...@@ -37,11 +40,12 @@
namespace pyston { namespace pyston {
void RefcountTracker::setType(llvm::Value* v, RefType reftype) { llvm::Value* RefcountTracker::setType(llvm::Value* v, RefType reftype) {
auto& var = this->vars[v]; auto& var = this->vars[v];
assert(var.reftype == reftype || var.reftype == RefType::UNKNOWN); assert(var.reftype == reftype || var.reftype == RefType::UNKNOWN);
var.reftype = reftype; var.reftype = reftype;
return v;
} }
void RefcountTracker::refConsumed(llvm::Value* v, llvm::Instruction* inst) { void RefcountTracker::refConsumed(llvm::Value* v, llvm::Instruction* inst) {
...@@ -49,6 +53,9 @@ void RefcountTracker::refConsumed(llvm::Value* v, llvm::Instruction* inst) { ...@@ -49,6 +53,9 @@ void RefcountTracker::refConsumed(llvm::Value* v, llvm::Instruction* inst) {
assert(var.reftype != RefType::UNKNOWN); assert(var.reftype != RefType::UNKNOWN);
var.ref_consumers.push_back(inst); var.ref_consumers.push_back(inst);
// Make sure that this instruction actually references v:
assert(std::find(inst->op_begin(), inst->op_end(), v) != inst->op_end());
} }
void RefcountTracker::addRefcounts(IRGenState* irstate) { void RefcountTracker::addRefcounts(IRGenState* irstate) {
...@@ -130,12 +137,213 @@ void RefcountTracker::addRefcounts(IRGenState* irstate) { ...@@ -130,12 +137,213 @@ void RefcountTracker::addRefcounts(IRGenState* irstate) {
for (auto&& u : inst.uses()) { for (auto&& u : inst.uses()) {
check_val_missed(u.get()); check_val_missed(u.get());
} }
for (auto&& op : inst.operands()) {
check_val_missed(op);
}
} }
} }
ASSERT(num_untracked == 0, ""); ASSERT(num_untracked == 0, "");
#endif #endif
assert(0 && "implement me"); std::deque<llvm::BasicBlock*> block_queue;
struct RefState {
llvm::DenseMap<llvm::Value*, int> refs;
//llvm::Instruction*
};
llvm::DenseMap<llvm::BasicBlock*, RefState> states;
for (auto&& BB : *f) {
if (llvm::succ_begin(&BB) == llvm::succ_end(&BB)) {
printf("%s is a terminator block\n", BB.getName().data());
block_queue.push_back(&BB);
}
}
auto addRefcountsToBlock = [](llvm::BasicBlock* BB, llvm::Value* v, int num_refs) {
llvm::Instruction* incref_pt;// = BB->getFirstInsertionPt();
if (llvm::isa<llvm::LandingPadInst>(*BB->begin())) {
// Don't split up the landingpad+extract+cxa_begin_catch
auto it = BB->begin();
++it;
++it;
++it;
incref_pt = &*it;
} else {
incref_pt = BB->getFirstInsertionPt();
}
#ifdef Py_REF_DEBUG
auto reftotal_gv = g.cur_module->getOrInsertGlobal("_Py_RefTotal", g.i64);
auto reftotal = new llvm::LoadInst(reftotal_gv, "", incref_pt);
auto new_reftotal = llvm::BinaryOperator::Create(
llvm::BinaryOperator::BinaryOps::Add, reftotal,
getConstantInt(num_refs, g.i64), "", incref_pt);
new llvm::StoreInst(new_reftotal, reftotal_gv, "", incref_pt);
#endif
llvm::ArrayRef<llvm::Value*> idxs({ getConstantInt(0, g.i32), getConstantInt(0, g.i32) });
auto refcount_ptr = llvm::GetElementPtrInst::CreateInBounds(v, idxs, "", incref_pt);
auto refcount = new llvm::LoadInst(refcount_ptr, "", incref_pt);
auto new_refcount = llvm::BinaryOperator::Create(
llvm::BinaryOperator::BinaryOps::Add, refcount,
getConstantInt(num_refs, g.i64), "", incref_pt);
new llvm::StoreInst(new_refcount, refcount_ptr, "", incref_pt);
};
while (!block_queue.empty()) {
llvm::BasicBlock& BB = *block_queue.front();
block_queue.pop_front();
#if 0
llvm::Instruction* term_inst = BB.getTerminator();
llvm::Instruction* insert_before = term_inst;
if (llvm::isa<llvm::UnreachableInst>(insert_before)) {
insert_before = &*(++BB.rbegin());
assert(llvm::isa<llvm::CallInst>(insert_before) || llvm::isa<llvm::IntrinsicInst>(insert_before));
}
#endif
llvm::outs() << "Processing " << BB.getName() << '\n';
assert(!states.count(&BB));
RefState& state = states[&BB];
llvm::SmallVector<llvm::BasicBlock*, 4> successors;
successors.insert(successors.end(), llvm::succ_begin(&BB), llvm::succ_end(&BB));
if (successors.size()) {
llvm::DenseSet<llvm::Value*> tracked_values;
for (auto SBB : successors) {
assert(states.count(SBB));
for (auto&& p : states[SBB].refs) {
tracked_values.insert(p.first);
}
}
for (auto v : tracked_values) {
assert(rt->vars.count(v));
auto refstate = rt->vars[v];
if (refstate.reftype == RefType::BORROWED) {
int min_refs = 1000000000;
for (auto SBB : successors) {
auto it = states[SBB].refs.find(v);
if (it != states[SBB].refs.end()) {
min_refs = std::min(it->second, min_refs);
} else
min_refs = 0;
}
for (auto SBB : successors) {
auto it = states[SBB].refs.find(v);
int this_refs = 0;
if (it != states[SBB].refs.end()) {
this_refs = it->second;
}
if (this_refs > min_refs) {
addRefcountsToBlock(SBB, v, this_refs - min_refs);
//llvm::outs() << *incref_pt << '\n';
//llvm::outs() << "Need to incref " << *v << " at beginning of " << SBB->getName() << "\n";
}
}
state.refs[v] = min_refs;
} else {
assert(0 && "finish implementing");
}
}
}
for (auto &I : llvm::iterator_range<llvm::BasicBlock::reverse_iterator>(BB.rbegin(), BB.rend())) {
llvm::DenseMap<llvm::Value*, int> num_consumed_by_inst;
llvm::DenseMap<llvm::Value*, int> num_times_as_op;
for (llvm::Value* op : I.operands()) {
auto it = rt->vars.find(op);
if (it == rt->vars.end())
continue;
int& nops = num_times_as_op[op];
nops++;
if (nops > 1)
continue;
auto&& var_state = it->second;
for (auto consuming_inst : var_state.ref_consumers) {
if (consuming_inst == &I)
num_consumed_by_inst[op]++;
}
}
for (auto&& p : num_times_as_op) {
auto& op = p.first;
auto&& it = num_consumed_by_inst.find(op);
int num_consumed = 0;
if (it != num_consumed_by_inst.end())
num_consumed = it->second;
if (num_times_as_op[op] > num_consumed) {
if (rt->vars[op].reftype == RefType::BORROWED) {
} else {
assert(0 && "handle this case");
}
}
if (num_consumed)
state.refs[op] += num_consumed;
}
}
// Handle variables that were defined in this BB:
for (auto it = state.refs.begin(), end = state.refs.end(); it != end; ) {
llvm::Instruction* inst = llvm::dyn_cast<llvm::Instruction>(it->first);
if (inst && inst->getParent() == &BB) {
assert(it->second == 1 && rt->vars[it->first].reftype == RefType::OWNED);
state.refs.erase(it);
++it;
} else {
++it;
}
}
if (&BB == &BB.getParent()->front()) {
for (auto&& p : state.refs) {
llvm::outs() << *p.first << " " << p.second << '\n';
assert(llvm::isa<llvm::GlobalVariable>(p.first));
assert(rt->vars[p.first].reftype == RefType::BORROWED);
addRefcountsToBlock(&BB, p.first, p.second);
}
state.refs.clear();
}
llvm::outs() << "End of " << BB.getName() << ":\n";
for (auto&& p : state.refs) {
llvm::outs() << "With " << p.second << " refs: " << *p.first << '\n';
}
llvm::outs() << '\n';
for (auto&& PBB : llvm::iterator_range<llvm::pred_iterator>(llvm::pred_begin(&BB), llvm::pred_end(&BB))) {
bool all_succ_done = true;
for (auto&& SBB : llvm::iterator_range<llvm::succ_iterator>(llvm::succ_begin(PBB), llvm::succ_end(PBB))) {
if (!states.count(SBB)) {
all_succ_done = false;
break;
}
}
if (all_succ_done) {
llvm::outs() << PBB->getName() << " is now ready to be processed\n";
block_queue.push_back(PBB);
}
}
}
fprintf(stderr, "After refcounts:\n");
fprintf(stderr, "\033[35m");
dumpPrettyIR(f);
fprintf(stderr, "\033[0m");
} }
} // namespace pyston } // namespace pyston
...@@ -466,14 +466,14 @@ BORROWED(BoxedString*) BoxedModule::getStringConstant(llvm::StringRef ast_str, b ...@@ -466,14 +466,14 @@ BORROWED(BoxedString*) BoxedModule::getStringConstant(llvm::StringRef ast_str, b
return r; return r;
} }
Box* BoxedModule::getUnicodeConstant(llvm::StringRef ast_str) { BORROWED(Box*) BoxedModule::getUnicodeConstant(llvm::StringRef ast_str) {
Box*& r = unicode_constants[ast_str]; Box*& r = unicode_constants[ast_str];
if (!r) if (!r)
r = decodeUTF8StringPtr(ast_str); r = decodeUTF8StringPtr(ast_str);
return r; return r;
} }
BoxedInt* BoxedModule::getIntConstant(int64_t n) { BORROWED(BoxedInt*) BoxedModule::getIntConstant(int64_t n) {
BoxedInt*& r = int_constants[n]; BoxedInt*& r = int_constants[n];
if (!r) if (!r)
r = new BoxedInt(n); r = new BoxedInt(n);
...@@ -487,21 +487,21 @@ static int64_t getDoubleBits(double d) { ...@@ -487,21 +487,21 @@ static int64_t getDoubleBits(double d) {
return rtn; return rtn;
} }
BoxedFloat* BoxedModule::getFloatConstant(double d) { BORROWED(BoxedFloat*) BoxedModule::getFloatConstant(double d) {
BoxedFloat*& r = float_constants[getDoubleBits(d)]; BoxedFloat*& r = float_constants[getDoubleBits(d)];
if (!r) if (!r)
r = static_cast<BoxedFloat*>(boxFloat(d)); r = static_cast<BoxedFloat*>(boxFloat(d));
return r; return r;
} }
Box* BoxedModule::getPureImaginaryConstant(double d) { BORROWED(Box*) BoxedModule::getPureImaginaryConstant(double d) {
Box*& r = imaginary_constants[getDoubleBits(d)]; Box*& r = imaginary_constants[getDoubleBits(d)];
if (!r) if (!r)
r = createPureImaginary(d); r = createPureImaginary(d);
return r; return r;
} }
Box* BoxedModule::getLongConstant(llvm::StringRef ast_str) { BORROWED(Box*) BoxedModule::getLongConstant(llvm::StringRef ast_str) {
Box*& r = long_constants[ast_str]; Box*& r = long_constants[ast_str];
if (!r) if (!r)
r = createLong(ast_str); r = createLong(ast_str);
......
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