summaryrefslogtreecommitdiffstats
path: root/vm/alloc/MarkSweep.c
diff options
context:
space:
mode:
authorCarl Shapiro <cshapiro@google.com>2011-03-30 19:35:34 -0700
committerCarl Shapiro <cshapiro@google.com>2011-03-30 19:53:34 -0700
commitce87bfed41bbe4248b2770fb1a90f34b2518f6fa (patch)
tree8a2ca573187d322f7202df68d36a1101294f5ee5 /vm/alloc/MarkSweep.c
parent6920d2c5b83f8ff0cdc8f33770cd658969dbd262 (diff)
downloadandroid_dalvik-ce87bfed41bbe4248b2770fb1a90f34b2518f6fa.tar.gz
android_dalvik-ce87bfed41bbe4248b2770fb1a90f34b2518f6fa.tar.bz2
android_dalvik-ce87bfed41bbe4248b2770fb1a90f34b2518f6fa.zip
Delegate the queuing of cleared references to managed code.
Previously, the garbage collector would pass each cleared reference to the heap worker thread for queuing. The heap worker thread would then perform a callback into managed code for each cleared reference which assigned the reference to its reference queue. With this change, the garbage collector instead links together all of the cleared references and calls back into managed code exactly once to hand off the references for processing. This change makes the heap worker thread and its data structures obsolete. Change-Id: I28e02638f0877a7fd2ac96b9c3f5597a38541ebb
Diffstat (limited to 'vm/alloc/MarkSweep.c')
-rw-r--r--vm/alloc/MarkSweep.c54
1 files changed, 27 insertions, 27 deletions
diff --git a/vm/alloc/MarkSweep.c b/vm/alloc/MarkSweep.c
index 69b75de3f..e26394c85 100644
--- a/vm/alloc/MarkSweep.c
+++ b/vm/alloc/MarkSweep.c
@@ -20,7 +20,6 @@
#include "alloc/HeapBitmapInlines.h"
#include "alloc/HeapInternal.h"
#include "alloc/HeapSource.h"
-#include "alloc/HeapWorker.h"
#include "alloc/MarkSweep.h"
#include "alloc/Visit.h"
#include "alloc/VisitInlines.h"
@@ -342,8 +341,6 @@ static void verifyImmuneObjects()
* - Primitive classes
* - Special objects
* - gDvm.outOfMemoryObj
- * - Objects allocated with ALLOC_NO_GC
- * - Objects pending finalization (but not yet finalized)
* - Objects in debugger object registry
*
* Don't need:
@@ -795,11 +792,7 @@ static void enqueueReference(Object *ref)
assert(ref != NULL);
assert(dvmGetFieldObject(ref, gDvm.offJavaLangRefReference_queue) != NULL);
assert(dvmGetFieldObject(ref, gDvm.offJavaLangRefReference_queueNext) == NULL);
- if (!dvmHeapAddRefToLargeTable(&gDvm.gcHeap->referenceOperations, ref)) {
- LOGE_HEAP("enqueueReference(): no room for any more "
- "reference operations\n");
- dvmAbort();
- }
+ enqueuePendingReference(ref, &gDvm.gcHeap->clearedReferences);
}
/*
@@ -857,11 +850,9 @@ static void clearWhiteReferences(Object **list)
GcMarkContext *ctx;
Object *ref, *referent;
size_t referentOffset;
- bool doSignal;
ctx = &gDvm.gcHeap->markContext;
referentOffset = gDvm.offJavaLangRefReference_referent;
- doSignal = false;
while (*list != NULL) {
ref = dequeuePendingReference(list);
referent = dvmGetFieldObject(ref, referentOffset);
@@ -870,31 +861,23 @@ static void clearWhiteReferences(Object **list)
clearReference(ref);
if (isEnqueuable(ref)) {
enqueueReference(ref);
- doSignal = true;
}
}
}
- /*
- * If we cleared a reference with a reference queue we must notify
- * the heap worker to append the reference.
- */
- if (doSignal) {
- dvmSignalHeapWorker(false);
- }
assert(*list == NULL);
}
/*
* Enqueues finalizer references with white referents. White
- * referents are blackened, moved to the pendingNext field, and the
+ * referents are blackened, moved to the zombie field, and the
* referent field is cleared.
*/
static void enqueueFinalizerReferences(Object **list)
{
GcMarkContext *ctx = &gDvm.gcHeap->markContext;
size_t referentOffset = gDvm.offJavaLangRefReference_referent;
- size_t pendingNextOffset = gDvm.offJavaLangRefReference_pendingNext;
- bool doSignal = false;
+ size_t zombieOffset = gDvm.offJavaLangRefFinalizerReference_zombie;
+ bool hasEnqueued = false;
while (*list != NULL) {
Object *ref = dequeuePendingReference(list);
Object *referent = dvmGetFieldObject(ref, referentOffset);
@@ -902,15 +885,14 @@ static void enqueueFinalizerReferences(Object **list)
markObject(referent, ctx);
/* If the referent is non-null the reference must queuable. */
assert(isEnqueuable(ref));
- dvmSetFieldObject(ref, pendingNextOffset, referent);
+ dvmSetFieldObject(ref, zombieOffset, referent);
clearReference(ref);
enqueueReference(ref);
- doSignal = true;
+ hasEnqueued = true;
}
}
- if (doSignal) {
+ if (hasEnqueued) {
processMarkStack(ctx);
- dvmSignalHeapWorker(false);
}
assert(*list == NULL);
}
@@ -928,8 +910,8 @@ void dvmSetFinalizable(Object *obj)
assert(self != NULL);
Method *meth = gDvm.methJavaLangRefFinalizerReferenceAdd;
assert(meth != NULL);
- JValue unused;
- dvmCallMethod(self, meth, obj, &unused, obj);
+ JValue unusedResult;
+ dvmCallMethod(self, meth, NULL, &unusedResult, obj);
}
/*
@@ -982,6 +964,24 @@ void dvmHeapProcessReferences(Object **softReferences, bool clearSoftRefs,
assert(*phantomReferences == NULL);
}
+/*
+ * Pushes a list of cleared references out to the managed heap.
+ */
+void dvmEnqueueClearedReferences(Object **cleared)
+{
+ assert(cleared != NULL);
+ if (*cleared != NULL) {
+ Thread *self = dvmThreadSelf();
+ assert(self != NULL);
+ Method *meth = gDvm.methJavaLangRefReferenceQueueAdd;
+ assert(meth != NULL);
+ JValue unused;
+ Object *reference = *cleared;
+ dvmCallMethod(self, meth, NULL, &unused, reference);
+ *cleared = NULL;
+ }
+}
+
void dvmHeapFinishMarkStep()
{
GcMarkContext *ctx;