1*795d594fSAndroid Build Coastguard Worker /* 2*795d594fSAndroid Build Coastguard Worker * Copyright (C) 2019 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 art.Redefinition; 18*795d594fSAndroid Build Coastguard Worker import java.lang.invoke.*; 19*795d594fSAndroid Build Coastguard Worker import java.lang.reflect.Field; 20*795d594fSAndroid Build Coastguard Worker import java.util.Base64; 21*795d594fSAndroid Build Coastguard Worker import java.util.concurrent.CountDownLatch; 22*795d594fSAndroid Build Coastguard Worker 23*795d594fSAndroid Build Coastguard Worker public class Main { 24*795d594fSAndroid Build Coastguard Worker public static final class Transform { 25*795d594fSAndroid Build Coastguard Worker static { 26*795d594fSAndroid Build Coastguard Worker } 27*795d594fSAndroid Build Coastguard Worker 28*795d594fSAndroid Build Coastguard Worker public static Object foo = null; 29*795d594fSAndroid Build Coastguard Worker } 30*795d594fSAndroid Build Coastguard Worker 31*795d594fSAndroid Build Coastguard Worker /** 32*795d594fSAndroid Build Coastguard Worker * Base64 encoded dex bytes for: 33*795d594fSAndroid Build Coastguard Worker * 34*795d594fSAndroid Build Coastguard Worker * public static final class Transform { 35*795d594fSAndroid Build Coastguard Worker * static {} 36*795d594fSAndroid Build Coastguard Worker * public static Object bar = null; 37*795d594fSAndroid Build Coastguard Worker * public static Object foo = null; 38*795d594fSAndroid Build Coastguard Worker * } 39*795d594fSAndroid Build Coastguard Worker */ 40*795d594fSAndroid Build Coastguard Worker public static final byte[] DEX_BYTES = 41*795d594fSAndroid Build Coastguard Worker Base64.getDecoder() 42*795d594fSAndroid Build Coastguard Worker .decode( 43*795d594fSAndroid Build Coastguard Worker "ZGV4CjAzNQCjkRjcSr1RJO8FnnCjHV/8h6keJP/+P3WQAwAAcAAAAHhWNBIAAAAAAAAAANgCAAAQ" 44*795d594fSAndroid Build Coastguard Worker + "AAAAcAAAAAYAAACwAAAAAQAAAMgAAAACAAAA1AAAAAMAAADkAAAAAQAAAPwAAAB0AgAAHAEAAFwB" 45*795d594fSAndroid Build Coastguard Worker + "AABmAQAAbgEAAIABAACIAQAArAEAAMwBAADgAQAA6wEAAPYBAAD5AQAABgIAAAsCAAAQAgAAFgIA" 46*795d594fSAndroid Build Coastguard Worker + "AB0CAAACAAAAAwAAAAQAAAAFAAAABgAAAAkAAAAJAAAABQAAAAAAAAAAAAQACwAAAAAABAAMAAAA" 47*795d594fSAndroid Build Coastguard Worker + "AAAAAAAAAAAAAAAAAQAAAAQAAAABAAAAAAAAABEAAAAEAAAAAAAAAAcAAADIAgAApAIAAAAAAAAB" 48*795d594fSAndroid Build Coastguard Worker + "AAAAAAAAAFABAAAGAAAAEgBpAAAAaQABAA4AAQABAAEAAABVAQAABAAAAHAQAgAAAA4ABwAOPAAF" 49*795d594fSAndroid Build Coastguard Worker + "AA4AAAAACDxjbGluaXQ+AAY8aW5pdD4AEExNYWluJFRyYW5zZm9ybTsABkxNYWluOwAiTGRhbHZp" 50*795d594fSAndroid Build Coastguard Worker + "ay9hbm5vdGF0aW9uL0VuY2xvc2luZ0NsYXNzOwAeTGRhbHZpay9hbm5vdGF0aW9uL0lubmVyQ2xh" 51*795d594fSAndroid Build Coastguard Worker + "c3M7ABJMamF2YS9sYW5nL09iamVjdDsACU1haW4uamF2YQAJVHJhbnNmb3JtAAFWAAthY2Nlc3NG" 52*795d594fSAndroid Build Coastguard Worker + "bGFncwADYmFyAANmb28ABG5hbWUABXZhbHVlAHZ+fkQ4eyJjb21waWxhdGlvbi1tb2RlIjoiZGVi" 53*795d594fSAndroid Build Coastguard Worker + "dWciLCJtaW4tYXBpIjoxLCJzaGEtMSI6IjI4YmNlZjUwYWM4NTk3Y2YyMmU4OTJiMWJjM2EzYjky" 54*795d594fSAndroid Build Coastguard Worker + "Yjc0ZTcwZTkiLCJ2ZXJzaW9uIjoiMS42LjMyLWRldiJ9AAICAQ4YAQIDAgoEGQ0XCAIAAgAACQEJ" 55*795d594fSAndroid Build Coastguard Worker + "AIiABJwCAYGABLgCAAAAAAIAAACVAgAAmwIAALwCAAAAAAAAAAAAAAAAAAAPAAAAAAAAAAEAAAAA" 56*795d594fSAndroid Build Coastguard Worker + "AAAAAQAAABAAAABwAAAAAgAAAAYAAACwAAAAAwAAAAEAAADIAAAABAAAAAIAAADUAAAABQAAAAMA" 57*795d594fSAndroid Build Coastguard Worker + "AADkAAAABgAAAAEAAAD8AAAAASAAAAIAAAAcAQAAAyAAAAIAAABQAQAAAiAAABAAAABcAQAABCAA" 58*795d594fSAndroid Build Coastguard Worker + "AAIAAACVAgAAACAAAAEAAACkAgAAAxAAAAIAAAC4AgAABiAAAAEAAADIAgAAABAAAAEAAADYAgAA"); 59*795d594fSAndroid Build Coastguard Worker assertEquals(Object a, Object b)60*795d594fSAndroid Build Coastguard Worker public static void assertEquals(Object a, Object b) { 61*795d594fSAndroid Build Coastguard Worker if (a != b) { 62*795d594fSAndroid Build Coastguard Worker throw new Error("Expected " + b + ", got " + a); 63*795d594fSAndroid Build Coastguard Worker } 64*795d594fSAndroid Build Coastguard Worker } 65*795d594fSAndroid Build Coastguard Worker assertAllEquals(Object[] a, Object b)66*795d594fSAndroid Build Coastguard Worker public static void assertAllEquals(Object[] a, Object b) { 67*795d594fSAndroid Build Coastguard Worker boolean failed = false; 68*795d594fSAndroid Build Coastguard Worker String msg = ""; 69*795d594fSAndroid Build Coastguard Worker for (int i = 0; i < a.length; i++) { 70*795d594fSAndroid Build Coastguard Worker if (a[i] != b) { 71*795d594fSAndroid Build Coastguard Worker failed = true; 72*795d594fSAndroid Build Coastguard Worker msg += "Expected " + b + ", got a[" + i + "] (" + a[i] + "), "; 73*795d594fSAndroid Build Coastguard Worker } 74*795d594fSAndroid Build Coastguard Worker } 75*795d594fSAndroid Build Coastguard Worker if (failed) { 76*795d594fSAndroid Build Coastguard Worker throw new Error(msg); 77*795d594fSAndroid Build Coastguard Worker } 78*795d594fSAndroid Build Coastguard Worker } 79*795d594fSAndroid Build Coastguard Worker main(String[] args)80*795d594fSAndroid Build Coastguard Worker public static void main(String[] args) throws Exception, Throwable { 81*795d594fSAndroid Build Coastguard Worker System.loadLibrary(args[0]); 82*795d594fSAndroid Build Coastguard Worker Field f = Transform.class.getDeclaredField("foo"); 83*795d594fSAndroid Build Coastguard Worker Transform.foo = "THIS IS A FOO VALUE"; 84*795d594fSAndroid Build Coastguard Worker assertEquals(f.get(null), Transform.foo); 85*795d594fSAndroid Build Coastguard Worker final int num_threads = 10; 86*795d594fSAndroid Build Coastguard Worker Object[] results = new Object[num_threads]; 87*795d594fSAndroid Build Coastguard Worker Thread[] threads = new Thread[num_threads]; 88*795d594fSAndroid Build Coastguard Worker CountDownLatch start_latch = new CountDownLatch(num_threads); 89*795d594fSAndroid Build Coastguard Worker CountDownLatch continue_latch = new CountDownLatch(1); 90*795d594fSAndroid Build Coastguard Worker for (int i = 0; i < num_threads; i++) { 91*795d594fSAndroid Build Coastguard Worker final int id = i; 92*795d594fSAndroid Build Coastguard Worker threads[id] = 93*795d594fSAndroid Build Coastguard Worker new Thread( 94*795d594fSAndroid Build Coastguard Worker () -> { 95*795d594fSAndroid Build Coastguard Worker try { 96*795d594fSAndroid Build Coastguard Worker MethodHandle mh = 97*795d594fSAndroid Build Coastguard Worker NativeFieldScopeCheck( 98*795d594fSAndroid Build Coastguard Worker f, 99*795d594fSAndroid Build Coastguard Worker () -> { 100*795d594fSAndroid Build Coastguard Worker try { 101*795d594fSAndroid Build Coastguard Worker start_latch.countDown(); 102*795d594fSAndroid Build Coastguard Worker continue_latch.await(); 103*795d594fSAndroid Build Coastguard Worker } catch (Exception e) { 104*795d594fSAndroid Build Coastguard Worker throw new Error("failed!", e); 105*795d594fSAndroid Build Coastguard Worker } 106*795d594fSAndroid Build Coastguard Worker }); 107*795d594fSAndroid Build Coastguard Worker results[id] = mh.invokeExact(); 108*795d594fSAndroid Build Coastguard Worker } catch (Throwable t) { 109*795d594fSAndroid Build Coastguard Worker throw new Error("Failed", t); 110*795d594fSAndroid Build Coastguard Worker } 111*795d594fSAndroid Build Coastguard Worker }, 112*795d594fSAndroid Build Coastguard Worker "Target thread " + id); 113*795d594fSAndroid Build Coastguard Worker threads[id].start(); 114*795d594fSAndroid Build Coastguard Worker } 115*795d594fSAndroid Build Coastguard Worker start_latch.await(); 116*795d594fSAndroid Build Coastguard Worker Redefinition.doCommonStructuralClassRedefinition(Transform.class, DEX_BYTES); 117*795d594fSAndroid Build Coastguard Worker continue_latch.countDown(); 118*795d594fSAndroid Build Coastguard Worker for (Thread t : threads) { 119*795d594fSAndroid Build Coastguard Worker t.join(); 120*795d594fSAndroid Build Coastguard Worker } 121*795d594fSAndroid Build Coastguard Worker assertAllEquals(results, Transform.foo); 122*795d594fSAndroid Build Coastguard Worker } 123*795d594fSAndroid Build Coastguard Worker 124*795d594fSAndroid Build Coastguard Worker // Hold the field as a ArtField, run the 'test' function, turn the ArtField into a MethodHandle 125*795d594fSAndroid Build Coastguard Worker // directly and return that. NativeFieldScopeCheck(Field in, Runnable test)126*795d594fSAndroid Build Coastguard Worker public static native MethodHandle NativeFieldScopeCheck(Field in, Runnable test); 127*795d594fSAndroid Build Coastguard Worker } 128