--- /dev/null
+Test complete
--- /dev/null
+This is a test to verify that native allocation successfully runs
+finalizers and prevents OOM.
--- /dev/null
+/*
+ * Copyright (C) 2013 The Android Open Source Project
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+import java.lang.reflect.*;
+
+class Main {
+ static Object nativeLock = new Object();
+ static int nativeBytes = 0;
+ static Object runtime;
+ static Method register_native_allocation;
+ static Method register_native_free;
+ static int maxMem = 64 * 1024 * 1024;
+
+ static class NativeAllocation {
+ private int bytes;
+
+ NativeAllocation(int bytes) throws Exception {
+ this.bytes = bytes;
+ register_native_allocation.invoke(runtime, bytes);
+ synchronized (nativeLock) {
+ nativeBytes += bytes;
+ if (nativeBytes > maxMem) {
+ throw new OutOfMemoryError();
+ }
+ }
+ }
+
+ protected void finalize() throws Exception {
+ synchronized (nativeLock) {
+ nativeBytes -= bytes;
+ }
+ register_native_free.invoke(runtime, bytes);
+ }
+ }
+
+ public static void main(String[] args) throws Exception {
+ Class<?> vm_runtime = Class.forName("dalvik.system.VMRuntime");
+ Method get_runtime = vm_runtime.getDeclaredMethod("getRuntime");
+ runtime = get_runtime.invoke(null);
+ register_native_allocation = vm_runtime.getDeclaredMethod("registerNativeAllocation", Integer.TYPE);
+ register_native_free = vm_runtime.getDeclaredMethod("registerNativeFree", Integer.TYPE);
+ int count = 16;
+ int size = 512 * 0x400;
+ int allocation_count = 256;
+ NativeAllocation[] allocations = new NativeAllocation[count];
+ for (int i = 0; i < allocation_count; ++i) {
+ allocations[i % count] = new NativeAllocation(size);
+ }
+ System.out.println("Test complete");
+ }
+}
+
Thread *self = dvmThreadSelf();
assert(self != NULL);
JValue unusedResult;
+ assert(gDvm.methJavaLangSystem_runFinalization != NULL);
dvmCallMethod(self, gDvm.methJavaLangSystem_runFinalization, NULL, &unusedResult);
}
#include "alloc/HeapBitmap.h"
#include "alloc/HeapBitmapInlines.h"
+static void dvmHeapSourceUpdateMaxNativeFootprint();
static void snapIdealFootprint();
static void setIdealFootprint(size_t max);
static size_t getMaximumSize(const HeapSource *hs);
int32_t nativeBytesAllocated;
size_t nativeFootprintGCWatermark;
size_t nativeFootprintLimit;
+ bool nativeNeedToRunFinalization;
/*
* State for the GC daemon.
hs->nativeBytesAllocated = 0;
hs->nativeFootprintGCWatermark = startSize;
hs->nativeFootprintLimit = startSize * 2;
+ hs->nativeNeedToRunFinalization = false;
hs->hasGcThread = false;
hs->heapBase = (char *)base;
hs->heapLength = length;
} else {
heap->concurrentStartBytes = freeBytes - CONCURRENT_START;
}
+
+ /* Mark that we need to run finalizers and update the native watermarks
+ * next time we attempt to register a native allocation.
+ */
+ gHs->nativeNeedToRunFinalization = true;
}
/*
void dvmHeapSourceRegisterNativeAllocation(int bytes)
{
+ /* If we have just done a GC, ensure that the finalizers are done and update
+ * the native watermarks.
+ */
+ if (gHs->nativeNeedToRunFinalization) {
+ dvmRunFinalization();
+ dvmHeapSourceUpdateMaxNativeFootprint();
+ gHs->nativeNeedToRunFinalization = false;
+ }
+
android_atomic_add(bytes, &gHs->nativeBytesAllocated);
if ((size_t)gHs->nativeBytesAllocated > gHs->nativeFootprintGCWatermark) {
*/
if ((size_t)gHs->nativeBytesAllocated > gHs->nativeFootprintLimit) {
Thread* self = dvmThreadSelf();
-
dvmRunFinalization();
- if (!dvmCheckException(self)) {
+ if (dvmCheckException(self)) {
return;
}
-
dvmLockHeap();
bool waited = dvmWaitForConcurrentGcToComplete();
dvmUnlockHeap();
if (waited) {
// Just finished a GC, attempt to run finalizers.
dvmRunFinalization();
- if (!dvmCheckException(self)) {
+ if (dvmCheckException(self)) {
return;
}
}
dvmCollectGarbageInternal(GC_FOR_MALLOC);
dvmUnlockHeap();
dvmRunFinalization();
-
- if (!dvmCheckException(self)) {
+ gHs->nativeNeedToRunFinalization = false;
+ if (dvmCheckException(self)) {
return;
}
}