| /* |
| * Copyright (C) 2016 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. |
| */ |
| |
| import java.lang.reflect.*; |
| import java.util.Base64; |
| |
| class Main { |
| public static String TEST_NAME = "935-non-retransformable"; |
| |
| /** |
| * base64 encoded class/dex file for |
| * class Transform { |
| * public void sayHi() { |
| * System.out.println("Hello"); |
| * } |
| * public void sayGoodbye() { |
| * System.out.println("Goodbye"); |
| * } |
| * } |
| */ |
| private static final byte[] CLASS_BYTES = Base64.getDecoder().decode( |
| "yv66vgAAADQAHwoABwAQCQARABIIABMKABQAFQgAFgcAFwcAGAEABjxpbml0PgEAAygpVgEABENv" + |
| "ZGUBAA9MaW5lTnVtYmVyVGFibGUBAAVzYXlIaQEACnNheUdvb2RieWUBAApTb3VyY2VGaWxlAQAO" + |
| "VHJhbnNmb3JtLmphdmEMAAgACQcAGQwAGgAbAQAFSGVsbG8HABwMAB0AHgEAB0dvb2RieWUBAAlU" + |
| "cmFuc2Zvcm0BABBqYXZhL2xhbmcvT2JqZWN0AQAQamF2YS9sYW5nL1N5c3RlbQEAA291dAEAFUxq" + |
| "YXZhL2lvL1ByaW50U3RyZWFtOwEAE2phdmEvaW8vUHJpbnRTdHJlYW0BAAdwcmludGxuAQAVKExq" + |
| "YXZhL2xhbmcvU3RyaW5nOylWACAABgAHAAAAAAADAAAACAAJAAEACgAAAB0AAQABAAAABSq3AAGx" + |
| "AAAAAQALAAAABgABAAAAAQABAAwACQABAAoAAAAlAAIAAQAAAAmyAAISA7YABLEAAAABAAsAAAAK" + |
| "AAIAAAADAAgABAABAA0ACQABAAoAAAAlAAIAAQAAAAmyAAISBbYABLEAAAABAAsAAAAKAAIAAAAG" + |
| "AAgABwABAA4AAAACAA8="); |
| private static final byte[] DEX_BYTES = Base64.getDecoder().decode( |
| "ZGV4CjAzNQDpaN+7jX/ZLl9Jr0HAEV7nqL1YDuakKakgAwAAcAAAAHhWNBIAAAAAAAAAAIACAAAQ" + |
| "AAAAcAAAAAYAAACwAAAAAgAAAMgAAAABAAAA4AAAAAUAAADoAAAAAQAAABABAADwAQAAMAEAAJYB" + |
| "AACeAQAApwEAAK4BAAC7AQAA0gEAAOYBAAD6AQAADgIAAB4CAAAhAgAAJQIAADkCAAA+AgAARwIA" + |
| "AFMCAAADAAAABAAAAAUAAAAGAAAABwAAAAkAAAAJAAAABQAAAAAAAAAKAAAABQAAAJABAAAEAAEA" + |
| "DAAAAAAAAAAAAAAAAAAAAA4AAAAAAAAADwAAAAEAAQANAAAAAgAAAAAAAAAAAAAAAAAAAAIAAAAA" + |
| "AAAACAAAAAAAAABrAgAAAAAAAAEAAQABAAAAWgIAAAQAAABwEAQAAAAOAAMAAQACAAAAXwIAAAkA" + |
| "AABiAAAAGwEBAAAAbiADABAADgAAAAMAAQACAAAAZQIAAAkAAABiAAAAGwECAAAAbiADABAADgAA" + |
| "AAEAAAADAAY8aW5pdD4AB0dvb2RieWUABUhlbGxvAAtMVHJhbnNmb3JtOwAVTGphdmEvaW8vUHJp" + |
| "bnRTdHJlYW07ABJMamF2YS9sYW5nL09iamVjdDsAEkxqYXZhL2xhbmcvU3RyaW5nOwASTGphdmEv" + |
| "bGFuZy9TeXN0ZW07AA5UcmFuc2Zvcm0uamF2YQABVgACVkwAEmVtaXR0ZXI6IGphY2stNC4yMgAD" + |
| "b3V0AAdwcmludGxuAApzYXlHb29kYnllAAVzYXlIaQABAAcOAAYABw6HAAMABw6HAAAAAQIAgIAE" + |
| "sAIBAcgCAQHsAgAAAA0AAAAAAAAAAQAAAAAAAAABAAAAEAAAAHAAAAACAAAABgAAALAAAAADAAAA" + |
| "AgAAAMgAAAAEAAAAAQAAAOAAAAAFAAAABQAAAOgAAAAGAAAAAQAAABABAAABIAAAAwAAADABAAAB" + |
| "EAAAAQAAAJABAAACIAAAEAAAAJYBAAADIAAAAwAAAFoCAAAAIAAAAQAAAGsCAAAAEAAAAQAAAIAC" + |
| "AAA="); |
| |
| |
| public static ClassLoader getClassLoaderFor(String location) throws Exception { |
| try { |
| Class<?> class_loader_class = Class.forName("dalvik.system.PathClassLoader"); |
| Constructor<?> ctor = class_loader_class.getConstructor(String.class, ClassLoader.class); |
| /* on Dalvik, this is a DexFile; otherwise, it's null */ |
| return (ClassLoader)ctor.newInstance(location + "/" + TEST_NAME + "-ex.jar", |
| Main.class.getClassLoader()); |
| } catch (ClassNotFoundException e) { |
| // Running on RI. Use URLClassLoader. |
| return new java.net.URLClassLoader( |
| new java.net.URL[] { new java.net.URL("file://" + location + "/classes-ex/") }); |
| } |
| } |
| |
| public static void main(String[] args) { |
| art.Main.bindAgentJNIForClass(Main.class); |
| setPopRetransformations(false); |
| addCommonTransformationResult("Transform", CLASS_BYTES, DEX_BYTES); |
| enableCommonRetransformation(true); |
| try { |
| /* this is the "alternate" DEX/Jar file */ |
| ClassLoader new_loader = getClassLoaderFor(System.getenv("DEX_LOCATION")); |
| Class<?> klass = (Class<?>)new_loader.loadClass("TestMain"); |
| if (klass == null) { |
| throw new AssertionError("loadClass failed"); |
| } |
| Method run_test = klass.getMethod("runTest"); |
| run_test.invoke(null); |
| |
| // Remove the original transformation. It has been used by now. |
| popTransformationFor("Transform"); |
| // Make sure we don't get called for transformation again. |
| addCommonTransformationResult("Transform", new byte[0], new byte[0]); |
| doCommonClassRetransformation(new_loader.loadClass("Transform")); |
| run_test.invoke(null); |
| } catch (Exception e) { |
| System.out.println(e.toString()); |
| e.printStackTrace(); |
| } |
| } |
| |
| // Transforms the class |
| private static native void doCommonClassRetransformation(Class<?>... classes); |
| private static native void enableCommonRetransformation(boolean enable); |
| private static native void addCommonTransformationResult(String target_name, |
| byte[] class_bytes, |
| byte[] dex_bytes); |
| private static native void setPopRetransformations(boolean should_pop); |
| private static native void popTransformationFor(String target_name); |
| } |