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 import dalvik.system.InMemoryDexClassLoader; 18*795d594fSAndroid Build Coastguard Worker 19*795d594fSAndroid Build Coastguard Worker import static java.lang.invoke.MethodType.methodType; 20*795d594fSAndroid Build Coastguard Worker 21*795d594fSAndroid Build Coastguard Worker import java.lang.invoke.MethodHandle; 22*795d594fSAndroid Build Coastguard Worker import java.lang.invoke.MethodHandles; 23*795d594fSAndroid Build Coastguard Worker import java.lang.ref.WeakReference; 24*795d594fSAndroid Build Coastguard Worker import java.lang.reflect.Constructor; 25*795d594fSAndroid Build Coastguard Worker import java.lang.reflect.Method; 26*795d594fSAndroid Build Coastguard Worker import java.nio.ByteBuffer; 27*795d594fSAndroid Build Coastguard Worker import java.util.Base64; 28*795d594fSAndroid Build Coastguard Worker import java.util.List; 29*795d594fSAndroid Build Coastguard Worker import java.util.function.Consumer; 30*795d594fSAndroid Build Coastguard Worker 31*795d594fSAndroid Build Coastguard Worker public class Main { 32*795d594fSAndroid Build Coastguard Worker // src-ex/ClassWithAMethod.java 33*795d594fSAndroid Build Coastguard Worker private static final String DEX_FILE = 34*795d594fSAndroid Build Coastguard Worker "ZGV4CjA0MQC4x9/g6TtIcd5240Zevdlr1D/wRBHV4Tv0BAAAeAAAAHhWNBIAAAAAAAAAADAEAAAW" + 35*795d594fSAndroid Build Coastguard Worker "AAAAeAAAAAcAAADQAAAABAAAAOwAAAAAAAAAAAAAAAcAAAAcAQAAAQAAAFQBAAAAAAAAAAAAAPQE" + 36*795d594fSAndroid Build Coastguard Worker "AAAAAAAAGgIAAB8CAAAnAgAAOQIAAFACAABTAgAAVgIAAGoCAACOAgAAqgIAAL4CAADfAgAA4gIA" + 37*795d594fSAndroid Build Coastguard Worker "AOYCAAD5AgAABQMAABgDAAAeAwAAJgMAADQDAAA7AwAASAMAAAQAAAAGAAAABwAAAAgAAAAJAAAA" + 38*795d594fSAndroid Build Coastguard Worker "CgAAAAsAAAAEAAAAAAAAAAAAAAAFAAAABQAAAAAAAAALAAAABgAAAAAAAAAMAAAABgAAABQCAAAB" + 39*795d594fSAndroid Build Coastguard Worker "AAIAAQAAAAEAAQANAAAAAQAAABEAAAABAAEAEgAAAAEAAgAUAAAAAwADAAEAAAAEAAIAAQAAAAEA" + 40*795d594fSAndroid Build Coastguard Worker "AAABAAAABAAAAAAAAAADAAAAGAQAAPYDAAAAAAAABAAAAAIAAAABAAAAAAAAAPwBAAACAAAAEhAP" + 41*795d594fSAndroid Build Coastguard Worker "AAEAAAAAAAAAAAIAAAMAAAD+AAAAEQAAAAEAAAAAAAAABAIAAAUAAABxAAQAAAASABEAAAABAAEA" + 42*795d594fSAndroid Build Coastguard Worker "AQAAAAkCAAAEAAAAcBAGAAAADgACAAAAAgAAAA0CAAAIAAAAIgADABoBFABwIAUAEAAnABoADgAs" + 43*795d594fSAndroid Build Coastguard Worker "AA4AJwAOPAAYAA4AHgAOAAAAAAEAAAAEAAMoKUkABjxpbml0PgAQQ2xhc3NXaXRoQU1ldGhvZAAV" + 44*795d594fSAndroid Build Coastguard Worker "Q2xhc3NXaXRoQU1ldGhvZC5qYXZhAAFJAAFMABJMQ2xhc3NXaXRoQU1ldGhvZDsAIkxhbm5vdGF0" + 45*795d594fSAndroid Build Coastguard Worker "aW9ucy9Db25zdGFudE1ldGhvZEhhbmRsZTsAGkxqYXZhL2xhbmcvQXNzZXJ0aW9uRXJyb3I7ABJM" + 46*795d594fSAndroid Build Coastguard Worker "amF2YS9sYW5nL09iamVjdDsAH0xqYXZhL2xhbmcvaW52b2tlL01ldGhvZEhhbmRsZTsAAVYAAlZM" + 47*795d594fSAndroid Build Coastguard Worker "ABFjb25zdE1ldGhvZEhhbmRsZQAKZGVzY3JpcHRvcgARZmllbGRPck1ldGhvZE5hbWUABGtpbmQA" + 48*795d594fSAndroid Build Coastguard Worker "Bm1ldGhvZAAMbWV0aG9kSGFuZGxlAAVvd25lcgALdW5yZWFjaGFibGUAnAF+fkQ4eyJiYWNrZW5k" + 49*795d594fSAndroid Build Coastguard Worker "IjoiZGV4IiwiY29tcGlsYXRpb24tbW9kZSI6ImRlYnVnIiwiaGFzLWNoZWNrc3VtcyI6ZmFsc2Us" + 50*795d594fSAndroid Build Coastguard Worker "Im1pbi1hcGkiOjI4LCJzaGEtMSI6IjdkZTM3N2E4MGI1MzQ1MGY1Y2ExYTNmYzUwZjZkYzk5M2U3" + 51*795d594fSAndroid Build Coastguard Worker "ZThhM2UiLCJ2ZXJzaW9uIjoiOC44LjQtZGV2In0AAQIEDhcADxcREAQEExcCAAAFAACBgATEAwEJ" + 52*795d594fSAndroid Build Coastguard Worker "kAMBCfwCAQqoAwEK3AMBAAAA5wMAAAAAAAAAAAAAAQAAAAAAAAADAAAAEAQAABAAAAAAAAAAAQAA" + 53*795d594fSAndroid Build Coastguard Worker "AAAAAAABAAAAFgAAAHgAAAACAAAABwAAANAAAAADAAAABAAAAOwAAAAFAAAABwAAABwBAAAGAAAA" + 54*795d594fSAndroid Build Coastguard Worker "AQAAAFQBAAAIAAAAAQAAAHQBAAABIAAABQAAAHwBAAADIAAABQAAAPwBAAABEAAAAQAAABQCAAAC" + 55*795d594fSAndroid Build Coastguard Worker "IAAAFgAAABoCAAAEIAAAAQAAAOcDAAAAIAAAAQAAAPYDAAADEAAAAQAAABAEAAAGIAAAAQAAABgE" + 56*795d594fSAndroid Build Coastguard Worker "AAAAEAAAAQAAADAEAAA="; 57*795d594fSAndroid Build Coastguard Worker 58*795d594fSAndroid Build Coastguard Worker private static final int ITERATIONS = 5; 59*795d594fSAndroid Build Coastguard Worker main(String[] args)60*795d594fSAndroid Build Coastguard Worker public static void main(String[] args) throws Throwable { 61*795d594fSAndroid Build Coastguard Worker verify($noinline$getJavaApiMethodHandle()); 62*795d594fSAndroid Build Coastguard Worker verify($noinline$getConstMethodHandle()); 63*795d594fSAndroid Build Coastguard Worker } 64*795d594fSAndroid Build Coastguard Worker verify(MethodHandle mh)65*795d594fSAndroid Build Coastguard Worker private static void verify(MethodHandle mh) { 66*795d594fSAndroid Build Coastguard Worker int result = 0; 67*795d594fSAndroid Build Coastguard Worker for (int i = 0; i < ITERATIONS; ++i) { 68*795d594fSAndroid Build Coastguard Worker try { 69*795d594fSAndroid Build Coastguard Worker result += (int) mh.invokeWithArguments(List.of()); 70*795d594fSAndroid Build Coastguard Worker } catch (Throwable t) { 71*795d594fSAndroid Build Coastguard Worker throw new RuntimeException(t); 72*795d594fSAndroid Build Coastguard Worker } 73*795d594fSAndroid Build Coastguard Worker Runtime.getRuntime().gc(); 74*795d594fSAndroid Build Coastguard Worker System.runFinalization(); 75*795d594fSAndroid Build Coastguard Worker } 76*795d594fSAndroid Build Coastguard Worker 77*795d594fSAndroid Build Coastguard Worker if (result != ITERATIONS) { 78*795d594fSAndroid Build Coastguard Worker throw new AssertionError(); 79*795d594fSAndroid Build Coastguard Worker } 80*795d594fSAndroid Build Coastguard Worker } 81*795d594fSAndroid Build Coastguard Worker $noinline$getJavaApiMethodHandle()82*795d594fSAndroid Build Coastguard Worker private static MethodHandle $noinline$getJavaApiMethodHandle() throws Throwable { 83*795d594fSAndroid Build Coastguard Worker ClassLoader loader = new InMemoryDexClassLoader( 84*795d594fSAndroid Build Coastguard Worker ByteBuffer.wrap(Base64.getDecoder().decode(DEX_FILE)), 85*795d594fSAndroid Build Coastguard Worker ClassLoader.getSystemClassLoader()); 86*795d594fSAndroid Build Coastguard Worker Class<?> clazz = loader.loadClass("ClassWithAMethod"); 87*795d594fSAndroid Build Coastguard Worker return MethodHandles.publicLookup().findStatic(clazz, "method", methodType(int.class)); 88*795d594fSAndroid Build Coastguard Worker } 89*795d594fSAndroid Build Coastguard Worker $noinline$getConstMethodHandle()90*795d594fSAndroid Build Coastguard Worker private static MethodHandle $noinline$getConstMethodHandle() throws Throwable { 91*795d594fSAndroid Build Coastguard Worker ClassLoader loader = new InMemoryDexClassLoader( 92*795d594fSAndroid Build Coastguard Worker ByteBuffer.wrap(Base64.getDecoder().decode(DEX_FILE)), 93*795d594fSAndroid Build Coastguard Worker ClassLoader.getSystemClassLoader()); 94*795d594fSAndroid Build Coastguard Worker Class<?> clazz = loader.loadClass("ClassWithAMethod"); 95*795d594fSAndroid Build Coastguard Worker 96*795d594fSAndroid Build Coastguard Worker return (MethodHandle) clazz.getDeclaredMethod("constMethodHandle").invoke(null); 97*795d594fSAndroid Build Coastguard Worker } 98*795d594fSAndroid Build Coastguard Worker }