mirror of
https://github.com/corda/corda.git
synced 2025-01-22 12:28:11 +00:00
avoid calling doCollect recursively
We must not allocate heap objects from doCollect, since it might trigger a GC while one is already in progress, which can cause trouble when we're still queuing up objects to finalize, among other things. To avoid this, I've added extra fields to the finalizer and cleaner types which we can use to link instances up during GC without allocating new memory.
This commit is contained in:
parent
b9f8188544
commit
838cf9fdd1
@ -364,8 +364,18 @@ finalizerTargetUnreachable(Thread* t, Heap::Visitor* v, object* p)
|
|||||||
|
|
||||||
object finalizer = *p;
|
object finalizer = *p;
|
||||||
*p = finalizerNext(t, finalizer);
|
*p = finalizerNext(t, finalizer);
|
||||||
|
|
||||||
|
void (*function)(Thread*, object);
|
||||||
|
memcpy(&function, &finalizerFinalize(t, finalizer), BytesPerWord);
|
||||||
|
|
||||||
|
if (function) {
|
||||||
finalizerNext(t, finalizer) = t->m->finalizeQueue;
|
finalizerNext(t, finalizer) = t->m->finalizeQueue;
|
||||||
t->m->finalizeQueue = finalizer;
|
t->m->finalizeQueue = finalizer;
|
||||||
|
} else {
|
||||||
|
set(t, finalizer, FinalizerQueueTarget, finalizerTarget(t, finalizer));
|
||||||
|
set(t, finalizer, FinalizerQueueNext, root(t, Machine::ObjectsToFinalize));
|
||||||
|
setRoot(t, Machine::ObjectsToFinalize, finalizer);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
@ -382,8 +392,9 @@ referenceTargetUnreachable(Thread* t, Heap::Visitor* v, object* p)
|
|||||||
if (objectClass(t, *p) == type(t, Machine::CleanerType)) {
|
if (objectClass(t, *p) == type(t, Machine::CleanerType)) {
|
||||||
object reference = *p;
|
object reference = *p;
|
||||||
*p = jreferenceVmNext(t, reference);
|
*p = jreferenceVmNext(t, reference);
|
||||||
jreferenceVmNext(t, reference) = t->m->cleanerQueue;
|
|
||||||
t->m->cleanerQueue = reference;
|
set(t, reference, CleanerQueueNext, root(t, Machine::ObjectsToClean));
|
||||||
|
setRoot(t, Machine::ObjectsToClean, reference);
|
||||||
} else {
|
} else {
|
||||||
if (jreferenceQueue(t, *p)
|
if (jreferenceQueue(t, *p)
|
||||||
and t->m->heap->status(jreferenceQueue(t, *p)) != Heap::Unreachable)
|
and t->m->heap->status(jreferenceQueue(t, *p)) != Heap::Unreachable)
|
||||||
@ -453,7 +464,6 @@ postVisit(Thread* t, Heap::Visitor* v)
|
|||||||
bool major = m->heap->collectionType() == Heap::MajorCollection;
|
bool major = m->heap->collectionType() == Heap::MajorCollection;
|
||||||
|
|
||||||
assert(t, m->finalizeQueue == 0);
|
assert(t, m->finalizeQueue == 0);
|
||||||
assert(t, m->cleanerQueue == 0);
|
|
||||||
|
|
||||||
object firstNewTenuredFinalizer = 0;
|
object firstNewTenuredFinalizer = 0;
|
||||||
object lastNewTenuredFinalizer = 0;
|
object lastNewTenuredFinalizer = 0;
|
||||||
@ -2230,6 +2240,11 @@ class HeapClient: public Heap::Client {
|
|||||||
void
|
void
|
||||||
doCollect(Thread* t, Heap::CollectionType type)
|
doCollect(Thread* t, Heap::CollectionType type)
|
||||||
{
|
{
|
||||||
|
expect(t, not t->m->collecting);
|
||||||
|
|
||||||
|
t->m->collecting = true;
|
||||||
|
THREAD_RESOURCE0(t, t->m->collecting = false);
|
||||||
|
|
||||||
#ifdef VM_STRESS
|
#ifdef VM_STRESS
|
||||||
bool stress = (t->flags & Thread::StressFlag) != 0;
|
bool stress = (t->flags & Thread::StressFlag) != 0;
|
||||||
if (not stress) atomicOr(&(t->flags), Thread::StressFlag);
|
if (not stress) atomicOr(&(t->flags), Thread::StressFlag);
|
||||||
@ -2263,36 +2278,16 @@ doCollect(Thread* t, Heap::CollectionType type)
|
|||||||
#endif
|
#endif
|
||||||
|
|
||||||
object finalizeQueue = t->m->finalizeQueue;
|
object finalizeQueue = t->m->finalizeQueue;
|
||||||
PROTECT(t, finalizeQueue);
|
|
||||||
t->m->finalizeQueue = 0;
|
t->m->finalizeQueue = 0;
|
||||||
for (; finalizeQueue; finalizeQueue = finalizerNext(t, finalizeQueue)) {
|
for (; finalizeQueue; finalizeQueue = finalizerNext(t, finalizeQueue)) {
|
||||||
void (*function)(Thread*, object);
|
void (*function)(Thread*, object);
|
||||||
memcpy(&function, &finalizerFinalize(t, finalizeQueue), BytesPerWord);
|
memcpy(&function, &finalizerFinalize(t, finalizeQueue), BytesPerWord);
|
||||||
if (function) {
|
|
||||||
function(t, finalizerTarget(t, finalizeQueue));
|
function(t, finalizerTarget(t, finalizeQueue));
|
||||||
} else {
|
|
||||||
setRoot(t, Machine::ObjectsToFinalize, makeFinalizeNode
|
|
||||||
(t, finalizerTarget(t, finalizeQueue),
|
|
||||||
const_cast<char*>("finalize"),
|
|
||||||
root(t, Machine::ObjectsToFinalize)));
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
object cleanerQueue = t->m->cleanerQueue;
|
if ((root(t, Machine::ObjectsToFinalize) or root(t, Machine::ObjectsToClean))
|
||||||
PROTECT(t, cleanerQueue);
|
and m->finalizeThread == 0)
|
||||||
t->m->cleanerQueue = 0;
|
{
|
||||||
while (cleanerQueue) {
|
|
||||||
setRoot(t, Machine::ObjectsToFinalize, makeFinalizeNode
|
|
||||||
(t, cleanerQueue,
|
|
||||||
const_cast<char*>("clean"),
|
|
||||||
root(t, Machine::ObjectsToFinalize)));
|
|
||||||
|
|
||||||
object tmp = cleanerQueue;
|
|
||||||
cleanerQueue = jreferenceVmNext(t, cleanerQueue);
|
|
||||||
jreferenceVmNext(t, tmp) = 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (root(t, Machine::ObjectsToFinalize) and m->finalizeThread == 0) {
|
|
||||||
m->finalizeThread = m->processor->makeThread
|
m->finalizeThread = m->processor->makeThread
|
||||||
(m, root(t, Machine::FinalizerThread), m->rootThread);
|
(m, root(t, Machine::FinalizerThread), m->rootThread);
|
||||||
|
|
||||||
@ -2355,10 +2350,10 @@ Machine::Machine(System* system, Heap* heap, Finder* bootFinder,
|
|||||||
finalizers(0),
|
finalizers(0),
|
||||||
tenuredFinalizers(0),
|
tenuredFinalizers(0),
|
||||||
finalizeQueue(0),
|
finalizeQueue(0),
|
||||||
cleanerQueue(0),
|
|
||||||
weakReferences(0),
|
weakReferences(0),
|
||||||
tenuredWeakReferences(0),
|
tenuredWeakReferences(0),
|
||||||
unsafe(false),
|
unsafe(false),
|
||||||
|
collecting(false),
|
||||||
triedBuiltinOnLoad(false),
|
triedBuiltinOnLoad(false),
|
||||||
dumpedHeapOnOOM(false),
|
dumpedHeapOnOOM(false),
|
||||||
heapPoolIndex(0)
|
heapPoolIndex(0)
|
||||||
@ -3810,7 +3805,7 @@ addFinalizer(Thread* t, object target, void (*finalize)(Thread*, object))
|
|||||||
void* function;
|
void* function;
|
||||||
memcpy(&function, &finalize, BytesPerWord);
|
memcpy(&function, &finalize, BytesPerWord);
|
||||||
|
|
||||||
object f = makeFinalizer(t, 0, function, 0);
|
object f = makeFinalizer(t, 0, function, 0, 0, 0);
|
||||||
finalizerTarget(t, f) = target;
|
finalizerTarget(t, f) = target;
|
||||||
finalizerNext(t, f) = t->m->finalizers;
|
finalizerNext(t, f) = t->m->finalizers;
|
||||||
t->m->finalizers = f;
|
t->m->finalizers = f;
|
||||||
@ -4104,14 +4099,18 @@ makeTrace(Thread* t, Thread* target)
|
|||||||
void
|
void
|
||||||
runFinalizeThread(Thread* t)
|
runFinalizeThread(Thread* t)
|
||||||
{
|
{
|
||||||
object list = 0;
|
object finalizeList = 0;
|
||||||
PROTECT(t, list);
|
PROTECT(t, finalizeList);
|
||||||
|
|
||||||
|
object cleanList = 0;
|
||||||
|
PROTECT(t, cleanList);
|
||||||
|
|
||||||
while (true) {
|
while (true) {
|
||||||
{ ACQUIRE(t, t->m->stateLock);
|
{ ACQUIRE(t, t->m->stateLock);
|
||||||
|
|
||||||
while (t->m->finalizeThread
|
while (t->m->finalizeThread
|
||||||
and root(t, Machine::ObjectsToFinalize) == 0)
|
and root(t, Machine::ObjectsToFinalize) == 0
|
||||||
|
and root(t, Machine::ObjectsToClean) == 0)
|
||||||
{
|
{
|
||||||
ENTER(t, Thread::IdleState);
|
ENTER(t, Thread::IdleState);
|
||||||
t->m->stateLock->wait(t->systemThread, 0);
|
t->m->stateLock->wait(t->systemThread, 0);
|
||||||
@ -4120,15 +4119,20 @@ runFinalizeThread(Thread* t)
|
|||||||
if (t->m->finalizeThread == 0) {
|
if (t->m->finalizeThread == 0) {
|
||||||
return;
|
return;
|
||||||
} else {
|
} else {
|
||||||
list = root(t, Machine::ObjectsToFinalize);
|
finalizeList = root(t, Machine::ObjectsToFinalize);
|
||||||
setRoot(t, Machine::ObjectsToFinalize, 0);
|
setRoot(t, Machine::ObjectsToFinalize, 0);
|
||||||
|
|
||||||
|
cleanList = root(t, Machine::ObjectsToClean);
|
||||||
|
setRoot(t, Machine::ObjectsToClean, 0);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
for (; list; list = finalizeNodeNext(t, list)) {
|
for (; finalizeList; finalizeList = finalizerQueueNext(t, finalizeList)) {
|
||||||
finalizeObject
|
finalizeObject(t, finalizerQueueTarget(t, finalizeList), "finalize");
|
||||||
(t, finalizeNodeTarget(t, list),
|
}
|
||||||
static_cast<const char*>(finalizeNodeMethodName(t, list)));
|
|
||||||
|
for (; cleanList; cleanList = cleanerQueueNext(t, cleanList)) {
|
||||||
|
finalizeObject(t, cleanList, "clean");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1260,6 +1260,7 @@ class Machine {
|
|||||||
ShutdownHooks,
|
ShutdownHooks,
|
||||||
FinalizerThread,
|
FinalizerThread,
|
||||||
ObjectsToFinalize,
|
ObjectsToFinalize,
|
||||||
|
ObjectsToClean,
|
||||||
NullPointerException,
|
NullPointerException,
|
||||||
ArithmeticException,
|
ArithmeticException,
|
||||||
ArrayIndexOutOfBoundsException,
|
ArrayIndexOutOfBoundsException,
|
||||||
@ -1311,10 +1312,10 @@ class Machine {
|
|||||||
object finalizers;
|
object finalizers;
|
||||||
object tenuredFinalizers;
|
object tenuredFinalizers;
|
||||||
object finalizeQueue;
|
object finalizeQueue;
|
||||||
object cleanerQueue;
|
|
||||||
object weakReferences;
|
object weakReferences;
|
||||||
object tenuredWeakReferences;
|
object tenuredWeakReferences;
|
||||||
bool unsafe;
|
bool unsafe;
|
||||||
|
bool collecting;
|
||||||
bool triedBuiltinOnLoad;
|
bool triedBuiltinOnLoad;
|
||||||
bool dumpedHeapOnOOM;
|
bool dumpedHeapOnOOM;
|
||||||
JavaVMVTable javaVMVTable;
|
JavaVMVTable javaVMVTable;
|
||||||
|
@ -104,12 +104,9 @@
|
|||||||
(type finalizer
|
(type finalizer
|
||||||
(nogc object target)
|
(nogc object target)
|
||||||
(void* finalize)
|
(void* finalize)
|
||||||
(nogc object next))
|
(nogc object next)
|
||||||
|
(object queueTarget)
|
||||||
(type finalizeNode
|
(object queueNext))
|
||||||
(object target)
|
|
||||||
(void* methodName)
|
|
||||||
(object next))
|
|
||||||
|
|
||||||
(type hashMap
|
(type hashMap
|
||||||
(uint32_t size)
|
(uint32_t size)
|
||||||
@ -299,7 +296,8 @@
|
|||||||
|
|
||||||
(type phantomReference java/lang/ref/PhantomReference)
|
(type phantomReference java/lang/ref/PhantomReference)
|
||||||
|
|
||||||
(type cleaner sun/misc/Cleaner)
|
(type cleaner sun/misc/Cleaner
|
||||||
|
(object queueNext))
|
||||||
|
|
||||||
(type byteArray [B
|
(type byteArray [B
|
||||||
(extends jobject)
|
(extends jobject)
|
||||||
|
Loading…
Reference in New Issue
Block a user