Commit 1b3bcf5e authored by Kevin Modzelewski's avatar Kevin Modzelewski

Handle statically-allocated PyTypeObjects by having the collector keep a list of them.

Even with some optimizations it seems to hurt perf a little bit...
we'll have to see if it's worth it.

Also, rename "static roots" to "permanent roots", since it's
slightly different than the "static allocation" problem we're
tackling here.
parent 7be0644b
...@@ -37,31 +37,46 @@ namespace pyston { ...@@ -37,31 +37,46 @@ namespace pyston {
namespace gc { namespace gc {
static TraceStack roots; static TraceStack roots;
void registerStaticRootObj(void* obj) { void registerPermanentRoot(void* obj) {
assert(global_heap.getAllocationFromInteriorPointer(obj)); assert(global_heap.getAllocationFromInteriorPointer(obj));
roots.push(obj); roots.push(obj);
#ifndef NDEBUG
// Check for double-registers. Wouldn't cause any problems, but we probably shouldn't be doing them.
static std::unordered_set<void*> roots;
ASSERT(roots.count(obj) == 0, "Please only register roots once");
roots.insert(obj);
#endif
} }
std::vector<std::pair<void*, void*> > static_root_memory; static std::unordered_set<void*> nonheap_roots;
void registerStaticRootMemory(void* start, void* end) { // Track the highest-addressed nonheap root; the assumption is that the nonheap roots will
assert(start < end); // typically all have lower addresses than the heap roots, so this can serve as a cheap
// way to verify it's not a nonheap root (the full check requires a hashtable lookup).
static void* max_nonheap_root = 0;
void registerNonheapRootObject(void* obj) {
// I suppose that things could work fine even if this were true, but why would it happen?
assert(global_heap.getAllocationFromInteriorPointer(obj) == NULL);
assert(nonheap_roots.count(obj) == 0);
// While these aren't necessary to work correctly, they are the anticipated use case: nonheap_roots.insert(obj);
assert(global_heap.getAllocationFromInteriorPointer(start) == NULL);
assert(global_heap.getAllocationFromInteriorPointer(end) == NULL); max_nonheap_root = std::max(obj, max_nonheap_root);
}
static_root_memory.push_back(std::make_pair(start, end)); static bool isNonheapRoot(void* p) {
return p <= max_nonheap_root && nonheap_roots.count(p) != 0;
} }
static std::unordered_set<StaticRootHandle*>* getRootHandles() { static std::unordered_set<GCRootHandle*>* getRootHandles() {
static std::unordered_set<StaticRootHandle*> root_handles; static std::unordered_set<GCRootHandle*> root_handles;
return &root_handles; return &root_handles;
} }
StaticRootHandle::StaticRootHandle() { GCRootHandle::GCRootHandle() {
getRootHandles()->insert(this); getRootHandles()->insert(this);
} }
StaticRootHandle::~StaticRootHandle() { GCRootHandle::~GCRootHandle() {
getRootHandles()->erase(this); getRootHandles()->erase(this);
} }
...@@ -72,19 +87,19 @@ bool TraceStackGCVisitor::isValid(void* p) { ...@@ -72,19 +87,19 @@ bool TraceStackGCVisitor::isValid(void* p) {
} }
void TraceStackGCVisitor::visit(void* p) { void TraceStackGCVisitor::visit(void* p) {
if (isNonheapRoot(p)) {
return;
} else {
assert(global_heap.getAllocationFromInteriorPointer(p)->user_data == p); assert(global_heap.getAllocationFromInteriorPointer(p)->user_data == p);
stack->push(p); stack->push(p);
}
} }
void TraceStackGCVisitor::visitRange(void* const* start, void* const* end) { void TraceStackGCVisitor::visitRange(void* const* start, void* const* end) {
#ifndef NDEBUG while (start < end) {
void* const* cur = start; visit(*start);
while (cur < end) { start++;
assert(isValid(*cur));
cur++;
} }
#endif
stack->pushall(start, end);
} }
void TraceStackGCVisitor::visitPotential(void* p) { void TraceStackGCVisitor::visitPotential(void* p) {
...@@ -113,8 +128,14 @@ static void markPhase() { ...@@ -113,8 +128,14 @@ static void markPhase() {
TraceStackGCVisitor visitor(&stack); TraceStackGCVisitor visitor(&stack);
for (const auto& p : static_root_memory) { for (void* p : nonheap_roots) {
visitor.visitPotentialRange((void**)p.first, (void**)p.second); Box* b = reinterpret_cast<Box*>(p);
BoxedClass* cls = b->cls;
if (cls) {
ASSERT(cls->gc_visit, "%s", getTypeName(b)->c_str());
cls->gc_visit(&visitor, b);
}
} }
for (auto h : *getRootHandles()) { for (auto h : *getRootHandles()) {
......
...@@ -27,7 +27,7 @@ private: ...@@ -27,7 +27,7 @@ private:
std::vector<void*> v; std::vector<void*> v;
public: public:
void pushall(void* const* start, void* const* end) { v.insert(v.end(), start, end); } template <typename T> void pushall(T start, T end) { v.insert(v.end(), start, end); }
void push(void* p) { v.push_back(p); } void push(void* p) { v.push_back(p); }
...@@ -62,25 +62,27 @@ public: ...@@ -62,25 +62,27 @@ public:
// Mark this gc-allocated object as being a root, even if there are no visible references to it. // Mark this gc-allocated object as being a root, even if there are no visible references to it.
// (Note: this marks the gc allocation itself, not the pointer that points to one. For that, use // (Note: this marks the gc allocation itself, not the pointer that points to one. For that, use
// a StaticRootHandle or registerStaticRootMemory) // a GCRootHandle)
void registerStaticRootObj(void* root_obj); void registerPermanentRoot(void* root_obj);
// Register a non-gc region of memory (such as statically-allocated memory) as a source of potential // Register an object that was not allocated through this collector, as a root for this collector.
// GC roots. // The motivating usecase is statically-allocated PyTypeObject objects, which are full Python objects
void registerStaticRootMemory(void* start, void* end); // even if they are not heap allocated.
void runCollection(); void registerNonheapRootObject(void* obj);
// If you want to have a static root "location" where multiple values could be stored, use this: // If you want to have a static root "location" where multiple values could be stored, use this:
class StaticRootHandle { class GCRootHandle {
public: public:
Box* value; Box* value;
StaticRootHandle(); GCRootHandle();
~StaticRootHandle(); ~GCRootHandle();
void operator=(Box* b) { value = b; } void operator=(Box* b) { value = b; }
operator Box*() { return value; } operator Box*() { return value; }
}; };
void runCollection();
} }
} }
......
...@@ -71,8 +71,8 @@ void setupBool() { ...@@ -71,8 +71,8 @@ void setupBool() {
True = new BoxedBool(true); True = new BoxedBool(true);
False = new BoxedBool(false); False = new BoxedBool(false);
gc::registerStaticRootObj(True); gc::registerPermanentRoot(True);
gc::registerStaticRootObj(False); gc::registerPermanentRoot(False);
} }
void teardownBool() { void teardownBool() {
......
...@@ -528,7 +528,7 @@ void setupBuiltins() { ...@@ -528,7 +528,7 @@ void setupBuiltins() {
notimplemented_cls->giveAttr("__repr__", new BoxedFunction(boxRTFunction((void*)notimplementedRepr, STR, 1))); notimplemented_cls->giveAttr("__repr__", new BoxedFunction(boxRTFunction((void*)notimplementedRepr, STR, 1)));
notimplemented_cls->freeze(); notimplemented_cls->freeze();
NotImplemented = new Box(notimplemented_cls); NotImplemented = new Box(notimplemented_cls);
gc::registerStaticRootObj(NotImplemented); gc::registerPermanentRoot(NotImplemented);
builtins_module->giveAttr("NotImplemented", NotImplemented); builtins_module->giveAttr("NotImplemented", NotImplemented);
builtins_module->giveAttr("NotImplementedType", notimplemented_cls); builtins_module->giveAttr("NotImplementedType", notimplemented_cls);
......
...@@ -101,7 +101,7 @@ public: ...@@ -101,7 +101,7 @@ public:
void setupSys() { void setupSys() {
sys_modules_dict = new BoxedDict(); sys_modules_dict = new BoxedDict();
gc::registerStaticRootObj(sys_modules_dict); gc::registerPermanentRoot(sys_modules_dict);
// This is ok to call here because we've already created the sys_modules_dict // This is ok to call here because we've already created the sys_modules_dict
sys_module = createModule("sys", "__builtin__"); sys_module = createModule("sys", "__builtin__");
......
...@@ -201,7 +201,7 @@ extern "C" void conservativeGCHandler(GCVisitor* v, Box* b) { ...@@ -201,7 +201,7 @@ extern "C" void conservativeGCHandler(GCVisitor* v, Box* b) {
} }
extern "C" int PyType_Ready(PyTypeObject* cls) { extern "C" int PyType_Ready(PyTypeObject* cls) {
gc::registerStaticRootMemory(cls, cls + 1); gc::registerNonheapRootObject(cls);
// unhandled fields: // unhandled fields:
RELEASE_ASSERT(cls->tp_print == NULL, ""); RELEASE_ASSERT(cls->tp_print == NULL, "");
......
...@@ -263,7 +263,6 @@ void setupDict() { ...@@ -263,7 +263,6 @@ void setupDict() {
dict_cls->freeze(); dict_cls->freeze();
gc::registerStaticRootObj(dict_iterator_cls);
dict_iterator_cls->giveAttr("__name__", boxStrConstant("dictiterator")); dict_iterator_cls->giveAttr("__name__", boxStrConstant("dictiterator"));
CLFunction* hasnext = boxRTFunction((void*)dictIterHasnextUnboxed, BOOL, 1); CLFunction* hasnext = boxRTFunction((void*)dictIterHasnextUnboxed, BOOL, 1);
......
...@@ -190,7 +190,6 @@ void setupGenerator() { ...@@ -190,7 +190,6 @@ void setupGenerator() {
generator_cls->giveAttr("send", new BoxedFunction(boxRTFunction((void*)generatorSend, UNKNOWN, 2))); generator_cls->giveAttr("send", new BoxedFunction(boxRTFunction((void*)generatorSend, UNKNOWN, 2)));
generator_cls->giveAttr("throw", new BoxedFunction(boxRTFunction((void*)generatorThrow, UNKNOWN, 2))); generator_cls->giveAttr("throw", new BoxedFunction(boxRTFunction((void*)generatorThrow, UNKNOWN, 2)));
gc::registerStaticRootObj(generator_cls);
generator_cls->freeze(); generator_cls->freeze();
} }
} }
...@@ -659,7 +659,7 @@ static void _addFuncIntUnknown(const char* name, ConcreteCompilerType* rtn_type, ...@@ -659,7 +659,7 @@ static void _addFuncIntUnknown(const char* name, ConcreteCompilerType* rtn_type,
void setupInt() { void setupInt() {
for (int i = 0; i < NUM_INTERNED_INTS; i++) { for (int i = 0; i < NUM_INTERNED_INTS; i++) {
interned_ints[i] = new BoxedInt(int_cls, i); interned_ints[i] = new BoxedInt(int_cls, i);
gc::registerStaticRootObj(interned_ints[i]); gc::registerPermanentRoot(interned_ints[i]);
} }
int_cls->giveAttr("__name__", boxStrConstant("int")); int_cls->giveAttr("__name__", boxStrConstant("int"));
......
...@@ -575,7 +575,6 @@ void setupList() { ...@@ -575,7 +575,6 @@ void setupList() {
list_cls->freeze(); list_cls->freeze();
gc::registerStaticRootObj(list_iterator_cls);
list_iterator_cls->giveAttr("__name__", boxStrConstant("listiterator")); list_iterator_cls->giveAttr("__name__", boxStrConstant("listiterator"));
CLFunction* hasnext = boxRTFunction((void*)listiterHasnextUnboxed, BOOL, 1); CLFunction* hasnext = boxRTFunction((void*)listiterHasnextUnboxed, BOOL, 1);
......
...@@ -413,7 +413,7 @@ BoxedClass::BoxedClass(BoxedClass* base, gcvisit_func gc_visit, int attrs_offset ...@@ -413,7 +413,7 @@ BoxedClass::BoxedClass(BoxedClass* base, gcvisit_func gc_visit, int attrs_offset
} }
if (!is_user_defined) if (!is_user_defined)
gc::registerStaticRootObj(this); gc::registerPermanentRoot(this);
} }
extern "C" const std::string* getNameOfClass(BoxedClass* cls) { extern "C" const std::string* getNameOfClass(BoxedClass* cls) {
...@@ -696,7 +696,7 @@ static Box* _handleClsAttr(Box* obj, Box* attr) { ...@@ -696,7 +696,7 @@ static Box* _handleClsAttr(Box* obj, Box* attr) {
return boxBool(rtn); return boxBool(rtn);
} }
case BoxedMemberDescriptor::INT: { case BoxedMemberDescriptor::INT: {
int rtn = reinterpret_cast<int*>(obj)[member_desc->offset/sizeof(int)]; int rtn = reinterpret_cast<int*>(obj)[member_desc->offset / sizeof(int)];
return boxInt(rtn); return boxInt(rtn);
} }
default: default:
......
...@@ -213,7 +213,6 @@ void setupSet() { ...@@ -213,7 +213,6 @@ void setupSet() {
new BoxedFunction(boxRTFunction((void*)setiteratorHasnext, BOXED_BOOL, 1))); new BoxedFunction(boxRTFunction((void*)setiteratorHasnext, BOXED_BOOL, 1)));
set_iterator_cls->giveAttr("next", new BoxedFunction(boxRTFunction((void*)setiteratorNext, UNKNOWN, 1))); set_iterator_cls->giveAttr("next", new BoxedFunction(boxRTFunction((void*)setiteratorNext, UNKNOWN, 1)));
set_iterator_cls->freeze(); set_iterator_cls->freeze();
gc::registerStaticRootObj(set_iterator_cls);
set_cls->giveAttr("__new__", set_cls->giveAttr("__new__",
new BoxedFunction(boxRTFunction((void*)setNew, UNKNOWN, 2, 1, false, false), { None })); new BoxedFunction(boxRTFunction((void*)setNew, UNKNOWN, 2, 1, false, false), { None }));
......
...@@ -102,7 +102,7 @@ void unwindExc(Box* exc_obj) { ...@@ -102,7 +102,7 @@ void unwindExc(Box* exc_obj) {
} }
static std::vector<const LineInfo*> getTracebackEntries(); static std::vector<const LineInfo*> getTracebackEntries();
static gc::StaticRootHandle last_exc; static gc::GCRootHandle last_exc;
static std::vector<const LineInfo*> last_tb; static std::vector<const LineInfo*> last_tb;
void raiseRaw(Box* exc_obj) __attribute__((__noreturn__)); void raiseRaw(Box* exc_obj) __attribute__((__noreturn__));
......
...@@ -784,7 +784,6 @@ Box* strCount2(BoxedString* self, Box* elt) { ...@@ -784,7 +784,6 @@ Box* strCount2(BoxedString* self, Box* elt) {
void setupStr() { void setupStr() {
str_iterator_cls = new BoxedClass(object_cls, &strIteratorGCHandler, 0, sizeof(BoxedString), false); str_iterator_cls = new BoxedClass(object_cls, &strIteratorGCHandler, 0, sizeof(BoxedString), false);
gc::registerStaticRootObj(str_iterator_cls);
str_iterator_cls->giveAttr("__name__", boxStrConstant("striterator")); str_iterator_cls->giveAttr("__name__", boxStrConstant("striterator"));
str_iterator_cls->giveAttr("__hasnext__", str_iterator_cls->giveAttr("__hasnext__",
new BoxedFunction(boxRTFunction((void*)BoxedStringIterator::hasnext, BOXED_BOOL, 1))); new BoxedFunction(boxRTFunction((void*)BoxedStringIterator::hasnext, BOXED_BOOL, 1)));
......
...@@ -329,7 +329,6 @@ void setupTuple() { ...@@ -329,7 +329,6 @@ void setupTuple() {
tuple_cls->freeze(); tuple_cls->freeze();
gc::registerStaticRootObj(tuple_iterator_cls);
tuple_iterator_cls->giveAttr("__name__", boxStrConstant("tupleiterator")); tuple_iterator_cls->giveAttr("__name__", boxStrConstant("tupleiterator"));
CLFunction* hasnext = boxRTFunction((void*)tupleiterHasnextUnboxed, BOOL, 1); CLFunction* hasnext = boxRTFunction((void*)tupleiterHasnextUnboxed, BOOL, 1);
......
...@@ -183,7 +183,7 @@ extern "C" void boxGCHandler(GCVisitor* v, Box* b) { ...@@ -183,7 +183,7 @@ extern "C" void boxGCHandler(GCVisitor* v, Box* b) {
HCAttrs::AttrList* attr_list = attrs->attr_list; HCAttrs::AttrList* attr_list = attrs->attr_list;
assert(attr_list); assert(attr_list);
v->visit(attr_list); v->visit(attr_list);
v->visitPotentialRange((void**)&attr_list->attrs[0], (void**)&attr_list->attrs[nattrs]); v->visitRange((void**)&attr_list->attrs[0], (void**)&attr_list->attrs[nattrs]);
} }
} }
} else { } else {
...@@ -584,7 +584,7 @@ Box* objectStr(Box* obj) { ...@@ -584,7 +584,7 @@ Box* objectStr(Box* obj) {
bool TRACK_ALLOCATIONS = false; bool TRACK_ALLOCATIONS = false;
void setupRuntime() { void setupRuntime() {
root_hcls = HiddenClass::makeRoot(); root_hcls = HiddenClass::makeRoot();
gc::registerStaticRootObj(root_hcls); gc::registerPermanentRoot(root_hcls);
object_cls = new BoxedClass(NULL, &boxGCHandler, 0, sizeof(Box), false); object_cls = new BoxedClass(NULL, &boxGCHandler, 0, sizeof(Box), false);
type_cls = new BoxedClass(object_cls, &typeGCHandler, offsetof(BoxedClass, attrs), sizeof(BoxedClass), false); type_cls = new BoxedClass(object_cls, &typeGCHandler, offsetof(BoxedClass, attrs), sizeof(BoxedClass), false);
...@@ -593,7 +593,7 @@ void setupRuntime() { ...@@ -593,7 +593,7 @@ void setupRuntime() {
none_cls = new BoxedClass(object_cls, NULL, 0, sizeof(Box), false); none_cls = new BoxedClass(object_cls, NULL, 0, sizeof(Box), false);
None = new Box(none_cls); None = new Box(none_cls);
gc::registerStaticRootObj(None); gc::registerPermanentRoot(None);
// TODO we leak all the string data! // TODO we leak all the string data!
str_cls = new BoxedClass(object_cls, NULL, 0, sizeof(BoxedString), false); str_cls = new BoxedClass(object_cls, NULL, 0, sizeof(BoxedString), false);
...@@ -607,7 +607,7 @@ void setupRuntime() { ...@@ -607,7 +607,7 @@ void setupRuntime() {
tuple_cls = new BoxedClass(object_cls, &tupleGCHandler, 0, sizeof(BoxedTuple), false); tuple_cls = new BoxedClass(object_cls, &tupleGCHandler, 0, sizeof(BoxedTuple), false);
EmptyTuple = new BoxedTuple({}); EmptyTuple = new BoxedTuple({});
gc::registerStaticRootObj(EmptyTuple); gc::registerPermanentRoot(EmptyTuple);
module_cls = new BoxedClass(object_cls, NULL, offsetof(BoxedModule, attrs), sizeof(BoxedModule), false); module_cls = new BoxedClass(object_cls, NULL, offsetof(BoxedModule, attrs), sizeof(BoxedModule), false);
......
...@@ -331,7 +331,8 @@ public: ...@@ -331,7 +331,8 @@ public:
int offset; int offset;
BoxedMemberDescriptor(MemberType type, int offset) : Box(member_cls), type(type), offset(offset) {} BoxedMemberDescriptor(MemberType type, int offset) : Box(member_cls), type(type), offset(offset) {}
BoxedMemberDescriptor(PyMemberDef* member) : Box(member_cls), type((MemberType)member->type), offset(member->offset) {} BoxedMemberDescriptor(PyMemberDef* member)
: Box(member_cls), type((MemberType)member->type), offset(member->offset) {}
}; };
// 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, ie a python-level object?
......
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