/*
 * 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;
}