1*795d594fSAndroid Build Coastguard Worker /* 2*795d594fSAndroid Build Coastguard Worker * Copyright (C) 2024 The Android Open Source Project 3*795d594fSAndroid Build Coastguard Worker * 4*795d594fSAndroid Build Coastguard Worker * Licensed under the Apache License, Version 2.0 (the "License"); 5*795d594fSAndroid Build Coastguard Worker * you may not use this file except in compliance with the License. 6*795d594fSAndroid Build Coastguard Worker * You may obtain a copy of the License at 7*795d594fSAndroid Build Coastguard Worker * 8*795d594fSAndroid Build Coastguard Worker * http://www.apache.org/licenses/LICENSE-2.0 9*795d594fSAndroid Build Coastguard Worker * 10*795d594fSAndroid Build Coastguard Worker * Unless required by applicable law or agreed to in writing, software 11*795d594fSAndroid Build Coastguard Worker * distributed under the License is distributed on an "AS IS" BASIS, 12*795d594fSAndroid Build Coastguard Worker * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13*795d594fSAndroid Build Coastguard Worker * See the License for the specific language governing permissions and 14*795d594fSAndroid Build Coastguard Worker * limitations under the License. 15*795d594fSAndroid Build Coastguard Worker */ 16*795d594fSAndroid Build Coastguard Worker 17*795d594fSAndroid Build Coastguard Worker public class Main { 18*795d594fSAndroid Build Coastguard Worker // Numbers have to be inlined for the bug to trigger: we cannot define this as a generic method 19*795d594fSAndroid Build Coastguard Worker // that takes e.g. 12 as a parameter. We can define some methods to test different cases. Each 20*795d594fSAndroid Build Coastguard Worker // method is the same as doing: 21*795d594fSAndroid Build Coastguard Worker // result = initial_value; 22*795d594fSAndroid Build Coastguard Worker // for (int i = start; i < end; ++i) { result += i * 10; } 23*795d594fSAndroid Build Coastguard Worker // return result; 24*795d594fSAndroid Build Coastguard Worker // As long as `end` is 5 or less 25*795d594fSAndroid Build Coastguard Worker $noinline$nestedLoopCalculation_0_2_12()26*795d594fSAndroid Build Coastguard Worker private static int $noinline$nestedLoopCalculation_0_2_12() { 27*795d594fSAndroid Build Coastguard Worker int result = 12; 28*795d594fSAndroid Build Coastguard Worker for (int outer = 0; outer < 2; ++outer) { 29*795d594fSAndroid Build Coastguard Worker int first_inner; 30*795d594fSAndroid Build Coastguard Worker for (first_inner = 5; first_inner > outer; first_inner--) {} 31*795d594fSAndroid Build Coastguard Worker for (int second_inner = 0; second_inner < 10; second_inner++) { 32*795d594fSAndroid Build Coastguard Worker result += first_inner; 33*795d594fSAndroid Build Coastguard Worker } 34*795d594fSAndroid Build Coastguard Worker } 35*795d594fSAndroid Build Coastguard Worker return result; 36*795d594fSAndroid Build Coastguard Worker } 37*795d594fSAndroid Build Coastguard Worker $noinline$nestedLoopCalculation_1_3_12()38*795d594fSAndroid Build Coastguard Worker private static int $noinline$nestedLoopCalculation_1_3_12() { 39*795d594fSAndroid Build Coastguard Worker int result = 12; 40*795d594fSAndroid Build Coastguard Worker for (int outer = 1; outer < 3; ++outer) { 41*795d594fSAndroid Build Coastguard Worker int first_inner; 42*795d594fSAndroid Build Coastguard Worker for (first_inner = 5; first_inner > outer; first_inner--) {} 43*795d594fSAndroid Build Coastguard Worker for (int second_inner = 0; second_inner < 10; second_inner++) { 44*795d594fSAndroid Build Coastguard Worker result += first_inner; 45*795d594fSAndroid Build Coastguard Worker } 46*795d594fSAndroid Build Coastguard Worker } 47*795d594fSAndroid Build Coastguard Worker return result; 48*795d594fSAndroid Build Coastguard Worker } 49*795d594fSAndroid Build Coastguard Worker $noinline$nestedLoopCalculation_minus2_2_12()50*795d594fSAndroid Build Coastguard Worker private static int $noinline$nestedLoopCalculation_minus2_2_12() { 51*795d594fSAndroid Build Coastguard Worker int result = 12; 52*795d594fSAndroid Build Coastguard Worker for (int outer = -2; outer < 2; ++outer) { 53*795d594fSAndroid Build Coastguard Worker int first_inner; 54*795d594fSAndroid Build Coastguard Worker for (first_inner = 5; first_inner > outer; first_inner--) {} 55*795d594fSAndroid Build Coastguard Worker for (int second_inner = 0; second_inner < 10; second_inner++) { 56*795d594fSAndroid Build Coastguard Worker result += first_inner; 57*795d594fSAndroid Build Coastguard Worker } 58*795d594fSAndroid Build Coastguard Worker } 59*795d594fSAndroid Build Coastguard Worker return result; 60*795d594fSAndroid Build Coastguard Worker } 61*795d594fSAndroid Build Coastguard Worker $noinline$nestedLoopCalculation_0_5_12()62*795d594fSAndroid Build Coastguard Worker private static int $noinline$nestedLoopCalculation_0_5_12() { 63*795d594fSAndroid Build Coastguard Worker int result = 12; 64*795d594fSAndroid Build Coastguard Worker for (int outer = 0; outer < 5; ++outer) { 65*795d594fSAndroid Build Coastguard Worker int first_inner; 66*795d594fSAndroid Build Coastguard Worker for (first_inner = 5; first_inner > outer; first_inner--) {} 67*795d594fSAndroid Build Coastguard Worker for (int second_inner = 0; second_inner < 10; second_inner++) { 68*795d594fSAndroid Build Coastguard Worker result += first_inner; 69*795d594fSAndroid Build Coastguard Worker } 70*795d594fSAndroid Build Coastguard Worker } 71*795d594fSAndroid Build Coastguard Worker return result; 72*795d594fSAndroid Build Coastguard Worker } 73*795d594fSAndroid Build Coastguard Worker main(String[] f)74*795d594fSAndroid Build Coastguard Worker public static void main(String[] f) { 75*795d594fSAndroid Build Coastguard Worker // 12 + 0 + 10 = 22 76*795d594fSAndroid Build Coastguard Worker assertIntEquals(22, $noinline$nestedLoopCalculation_0_2_12()); 77*795d594fSAndroid Build Coastguard Worker // 12 + 10 + 20 = 42 78*795d594fSAndroid Build Coastguard Worker assertIntEquals(42, $noinline$nestedLoopCalculation_1_3_12()); 79*795d594fSAndroid Build Coastguard Worker // 12 + (-20) + (-10) + 0 + 10 = -8 80*795d594fSAndroid Build Coastguard Worker assertIntEquals(-8, $noinline$nestedLoopCalculation_minus2_2_12()); 81*795d594fSAndroid Build Coastguard Worker // 12 + 0 + 10 + 20 + 30 + 40 = 112 82*795d594fSAndroid Build Coastguard Worker assertIntEquals(112, $noinline$nestedLoopCalculation_0_5_12()); 83*795d594fSAndroid Build Coastguard Worker } 84*795d594fSAndroid Build Coastguard Worker assertIntEquals(int expected, int result)85*795d594fSAndroid Build Coastguard Worker public static void assertIntEquals(int expected, int result) { 86*795d594fSAndroid Build Coastguard Worker if (expected != result) { 87*795d594fSAndroid Build Coastguard Worker throw new Error("Expected: " + expected + ", found: " + result); 88*795d594fSAndroid Build Coastguard Worker } 89*795d594fSAndroid Build Coastguard Worker } 90*795d594fSAndroid Build Coastguard Worker } 91