| /* |
| * 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; |
| |
| 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); |
| synchronized (nativeLock) { |
| if (!testingDeadlock) { |
| nativeBytes += bytes; |
| if (nativeBytes > 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"); |
| } |
| } |
| |