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 RacyLoader 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 28*795d594fSAndroid Build Coastguard Worker private DefiningLoader[] defining_loaders; 29*795d594fSAndroid Build Coastguard Worker RacyLoader(ClassLoader parent, int count)30*795d594fSAndroid Build Coastguard Worker public RacyLoader(ClassLoader parent, int count) { 31*795d594fSAndroid Build Coastguard Worker super(parent); 32*795d594fSAndroid Build Coastguard Worker this.count = count; 33*795d594fSAndroid Build Coastguard Worker defining_loaders = new DefiningLoader[2]; 34*795d594fSAndroid Build Coastguard Worker for (int i = 0; i != defining_loaders.length; ++i) { 35*795d594fSAndroid Build Coastguard Worker defining_loaders[i] = new DefiningLoader(parent); 36*795d594fSAndroid Build Coastguard Worker } 37*795d594fSAndroid Build Coastguard Worker } 38*795d594fSAndroid Build Coastguard Worker findClass(String name)39*795d594fSAndroid Build Coastguard Worker protected Class<?> findClass(String name) throws ClassNotFoundException 40*795d594fSAndroid Build Coastguard Worker { 41*795d594fSAndroid Build Coastguard Worker if (name.equals("Test") || name.equals("Test3")) { 42*795d594fSAndroid Build Coastguard Worker throw new Error("Unexpected RacyLoader.findClass(\"" + name + "\")"); 43*795d594fSAndroid Build Coastguard Worker } 44*795d594fSAndroid Build Coastguard Worker return super.findClass(name); 45*795d594fSAndroid Build Coastguard Worker } 46*795d594fSAndroid Build Coastguard Worker loadClass(String name, boolean resolve)47*795d594fSAndroid Build Coastguard Worker protected Class<?> loadClass(String name, boolean resolve) 48*795d594fSAndroid Build Coastguard Worker throws ClassNotFoundException 49*795d594fSAndroid Build Coastguard Worker { 50*795d594fSAndroid Build Coastguard Worker if (name.equals("Test") || name.equals("Test3")) { 51*795d594fSAndroid Build Coastguard Worker int my_index = syncWithOtherInstances(count); 52*795d594fSAndroid Build Coastguard Worker Class<?> result = defining_loaders[my_index & 1].loadClass(name, resolve); 53*795d594fSAndroid Build Coastguard Worker syncWithOtherInstances(2 * count); 54*795d594fSAndroid Build Coastguard Worker return result; 55*795d594fSAndroid Build Coastguard Worker } 56*795d594fSAndroid Build Coastguard Worker return super.loadClass(name, resolve); 57*795d594fSAndroid Build Coastguard Worker } 58*795d594fSAndroid Build Coastguard Worker syncWithOtherInstances(int limit)59*795d594fSAndroid Build Coastguard Worker private int syncWithOtherInstances(int limit) { 60*795d594fSAndroid Build Coastguard Worker int my_index; 61*795d594fSAndroid Build Coastguard Worker synchronized (lock) { 62*795d594fSAndroid Build Coastguard Worker my_index = index; 63*795d594fSAndroid Build Coastguard Worker ++index; 64*795d594fSAndroid Build Coastguard Worker if (index != limit) { 65*795d594fSAndroid Build Coastguard Worker do { 66*795d594fSAndroid Build Coastguard Worker try { 67*795d594fSAndroid Build Coastguard Worker lock.wait(); 68*795d594fSAndroid Build Coastguard Worker } catch (InterruptedException ie) { 69*795d594fSAndroid Build Coastguard Worker throw new Error(ie); 70*795d594fSAndroid Build Coastguard Worker } 71*795d594fSAndroid Build Coastguard Worker } while (index < limit); 72*795d594fSAndroid Build Coastguard Worker } else { 73*795d594fSAndroid Build Coastguard Worker lock.notifyAll(); 74*795d594fSAndroid Build Coastguard Worker } 75*795d594fSAndroid Build Coastguard Worker } 76*795d594fSAndroid Build Coastguard Worker return my_index; 77*795d594fSAndroid Build Coastguard Worker } 78*795d594fSAndroid Build Coastguard Worker } 79