| /* |
| * Copyright (C) 2016 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.util.IdentityHashMap; |
| import dalvik.system.VMRuntime; |
| |
| public class Main { |
| public static void main(String[] args) { |
| System.loadLibrary(args[0]); |
| int initialSize = monitorListSize(); |
| IdentityHashMap<Object, Integer> all = new IdentityHashMap(); |
| for (int i = 0; i < 5000; ++i) { |
| Object obj = new Object(); |
| synchronized(obj) { |
| // Should force inflation. |
| all.put(obj, obj.hashCode()); |
| } |
| } |
| // Since monitor deflation is delayed significantly, we believe that even with an intervening |
| // GC, monitors should remain inflated. We allow some slop for unrelated concurrent runtime |
| // actions. |
| int inflatedSize = monitorListSize(); |
| if (inflatedSize >= initialSize + 4000) { |
| System.out.println("Monitor list grew by at least 4000 monitors"); |
| } else { |
| System.out.println("Monitor list did not grow as expected"); |
| } |
| // Encourage monitor deflation. |
| // trim() (Heap::Trim()) deflates only in JANK_IMPERCEPTIBLE state. |
| // Some of this mirrors code in ActivityThread.java. |
| final int DALVIK_PROCESS_STATE_JANK_PERCEPTIBLE = 0; |
| final int DALVIK_PROCESS_STATE_JANK_IMPERCEPTIBLE = 1; |
| VMRuntime.getRuntime().updateProcessState(DALVIK_PROCESS_STATE_JANK_IMPERCEPTIBLE); |
| System.gc(); |
| System.runFinalization(); |
| trim(); |
| VMRuntime.getRuntime().updateProcessState(DALVIK_PROCESS_STATE_JANK_PERCEPTIBLE); |
| int finalSize = monitorListSize(); |
| if (finalSize > initialSize + 1000) { |
| System.out.println("Monitor list failed to shrink properly"); |
| } else { |
| System.out.println("Monitor list shrank correctly"); |
| } |
| int j = 0; |
| for (Object obj: all.keySet()) { |
| ++j; |
| if (obj.hashCode() != all.get(obj)) { |
| throw new AssertionError("Failed hashcode test!"); |
| } |
| } |
| System.out.println("Finished first check"); |
| for (Object obj: all.keySet()) { |
| ++j; |
| synchronized(obj) { |
| if (obj.hashCode() != all.get(obj)) { |
| throw new AssertionError("Failed hashcode test!"); |
| } |
| } |
| } |
| System.out.println("Finished second check"); |
| System.out.println("Total checks: " + j); |
| } |
| |
| private static native void trim(); |
| |
| private static native int monitorListSize(); |
| } |