Commit 4f03c2d0 authored by Kevin Modzelewski's avatar Kevin Modzelewski

Add a bunch more kill flags

and switch the llvm tier to use them instead of its own analysis
parent 85da6551
...@@ -52,28 +52,17 @@ private: ...@@ -52,28 +52,17 @@ private:
} }
}; };
llvm::DenseSet<AST_Name*> kills;
llvm::DenseMap<InternedString, AST_Name*> last_uses;
llvm::DenseMap<InternedString, Status> statuses; llvm::DenseMap<InternedString, Status> statuses;
LivenessAnalysis* analysis; LivenessAnalysis* analysis;
void _doLoad(InternedString name, AST_Name* node) { void _doLoad(InternedString name, AST_Name* node) {
Status& status = statuses[name]; Status& status = statuses[name];
status.addUsage(Status::USED); status.addUsage(Status::USED);
last_uses[name] = node;
} }
void _doStore(InternedString name) { void _doStore(InternedString name) {
Status& status = statuses[name]; Status& status = statuses[name];
status.addUsage(Status::DEFINED); status.addUsage(Status::DEFINED);
auto it = last_uses.find(name);
if (it != last_uses.end()) {
kills.insert(it->second);
last_uses.erase(it);
}
} }
Status::Usage getStatusFirst(InternedString name) const { Status::Usage getStatusFirst(InternedString name) const {
...@@ -90,22 +79,7 @@ public: ...@@ -90,22 +79,7 @@ public:
bool firstIsDef(InternedString name) const { return getStatusFirst(name) == Status::DEFINED; } bool firstIsDef(InternedString name) const { return getStatusFirst(name) == Status::DEFINED; }
bool isKilledAt(AST_Name* node, bool is_live_at_end) const { bool isKilledAt(AST_Name* node, bool is_live_at_end) { return node->is_kill; }
if (kills.count(node))
return true;
// If it's not live at the end, then the last use is a kill
// even though we weren't able to determine that in a single
// pass
if (!is_live_at_end) {
auto it = last_uses.find(node->id);
if (it != last_uses.end() && node == it->second)
return true;
}
return false;
}
bool visit_classdef(AST_ClassDef* node) { bool visit_classdef(AST_ClassDef* node) {
......
...@@ -1691,7 +1691,6 @@ Value ASTInterpreter::visit_name(AST_Name* node) { ...@@ -1691,7 +1691,6 @@ Value ASTInterpreter::visit_name(AST_Name* node) {
bool is_live = true; bool is_live = true;
if (node->is_kill) { if (node->is_kill) {
is_live = false; is_live = false;
assert(!source_info->getLiveness()->isLiveAtEnd(node->id, current_block));
} else if (node->lookup_type == ScopeInfo::VarScopeType::FAST) } else if (node->lookup_type == ScopeInfo::VarScopeType::FAST)
is_live = source_info->getLiveness()->isLiveAtEnd(node->id, current_block); is_live = source_info->getLiveness()->isLiveAtEnd(node->id, current_block);
......
...@@ -309,7 +309,7 @@ private: ...@@ -309,7 +309,7 @@ private:
auto name = nodeName(); auto name = nodeName();
pushAssign(name, call); pushAssign(name, call);
return makeLoad(name, e); return makeLoad(name, e, true);
} }
AST_Name* remapName(AST_Name* name) { return name; } AST_Name* remapName(AST_Name* name) { return name; }
...@@ -432,7 +432,7 @@ private: ...@@ -432,7 +432,7 @@ private:
// printf("Exit block for comp %d is %d\n", i, exit_blocks[i]->idx); // printf("Exit block for comp %d is %d\n", i, exit_blocks[i]->idx);
} }
return makeLoad(rtn_name, node); return makeLoad(rtn_name, node, /* is_kill */ true);
} }
AST_expr* makeNum(int n) { AST_expr* makeNum(int n) {
...@@ -476,9 +476,9 @@ private: ...@@ -476,9 +476,9 @@ private:
void pushReraise(AST* node, InternedString exc_type_name, InternedString exc_value_name, void pushReraise(AST* node, InternedString exc_type_name, InternedString exc_value_name,
InternedString exc_traceback_name) { InternedString exc_traceback_name) {
auto raise = new AST_Raise(); auto raise = new AST_Raise();
raise->arg0 = makeLoad(exc_type_name, node); raise->arg0 = makeLoad(exc_type_name, node, true);
raise->arg1 = makeLoad(exc_value_name, node); raise->arg1 = makeLoad(exc_value_name, node, true);
raise->arg2 = makeLoad(exc_traceback_name, node); raise->arg2 = makeLoad(exc_traceback_name, node, true);
push_back(raise); push_back(raise);
curblock = nullptr; curblock = nullptr;
} }
...@@ -602,7 +602,7 @@ private: ...@@ -602,7 +602,7 @@ private:
InternedString tmp_name = nodeName("", i); InternedString tmp_name = nodeName("", i);
new_target->elts.push_back(makeName(tmp_name, AST_TYPE::Store, target->lineno)); new_target->elts.push_back(makeName(tmp_name, AST_TYPE::Store, target->lineno));
pushAssign((*elts)[i], makeLoad(tmp_name, target)); pushAssign((*elts)[i], makeLoad(tmp_name, target, /* is_kill */ true));
} }
} else { } else {
RELEASE_ASSERT(0, "%d", target->type); RELEASE_ASSERT(0, "%d", target->type);
...@@ -773,7 +773,7 @@ private: ...@@ -773,7 +773,7 @@ private:
cfg->placeBlock(exit_block); cfg->placeBlock(exit_block);
curblock = exit_block; curblock = exit_block;
return makeLoad(name, node); return makeLoad(name, node, true);
} }
AST_expr* remapCall(AST_Call* node) { AST_expr* remapCall(AST_Call* node) {
...@@ -843,17 +843,24 @@ private: ...@@ -843,17 +843,24 @@ private:
AST_expr* left = remapExpr(node->left); AST_expr* left = remapExpr(node->left);
for (int i = 0; i < node->ops.size(); i++) { for (int i = 0; i < node->ops.size(); i++) {
if (i > 0)
push_back(makeKill(name));
AST_expr* right = remapExpr(node->comparators[i]); AST_expr* right = remapExpr(node->comparators[i]);
AST_Compare* val = new AST_Compare; AST_Compare* val = new AST_Compare;
val->col_offset = node->col_offset; val->col_offset = node->col_offset;
val->lineno = node->lineno; val->lineno = node->lineno;
val->left = left; val->left = left;
val->comparators.push_back(_dup(right)); if (i < node->ops.size() - 1)
val->comparators.push_back(_dup(right));
else
val->comparators.push_back(right);
val->ops.push_back(node->ops[i]); val->ops.push_back(node->ops[i]);
pushAssign(name, val); pushAssign(name, val);
// TODO: this branch is unused for the last comparison
AST_Branch* br = new AST_Branch(); AST_Branch* br = new AST_Branch();
br->test = callNonzero(makeLoad(name, node)); br->test = callNonzero(makeLoad(name, node));
push_back(br); push_back(br);
...@@ -879,7 +886,7 @@ private: ...@@ -879,7 +886,7 @@ private:
cfg->placeBlock(exit_block); cfg->placeBlock(exit_block);
curblock = exit_block; curblock = exit_block;
return makeLoad(name, node); return makeLoad(name, node, true);
} }
} }
...@@ -971,7 +978,7 @@ private: ...@@ -971,7 +978,7 @@ private:
AST_MakeFunction* func = makeFunctionForScope(node); AST_MakeFunction* func = makeFunctionForScope(node);
func->function_def->args->args.push_back(makeName(arg_name, AST_TYPE::Param, node->lineno)); func->function_def->args->args.push_back(makeName(arg_name, AST_TYPE::Param, node->lineno));
emitComprehensionLoops(&func->function_def->body, node->generators, emitComprehensionLoops(&func->function_def->body, node->generators,
makeName(arg_name, AST_TYPE::Load, node->lineno), makeName(arg_name, AST_TYPE::Load, node->lineno, /* col_offset */ 0, /* is_kill */ true),
[this, node](std::vector<AST_stmt*>* insert_point) { [this, node](std::vector<AST_stmt*>* insert_point) {
auto y = new AST_Yield(); auto y = new AST_Yield();
y->value = node->elt; y->value = node->elt;
...@@ -981,7 +988,7 @@ private: ...@@ -981,7 +988,7 @@ private:
InternedString func_var_name = nodeName(); InternedString func_var_name = nodeName();
pushAssign(func_var_name, func); pushAssign(func_var_name, func);
return makeCall(makeLoad(func_var_name, node), first); return makeCall(makeLoad(func_var_name, node, true), first);
} }
void emitComprehensionYield(AST_DictComp* node, InternedString dict_name, std::vector<AST_stmt*>* insert_point) { void emitComprehensionYield(AST_DictComp* node, InternedString dict_name, std::vector<AST_stmt*>* insert_point) {
...@@ -1013,17 +1020,19 @@ private: ...@@ -1013,17 +1020,19 @@ private:
auto lambda = auto lambda =
[&](std::vector<AST_stmt*>* insert_point) { emitComprehensionYield(node, rtn_name, insert_point); }; [&](std::vector<AST_stmt*>* insert_point) { emitComprehensionYield(node, rtn_name, insert_point); };
AST_Name* first_name = makeName(internString("#arg"), AST_TYPE::Load, node->lineno); AST_Name* first_name
= makeName(internString("#arg"), AST_TYPE::Load, node->lineno, /* col_offset */ 0, /* is_kill */ true);
emitComprehensionLoops(&func->function_def->body, node->generators, first_name, lambda); emitComprehensionLoops(&func->function_def->body, node->generators, first_name, lambda);
auto rtn = new AST_Return(); auto rtn = new AST_Return();
rtn->value = makeName(rtn_name, AST_TYPE::Load, node->lineno); rtn->value = makeName(rtn_name, AST_TYPE::Load, node->lineno, /* col_offset */ 0, /* is_kill */ true);
func->function_def->body.push_back(rtn); func->function_def->body.push_back(rtn);
InternedString func_var_name = nodeName(); InternedString func_var_name = nodeName();
pushAssign(func_var_name, func); pushAssign(func_var_name, func);
return makeCall(makeName(func_var_name, AST_TYPE::Load, node->lineno), first); return makeCall(makeName(func_var_name, AST_TYPE::Load, node->lineno, /* col_offset */ 0, /* is_kill */ true),
first);
} }
AST_expr* remapIfExp(AST_IfExp* node) { AST_expr* remapIfExp(AST_IfExp* node) {
...@@ -1054,7 +1063,7 @@ private: ...@@ -1054,7 +1063,7 @@ private:
cfg->placeBlock(exit_block); cfg->placeBlock(exit_block);
curblock = exit_block; curblock = exit_block;
return makeLoad(rtn_name, node); return makeLoad(rtn_name, node, true);
} }
AST_slice* remapIndex(AST_Index* node) { AST_slice* remapIndex(AST_Index* node) {
...@@ -1190,7 +1199,7 @@ private: ...@@ -1190,7 +1199,7 @@ private:
if (root_type != AST_TYPE::FunctionDef && root_type != AST_TYPE::Lambda) if (root_type != AST_TYPE::FunctionDef && root_type != AST_TYPE::Lambda)
raiseExcHelper(SyntaxError, "'yield' outside function"); raiseExcHelper(SyntaxError, "'yield' outside function");
return makeLoad(node_name, node); return makeLoad(node_name, node, /* is_kill */ true);
} }
AST_slice* remapSlice(AST_slice* node) { AST_slice* remapSlice(AST_slice* node) {
...@@ -1328,7 +1337,7 @@ private: ...@@ -1328,7 +1337,7 @@ private:
void exitFinally(AST* node, Why why, CFGBlock* exit_block = nullptr) { void exitFinally(AST* node, Why why, CFGBlock* exit_block = nullptr) {
switch (why) { switch (why) {
case Why::RETURN: case Why::RETURN:
doReturn(makeLoad(internString(RETURN_NAME), node)); doReturn(makeLoad(internString(RETURN_NAME), node, true));
break; break;
case Why::BREAK: case Why::BREAK:
doBreak(node); doBreak(node);
...@@ -1350,14 +1359,14 @@ private: ...@@ -1350,14 +1359,14 @@ private:
// helper for visit_{with,tryfinally}. Generates a branch testing the value of `whyexpr' against `why', and // helper for visit_{with,tryfinally}. Generates a branch testing the value of `whyexpr' against `why', and
// performing the appropriate exit from the with-block if they are equal. // performing the appropriate exit from the with-block if they are equal.
// NB. `exit_block' is only used if `why' is FALLTHROUGH. // NB. `exit_block' is only used if `why' is FALLTHROUGH.
void exitFinallyIf(AST* node, Why why, InternedString whyname, CFGBlock* exit_block = nullptr) { void exitFinallyIf(AST* node, Why why, InternedString whyname, bool is_kill = false) {
CFGBlock* do_exit = cfg->addDeferredBlock(); CFGBlock* do_exit = cfg->addDeferredBlock();
do_exit->info = "with_exit_if"; do_exit->info = "with_exit_if";
CFGBlock* otherwise = makeFinallyCont(why, makeLoad(whyname, node), do_exit); CFGBlock* otherwise = makeFinallyCont(why, makeLoad(whyname, node, is_kill), do_exit);
cfg->placeBlock(do_exit); cfg->placeBlock(do_exit);
curblock = do_exit; curblock = do_exit;
exitFinally(node, why, exit_block); exitFinally(node, why);
curblock = otherwise; curblock = otherwise;
} }
...@@ -1730,7 +1739,7 @@ public: ...@@ -1730,7 +1739,7 @@ public:
InternedString n_name(nodeName()); InternedString n_name(nodeName());
pushAssign(n_name, makeLoad(n->id, node)); pushAssign(n_name, makeLoad(n->id, node));
remapped_target = n; remapped_target = n;
remapped_lhs = makeLoad(n_name, node); remapped_lhs = makeLoad(n_name, node, /* is_kill */ true);
break; break;
} }
case AST_TYPE::Subscript: { case AST_TYPE::Subscript: {
...@@ -1883,13 +1892,14 @@ public: ...@@ -1883,13 +1892,14 @@ public:
AST_Print* remapped = new AST_Print(); AST_Print* remapped = new AST_Print();
remapped->col_offset = node->col_offset; remapped->col_offset = node->col_offset;
remapped->lineno = node->lineno; remapped->lineno = node->lineno;
// TODO not good to reuse 'dest' like this
remapped->dest = _dup(dest);
if (i < node->values.size() - 1) if (i < node->values.size() - 1) {
remapped->dest = _dup(dest);
remapped->nl = false; remapped->nl = false;
else } else {
remapped->dest = dest;
remapped->nl = node->nl; remapped->nl = node->nl;
}
remapped->values.push_back(remapExpr(v)); remapped->values.push_back(remapExpr(v));
push_back(remapped); push_back(remapped);
...@@ -2384,7 +2394,7 @@ public: ...@@ -2384,7 +2394,7 @@ public:
exitFinallyIf(node, Why::CONTINUE, exc_why_name); exitFinallyIf(node, Why::CONTINUE, exc_why_name);
CFGBlock* reraise = cfg->addDeferredBlock(); CFGBlock* reraise = cfg->addDeferredBlock();
CFGBlock* noexc = makeFinallyCont(Why::EXCEPTION, makeLoad(exc_why_name, node), reraise); CFGBlock* noexc = makeFinallyCont(Why::EXCEPTION, makeLoad(exc_why_name, node, true), reraise);
cfg->placeBlock(reraise); cfg->placeBlock(reraise);
curblock = reraise; curblock = reraise;
...@@ -2441,7 +2451,7 @@ public: ...@@ -2441,7 +2451,7 @@ public:
// Oddly, this acces to __enter__ doesn't suffer from the same bug. Perhaps it has something to do with // Oddly, this acces to __enter__ doesn't suffer from the same bug. Perhaps it has something to do with
// __enter__ being called immediately? // __enter__ being called immediately?
AST_expr* enter = makeLoadAttribute(makeLoad(ctxmgrname, node), internString("__enter__"), true); AST_expr* enter = makeLoadAttribute(makeLoad(ctxmgrname, node, true), internString("__enter__"), true);
enter = remapExpr(makeCall(enter)); enter = remapExpr(makeCall(enter));
if (node->optional_vars) if (node->optional_vars)
pushAssign(node->optional_vars, enter); pushAssign(node->optional_vars, enter);
...@@ -2483,7 +2493,7 @@ public: ...@@ -2483,7 +2493,7 @@ public:
// call the context-manager's exit method // call the context-manager's exit method
InternedString suppressname = nodeName("suppress"); InternedString suppressname = nodeName("suppress");
pushAssign(suppressname, makeCall(makeLoad(exitname, node), makeLoad(exc_type_name, node), pushAssign(suppressname, makeCall(makeLoad(exitname, node, true), makeLoad(exc_type_name, node),
makeLoad(exc_value_name, node), makeLoad(exc_traceback_name, node))); makeLoad(exc_value_name, node), makeLoad(exc_traceback_name, node)));
// if it returns true, suppress the error and go to our exit block // if it returns true, suppress the error and go to our exit block
...@@ -2492,7 +2502,7 @@ public: ...@@ -2492,7 +2502,7 @@ public:
// break potential critical edge // break potential critical edge
CFGBlock* exiter = cfg->addDeferredBlock(); CFGBlock* exiter = cfg->addDeferredBlock();
exiter->info = "with_exiter"; exiter->info = "with_exiter";
pushBranch(makeLoad(suppressname, node), exiter, reraise_block); pushBranch(makeLoad(suppressname, node, true), exiter, reraise_block);
cfg->placeBlock(exiter); cfg->placeBlock(exiter);
curblock = exiter; curblock = exiter;
...@@ -2512,15 +2522,15 @@ public: ...@@ -2512,15 +2522,15 @@ public:
cfg->placeBlock(finally_block); cfg->placeBlock(finally_block);
curblock = finally_block; curblock = finally_block;
// call the context-manager's exit method, ignoring result // call the context-manager's exit method, ignoring result
push_back(makeExpr(makeCall(makeLoad(exitname, node), makeLoad(nonename, node), makeLoad(nonename, node), push_back(makeExpr(makeCall(makeLoad(exitname, node, true), makeLoad(nonename, node),
makeLoad(nonename, node)))); makeLoad(nonename, node), makeLoad(nonename, node))));
if (finally_did_why & (1 << Why::CONTINUE)) if (finally_did_why & (1 << Why::CONTINUE))
exitFinallyIf(node, Why::CONTINUE, whyname); exitFinallyIf(node, Why::CONTINUE, whyname, /* is_kill */ finally_did_why == (1 << Why::CONTINUE));
if (finally_did_why & (1 << Why::BREAK)) if (finally_did_why & (1 << Why::BREAK))
exitFinallyIf(node, Why::BREAK, whyname); exitFinallyIf(node, Why::BREAK, whyname, /* is_kill */ !(finally_did_why & (1 << Why::RETURN)));
if (finally_did_why & (1 << Why::RETURN)) if (finally_did_why & (1 << Why::RETURN))
exitFinallyIf(node, Why::RETURN, whyname); exitFinallyIf(node, Why::RETURN, whyname, /* is_kill */ true);
exitFinally(node, Why::FALLTHROUGH, exit_block); exitFinally(node, Why::FALLTHROUGH, exit_block);
} }
...@@ -2883,4 +2893,8 @@ CFG* computeCFG(SourceInfo* source, std::vector<AST_stmt*> body) { ...@@ -2883,4 +2893,8 @@ CFG* computeCFG(SourceInfo* source, std::vector<AST_stmt*> body) {
return rtn; return rtn;
} }
void printCFG(CFG* cfg) {
cfg->print();
}
} }
...@@ -120,6 +120,7 @@ public: ...@@ -120,6 +120,7 @@ public:
class SourceInfo; class SourceInfo;
CFG* computeCFG(SourceInfo* source, std::vector<AST_stmt*> body); CFG* computeCFG(SourceInfo* source, std::vector<AST_stmt*> body);
void printCFG(CFG* cfg);
} }
#endif #endif
# expected: fail
# - we have an extra evaluation of __nonzero__ at the end
class A(object):
def __init__(self, n):
print "init", n
self.n = n
def __lt__(self, rhs):
print "lt", self.n, rhs.n
return self
def __nonzero__(self):
print "nonzero", self.n
return True
def __repr__(self):
return "<A (%d)>" % self.n
print A(1) < A(2) < A(3)
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