Commit c524f07e authored by Kevin Modzelewski's avatar Kevin Modzelewski

Add a freelist for attribute arrays

Now that they are only a small number of sizes (powers of two),
can use a freelist pretty easily.  Especially because most of the
allocations will come from places where we can precompute the freelist
index.

About a 5% improvement on raytrace.py
parent 843ad031
...@@ -1119,8 +1119,21 @@ template Box* Box::getattr<NOT_REWRITABLE>(BoxedString*, GetattrRewriteArgs*); ...@@ -1119,8 +1119,21 @@ template Box* Box::getattr<NOT_REWRITABLE>(BoxedString*, GetattrRewriteArgs*);
// overhead, so the resulting size might not end up fitting that efficiently. // overhead, so the resulting size might not end up fitting that efficiently.
#define INITIAL_ARRAY_SIZE 4 #define INITIAL_ARRAY_SIZE 4
// Freelist for attribute arrays. Parameters have not been tuned.
#define ARRAYLIST_FREELIST_SIZE 100
#define ARRAYLIST_NUM_FREELISTS 4
#define MAX_FREELIST_SIZE (INITIAL_ARRAY_SIZE * (1 << (ARRAYLIST_NUM_FREELISTS - 1)))
HCAttrs::AttrList* attrlist_freelist[ARRAYLIST_NUM_FREELISTS][ARRAYLIST_FREELIST_SIZE];
int attrlist_freelist_size[ARRAYLIST_NUM_FREELISTS];
int freelist_index[]
= { 0, 0, 0, 0, 0, 1, 1, 1, 1, 2, 2, 2, 2, 2, 2, 2, 2, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3 };
static_assert(sizeof(freelist_index) / sizeof(freelist_index[0]) == MAX_FREELIST_SIZE + 1, "");
static bool isPowerOfTwo(int n) {
return __builtin_popcountll(n) == 1;
}
static bool arrayIsAtCapacity(int n) { static bool arrayIsAtCapacity(int n) {
return n >= INITIAL_ARRAY_SIZE && __builtin_popcountll(n) == 1; return n >= INITIAL_ARRAY_SIZE && isPowerOfTwo(n);
} }
static int nextAttributeArraySize(int n) { static int nextAttributeArraySize(int n) {
...@@ -1128,6 +1141,106 @@ static int nextAttributeArraySize(int n) { ...@@ -1128,6 +1141,106 @@ static int nextAttributeArraySize(int n) {
return n * 2; return n * 2;
} }
static int freelistIndex(int n) {
assert(n <= sizeof(freelist_index) / sizeof(freelist_index[0]));
return freelist_index[n];
}
static HCAttrs::AttrList* allocFromFreelist(int freelist_idx) {
int size = attrlist_freelist_size[freelist_idx];
if (size) {
auto rtn = attrlist_freelist[freelist_idx][size - 1];
attrlist_freelist_size[freelist_idx] = size - 1;
#ifndef NDEBUG
int nattrs = (1 << freelist_idx) * INITIAL_ARRAY_SIZE;
memset(rtn, 0xcb, sizeof(HCAttrs::AttrList) + nattrs * sizeof(Box*));
attrlist_freelist[freelist_idx][size - 1] = NULL;
#endif
return rtn;
}
int nattrs = (1 << freelist_idx) * INITIAL_ARRAY_SIZE;
return (HCAttrs::AttrList*)PyObject_MALLOC(sizeof(HCAttrs::AttrList) + nattrs * sizeof(Box*));
}
static HCAttrs::AttrList* allocAttrs(int nattrs) {
assert(arrayIsAtCapacity(nattrs));
if (nattrs <= MAX_FREELIST_SIZE)
return allocFromFreelist(freelistIndex(nattrs));
return (HCAttrs::AttrList*)PyObject_MALLOC(sizeof(HCAttrs::AttrList) + nattrs * sizeof(Box*));
}
static void freeAttrs(HCAttrs::AttrList* attrs, int nattrs) {
if (nattrs <= MAX_FREELIST_SIZE) {
int idx = freelistIndex(nattrs);
int size = attrlist_freelist_size[idx];
// TODO: should drop an old item from the freelist, not a new one
if (size == ARRAYLIST_FREELIST_SIZE) {
PyObject_FREE(attrs);
} else {
#ifndef NDEBUG
memset(attrs, 0xdb, sizeof(HCAttrs::AttrList) + nattrs * sizeof(Box*));
#endif
attrlist_freelist[idx][size] = attrs;
attrlist_freelist_size[idx]++;
return;
}
}
PyObject_FREE(attrs);
}
static HCAttrs::AttrList* reallocAttrs(HCAttrs::AttrList* attrs, int old_nattrs, int new_nattrs) {
assert(arrayIsAtCapacity(old_nattrs));
assert(new_nattrs > old_nattrs);
HCAttrs::AttrList* rtn = allocAttrs(new_nattrs);
memcpy(rtn, attrs, sizeof(HCAttrs::AttrList) + sizeof(Box*) * old_nattrs);
#ifndef NDEBUG
memset(&rtn->attrs[old_nattrs], 0xcb, sizeof(Box*) * (new_nattrs - old_nattrs));
#endif
freeAttrs(attrs, old_nattrs);
return rtn;
}
void HCAttrs::clear() noexcept {
HiddenClass* hcls = this->hcls;
if (!hcls)
return;
if (unlikely(hcls->type == HiddenClass::DICT_BACKED)) {
Box* d = this->attr_list->attrs[0];
Py_DECREF(d);
// Skips the attrlist freelist
PyObject_FREE(this->attr_list);
this->attr_list = NULL;
return;
}
assert(hcls->type == HiddenClass::NORMAL || hcls->type == HiddenClass::SINGLETON);
auto old_attr_list = this->attr_list;
auto old_attr_list_size = hcls->attributeArraySize();
new ((void*)this) HCAttrs(NULL);
if (old_attr_list) {
for (int i = 0; i < old_attr_list_size; i++) {
Py_DECREF(old_attr_list->attrs[i]);
}
freeAttrs(old_attr_list, old_attr_list_size);
}
}
void Box::appendNewHCAttr(BORROWED(Box*) new_attr, SetattrRewriteArgs* rewrite_args) { void Box::appendNewHCAttr(BORROWED(Box*) new_attr, SetattrRewriteArgs* rewrite_args) {
assert(cls->instancesHaveHCAttrs()); assert(cls->instancesHaveHCAttrs());
HCAttrs* attrs = getHCAttrsPtr(); HCAttrs* attrs = getHCAttrsPtr();
...@@ -1142,15 +1255,14 @@ void Box::appendNewHCAttr(BORROWED(Box*) new_attr, SetattrRewriteArgs* rewrite_a ...@@ -1142,15 +1255,14 @@ void Box::appendNewHCAttr(BORROWED(Box*) new_attr, SetattrRewriteArgs* rewrite_a
RewriterVar* r_array = NULL; RewriterVar* r_array = NULL;
if (numattrs == 0 || arrayIsAtCapacity(numattrs)) { if (numattrs == 0 || arrayIsAtCapacity(numattrs)) {
if (numattrs == 0) { if (numattrs == 0) {
int new_size = sizeof(HCAttrs::AttrList) + sizeof(Box*) * INITIAL_ARRAY_SIZE; attrs->attr_list = allocFromFreelist(0);
attrs->attr_list = (HCAttrs::AttrList*)PyMem_MALLOC(new_size);
if (rewrite_args) { if (rewrite_args) {
RewriterVar* r_newsize = rewrite_args->rewriter->loadConst(new_size, Location::forArg(0)); RewriterVar* r_newsize = rewrite_args->rewriter->loadConst(0, Location::forArg(0));
r_array = rewrite_args->rewriter->call(true, (void*)PyMem_Malloc, r_newsize); r_array = rewrite_args->rewriter->call(true, (void*)allocFromFreelist, r_newsize);
} }
} else { } else {
int new_size = sizeof(HCAttrs::AttrList) + sizeof(Box*) * nextAttributeArraySize(numattrs); int new_size = nextAttributeArraySize(numattrs);
attrs->attr_list = (HCAttrs::AttrList*)PyMem_REALLOC(attrs->attr_list, new_size); attrs->attr_list = (HCAttrs::AttrList*)reallocAttrs(attrs->attr_list, numattrs, new_size);
if (rewrite_args) { if (rewrite_args) {
if (cls->attrs_offset < 0) { if (cls->attrs_offset < 0) {
REWRITE_ABORTED(""); REWRITE_ABORTED("");
...@@ -1158,8 +1270,9 @@ void Box::appendNewHCAttr(BORROWED(Box*) new_attr, SetattrRewriteArgs* rewrite_a ...@@ -1158,8 +1270,9 @@ void Box::appendNewHCAttr(BORROWED(Box*) new_attr, SetattrRewriteArgs* rewrite_a
} else { } else {
RewriterVar* r_oldarray RewriterVar* r_oldarray
= rewrite_args->obj->getAttr(cls->attrs_offset + offsetof(HCAttrs, attr_list), Location::forArg(0)); = rewrite_args->obj->getAttr(cls->attrs_offset + offsetof(HCAttrs, attr_list), Location::forArg(0));
RewriterVar* r_newsize = rewrite_args->rewriter->loadConst(new_size, Location::forArg(1)); RewriterVar* r_oldsize = rewrite_args->rewriter->loadConst(numattrs, Location::forArg(1));
r_array = rewrite_args->rewriter->call(true, (void*)PyMem_Realloc, r_oldarray, r_newsize); RewriterVar* r_newsize = rewrite_args->rewriter->loadConst(new_size, Location::forArg(2));
r_array = rewrite_args->rewriter->call(true, (void*)reallocAttrs, r_oldarray, r_oldsize, r_newsize);
} }
} }
} }
...@@ -6103,7 +6216,7 @@ void Box::delattr(BoxedString* attr, DelattrRewriteArgs* rewrite_args) { ...@@ -6103,7 +6216,7 @@ void Box::delattr(BoxedString* attr, DelattrRewriteArgs* rewrite_args) {
// guarantee the size of the attr_list equals the number of attrs // guarantee the size of the attr_list equals the number of attrs
int new_size = sizeof(HCAttrs::AttrList) + sizeof(Box*) * (num_attrs - 1); int new_size = sizeof(HCAttrs::AttrList) + sizeof(Box*) * (num_attrs - 1);
// TODO: we might want to free some of this memory eventually // TODO: we might want to free some of this memory eventually
// attrs->attr_list = (HCAttrs::AttrList*)PyMem_REALLOC(attrs->attr_list, new_size); // attrs->attr_list = (HCAttrs::AttrList*)reallocAttrs(attrs->attr_list, num_attrs, new_size);
Py_DECREF(removed_object); Py_DECREF(removed_object);
return; return;
......
...@@ -2137,7 +2137,8 @@ private: ...@@ -2137,7 +2137,8 @@ private:
BoxedDict* d = (BoxedDict*)AttrWrapper::copy(this); BoxedDict* d = (BoxedDict*)AttrWrapper::copy(this);
b->clearAttrs(); b->clearAttrs();
HCAttrs* hcattrs = b->getHCAttrsPtr(); HCAttrs* hcattrs = b->getHCAttrsPtr();
auto new_attr_list = (HCAttrs::AttrList*)PyMem_MALLOC(sizeof(HCAttrs::AttrList) + sizeof(Box*)); // Skips the attrlist freelist:
auto new_attr_list = (HCAttrs::AttrList*)PyObject_MALLOC(sizeof(HCAttrs::AttrList) + sizeof(Box*));
new_attr_list->attrs[0] = d; new_attr_list->attrs[0] = d;
hcattrs->hcls = HiddenClass::dict_backed; hcattrs->hcls = HiddenClass::dict_backed;
...@@ -2802,7 +2803,8 @@ void Box::setDictBacked(STOLEN(Box*) val) { ...@@ -2802,7 +2803,8 @@ void Box::setDictBacked(STOLEN(Box*) val) {
} }
// assign the dict to the attribute list and switch to the dict backed strategy // assign the dict to the attribute list and switch to the dict backed strategy
auto new_attr_list = (HCAttrs::AttrList*)PyMem_MALLOC(sizeof(HCAttrs::AttrList) + sizeof(Box*)); // Skips the attrlist freelist
auto new_attr_list = (HCAttrs::AttrList*)PyObject_MALLOC(sizeof(HCAttrs::AttrList) + sizeof(Box*));
new_attr_list->attrs[0] = val; new_attr_list->attrs[0] = val;
hcattrs->hcls = HiddenClass::dict_backed; hcattrs->hcls = HiddenClass::dict_backed;
...@@ -3712,34 +3714,6 @@ void HiddenClass::dump() noexcept { ...@@ -3712,34 +3714,6 @@ void HiddenClass::dump() noexcept {
} }
} }
void HCAttrs::clear() noexcept {
HiddenClass* hcls = this->hcls;
if (!hcls)
return;
if (unlikely(hcls->type == HiddenClass::DICT_BACKED)) {
Box* d = this->attr_list->attrs[0];
Py_DECREF(d);
return;
}
assert(hcls->type == HiddenClass::NORMAL || hcls->type == HiddenClass::SINGLETON);
auto old_attr_list = this->attr_list;
auto old_attr_list_size = hcls->attributeArraySize();
new ((void*)this) HCAttrs(NULL);
if (old_attr_list) {
for (int i = 0; i < old_attr_list_size; i++) {
Py_DECREF(old_attr_list->attrs[i]);
}
PyMem_FREE(old_attr_list);
}
}
static void tupledealloc(PyTupleObject* op) noexcept { static void tupledealloc(PyTupleObject* op) noexcept {
Py_ssize_t i; Py_ssize_t i;
Py_ssize_t len = Py_SIZE(op); Py_ssize_t len = Py_SIZE(op);
......
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