/*
* Copyright (C) 2019 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 class InnerInitialized {
static int staticValue1 = 0;
static int staticValue2 = 1;
static int $noinline$runHotMethod(boolean doComputation) {
if (doComputation) {
for (int i = 0; i < 100000; i++) {
staticValue1 += staticValue2;
}
}
return staticValue1;
}
static {
// Make $noinline$runHotMethod hot so it gets compiled. The
// regression used to be that the JIT would incorrectly update the
// resolution entrypoint of the method. The entrypoint needs to stay
// the resolution entrypoint otherwise other threads may incorrectly
// execute static methods of the class while the class is still being
// initialized.
for (int i = 0; i < 10; i++) {
$noinline$runHotMethod(true);
}
// Start another thread that will invoke a static method of InnerInitialized.
new Thread(new Runnable() {
public void run() {
for (int i = 0; i < 100000; i++) {
$noinline$runInternalHotMethod(false);
}
// Give some time for the JIT compiler to compile $noinline$runInternalHotMethod.
// Only compiled code invoke the entrypoint of an ArtMethod.
try {
Thread.sleep(1000);
} catch (Exception e) {
}
int value = $noinline$runInternalHotMethod(true);
if (value != 42) {
throw new Error("Expected 42, got " + value);
}
}
public int $noinline$runInternalHotMethod(boolean invokeStaticMethod) {
if (invokeStaticMethod) {
// The bug used to be here: the compiled code of $noinline$runInternalHotMethod
// would invoke the entrypoint of $noinline$runHotMethod, which was incorrectly
// updated to the JIT entrypoint and therefore not hitting the resolution
// trampoline which would have waited for the class to be initialized.
return $noinline$runHotMethod(false);
}
return 0;
}
}).start();
// Give some time for the JIT compiler to compile runHotMethod, and also for the
// other thread to invoke $noinline$runHotMethod.
// This wait should be longer than the other thread's wait to make sure the other
// thread hits the $noinline$runHotMethod call before the initialization of
// InnerInitialized is finished.
try {
Thread.sleep(5000);
} catch (Exception e) {
}
staticValue1 = 42;
}
}
public static void main(String[] args) throws Exception {
System.out.println(InnerInitialized.staticValue1);
}
}