/*
* Copyright (C) 2017 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.
*/
public class Main {
public static void main(String[] args) {
System.loadLibrary(args[0]);
if (!hasJit()) {
// Make the test pass if not using JIT.
return;
}
if (hasImage()) {
throw new Error("The `run` script should prevent this test from running with an image!");
}
if (!isClassMoveable(String.class)) {
throw new Error("String.class not moveable despite running without image!");
}
// Make sure the Main.test() is JIT-compiled and then call it.
ensureJitCompiled(Main.class, "test");
test();
}
public static void test() {
int length = 5;
// Hide the type of these strings in an Object array,
// so that we treat them as Object for the String.equals() below.
Object[] array = new Object[length];
for (int i = 0; i != length; ++i) {
array[i] = "V" + i;
}
// Continually check string equality between a newly allocated String and an
// already allocated String with the same contents while allocating over 128MiB
// memory (with heap size limited to 16MiB), ensuring we run GC and stress the
// instanceof check in the String.equals() implementation.
for (int count = 0; count != 128 * 1024; ++count) {
for (int i = 0; i != length; ++i) {
allocateAtLeast1KiB();
assertTrue(("V" + i).equals(array[i]));
}
}
}
public static void allocateAtLeast1KiB() {
// Give GC more work by allocating Object arrays.
memory[allocationIndex] = new Object[1024 / 4];
++allocationIndex;
if (allocationIndex == memory.length) {
allocationIndex = 0;
}
}
public static void assertTrue(boolean value) {
if (!value) {
throw new Error("Assertion failed!");
}
}
private native static boolean hasJit();
private native static boolean hasImage();
private native static boolean isClassMoveable(Class<?> cls);
private static native void ensureJitCompiled(Class<?> itf, String method_name);
// We shall retain some allocated memory and release old allocations
// so that the GC has something to do.
public static Object[] memory = new Object[4096];
public static int allocationIndex = 0;
}