/* * 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.*; import java.lang.Runtime; import dalvik.system.VMRuntime; public class Main { static Object nativeLock = new Object(); static Object deadlockLock = new Object(); static boolean aboutToDeadlockLock = false; static int nativeBytes = 0; static Object runtime; static Method register_native_allocation; static Method register_native_free; static long maxMem = 0; static class NativeAllocation { private int bytes; NativeAllocation(int bytes, boolean testingDeadlock) throws Exception { this.bytes = bytes; register_native_allocation.invoke(runtime, bytes); // Register native allocation can only provide guarantees bounding // the maximum outstanding allocations if finalizers don't time // out. In case finalizers have timed out, wait longer for them // now to complete so we can test the guarantees. if (!testingDeadlock) { VMRuntime.runFinalization(0); } synchronized (nativeLock) { if (!testingDeadlock) { nativeBytes += bytes; if (nativeBytes > 2 * maxMem) { throw new OutOfMemoryError(); } } } } protected void finalize() throws Exception { synchronized (nativeLock) { nativeBytes -= bytes; } register_native_free.invoke(runtime, bytes); aboutToDeadlockLock = true; synchronized (deadlockLock) { } } } 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); maxMem = Runtime.getRuntime().maxMemory(); int count = 16; int size = (int)(maxMem / 2 / count); int allocation_count = 256; NativeAllocation[] allocations = new NativeAllocation[count]; for (int i = 0; i < allocation_count; ++i) { allocations[i % count] = new NativeAllocation(size, false); } // Test that we don't get a deadlock if we are holding nativeLock. If there is no timeout, // then we will get a finalizer timeout exception. aboutToDeadlockLock = false; synchronized (deadlockLock) { for (int i = 0; aboutToDeadlockLock != true; ++i) { allocations[i % count] = new NativeAllocation(size, true); } // Do more allocations now that the finalizer thread is deadlocked so that we force // finalization and timeout. for (int i = 0; i < 10; ++i) { allocations[i % count] = new NativeAllocation(size, true); } } System.out.println("Test complete"); } }