mirror of
https://github.com/pocketpy/pocketpy
synced 2025-11-06 19:50:16 +00:00
114 lines
3.3 KiB
C
114 lines
3.3 KiB
C
#include "pocketpy/interpreter/heap.h"
|
|
#include "pocketpy/common/memorypool.h"
|
|
#include "pocketpy/objects/base.h"
|
|
|
|
void ManagedHeap__ctor(ManagedHeap* self, VM* vm) {
|
|
c11_vector__ctor(&self->no_gc, sizeof(PyObject*));
|
|
c11_vector__ctor(&self->gen, sizeof(PyObject*));
|
|
|
|
self->gc_threshold = PK_GC_MIN_THRESHOLD;
|
|
self->gc_counter = 0;
|
|
self->vm = vm;
|
|
|
|
self->gc_on_delete = NULL;
|
|
}
|
|
|
|
void ManagedHeap__dtor(ManagedHeap* self) {
|
|
for(int i = 0; i < self->gen.length; i++) {
|
|
PyObject* obj = c11__getitem(PyObject*, &self->gen, i);
|
|
PyObject__delete(obj);
|
|
}
|
|
for(int i = 0; i < self->no_gc.length; i++) {
|
|
PyObject* obj = c11__getitem(PyObject*, &self->no_gc, i);
|
|
PyObject__delete(obj);
|
|
}
|
|
c11_vector__dtor(&self->no_gc);
|
|
c11_vector__dtor(&self->gen);
|
|
}
|
|
|
|
void ManagedHeap__collect_if_needed(ManagedHeap* self) {
|
|
if(self->gc_counter < self->gc_threshold) return;
|
|
self->gc_counter = 0;
|
|
ManagedHeap__collect(self);
|
|
self->gc_threshold = self->gen.length * 2;
|
|
if(self->gc_threshold < PK_GC_MIN_THRESHOLD) { self->gc_threshold = PK_GC_MIN_THRESHOLD; }
|
|
}
|
|
|
|
int ManagedHeap__collect(ManagedHeap* self) {
|
|
ManagedHeap__mark(self);
|
|
int freed = ManagedHeap__sweep(self);
|
|
return freed;
|
|
}
|
|
|
|
int ManagedHeap__sweep(ManagedHeap* self) {
|
|
c11_vector alive;
|
|
c11_vector__ctor(&alive, sizeof(PyObject*));
|
|
c11_vector__reserve(&alive, self->gen.length / 2);
|
|
|
|
for(int i = 0; i < self->gen.length; i++) {
|
|
PyObject* obj = c11__getitem(PyObject*, &self->gen, i);
|
|
if(obj->gc_marked) {
|
|
obj->gc_marked = false;
|
|
c11_vector__push(PyObject*, &alive, obj);
|
|
} else {
|
|
if(self->gc_on_delete) { self->gc_on_delete(self->vm, obj); }
|
|
PyObject__delete(obj);
|
|
}
|
|
}
|
|
|
|
// clear _no_gc marked flag
|
|
for(int i = 0; i < self->no_gc.length; i++) {
|
|
PyObject* obj = c11__getitem(PyObject*, &self->no_gc, i);
|
|
obj->gc_marked = false;
|
|
}
|
|
|
|
int freed = self->gen.length - alive.length;
|
|
|
|
// destroy old gen
|
|
c11_vector__dtor(&self->gen);
|
|
// move alive to gen
|
|
self->gen = alive;
|
|
|
|
PoolObject_shrink_to_fit();
|
|
return freed;
|
|
}
|
|
|
|
PyObject* ManagedHeap__new(ManagedHeap* self, py_Type type, int slots, int udsize) {
|
|
PyObject* obj = PyObject__new(type, slots, udsize);
|
|
c11_vector__push(PyObject*, &self->no_gc, obj);
|
|
return obj;
|
|
}
|
|
|
|
PyObject* ManagedHeap__gcnew(ManagedHeap* self, py_Type type, int slots, int udsize) {
|
|
PyObject* obj = PyObject__new(type, slots, udsize);
|
|
c11_vector__push(PyObject*, &self->gen, obj);
|
|
self->gc_counter++;
|
|
return obj;
|
|
}
|
|
|
|
PyObject* PyObject__new(py_Type type, int slots, int size) {
|
|
assert(slots >= 0 || slots == -1);
|
|
PyObject* self;
|
|
// header + slots + udsize
|
|
size = sizeof(PyObject) + PK_OBJ_SLOTS_SIZE(slots) + size;
|
|
if(size <= kPoolObjectBlockSize) {
|
|
self = PoolObject_alloc();
|
|
self->gc_is_large = false;
|
|
} else {
|
|
self = malloc(size);
|
|
self->gc_is_large = true;
|
|
}
|
|
self->type = type;
|
|
self->gc_marked = false;
|
|
self->slots = slots;
|
|
|
|
// initialize slots or dict
|
|
void* p = (char*)self + 8;
|
|
if(slots >= 0) {
|
|
memset(p, 0, slots * sizeof(py_TValue));
|
|
} else {
|
|
NameDict__ctor(p);
|
|
}
|
|
return self;
|
|
}
|