blob: 6d7d1ed44b2f147d60e3d6bcf7cc02c73568690c [file] [log] [blame]
/*
* Copyright 2000-2014 JetBrains s.r.o.
*
* 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.
*/
package com.intellij.psi.impl.source.tree.injected;
import com.intellij.util.containers.ConcurrentHashMap;
import org.jetbrains.annotations.NotNull;
import org.jetbrains.annotations.Nullable;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;
import java.util.Map;
public class ClassMapCachingNulls<T> {
private final Map<Class, T[]> myBackingMap;
private final T[] myEmptyArray;
private final Map<Class, T[]> myMap = new ConcurrentHashMap<Class, T[]>();
public ClassMapCachingNulls(@NotNull Map<Class, T[]> backingMap, T[] emptyArray) {
myBackingMap = backingMap;
myEmptyArray = emptyArray;
}
@Nullable
public T[] get(Class aClass) {
T[] value = myMap.get(aClass);
if (value != null) {
if (value == myEmptyArray) {
return null;
}
else {
assert value.length != 0;
return value;
}
}
List<T> result = getFromBackingMap(aClass);
return cache(aClass, result);
}
private T[] cache(Class aClass, List<T> result) {
T[] value;
if (result == null) {
myMap.put(aClass, myEmptyArray);
value = null;
}
else {
assert !result.isEmpty();
value = result.toArray(myEmptyArray);
myMap.put(aClass, value);
}
return value;
}
private List<T> getFromBackingMap(Class aClass) {
T[] value = myBackingMap.get(aClass);
List<T> result = null;
if (value != null) {
assert value.length != 0;
result = new ArrayList<T>(Arrays.asList(value));
}
for (final Class aClass1 : aClass.getInterfaces()) {
result = addFromUpper(result, aClass1);
}
final Class superclass = aClass.getSuperclass();
if (superclass != null) {
result = addFromUpper(result, superclass);
}
return result;
}
private List<T> addFromUpper(List<T> value, Class superclass) {
T[] fromUpper = get(superclass);
if (fromUpper != null) {
assert fromUpper.length != 0;
if (value == null) {
value = new ArrayList<T>(fromUpper.length);
}
for (T t : fromUpper) {
if (!value.contains(t)) {
value.add(t);
}
}
assert !value.isEmpty();
}
return value;
}
public Map<Class, T[]> getBackingMap() {
return myBackingMap;
}
public void clearCache() {
myMap.clear();
}
}