1*795d594fSAndroid Build Coastguard Worker /* 2*795d594fSAndroid Build Coastguard Worker * Copyright (C) 2017 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 java.util.Arrays; 18*795d594fSAndroid Build Coastguard Worker import java.lang.reflect.Executable; 19*795d594fSAndroid Build Coastguard Worker import java.lang.reflect.Method; 20*795d594fSAndroid Build Coastguard Worker import java.util.Base64; 21*795d594fSAndroid Build Coastguard Worker import art.Breakpoint; 22*795d594fSAndroid Build Coastguard Worker import art.Redefinition; 23*795d594fSAndroid Build Coastguard Worker 24*795d594fSAndroid Build Coastguard Worker public class Main { 25*795d594fSAndroid Build Coastguard Worker static class Transform { sayHi()26*795d594fSAndroid Build Coastguard Worker public void sayHi() { 27*795d594fSAndroid Build Coastguard Worker System.out.println("Hello"); 28*795d594fSAndroid Build Coastguard Worker } 29*795d594fSAndroid Build Coastguard Worker } 30*795d594fSAndroid Build Coastguard Worker 31*795d594fSAndroid Build Coastguard Worker /** 32*795d594fSAndroid Build Coastguard Worker * base64 encoded class/dex file for 33*795d594fSAndroid Build Coastguard Worker * class Transform { 34*795d594fSAndroid Build Coastguard Worker * public void sayHi() { 35*795d594fSAndroid Build Coastguard Worker * System.out.println("Goodbye"); 36*795d594fSAndroid Build Coastguard Worker * } 37*795d594fSAndroid Build Coastguard Worker * } 38*795d594fSAndroid Build Coastguard Worker */ 39*795d594fSAndroid Build Coastguard Worker private static final byte[] DEX_BYTES = Base64.getDecoder().decode( 40*795d594fSAndroid Build Coastguard Worker "ZGV4CjAzNQA7jFommHUzfbuvjq/I2cDcwdjqQk6KPfqYAwAAcAAAAHhWNBIAAAAAAAAAANQCAAAU" + 41*795d594fSAndroid Build Coastguard Worker "AAAAcAAAAAkAAADAAAAAAgAAAOQAAAABAAAA/AAAAAQAAAAEAQAAAQAAACQBAABUAgAARAEAAJ4B" + 42*795d594fSAndroid Build Coastguard Worker "AACmAQAArwEAAMEBAADJAQAA7QEAAA0CAAAkAgAAOAIAAEwCAABgAgAAawIAAHYCAAB5AgAAfQIA" + 43*795d594fSAndroid Build Coastguard Worker "AIoCAACQAgAAlQIAAJ4CAAClAgAAAgAAAAMAAAAEAAAABQAAAAYAAAAHAAAACAAAAAkAAAAMAAAA" + 44*795d594fSAndroid Build Coastguard Worker "DAAAAAgAAAAAAAAADQAAAAgAAACYAQAABwAEABAAAAAAAAAAAAAAAAAAAAASAAAABAABABEAAAAF" + 45*795d594fSAndroid Build Coastguard Worker "AAAAAAAAAAAAAAAAAAAABQAAAAAAAAAKAAAAiAEAAMYCAAAAAAAAAgAAALcCAAC9AgAAAQABAAEA" + 46*795d594fSAndroid Build Coastguard Worker "AACsAgAABAAAAHAQAwAAAA4AAwABAAIAAACxAgAACAAAAGIAAAAaAQEAbiACABAADgBEAQAAAAAA" + 47*795d594fSAndroid Build Coastguard Worker "AAAAAAAAAAAAAQAAAAYABjxpbml0PgAHR29vZGJ5ZQAQTE1haW4kVHJhbnNmb3JtOwAGTE1haW47" + 48*795d594fSAndroid Build Coastguard Worker "ACJMZGFsdmlrL2Fubm90YXRpb24vRW5jbG9zaW5nQ2xhc3M7AB5MZGFsdmlrL2Fubm90YXRpb24v" + 49*795d594fSAndroid Build Coastguard Worker "SW5uZXJDbGFzczsAFUxqYXZhL2lvL1ByaW50U3RyZWFtOwASTGphdmEvbGFuZy9PYmplY3Q7ABJM" + 50*795d594fSAndroid Build Coastguard Worker "amF2YS9sYW5nL1N0cmluZzsAEkxqYXZhL2xhbmcvU3lzdGVtOwAJTWFpbi5qYXZhAAlUcmFuc2Zv" + 51*795d594fSAndroid Build Coastguard Worker "cm0AAVYAAlZMAAthY2Nlc3NGbGFncwAEbmFtZQADb3V0AAdwcmludGxuAAVzYXlIaQAFdmFsdWUA" + 52*795d594fSAndroid Build Coastguard Worker "EgAHDgAUAAcOeAACAgETGAECAwIOBAgPFwsAAAEBAICABNACAQHoAhAAAAAAAAAAAQAAAAAAAAAB" + 53*795d594fSAndroid Build Coastguard Worker "AAAAFAAAAHAAAAACAAAACQAAAMAAAAADAAAAAgAAAOQAAAAEAAAAAQAAAPwAAAAFAAAABAAAAAQB" + 54*795d594fSAndroid Build Coastguard Worker "AAAGAAAAAQAAACQBAAADEAAAAQAAAEQBAAABIAAAAgAAAFABAAAGIAAAAQAAAIgBAAABEAAAAQAA" + 55*795d594fSAndroid Build Coastguard Worker "AJgBAAACIAAAFAAAAJ4BAAADIAAAAgAAAKwCAAAEIAAAAgAAALcCAAAAIAAAAQAAAMYCAAAAEAAA" + 56*795d594fSAndroid Build Coastguard Worker "AQAAANQCAAA="); 57*795d594fSAndroid Build Coastguard Worker notifyBreakpointReached(Thread thr, Executable e, long loc)58*795d594fSAndroid Build Coastguard Worker public static void notifyBreakpointReached(Thread thr, Executable e, long loc) { 59*795d594fSAndroid Build Coastguard Worker System.out.println( 60*795d594fSAndroid Build Coastguard Worker "\tBreakpoint reached: " + e + " @ line=" + Breakpoint.locationToLine(e, loc)); 61*795d594fSAndroid Build Coastguard Worker } 62*795d594fSAndroid Build Coastguard Worker check(boolean b, String msg)63*795d594fSAndroid Build Coastguard Worker public static void check(boolean b, String msg) { 64*795d594fSAndroid Build Coastguard Worker if (!b) { 65*795d594fSAndroid Build Coastguard Worker throw new Error("FAILED: " + msg); 66*795d594fSAndroid Build Coastguard Worker } 67*795d594fSAndroid Build Coastguard Worker } main(String[] args)68*795d594fSAndroid Build Coastguard Worker public static void main(String[] args) throws Exception { 69*795d594fSAndroid Build Coastguard Worker System.loadLibrary(args[0]); 70*795d594fSAndroid Build Coastguard Worker // Set up breakpoints 71*795d594fSAndroid Build Coastguard Worker Breakpoint.stopBreakpointWatch(Thread.currentThread()); 72*795d594fSAndroid Build Coastguard Worker Breakpoint.startBreakpointWatch( 73*795d594fSAndroid Build Coastguard Worker Main.class, 74*795d594fSAndroid Build Coastguard Worker Main.class.getDeclaredMethod( 75*795d594fSAndroid Build Coastguard Worker "notifyBreakpointReached", Thread.class, Executable.class, Long.TYPE), 76*795d594fSAndroid Build Coastguard Worker Thread.currentThread()); 77*795d594fSAndroid Build Coastguard Worker 78*795d594fSAndroid Build Coastguard Worker Method targetMethod = Transform.class.getDeclaredMethod("sayHi"); 79*795d594fSAndroid Build Coastguard Worker Transform t = new Transform(); 80*795d594fSAndroid Build Coastguard Worker check(isInterpretOnly() || !isMethodDeoptimized(targetMethod), 81*795d594fSAndroid Build Coastguard Worker "method should not be deoptimized"); 82*795d594fSAndroid Build Coastguard Worker t.sayHi(); 83*795d594fSAndroid Build Coastguard Worker 84*795d594fSAndroid Build Coastguard Worker // Set a breakpoint at the start of the function. 85*795d594fSAndroid Build Coastguard Worker Breakpoint.setBreakpoint(targetMethod, 0); 86*795d594fSAndroid Build Coastguard Worker check(isInterpretOnly() || isMethodDeoptimized(targetMethod), 87*795d594fSAndroid Build Coastguard Worker "method should be deoptimized"); 88*795d594fSAndroid Build Coastguard Worker t.sayHi(); 89*795d594fSAndroid Build Coastguard Worker 90*795d594fSAndroid Build Coastguard Worker System.out.println("Redefining transform!"); 91*795d594fSAndroid Build Coastguard Worker Redefinition.doCommonClassRedefinition(Transform.class, new byte[0], DEX_BYTES); 92*795d594fSAndroid Build Coastguard Worker check(isInterpretOnly() || !isMethodDeoptimized(targetMethod), 93*795d594fSAndroid Build Coastguard Worker "method should not be deoptimized"); 94*795d594fSAndroid Build Coastguard Worker t.sayHi(); 95*795d594fSAndroid Build Coastguard Worker 96*795d594fSAndroid Build Coastguard Worker Breakpoint.setBreakpoint(targetMethod, 0); 97*795d594fSAndroid Build Coastguard Worker check(isInterpretOnly() || isMethodDeoptimized(targetMethod), 98*795d594fSAndroid Build Coastguard Worker "method should be deoptimized"); 99*795d594fSAndroid Build Coastguard Worker t.sayHi(); 100*795d594fSAndroid Build Coastguard Worker } 101*795d594fSAndroid Build Coastguard Worker isMethodDeoptimized(Method m)102*795d594fSAndroid Build Coastguard Worker static native boolean isMethodDeoptimized(Method m); isInterpretOnly()103*795d594fSAndroid Build Coastguard Worker static native boolean isInterpretOnly(); 104*795d594fSAndroid Build Coastguard Worker } 105