Commit 71840bc9 authored by Travis Hance's avatar Travis Hance

Merge pull request #415 from tjhance/static-closures

implemented kmod's suggestions
parents ec5a99c9 6fecfd16
...@@ -103,10 +103,14 @@ public: ...@@ -103,10 +103,14 @@ public:
bool usesNameLookup() override { return false; } bool usesNameLookup() override { return false; }
bool isPassedToViaClosure(InternedString name) override { return false; }
bool areLocalsFromModule() override { return true; } bool areLocalsFromModule() override { return true; }
DerefInfo getDerefInfo(InternedString) override { RELEASE_ASSERT(0, "This should never get called"); }
size_t getClosureOffset(InternedString) override { RELEASE_ASSERT(0, "This should never get called"); }
size_t getClosureSize() override { RELEASE_ASSERT(0, "This should never get called"); }
std::vector<std::pair<InternedString, DerefInfo>> v;
const std::vector<std::pair<InternedString, DerefInfo>>& getAllDerefVarsAndInfo() override { return v; }
InternedString mangleName(InternedString id) override { return id; } InternedString mangleName(InternedString id) override { return id; }
InternedString internString(llvm::StringRef s) override { abort(); } InternedString internString(llvm::StringRef s) override { abort(); }
}; };
...@@ -165,10 +169,14 @@ public: ...@@ -165,10 +169,14 @@ public:
bool usesNameLookup() override { return true; } bool usesNameLookup() override { return true; }
bool isPassedToViaClosure(InternedString name) override { return false; }
bool areLocalsFromModule() override { return false; } bool areLocalsFromModule() override { return false; }
DerefInfo getDerefInfo(InternedString) override { RELEASE_ASSERT(0, "This should never get called"); }
size_t getClosureOffset(InternedString) override { RELEASE_ASSERT(0, "This should never get called"); }
size_t getClosureSize() override { RELEASE_ASSERT(0, "This should never get called"); }
std::vector<std::pair<InternedString, DerefInfo>> v;
const std::vector<std::pair<InternedString, DerefInfo>>& getAllDerefVarsAndInfo() override { return v; }
InternedString mangleName(InternedString id) override { return id; } InternedString mangleName(InternedString id) override { return id; }
InternedString internString(llvm::StringRef s) override { abort(); } InternedString internString(llvm::StringRef s) override { abort(); }
}; };
...@@ -258,11 +266,22 @@ private: ...@@ -258,11 +266,22 @@ private:
AST* ast; AST* ast;
bool usesNameLookup_; bool usesNameLookup_;
llvm::DenseMap<InternedString, size_t> closure_offsets;
std::vector<std::pair<InternedString, DerefInfo>> allDerefVarsAndInfo;
bool allDerefVarsAndInfoCached;
public: public:
ScopeInfoBase(ScopeInfo* parent, ScopingAnalysis::ScopeNameUsage* usage, AST* ast, bool usesNameLookup) ScopeInfoBase(ScopeInfo* parent, ScopingAnalysis::ScopeNameUsage* usage, AST* ast, bool usesNameLookup)
: parent(parent), usage(usage), ast(ast), usesNameLookup_(usesNameLookup) { : parent(parent), usage(usage), ast(ast), usesNameLookup_(usesNameLookup), allDerefVarsAndInfoCached(false) {
assert(usage); assert(usage);
assert(ast); assert(ast);
int i = 0;
for (auto& p : usage->referenced_from_nested) {
closure_offsets[p] = i;
i++;
}
} }
~ScopeInfoBase() override { delete this->usage; } ~ScopeInfoBase() override { delete this->usage; }
...@@ -301,20 +320,79 @@ public: ...@@ -301,20 +320,79 @@ public:
bool usesNameLookup() override { return usesNameLookup_; } bool usesNameLookup() override { return usesNameLookup_; }
bool isPassedToViaClosure(InternedString name) override { bool areLocalsFromModule() override { return false; }
if (isCompilerCreatedName(name))
return false; DerefInfo getDerefInfo(InternedString name) override {
assert(getScopeTypeOfName(name) == VarScopeType::DEREF);
// TODO pre-compute this?
return usage->got_from_closure.count(name) > 0 || usage->passthrough_accesses.count(name) > 0; size_t parentCounter = 0;
// Casting to a ScopeInfoBase* is okay because only a ScopeInfoBase can have a closure.
// We just walk up the scopes until we find the scope with this name. Count the number
// of parent links we follow, and then get the offset of the name.
for (ScopeInfoBase* parent = static_cast<ScopeInfoBase*>(this->parent); parent != NULL;
parent = static_cast<ScopeInfoBase*>(parent->parent)) {
if (parent->createsClosure()) {
auto it = parent->closure_offsets.find(name);
if (it != parent->closure_offsets.end()) {
return DerefInfo{.num_parents_from_passed_closure = parentCounter, .offset = it->second };
}
parentCounter++;
}
} }
bool areLocalsFromModule() override { return false; } RELEASE_ASSERT(0, "Should not get here");
}
size_t getClosureOffset(InternedString name) override {
assert(getScopeTypeOfName(name) == VarScopeType::CLOSURE);
return closure_offsets[name];
}
size_t getClosureSize() override {
assert(createsClosure());
return closure_offsets.size();
}
InternedString mangleName(const InternedString id) override { InternedString mangleName(const InternedString id) override {
return pyston::mangleName(id, usage->private_name, usage->scoping->getInternedStrings()); return pyston::mangleName(id, usage->private_name, usage->scoping->getInternedStrings());
} }
InternedString internString(llvm::StringRef s) override { return usage->scoping->getInternedStrings().get(s); } InternedString internString(llvm::StringRef s) override { return usage->scoping->getInternedStrings().get(s); }
const std::vector<std::pair<InternedString, DerefInfo>>& getAllDerefVarsAndInfo() override {
if (!allDerefVarsAndInfoCached) {
allDerefVarsAndInfoCached = true;
// TODO this could probably be implemented faster
// Get all the variables that we need to return: any variable from the
// passed-in closure that is accessed in this scope or in a child scope.
StrSet allDerefs = usage->got_from_closure;
for (InternedString name : usage->passthrough_accesses) {
if (allDerefs.find(name) != allDerefs.end()) {
allDerefs.insert(name);
}
}
// Call `getDerefInfo` on all of these variables and put the results in
// `allDerefVarsAndInfo`
for (InternedString name : allDerefs) {
allDerefVarsAndInfo.push_back({ name, getDerefInfo(name) });
}
// Sort in order of `num_parents_from_passed_closure`
std::sort(allDerefVarsAndInfo.begin(), allDerefVarsAndInfo.end(), derefComparator);
}
return allDerefVarsAndInfo;
}
private:
static bool derefComparator(const std::pair<InternedString, DerefInfo>& p1,
const std::pair<InternedString, DerefInfo>& p2) {
return p1.second.num_parents_from_passed_closure < p2.second.num_parents_from_passed_closure;
};
}; };
class NameCollectorVisitor : public ASTVisitor { class NameCollectorVisitor : public ASTVisitor {
......
...@@ -25,6 +25,16 @@ class AST_Module; ...@@ -25,6 +25,16 @@ class AST_Module;
class AST_Expression; class AST_Expression;
class AST_Suite; class AST_Suite;
// Each closure has an array (fixed-size for that particular scope) of variables
// and a parent pointer to a parent closure. To look up a variable from the passed-in
// closure (i.e., DEREF), you just need to know (i) how many parents up to go and
// (ii) what offset into the array to find the variable. This struct stores that
// information. You can query the ScopeInfo with a name to get this info.
struct DerefInfo {
size_t num_parents_from_passed_closure;
size_t offset;
};
class ScopeInfo { class ScopeInfo {
public: public:
ScopeInfo() {} ScopeInfo() {}
...@@ -75,11 +85,28 @@ public: ...@@ -75,11 +85,28 @@ public:
}; };
virtual VarScopeType getScopeTypeOfName(InternedString name) = 0; virtual VarScopeType getScopeTypeOfName(InternedString name) = 0;
// Returns true if the variable should be passed via a closure to this scope. // Returns true if the scope may contain NAME variables.
// In particular, it returns true for ClassDef scope, for any scope
// with an `exec` statement or `import *` statement in it, or for any
// `exec` or `eval` scope.
virtual bool usesNameLookup() = 0;
virtual bool areLocalsFromModule() = 0;
// For a variable with DEREF lookup, return the DerefInfo used to lookup
// the variable in a passed closure.
virtual DerefInfo getDerefInfo(InternedString name) = 0;
// Gets the DerefInfo for each DEREF variable accessible in the scope.
// The returned vector is in SORTED ORDER by the `num_parents_from_passed_closure` field
// (ascending). This allows the caller to iterate through the vector while also walking up
// the closure chain to collect all the DEREF variable values. This is useful, for example,
// in the implementation of locals().
//
// Note that: // Note that:
// (a) This can be false even if there is an entry in the closure object // (a) This may not return a variable even if it is in the passed-in scope,
// passed to the scope, if the variable is not actually used in this // if the variable is not actually used in this scope or any child
// scope or any child scopes. This can happen, because the variable // scopes. This can happen, because the variable
// could be in the closure to be accessed by a different function, e.g. // could be in the closure to be accessed by a different function, e.g.
// //
// def f(); // def f();
...@@ -93,8 +120,8 @@ public: ...@@ -93,8 +120,8 @@ public:
// # passed a closure object with `a` in it // # passed a closure object with `a` in it
// print locals() // print locals()
// //
// (b) This can be true even if it is not used in this scope, if it // (b) This can contain a variable even if it is not access in this scope,
// is used in a child scope. For example: // if it used in a child scope instead. For example:
// //
// def f(): // def f():
// a = 0 // a = 0
...@@ -102,19 +129,15 @@ public: ...@@ -102,19 +129,15 @@ public:
// def h(): // def h():
// print a // print a
// print locals() # should contain `a` // print locals() # should contain `a`
// virtual const std::vector<std::pair<InternedString, DerefInfo>>& getAllDerefVarsAndInfo() = 0;
// This is useful because it determines whether a variable from a closure
// into the locals() dictionary.
virtual bool isPassedToViaClosure(InternedString name) = 0; // For a variable with CLOSURE lookup, returns the offset within the `elts`
// array of a closure that this variable is stored.
// Returns true if the scope may contain NAME variables. virtual size_t getClosureOffset(InternedString name) = 0;
// In particular, it returns true for ClassDef scope, for any scope
// with an `exec` statement or `import *` statement in it, or for any
// `exec` or `eval` scope.
virtual bool usesNameLookup() = 0;
virtual bool areLocalsFromModule() = 0; // Returns the size of the `elts` array for a closure created by this scope.
// Should only be called if this scope creates a closure.
virtual size_t getClosureSize() = 0;
virtual InternedString mangleName(InternedString id) = 0; virtual InternedString mangleName(InternedString id) = 0;
virtual InternedString internString(llvm::StringRef) = 0; virtual InternedString internString(llvm::StringRef) = 0;
......
...@@ -228,7 +228,7 @@ void ASTInterpreter::initArguments(int nargs, BoxedClosure* _closure, BoxedGener ...@@ -228,7 +228,7 @@ void ASTInterpreter::initArguments(int nargs, BoxedClosure* _closure, BoxedGener
generator = _generator; generator = _generator;
if (scope_info->createsClosure()) if (scope_info->createsClosure())
created_closure = createClosure(passed_closure); created_closure = createClosure(passed_closure, scope_info->getClosureSize());
std::vector<Box*, StlCompatAllocator<Box*>> argsArray{ arg1, arg2, arg3 }; std::vector<Box*, StlCompatAllocator<Box*>> argsArray{ arg1, arg2, arg3 };
for (int i = 3; i < nargs; ++i) for (int i = 3; i < nargs; ++i)
...@@ -354,8 +354,9 @@ void ASTInterpreter::doStore(InternedString name, Value value) { ...@@ -354,8 +354,9 @@ void ASTInterpreter::doStore(InternedString name, Value value) {
setitem(frame_info.boxedLocals, boxString(name.str()), value.o); setitem(frame_info.boxedLocals, boxString(name.str()), value.o);
} else { } else {
sym_table[name] = value.o; sym_table[name] = value.o;
if (vst == ScopeInfo::VarScopeType::CLOSURE) if (vst == ScopeInfo::VarScopeType::CLOSURE) {
setattr(created_closure, name.c_str(), value.o); created_closure->elts[scope_info->getClosureOffset(name)] = value.o;
}
} }
} }
...@@ -1086,8 +1087,20 @@ Value ASTInterpreter::visit_name(AST_Name* node) { ...@@ -1086,8 +1087,20 @@ Value ASTInterpreter::visit_name(AST_Name* node) {
switch (node->lookup_type) { switch (node->lookup_type) {
case ScopeInfo::VarScopeType::GLOBAL: case ScopeInfo::VarScopeType::GLOBAL:
return getGlobal(source_info->parent_module, &node->id.str()); return getGlobal(source_info->parent_module, &node->id.str());
case ScopeInfo::VarScopeType::DEREF: case ScopeInfo::VarScopeType::DEREF: {
return getattr(passed_closure, node->id.c_str()); DerefInfo deref_info = scope_info->getDerefInfo(node->id);
assert(passed_closure);
BoxedClosure* closure = passed_closure;
for (int i = 0; i < deref_info.num_parents_from_passed_closure; i++) {
closure = closure->parent;
}
Box* val = closure->elts[deref_info.offset];
if (val == NULL) {
raiseExcHelper(NameError, "free variable '%s' referenced before assignment in enclosing scope",
node->id.c_str());
}
return val;
}
case ScopeInfo::VarScopeType::FAST: case ScopeInfo::VarScopeType::FAST:
case ScopeInfo::VarScopeType::CLOSURE: { case ScopeInfo::VarScopeType::CLOSURE: {
SymMap::iterator it = sym_table.find(node->id); SymMap::iterator it = sym_table.find(node->id);
......
...@@ -1781,15 +1781,17 @@ public: ...@@ -1781,15 +1781,17 @@ public:
CompilerVariable* getattr(IREmitter& emitter, const OpInfo& info, ConcreteCompilerVariable* var, CompilerVariable* getattr(IREmitter& emitter, const OpInfo& info, ConcreteCompilerVariable* var,
const std::string* attr, bool cls_only) override { const std::string* attr, bool cls_only) override {
RELEASE_ASSERT(0, "should not be called\n");
/*
assert(!cls_only); assert(!cls_only);
llvm::Value* bitcast = emitter.getBuilder()->CreateBitCast(var->getValue(), g.llvm_value_type_ptr); llvm::Value* bitcast = emitter.getBuilder()->CreateBitCast(var->getValue(), g.llvm_value_type_ptr);
return ConcreteCompilerVariable(UNKNOWN, bitcast, true).getattr(emitter, info, attr, cls_only); return ConcreteCompilerVariable(UNKNOWN, bitcast, true).getattr(emitter, info, attr, cls_only);
*/
} }
void setattr(IREmitter& emitter, const OpInfo& info, ConcreteCompilerVariable* var, const std::string* attr, void setattr(IREmitter& emitter, const OpInfo& info, ConcreteCompilerVariable* var, const std::string* attr,
CompilerVariable* v) override { CompilerVariable* v) override {
llvm::Value* bitcast = emitter.getBuilder()->CreateBitCast(var->getValue(), g.llvm_value_type_ptr); RELEASE_ASSERT(0, "should not be called\n");
ConcreteCompilerVariable(UNKNOWN, bitcast, true).setattr(emitter, info, attr, v);
} }
ConcreteCompilerType* getConcreteType() override { return this; } ConcreteCompilerType* getConcreteType() override { return this; }
......
...@@ -72,6 +72,21 @@ llvm::Value* IRGenState::getScratchSpace(int min_bytes) { ...@@ -72,6 +72,21 @@ llvm::Value* IRGenState::getScratchSpace(int min_bytes) {
return scratch_space; return scratch_space;
} }
static llvm::Value* getClosureParentGep(IREmitter& emitter, llvm::Value* closure) {
static_assert(sizeof(Box) == offsetof(BoxedClosure, parent), "");
static_assert(offsetof(BoxedClosure, parent) + sizeof(BoxedClosure*) == offsetof(BoxedClosure, nelts), "");
return emitter.getBuilder()->CreateConstInBoundsGEP2_32(closure, 0, 1);
}
static llvm::Value* getClosureElementGep(IREmitter& emitter, llvm::Value* closure, size_t index) {
static_assert(sizeof(Box) == offsetof(BoxedClosure, parent), "");
static_assert(offsetof(BoxedClosure, parent) + sizeof(BoxedClosure*) == offsetof(BoxedClosure, nelts), "");
static_assert(offsetof(BoxedClosure, nelts) + sizeof(size_t) == offsetof(BoxedClosure, elts), "");
return emitter.getBuilder()->CreateGEP(
closure,
{ llvm::ConstantInt::get(g.i32, 0), llvm::ConstantInt::get(g.i32, 3), llvm::ConstantInt::get(g.i32, index) });
}
static llvm::Value* getBoxedLocalsGep(llvm::IRBuilder<true>& builder, llvm::Value* v) { static llvm::Value* getBoxedLocalsGep(llvm::IRBuilder<true>& builder, llvm::Value* v) {
static_assert(offsetof(FrameInfo, exc) == 0, ""); static_assert(offsetof(FrameInfo, exc) == 0, "");
static_assert(sizeof(ExcInfo) == 24, ""); static_assert(sizeof(ExcInfo) == 24, "");
...@@ -898,10 +913,51 @@ private: ...@@ -898,10 +913,51 @@ private:
assert(!is_kill); assert(!is_kill);
assert(scope_info->takesClosure()); assert(scope_info->takesClosure());
// This is the information on how to look up the variable in the closure object.
DerefInfo deref_info = scope_info->getDerefInfo(node->id);
// This code is basically:
// closure = created_closure;
// closure = closure->parent;
// [...]
// closure = closure->parent;
// closure->elts[deref_info.offset]
// Where the parent lookup is done `deref_info.num_parents_from_passed_closure` times
CompilerVariable* closure = symbol_table[internString(PASSED_CLOSURE_NAME)]; CompilerVariable* closure = symbol_table[internString(PASSED_CLOSURE_NAME)];
assert(closure); llvm::Value* closureValue = closure->makeConverted(emitter, CLOSURE)->getValue();
closure->decvref(emitter);
for (int i = 0; i < deref_info.num_parents_from_passed_closure; i++) {
closureValue = emitter.getBuilder()->CreateLoad(getClosureParentGep(emitter, closureValue));
}
llvm::Value* lookupResult
= emitter.getBuilder()->CreateLoad(getClosureElementGep(emitter, closureValue, deref_info.offset));
// If the value is NULL, the variable is undefined.
// Create a branch on if the value is NULL.
llvm::BasicBlock* success_bb
= llvm::BasicBlock::Create(g.context, "deref_defined", irstate->getLLVMFunction());
success_bb->moveAfter(curblock);
llvm::BasicBlock* fail_bb
= llvm::BasicBlock::Create(g.context, "deref_undefined", irstate->getLLVMFunction());
llvm::Value* check_val
= emitter.getBuilder()->CreateICmpEQ(lookupResult, embedConstantPtr(NULL, g.llvm_value_type_ptr));
llvm::BranchInst* non_null_check = emitter.getBuilder()->CreateCondBr(check_val, fail_bb, success_bb);
return closure->getattr(emitter, getEmptyOpInfo(unw_info), &node->id.str(), false); // Case that it is undefined: call the assert fail function.
curblock = fail_bb;
emitter.getBuilder()->SetInsertPoint(curblock);
llvm::CallSite call = emitter.createCall(unw_info, g.funcs.assertFailDerefNameDefined,
getStringConstantPtr(node->id.str() + '\0'));
call.setDoesNotReturn();
emitter.getBuilder()->CreateUnreachable();
// Case that it is defined: carry on in with the retrieved value.
curblock = success_bb;
emitter.getBuilder()->SetInsertPoint(curblock);
return new ConcreteCompilerVariable(UNKNOWN, lookupResult, true);
} else if (vst == ScopeInfo::VarScopeType::NAME) { } else if (vst == ScopeInfo::VarScopeType::NAME) {
llvm::Value* boxedLocals = irstate->getBoxedLocalsVar(); llvm::Value* boxedLocals = irstate->getBoxedLocalsVar();
llvm::Value* attr = getStringConstantPtr(node->id.str() + '\0'); llvm::Value* attr = getStringConstantPtr(node->id.str() + '\0');
...@@ -1435,10 +1491,14 @@ private: ...@@ -1435,10 +1491,14 @@ private:
_popFake(defined_name, true); _popFake(defined_name, true);
if (vst == ScopeInfo::VarScopeType::CLOSURE) { if (vst == ScopeInfo::VarScopeType::CLOSURE) {
CompilerVariable* closure = symbol_table[internString(CREATED_CLOSURE_NAME)]; size_t offset = scope_info->getClosureOffset(name);
assert(closure);
closure->setattr(emitter, getEmptyOpInfo(unw_info), &name.str(), val); // This is basically `closure->elts[offset] = val;`
CompilerVariable* closure = symbol_table[internString(CREATED_CLOSURE_NAME)];
llvm::Value* closureValue = closure->makeConverted(emitter, CLOSURE)->getValue();
closure->decvref(emitter);
llvm::Value* gep = getClosureElementGep(emitter, closureValue, offset);
emitter.getBuilder()->CreateStore(val->makeConverted(emitter, UNKNOWN)->getValue(), gep);
} }
} }
} }
...@@ -1897,7 +1957,8 @@ private: ...@@ -1897,7 +1957,8 @@ private:
assert(var->getType() != BOXED_FLOAT assert(var->getType() != BOXED_FLOAT
&& "should probably unbox it, but why is it boxed in the first place?"); && "should probably unbox it, but why is it boxed in the first place?");
// This line can never get hit right now for the same reason that the variables must already be concrete, // This line can never get hit right now for the same reason that the variables must already be
// concrete,
// because we're over-generating phis. // because we're over-generating phis.
ASSERT(var->isGrabbed(), "%s", p.first.c_str()); ASSERT(var->isGrabbed(), "%s", p.first.c_str());
// var->ensureGrabbed(emitter); // var->ensureGrabbed(emitter);
...@@ -2157,7 +2218,8 @@ private: ...@@ -2157,7 +2218,8 @@ private:
} else { } else {
#ifndef NDEBUG #ifndef NDEBUG
if (myblock->successors.size()) { if (myblock->successors.size()) {
// TODO getTypeAtBlockEnd will automatically convert up to the concrete type, which we don't want // TODO getTypeAtBlockEnd will automatically convert up to the concrete type, which we don't
// want
// here, but this is just for debugging so I guess let it happen for now: // here, but this is just for debugging so I guess let it happen for now:
ConcreteCompilerType* ending_type = types->getTypeAtBlockEnd(it->first, myblock); ConcreteCompilerType* ending_type = types->getTypeAtBlockEnd(it->first, myblock);
ASSERT(it->second->canConvertTo(ending_type), "%s is supposed to be %s, but somehow is %s", ASSERT(it->second->canConvertTo(ending_type), "%s is supposed to be %s, but somehow is %s",
...@@ -2357,7 +2419,8 @@ public: ...@@ -2357,7 +2419,8 @@ public:
if (!passed_closure) if (!passed_closure)
passed_closure = embedConstantPtr(nullptr, g.llvm_closure_type_ptr); passed_closure = embedConstantPtr(nullptr, g.llvm_closure_type_ptr);
llvm::Value* new_closure = emitter.getBuilder()->CreateCall(g.funcs.createClosure, passed_closure); llvm::Value* new_closure = emitter.getBuilder()->CreateCall2(
g.funcs.createClosure, passed_closure, getConstantInt(scope_info->getClosureSize(), g.i64));
symbol_table[internString(CREATED_CLOSURE_NAME)] symbol_table[internString(CREATED_CLOSURE_NAME)]
= new ConcreteCompilerVariable(getCreatedClosureType(), new_closure, true); = new ConcreteCompilerVariable(getCreatedClosureType(), new_closure, true);
} }
......
...@@ -216,6 +216,7 @@ void initGlobalFuncs(GlobalState& g) { ...@@ -216,6 +216,7 @@ void initGlobalFuncs(GlobalState& g) {
GET(raiseAttributeErrorStr); GET(raiseAttributeErrorStr);
GET(raiseNotIterableError); GET(raiseNotIterableError);
GET(assertNameDefined); GET(assertNameDefined);
GET(assertFailDerefNameDefined);
GET(assertFail); GET(assertFail);
GET(printFloat); GET(printFloat);
......
...@@ -41,7 +41,7 @@ struct GlobalFuncs { ...@@ -41,7 +41,7 @@ struct GlobalFuncs {
*exceptionMatches, *yield, *getiterHelper, *hasnext; *exceptionMatches, *yield, *getiterHelper, *hasnext;
llvm::Value* unpackIntoArray, *raiseAttributeError, *raiseAttributeErrorStr, *raiseNotIterableError, llvm::Value* unpackIntoArray, *raiseAttributeError, *raiseAttributeErrorStr, *raiseNotIterableError,
*assertNameDefined, *assertFail; *assertNameDefined, *assertFail, *assertFailDerefNameDefined;
llvm::Value* printFloat, *listAppendInternal, *getSysStdout; llvm::Value* printFloat, *listAppendInternal, *getSysStdout;
llvm::Value* runtimeCall0, *runtimeCall1, *runtimeCall2, *runtimeCall3, *runtimeCall, *runtimeCallN; llvm::Value* runtimeCall0, *runtimeCall1, *runtimeCall2, *runtimeCall3, *runtimeCall, *runtimeCallN;
llvm::Value* callattr0, *callattr1, *callattr2, *callattr3, *callattr, *callattrN; llvm::Value* callattr0, *callattr1, *callattr2, *callattr3, *callattr, *callattrN;
......
...@@ -758,18 +758,21 @@ Box* fastLocalsToBoxedLocals() { ...@@ -758,18 +758,21 @@ Box* fastLocalsToBoxedLocals() {
// Add the locals from the closure // Add the locals from the closure
// TODO in a ClassDef scope, we aren't supposed to add these // TODO in a ClassDef scope, we aren't supposed to add these
for (; closure != NULL; closure = closure->parent) { size_t depth = 0;
assert(closure->cls == closure_cls); for (auto& p : scope_info->getAllDerefVarsAndInfo()) {
for (auto& attr_offset : closure->attrs.hcls->getAttrOffsets()) { InternedString name = p.first;
const std::string& name = attr_offset.first(); DerefInfo derefInfo = p.second;
int offset = attr_offset.second; while (depth < derefInfo.num_parents_from_passed_closure) {
Box* val = closure->attrs.attr_list->attrs[offset]; depth++;
if (val != NULL && scope_info->isPassedToViaClosure(scope_info->internString(name))) { closure = closure->parent;
Box* boxedName = boxString(name); }
if (d->d.count(boxedName) == 0) { assert(closure != NULL);
d->d[boxString(name)] = val; Box* val = closure->elts[derefInfo.offset];
} Box* boxedName = boxString(name.str());
} if (val != NULL) {
d->d[boxedName] = val;
} else {
d->d.erase(boxedName);
} }
} }
......
...@@ -98,6 +98,7 @@ void force() { ...@@ -98,6 +98,7 @@ void force() {
FORCE(raiseAttributeErrorStr); FORCE(raiseAttributeErrorStr);
FORCE(raiseNotIterableError); FORCE(raiseNotIterableError);
FORCE(assertNameDefined); FORCE(assertNameDefined);
FORCE(assertFailDerefNameDefined);
FORCE(assertFail); FORCE(assertFail);
FORCE(printFloat); FORCE(printFloat);
......
...@@ -233,6 +233,10 @@ extern "C" void assertNameDefined(bool b, const char* name, BoxedClass* exc_cls, ...@@ -233,6 +233,10 @@ extern "C" void assertNameDefined(bool b, const char* name, BoxedClass* exc_cls,
} }
} }
extern "C" void assertFailDerefNameDefined(const char* name) {
raiseExcHelper(NameError, "free variable '%s' referenced before assignment in enclosing scope", name);
}
extern "C" void raiseAttributeErrorStr(const char* typeName, const char* attr) { extern "C" void raiseAttributeErrorStr(const char* typeName, const char* attr) {
raiseExcHelper(AttributeError, "'%s' object has no attribute '%s'", typeName, attr); raiseExcHelper(AttributeError, "'%s' object has no attribute '%s'", typeName, attr);
} }
...@@ -1273,40 +1277,7 @@ Box* getattrInternalGeneric(Box* obj, const std::string& attr, GetattrRewriteArg ...@@ -1273,40 +1277,7 @@ Box* getattrInternalGeneric(Box* obj, const std::string& attr, GetattrRewriteArg
*bind_obj_out = NULL; *bind_obj_out = NULL;
} }
// TODO this should be a custom getattr assert(obj->cls != closure_cls);
if (obj->cls == closure_cls) {
Box* val = NULL;
if (rewrite_args) {
GetattrRewriteArgs hrewrite_args(rewrite_args->rewriter, rewrite_args->obj, rewrite_args->destination);
val = obj->getattr(attr, &hrewrite_args);
if (!hrewrite_args.out_success) {
rewrite_args = NULL;
} else if (val) {
rewrite_args->out_rtn = hrewrite_args.out_rtn;
rewrite_args->out_success = true;
return val;
}
} else {
val = obj->getattr(attr, NULL);
if (val) {
return val;
}
}
// If val doesn't exist, then we move up to the parent closure
// TODO closures should get their own treatment, but now just piggy-back on the
// normal hidden-class IC logic.
// Can do better since we don't need to guard on the cls (always going to be closure)
BoxedClosure* closure = static_cast<BoxedClosure*>(obj);
if (closure->parent) {
if (rewrite_args) {
rewrite_args->obj = rewrite_args->obj->getAttr(offsetof(BoxedClosure, parent));
}
return getattrInternal(closure->parent, attr, rewrite_args);
}
raiseExcHelper(NameError, "free variable '%s' referenced before assignment in enclosing scope", attr.c_str());
}
// Handle descriptor logic here. // Handle descriptor logic here.
// A descriptor is either a data descriptor or a non-data descriptor. // A descriptor is either a data descriptor or a non-data descriptor.
......
...@@ -83,9 +83,10 @@ extern "C" Box* importFrom(Box* obj, const std::string* attr); ...@@ -83,9 +83,10 @@ extern "C" Box* importFrom(Box* obj, const std::string* attr);
extern "C" Box* importStar(Box* from_module, BoxedModule* to_module); extern "C" Box* importStar(Box* from_module, BoxedModule* to_module);
extern "C" Box** unpackIntoArray(Box* obj, int64_t expected_size); extern "C" Box** unpackIntoArray(Box* obj, int64_t expected_size);
extern "C" void assertNameDefined(bool b, const char* name, BoxedClass* exc_cls, bool local_var_msg); extern "C" void assertNameDefined(bool b, const char* name, BoxedClass* exc_cls, bool local_var_msg);
extern "C" void assertFailDerefNameDefined(const char* name);
extern "C" void assertFail(BoxedModule* inModule, Box* msg); extern "C" void assertFail(BoxedModule* inModule, Box* msg);
extern "C" bool isSubclass(BoxedClass* child, BoxedClass* parent); extern "C" bool isSubclass(BoxedClass* child, BoxedClass* parent);
extern "C" BoxedClosure* createClosure(BoxedClosure* parent_closure); extern "C" BoxedClosure* createClosure(BoxedClosure* parent_closure, size_t size);
Box* getiter(Box* o); Box* getiter(Box* o);
extern "C" Box* getPystonIter(Box* o); extern "C" Box* getPystonIter(Box* o);
......
...@@ -636,6 +636,11 @@ extern "C" void closureGCHandler(GCVisitor* v, Box* b) { ...@@ -636,6 +636,11 @@ extern "C" void closureGCHandler(GCVisitor* v, Box* b) {
BoxedClosure* c = (BoxedClosure*)b; BoxedClosure* c = (BoxedClosure*)b;
if (c->parent) if (c->parent)
v->visit(c->parent); v->visit(c->parent);
for (int i = 0; i < c->nelts; i++) {
if (c->elts[i])
v->visit(c->elts[i]);
}
} }
extern "C" { extern "C" {
...@@ -849,10 +854,12 @@ extern "C" Box* createSlice(Box* start, Box* stop, Box* step) { ...@@ -849,10 +854,12 @@ extern "C" Box* createSlice(Box* start, Box* stop, Box* step) {
return rtn; return rtn;
} }
extern "C" BoxedClosure* createClosure(BoxedClosure* parent_closure) { extern "C" BoxedClosure* createClosure(BoxedClosure* parent_closure, size_t n) {
if (parent_closure) if (parent_closure)
assert(parent_closure->cls == closure_cls); assert(parent_closure->cls == closure_cls);
return new BoxedClosure(parent_closure); BoxedClosure* closure = new (n) BoxedClosure(parent_closure);
assert(closure->cls == closure_cls);
return closure;
} }
extern "C" Box* sliceNew(Box* cls, Box* start, Box* stop, Box** args) { extern "C" Box* sliceNew(Box* cls, Box* start, Box* stop, Box** args) {
...@@ -2021,8 +2028,8 @@ void setupRuntime() { ...@@ -2021,8 +2028,8 @@ void setupRuntime() {
sizeof(BoxedSet), false, "frozenset"); sizeof(BoxedSet), false, "frozenset");
capi_getset_cls capi_getset_cls
= BoxedHeapClass::create(type_cls, object_cls, NULL, 0, 0, sizeof(BoxedGetsetDescriptor), false, "getset"); = BoxedHeapClass::create(type_cls, object_cls, NULL, 0, 0, sizeof(BoxedGetsetDescriptor), false, "getset");
closure_cls = BoxedHeapClass::create(type_cls, object_cls, &closureGCHandler, offsetof(BoxedClosure, attrs), 0, closure_cls
sizeof(BoxedClosure), false, "closure"); = BoxedHeapClass::create(type_cls, object_cls, &closureGCHandler, 0, 0, sizeof(BoxedClosure), false, "closure");
property_cls = BoxedHeapClass::create(type_cls, object_cls, &propertyGCHandler, 0, 0, sizeof(BoxedProperty), false, property_cls = BoxedHeapClass::create(type_cls, object_cls, &propertyGCHandler, 0, 0, sizeof(BoxedProperty), false,
"property"); "property");
staticmethod_cls = BoxedHeapClass::create(type_cls, object_cls, &staticmethodGCHandler, 0, 0, staticmethod_cls = BoxedHeapClass::create(type_cls, object_cls, &staticmethodGCHandler, 0, 0,
......
...@@ -623,15 +623,27 @@ public: ...@@ -623,15 +623,27 @@ public:
DEFAULT_CLASS_SIMPLE(classmethod_cls); DEFAULT_CLASS_SIMPLE(classmethod_cls);
}; };
// TODO is there any particular reason to make this a Box, ie a python-level object? // TODO is there any particular reason to make this a Box, i.e. a python-level object?
class BoxedClosure : public Box { class BoxedClosure : public Box {
public: public:
HCAttrs attrs;
BoxedClosure* parent; BoxedClosure* parent;
size_t nelts;
Box* elts[0];
BoxedClosure(BoxedClosure* parent) : parent(parent) {} BoxedClosure(BoxedClosure* parent) : parent(parent) {}
DEFAULT_CLASS(closure_cls); void* operator new(size_t size, size_t nelts) __attribute__((visibility("default"))) {
/*
BoxedClosure* rtn
= static_cast<BoxedClosure*>(gc_alloc(_PyObject_VAR_SIZE(closure_cls, nelts), gc::GCKind::PYTHON));
*/
BoxedClosure* rtn
= static_cast<BoxedClosure*>(gc_alloc(sizeof(BoxedClosure) + nelts * sizeof(Box*), gc::GCKind::PYTHON));
rtn->nelts = nelts;
rtn->cls = closure_cls;
memset((void*)rtn->elts, 0, sizeof(Box*) * nelts);
return rtn;
}
}; };
class BoxedGenerator : public Box { class BoxedGenerator : public Box {
......
# The use of c makes sure a closure gets passed through all 4 functions.
# The use of a in g makes sure that a is in f's closure.
# The a in j should refer to the a in h, thus throwing an exception since
# it is undefined (that is, it should *not* access the a from f even
# though it access via the closure).
try:
def f():
c = 0
a = 0
def g():
print c
print a
def h():
print c
def j():
print c
print a
j()
a = 1
h()
g()
f()
except NameError as ne:
print ne.message
...@@ -162,6 +162,8 @@ def run_test(fn, check_stats, run_memcheck): ...@@ -162,6 +162,8 @@ def run_test(fn, check_stats, run_memcheck):
skip = eval(skip_if) skip = eval(skip_if)
if skip: if skip:
return r + " (skipped due to 'skip-if: %s')" % skip_if[:30] return r + " (skipped due to 'skip-if: %s')" % skip_if[:30]
elif fn.split('.')[0] in TESTS_TO_SKIP:
return r + " (skipped due to command line option)"
elif l.startswith("# allow-warning:"): elif l.startswith("# allow-warning:"):
allow_warnings.append("Warning: " + l.split(':', 1)[1].strip()) allow_warnings.append("Warning: " + l.split(':', 1)[1].strip())
elif l.startswith("# no-collect-stats"): elif l.startswith("# no-collect-stats"):
......
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