/* * 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. */ public class RacyMisbehavingLoader extends DefiningLoader { static { // For JVM, register as parallel capable. // Android treats all class loaders as parallel capable and makes this a no-op. registerAsParallelCapable(); } private Object lock = new Object(); private int index = 0; private int count; private boolean throw_error; private DefiningLoader[] defining_loaders; public RacyMisbehavingLoader(ClassLoader parent, int count, boolean throw_error) { super(parent); this.count = count; this.throw_error = throw_error; defining_loaders = new DefiningLoader[2]; for (int i = 0; i != defining_loaders.length; ++i) { defining_loaders[i] = new DefiningLoader(parent); } } public void reportAfterLoading() { synchronized (lock) { ++index; if (index == 2 * count) { lock.notifyAll(); } } } protected Class findClass(String name) throws ClassNotFoundException { if (name.equals("Test")) { throw new Error("Unexpected RacyLoader.findClass(\"" + name + "\")"); } return super.findClass(name); } protected Class loadClass(String name, boolean resolve) throws ClassNotFoundException { if (name.equals("Test")) { int my_index = syncWithOtherInstances(count); Class result; if ((my_index & 1) == 0) { // Do not delay loading the correct class. result = defining_loaders[my_index & 1].loadClass(name, resolve); } else { // Delay loading the wrong class. syncWithOtherInstances(2 * count); if (throw_error) { throw new Error("RacyMisbehavingLoader throw_error=true"); } result = defining_loaders[my_index & 1].loadClass("Test3", resolve); } return result; } return super.loadClass(name, resolve); } private int syncWithOtherInstances(int limit) { int my_index; synchronized (lock) { my_index = index; ++index; if (index != limit) { do { try { lock.wait(); } catch (InterruptedException ie) { throw new Error(ie); } } while (index < limit); } else { lock.notifyAll(); } } return my_index; } }