blob: 87127551258584ffd460192fcdafd0e2480be12d [file] [log] [blame]
Mathieu Chartier987ccff2013-07-08 11:05:21 -07001/*
2 * Copyright (C) 2013 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17import java.lang.reflect.*;
Mathieu Chartiere03df652014-09-02 17:36:08 -070018import java.lang.Runtime;
Richard Uhler6bfc37d2016-11-24 11:21:41 +000019import dalvik.system.VMRuntime;
Mathieu Chartier987ccff2013-07-08 11:05:21 -070020
Andreas Gampe1c83cbc2014-07-22 18:52:29 -070021public class Main {
Mathieu Chartier987ccff2013-07-08 11:05:21 -070022 static Object nativeLock = new Object();
Mathieu Chartier3b532d72015-06-05 13:21:05 -070023 static Object deadlockLock = new Object();
24 static boolean aboutToDeadlockLock = false;
Mathieu Chartier987ccff2013-07-08 11:05:21 -070025 static int nativeBytes = 0;
26 static Object runtime;
27 static Method register_native_allocation;
28 static Method register_native_free;
Mathieu Chartiere03df652014-09-02 17:36:08 -070029 static long maxMem = 0;
Mathieu Chartier987ccff2013-07-08 11:05:21 -070030
31 static class NativeAllocation {
32 private int bytes;
33
Mathieu Chartier3b532d72015-06-05 13:21:05 -070034 NativeAllocation(int bytes, boolean testingDeadlock) throws Exception {
Mathieu Chartier987ccff2013-07-08 11:05:21 -070035 this.bytes = bytes;
36 register_native_allocation.invoke(runtime, bytes);
Richard Uhler6bfc37d2016-11-24 11:21:41 +000037
38 // Register native allocation can only provide guarantees bounding
39 // the maximum outstanding allocations if finalizers don't time
40 // out. In case finalizers have timed out, wait longer for them
41 // now to complete so we can test the guarantees.
42 if (!testingDeadlock) {
43 VMRuntime.runFinalization(0);
44 }
45
Mathieu Chartier987ccff2013-07-08 11:05:21 -070046 synchronized (nativeLock) {
Mathieu Chartier3b532d72015-06-05 13:21:05 -070047 if (!testingDeadlock) {
48 nativeBytes += bytes;
Richard Uhler6bfc37d2016-11-24 11:21:41 +000049 if (nativeBytes > 2 * maxMem) {
Mathieu Chartier3b532d72015-06-05 13:21:05 -070050 throw new OutOfMemoryError();
51 }
Mathieu Chartier987ccff2013-07-08 11:05:21 -070052 }
53 }
54 }
55
56 protected void finalize() throws Exception {
57 synchronized (nativeLock) {
58 nativeBytes -= bytes;
59 }
60 register_native_free.invoke(runtime, bytes);
Mathieu Chartier3b532d72015-06-05 13:21:05 -070061 aboutToDeadlockLock = true;
62 synchronized (deadlockLock) {
63 }
Mathieu Chartier987ccff2013-07-08 11:05:21 -070064 }
65 }
66
67 public static void main(String[] args) throws Exception {
68 Class<?> vm_runtime = Class.forName("dalvik.system.VMRuntime");
69 Method get_runtime = vm_runtime.getDeclaredMethod("getRuntime");
70 runtime = get_runtime.invoke(null);
71 register_native_allocation = vm_runtime.getDeclaredMethod("registerNativeAllocation", Integer.TYPE);
72 register_native_free = vm_runtime.getDeclaredMethod("registerNativeFree", Integer.TYPE);
Mathieu Chartiere03df652014-09-02 17:36:08 -070073 maxMem = Runtime.getRuntime().maxMemory();
Mathieu Chartier987ccff2013-07-08 11:05:21 -070074 int count = 16;
Mathieu Chartiere03df652014-09-02 17:36:08 -070075 int size = (int)(maxMem / 2 / count);
Mathieu Chartier987ccff2013-07-08 11:05:21 -070076 int allocation_count = 256;
77 NativeAllocation[] allocations = new NativeAllocation[count];
78 for (int i = 0; i < allocation_count; ++i) {
Mathieu Chartier3b532d72015-06-05 13:21:05 -070079 allocations[i % count] = new NativeAllocation(size, false);
80 }
81 // Test that we don't get a deadlock if we are holding nativeLock. If there is no timeout,
82 // then we will get a finalizer timeout exception.
83 aboutToDeadlockLock = false;
84 synchronized (deadlockLock) {
85 for (int i = 0; aboutToDeadlockLock != true; ++i) {
86 allocations[i % count] = new NativeAllocation(size, true);
87 }
88 // Do more allocations now that the finalizer thread is deadlocked so that we force
89 // finalization and timeout.
90 for (int i = 0; i < 10; ++i) {
91 allocations[i % count] = new NativeAllocation(size, true);
92 }
Mathieu Chartier987ccff2013-07-08 11:05:21 -070093 }
94 System.out.println("Test complete");
95 }
96}
97