WALA/com.ibm.wala.core/src/com/ibm/wala/ipa/callgraph/propagation/ConcreteTypeKey.java

128 lines
4.0 KiB
Java

/*******************************************************************************
* Copyright (c) 2002 - 2006 IBM Corporation.
* All rights reserved. This program and the accompanying materials
* are made available under the terms of the Eclipse Public License v1.0
* which accompanies this distribution, and is available at
* http://www.eclipse.org/legal/epl-v10.html
*
* Contributors:
* IBM Corporation - initial API and implementation
*******************************************************************************/
package com.ibm.wala.ipa.callgraph.propagation;
import java.util.Collection;
import java.util.Iterator;
import java.util.function.Function;
import java.util.function.Predicate;
import com.ibm.wala.classLoader.IClass;
import com.ibm.wala.classLoader.NewSiteReference;
import com.ibm.wala.ipa.callgraph.CGNode;
import com.ibm.wala.ipa.callgraph.CallGraph;
import com.ibm.wala.ipa.cha.IClassHierarchy;
import com.ibm.wala.ssa.SSAInstruction;
import com.ibm.wala.types.TypeReference;
import com.ibm.wala.util.collections.ComposedIterator;
import com.ibm.wala.util.collections.FilterIterator;
import com.ibm.wala.util.collections.MapIterator;
import com.ibm.wala.util.collections.Pair;
import com.ibm.wala.util.debug.Assertions;
/**
* An instance key which represents a unique set for each concrete type.
*/
public final class ConcreteTypeKey implements InstanceKey {
private final IClass type;
public ConcreteTypeKey(IClass type) {
if (type == null) {
throw new IllegalArgumentException("type is null");
}
if (type.isInterface()) {
Assertions.UNREACHABLE("unexpected interface: " + type);
}
this.type = type;
}
@Override
public boolean equals(Object obj) {
if (obj instanceof ConcreteTypeKey) {
ConcreteTypeKey other = (ConcreteTypeKey) obj;
return type.equals(other.type);
} else {
return false;
}
}
@Override
public int hashCode() {
return 461 * type.hashCode();
}
@Override
public String toString() {
return "[" + type + "]";
}
public IClass getType() {
return type;
}
/*
* @see com.ibm.wala.ipa.callgraph.propagation.InstanceKey#getConcreteType()
*/
@Override
public IClass getConcreteType() {
return type;
}
/**
* @param pei a PEI instruction
* @param cha governing class hierarchy
* @return a set of ConcreteTypeKeys that represent the exceptions the PEI may throw.
* @throws IllegalArgumentException if pei is null
*/
public static InstanceKey[] getInstanceKeysForPEI(SSAInstruction pei, IClassHierarchy cha) {
if (pei == null) {
throw new IllegalArgumentException("pei is null");
}
Collection<TypeReference> types = pei.getExceptionTypes();
// TODO: institute a cache?
if (types == null) {
return null;
}
InstanceKey[] result = new InstanceKey[types.size()];
int i = 0;
for (TypeReference type : types) {
assert type != null;
IClass klass = cha.lookupClass(type);
result[i++] = new ConcreteTypeKey(klass);
}
return result;
}
@Override
public Iterator<Pair<CGNode, NewSiteReference>> getCreationSites(CallGraph CG) {
return new ComposedIterator<CGNode, Pair<CGNode, NewSiteReference>>(CG.iterator()) {
@Override
public Iterator<? extends Pair<CGNode, NewSiteReference>> makeInner(final CGNode outer) {
return new MapIterator<NewSiteReference, Pair<CGNode, NewSiteReference>>(
new FilterIterator<NewSiteReference>(
outer.iterateNewSites(),
new Predicate<NewSiteReference>() {
@Override public boolean test(NewSiteReference o) {
return o.getDeclaredType().equals(type.getReference());
}
}
),
new Function<NewSiteReference, Pair<CGNode, NewSiteReference>>() {
@Override
public Pair<CGNode, NewSiteReference> apply(NewSiteReference object) {
return Pair.make(outer, object);
}
});
}
};
}
}