1*795d594fSAndroid Build Coastguard Worker /* 2*795d594fSAndroid Build Coastguard Worker * Copyright (C) 2016 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 RacyMisbehavingLoader extends DefiningLoader { 18*795d594fSAndroid Build Coastguard Worker static { 19*795d594fSAndroid Build Coastguard Worker // For JVM, register as parallel capable. 20*795d594fSAndroid Build Coastguard Worker // Android treats all class loaders as parallel capable and makes this a no-op. registerAsParallelCapable()21*795d594fSAndroid Build Coastguard Worker registerAsParallelCapable(); 22*795d594fSAndroid Build Coastguard Worker } 23*795d594fSAndroid Build Coastguard Worker 24*795d594fSAndroid Build Coastguard Worker private Object lock = new Object(); 25*795d594fSAndroid Build Coastguard Worker private int index = 0; 26*795d594fSAndroid Build Coastguard Worker private int count; 27*795d594fSAndroid Build Coastguard Worker private boolean throw_error; 28*795d594fSAndroid Build Coastguard Worker 29*795d594fSAndroid Build Coastguard Worker private DefiningLoader[] defining_loaders; 30*795d594fSAndroid Build Coastguard Worker RacyMisbehavingLoader(ClassLoader parent, int count, boolean throw_error)31*795d594fSAndroid Build Coastguard Worker public RacyMisbehavingLoader(ClassLoader parent, int count, boolean throw_error) { 32*795d594fSAndroid Build Coastguard Worker super(parent); 33*795d594fSAndroid Build Coastguard Worker this.count = count; 34*795d594fSAndroid Build Coastguard Worker this.throw_error = throw_error; 35*795d594fSAndroid Build Coastguard Worker defining_loaders = new DefiningLoader[2]; 36*795d594fSAndroid Build Coastguard Worker for (int i = 0; i != defining_loaders.length; ++i) { 37*795d594fSAndroid Build Coastguard Worker defining_loaders[i] = new DefiningLoader(parent); 38*795d594fSAndroid Build Coastguard Worker } 39*795d594fSAndroid Build Coastguard Worker } 40*795d594fSAndroid Build Coastguard Worker reportAfterLoading()41*795d594fSAndroid Build Coastguard Worker public void reportAfterLoading() { 42*795d594fSAndroid Build Coastguard Worker synchronized (lock) { 43*795d594fSAndroid Build Coastguard Worker ++index; 44*795d594fSAndroid Build Coastguard Worker if (index == 2 * count) { 45*795d594fSAndroid Build Coastguard Worker lock.notifyAll(); 46*795d594fSAndroid Build Coastguard Worker } 47*795d594fSAndroid Build Coastguard Worker } 48*795d594fSAndroid Build Coastguard Worker } 49*795d594fSAndroid Build Coastguard Worker findClass(String name)50*795d594fSAndroid Build Coastguard Worker protected Class<?> findClass(String name) throws ClassNotFoundException 51*795d594fSAndroid Build Coastguard Worker { 52*795d594fSAndroid Build Coastguard Worker if (name.equals("Test")) { 53*795d594fSAndroid Build Coastguard Worker throw new Error("Unexpected RacyLoader.findClass(\"" + name + "\")"); 54*795d594fSAndroid Build Coastguard Worker } 55*795d594fSAndroid Build Coastguard Worker return super.findClass(name); 56*795d594fSAndroid Build Coastguard Worker } 57*795d594fSAndroid Build Coastguard Worker loadClass(String name, boolean resolve)58*795d594fSAndroid Build Coastguard Worker protected Class<?> loadClass(String name, boolean resolve) 59*795d594fSAndroid Build Coastguard Worker throws ClassNotFoundException 60*795d594fSAndroid Build Coastguard Worker { 61*795d594fSAndroid Build Coastguard Worker if (name.equals("Test")) { 62*795d594fSAndroid Build Coastguard Worker int my_index = syncWithOtherInstances(count); 63*795d594fSAndroid Build Coastguard Worker Class<?> result; 64*795d594fSAndroid Build Coastguard Worker if ((my_index & 1) == 0) { 65*795d594fSAndroid Build Coastguard Worker // Do not delay loading the correct class. 66*795d594fSAndroid Build Coastguard Worker result = defining_loaders[my_index & 1].loadClass(name, resolve); 67*795d594fSAndroid Build Coastguard Worker } else { 68*795d594fSAndroid Build Coastguard Worker // Delay loading the wrong class. 69*795d594fSAndroid Build Coastguard Worker syncWithOtherInstances(2 * count); 70*795d594fSAndroid Build Coastguard Worker if (throw_error) { 71*795d594fSAndroid Build Coastguard Worker throw new Error("RacyMisbehavingLoader throw_error=true"); 72*795d594fSAndroid Build Coastguard Worker } 73*795d594fSAndroid Build Coastguard Worker result = defining_loaders[my_index & 1].loadClass("Test3", resolve); 74*795d594fSAndroid Build Coastguard Worker } 75*795d594fSAndroid Build Coastguard Worker return result; 76*795d594fSAndroid Build Coastguard Worker } 77*795d594fSAndroid Build Coastguard Worker return super.loadClass(name, resolve); 78*795d594fSAndroid Build Coastguard Worker } 79*795d594fSAndroid Build Coastguard Worker syncWithOtherInstances(int limit)80*795d594fSAndroid Build Coastguard Worker private int syncWithOtherInstances(int limit) { 81*795d594fSAndroid Build Coastguard Worker int my_index; 82*795d594fSAndroid Build Coastguard Worker synchronized (lock) { 83*795d594fSAndroid Build Coastguard Worker my_index = index; 84*795d594fSAndroid Build Coastguard Worker ++index; 85*795d594fSAndroid Build Coastguard Worker if (index != limit) { 86*795d594fSAndroid Build Coastguard Worker do { 87*795d594fSAndroid Build Coastguard Worker try { 88*795d594fSAndroid Build Coastguard Worker lock.wait(); 89*795d594fSAndroid Build Coastguard Worker } catch (InterruptedException ie) { 90*795d594fSAndroid Build Coastguard Worker throw new Error(ie); 91*795d594fSAndroid Build Coastguard Worker } 92*795d594fSAndroid Build Coastguard Worker } while (index < limit); 93*795d594fSAndroid Build Coastguard Worker } else { 94*795d594fSAndroid Build Coastguard Worker lock.notifyAll(); 95*795d594fSAndroid Build Coastguard Worker } 96*795d594fSAndroid Build Coastguard Worker } 97*795d594fSAndroid Build Coastguard Worker return my_index; 98*795d594fSAndroid Build Coastguard Worker } 99*795d594fSAndroid Build Coastguard Worker } 100