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

Fix a gc bug with codegen strings

If we ask for a string as non-interned, and then ask for an interned
version of the string, we will discard the non-interned version and
generated a new interned version.  Now, keep the old version alive
as well.
parent c19b8cdb
...@@ -474,9 +474,19 @@ std::string BoxedModule::name() { ...@@ -474,9 +474,19 @@ std::string BoxedModule::name() {
BoxedString* BoxedModule::getStringConstant(llvm::StringRef ast_str, bool intern) { BoxedString* BoxedModule::getStringConstant(llvm::StringRef ast_str, bool intern) {
BoxedString*& r = str_constants[ast_str]; BoxedString*& r = str_constants[ast_str];
if (intern) if (intern) {
// If we had previously created a box for this string, we have to create a new
// string (or at least, be prepared to return a different value that we had already
// interned). This is fine, except we have to be careful because we promised
// that we would keep the previously-created string alive.
// So, make sure to put it onto the keep_alive list.
if (r && !PyString_CHECK_INTERNED(r)) {
keep_alive.push_back(r);
r = NULL;
}
if (!r)
r = internStringMortal(ast_str); r = internStringMortal(ast_str);
else if (!r) } else if (!r)
r = boxString(ast_str); r = boxString(ast_str);
return r; return r;
} }
...@@ -538,6 +548,8 @@ void BoxedModule::gcHandler(GCVisitor* v, Box* b) { ...@@ -538,6 +548,8 @@ void BoxedModule::gcHandler(GCVisitor* v, Box* b) {
visitContiguousMap(v, d->float_constants); visitContiguousMap(v, d->float_constants);
visitContiguousMap(v, d->imaginary_constants); visitContiguousMap(v, d->imaginary_constants);
visitContiguousMap(v, d->long_constants); visitContiguousMap(v, d->long_constants);
if (!d->keep_alive.empty())
v->visitRange((void**)&d->keep_alive[0], (void**)((&d->keep_alive[0]) + d->keep_alive.size()));
} }
// This mustn't throw; our IR generator generates calls to it without "invoke" even when there are exception handlers / // This mustn't throw; our IR generator generates calls to it without "invoke" even when there are exception handlers /
......
...@@ -879,6 +879,8 @@ private: ...@@ -879,6 +879,8 @@ private:
ContiguousMap<int64_t, BoxedFloat*, std::unordered_map<int64_t, int>> float_constants; ContiguousMap<int64_t, BoxedFloat*, std::unordered_map<int64_t, int>> float_constants;
ContiguousMap<int64_t, Box*, std::unordered_map<int64_t, int>> imaginary_constants; ContiguousMap<int64_t, Box*, std::unordered_map<int64_t, int>> imaginary_constants;
ContiguousMap<llvm::StringRef, Box*, llvm::StringMap<int>> long_constants; ContiguousMap<llvm::StringRef, Box*, llvm::StringMap<int>> long_constants;
// Other objects that this module needs to keep alive; see getStringConstant.
llvm::SmallVector<Box*, 8> keep_alive;
public: public:
DEFAULT_CLASS(module_cls); DEFAULT_CLASS(module_cls);
......
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