soot.jimple.infoflow.sourcesSinks.definitions.MethodSourceSinkDefinition Maven / Gradle / Ivy
package soot.jimple.infoflow.sourcesSinks.definitions;
import java.util.Arrays;
import java.util.Collection;
import java.util.Collections;
import java.util.HashSet;
import java.util.Set;
import soot.jimple.infoflow.data.SootMethodAndClass;
/**
* A class to handle all access paths of sources and sinks for a certain method.
*
* @author Daniel Magin
* @author Steven Arzt
*
*/
public class MethodSourceSinkDefinition extends AbstractSourceSinkDefinition
implements IAccessPathBasedSourceSinkDefinition {
private static MethodSourceSinkDefinition BASE_OBJ_SOURCE;
private static MethodSourceSinkDefinition BASE_OBJ_SINK;
private static MethodSourceSinkDefinition[] PARAM_OBJ_SOURCE = new MethodSourceSinkDefinition[5];
protected final SootMethodAndClass method;
protected final CallType callType;
protected Set baseObjects;
protected Set[] parameters;
protected Set returnValues;
/**
* Enumeration containing the different types of method invocations that can be
* defined as sources or sinks
*
* @author Steven Arzt
*
*/
public enum CallType {
/**
* The app calls the method
*/
MethodCall,
/**
* The method is a callback that is invoked by the Android operating system
*/
Callback,
/**
* All return values in the method are considered (only supported as sink)
*/
Return
}
/**
* Creates a new instance of the {@link MethodSourceSinkDefinition} class
* without a method. This constructor is intended to be used for sources that
* arise from UI elements or other programming constructs that are not directly
* associated with a single method being called.
*/
public MethodSourceSinkDefinition(Set baseObjects, Set[] parameters,
Set returnValues, CallType callType) {
this(null, baseObjects, parameters, returnValues, callType);
}
/**
* Creates a new instance of the {@link MethodSourceSinkDefinition} class
*
* @param am The method for which this object defines sources and sinks
*/
public MethodSourceSinkDefinition(SootMethodAndClass am) {
this(am, null, null, null, CallType.MethodCall);
}
/**
* Creates a new instance of the {@link MethodSourceSinkDefinition} class
*
* @param am The method for which this object defines sources and sinks
* @param callType The type of calls to define as sources or sinks
*/
public MethodSourceSinkDefinition(SootMethodAndClass am, CallType callType) {
this(am, null, null, null, callType);
}
/**
* Creates a new instance of the MethodSourceSinkDefinition class
*
* @param am The method for which this object defines sources and
* sinks
* @param baseObjects The source and sink definitions for the base object on
* which a method of this class is invoked
* @param parameters The source and sink definitions for parameters of the
* current method
* @param callType The type of calls to define as sources or sinks
* @param returnValues The source definitions for the return value of the
* current method
*/
public MethodSourceSinkDefinition(SootMethodAndClass am, Set baseObjects,
Set[] parameters, Set returnValues, CallType callType) {
this(am, baseObjects, parameters, returnValues, callType, null);
}
/**
* Creates a new instance of the MethodSourceSinkDefinition class
*
* @param am The method for which this object defines sources and
* sinks
* @param baseObjects The source and sink definitions for the base object on
* which a method of this class is invoked
* @param parameters The source and sink definitions for parameters of the
* current method
* @param returnValues The source definitions for the return value of the
* current method
* @param callType The type of calls to define as sources or sinks
* @param category The category to which this source or sink belongs
*/
public MethodSourceSinkDefinition(SootMethodAndClass am, Set baseObjects,
Set[] parameters, Set returnValues, CallType callType,
ISourceSinkCategory category) {
super(category);
this.method = am;
this.baseObjects = baseObjects == null || baseObjects.isEmpty() ? null : baseObjects;
this.parameters = parameters;
this.returnValues = returnValues == null || returnValues.isEmpty() ? null : returnValues;
this.callType = callType;
}
/**
* Gets the method for which this object defines sources and sinks
*
* @return The method for which this object defines sources and sinks
*/
public SootMethodAndClass getMethod() {
return this.method;
}
/**
* Gets the type of method invocations that are denoted by this source or sink
*
* @return The type of method invocations that are denoted by this source or
* sink
*/
public CallType getCallType() {
return this.callType;
}
/**
* Gets the source and sink definitions for the base object on which a method of
* this class is invoked
*
* @return The source and sink definitions for the base object
*/
public Set getBaseObjects() {
return this.baseObjects;
}
/**
* Gets the number of access paths defined as sources or sinks on base objects
*
* @return The number of access paths defined as sources or sinks on base
* objects
*/
public int getBaseObjectCount() {
return this.baseObjects == null ? 0 : this.baseObjects.size();
}
/**
* Gets the source and sink definitions for parameters of the current method
*
* @return The source and sink definitions for parameters
*/
public Set[] getParameters() {
return this.parameters;
}
/**
* Gets the number of access paths defined as sources or sinks on parameters
*
* @return The number of access paths defined as sources or sinks on parameters
*/
public int getParameterCount() {
if (this.parameters == null || this.parameters.length == 0)
return 0;
int cnt = 0;
for (Set apt : this.parameters)
cnt += apt.size();
return cnt;
}
/**
* Gets the source definitions for the return value of the current method
*
* @return The source definitions for the return value
*/
public Set getReturnValues() {
return this.returnValues;
}
/**
* Gets the number of access paths defined as sources or sinks on return values
*
* @return The number of access paths defined as sources or sinks on return
* values
*/
public int getReturnValueCount() {
return this.returnValues == null ? 0 : this.returnValues.size();
}
@Override
public boolean isEmpty() {
boolean parametersEmpty = true;
if (parameters != null)
for (Set paramSet : this.parameters)
if (paramSet != null && !paramSet.isEmpty()) {
parametersEmpty = false;
break;
}
return (baseObjects == null || baseObjects.isEmpty()) && parametersEmpty
&& (returnValues == null || returnValues.isEmpty());
}
@Override
public String toString() {
return method == null ? "" : method.getSignature();
}
@SuppressWarnings("unchecked")
@Override
public MethodSourceSinkDefinition getSourceOnlyDefinition() {
// Collect all base sources
Set baseSources = null;
if (baseObjects != null) {
baseSources = new HashSet<>(baseObjects.size());
for (AccessPathTuple apt : baseObjects)
if (apt.getSourceSinkType().isSource())
baseSources.add(apt);
}
// Collect all parameter sources
Set[] paramSources = null;
if (parameters != null && parameters.length > 0) {
paramSources = new Set[parameters.length];
for (int i = 0; i < parameters.length; i++) {
Set aptSet = parameters[i];
if (aptSet != null) {
Set thisParam = new HashSet<>(aptSet.size());
paramSources[i] = thisParam;
for (AccessPathTuple apt : aptSet)
if (apt.getSourceSinkType().isSource())
thisParam.add(apt);
}
}
}
// Collect all return sources
Set returnSources = null;
if (returnValues != null) {
returnSources = new HashSet<>(returnValues.size());
for (AccessPathTuple apt : returnValues)
if (apt.getSourceSinkType().isSource())
returnSources.add(apt);
}
MethodSourceSinkDefinition mssd = buildNewDefinition(baseSources, paramSources, returnSources);
return mssd;
}
@SuppressWarnings("unchecked")
@Override
public MethodSourceSinkDefinition getSinkOnlyDefinition() {
// Collect all base sinks
Set baseSinks = null;
if (baseObjects != null) {
baseSinks = new HashSet<>(baseObjects.size());
for (AccessPathTuple apt : baseObjects)
if (apt.getSourceSinkType().isSink())
baseSinks.add(apt);
}
// Collect all parameter sinks
Set[] paramSinks = null;
if (parameters != null) {
paramSinks = new Set[parameters.length];
for (int i = 0; i < parameters.length; i++) {
Set aptSet = parameters[i];
if (aptSet != null) {
Set thisParam = new HashSet<>(aptSet.size());
paramSinks[i] = thisParam;
for (AccessPathTuple apt : aptSet)
if (apt.getSourceSinkType().isSink())
thisParam.add(apt);
}
}
}
// Collect all return sinks
Set returnSinks = null;
if (returnValues != null) {
returnSinks = new HashSet<>(returnValues.size());
for (AccessPathTuple apt : returnValues)
if (apt.getSourceSinkType().isSink())
returnSinks.add(apt);
}
MethodSourceSinkDefinition mssd = buildNewDefinition(baseSinks, paramSinks, returnSinks);
return mssd;
}
/**
* Factory method for creating a new method-based source/sink definition based
* on the current one. This method is used when transforming the current
* definition. Derived classes can override this method to create instances of
* the correct class.
*
* @param baseAPTs The access paths rooted in the base object
* @param paramAPTs The access paths rooted in the method's parameters
* @param returnAPTs The access paths rooted in the return value
* @return The new source/sink definition object
*/
protected MethodSourceSinkDefinition buildNewDefinition(Set baseAPTs,
Set[] paramAPTs, Set returnAPTs) {
MethodSourceSinkDefinition def = buildNewDefinition(method, baseAPTs, paramAPTs, returnAPTs, callType);
def.category = category;
return def;
}
protected MethodSourceSinkDefinition buildNewDefinition(SootMethodAndClass methodAndclass,
Set filteredBaseObjects, Set[] filteredParameters,
Set filteredReturnValues, CallType callType) {
return new MethodSourceSinkDefinition(methodAndclass, filteredBaseObjects, filteredParameters,
filteredReturnValues, callType);
}
@Override
@SuppressWarnings("unchecked")
public void merge(ISourceSinkDefinition other) {
if (other instanceof MethodSourceSinkDefinition) {
MethodSourceSinkDefinition otherMethod = (MethodSourceSinkDefinition) other;
// Merge the base object definitions
if (otherMethod.baseObjects != null && !otherMethod.baseObjects.isEmpty()) {
if (this.baseObjects == null)
this.baseObjects = new HashSet<>();
for (AccessPathTuple apt : otherMethod.baseObjects)
this.baseObjects.add(apt);
}
// Merge the parameter definitions
if (otherMethod.parameters != null && otherMethod.parameters.length > 0) {
if (this.parameters == null)
this.parameters = new Set[this.method.getParameters().size()];
for (int i = 0; i < otherMethod.parameters.length; i++) {
addParameterDefinition(i, otherMethod.parameters[i]);
}
}
// Merge the return value definitions
if (otherMethod.returnValues != null && !otherMethod.returnValues.isEmpty()) {
if (this.returnValues == null)
this.returnValues = new HashSet<>();
for (AccessPathTuple apt : otherMethod.returnValues)
this.returnValues.add(apt);
}
}
}
/**
* Adds the given access path tuples to this source/sink definition for the
* given parameter index
*
* @param paramIdx The parameter index
* @param paramDefs The access path tuples
*/
@SuppressWarnings("unchecked")
public void addParameterDefinition(int paramIdx, Set paramDefs) {
if (paramDefs != null && !paramDefs.isEmpty()) {
// We may need to widen our parameter array
Set[] oldSet = this.parameters;
if (oldSet.length <= paramIdx) {
Set[] newSet = (Set[]) new Set>[paramIdx + 1];
System.arraycopy(oldSet, 0, newSet, 0, paramIdx);
this.parameters = newSet;
}
// We may not have a set of access path tuples yet
Set aps = this.parameters[paramIdx];
if (aps == null) {
aps = new HashSet<>(paramDefs.size());
this.parameters[paramIdx] = aps;
}
aps.addAll(paramDefs);
}
}
/**
* Gets the shared source definition that is not associated with any method and
* taints the base object
*
* @return The shared blank source definition that is not associated with any
* method and taints the base object
*/
public MethodSourceSinkDefinition getBaseObjectSource() {
if (BASE_OBJ_SOURCE == null)
BASE_OBJ_SOURCE = new MethodSourceSinkDefinition(
Collections.singleton(AccessPathTuple.getBlankSourceTuple()), null, null, CallType.MethodCall);
return BASE_OBJ_SOURCE;
}
/**
* Gets the shared sink definition that is not associated with any method and
* taints the base object
*
* @return The shared blank sink definition that is not associated with any
* method and taints the base object
*/
public MethodSourceSinkDefinition getBaseObjectSink() {
if (BASE_OBJ_SINK == null)
BASE_OBJ_SINK = new MethodSourceSinkDefinition(Collections.singleton(AccessPathTuple.getBlankSinkTuple()),
null, null, CallType.MethodCall);
return BASE_OBJ_SINK;
}
@Override
public int hashCode() {
final int prime = 31;
int result = super.hashCode();
result = prime * result + ((baseObjects == null) ? 0 : baseObjects.hashCode());
result = prime * result + ((callType == null) ? 0 : callType.hashCode());
result = prime * result + ((method == null) ? 0 : method.hashCode());
result = prime * result + Arrays.hashCode(parameters);
result = prime * result + ((returnValues == null) ? 0 : returnValues.hashCode());
return result;
}
@Override
public boolean equals(Object obj) {
if (this == obj)
return true;
if (!super.equals(obj))
return false;
if (getClass() != obj.getClass())
return false;
MethodSourceSinkDefinition other = (MethodSourceSinkDefinition) obj;
if (baseObjects == null) {
if (other.baseObjects != null)
return false;
} else if (!baseObjects.equals(other.baseObjects))
return false;
if (callType != other.callType)
return false;
if (method == null) {
if (other.method != null)
return false;
} else if (!method.equals(other.method))
return false;
if (!Arrays.equals(parameters, other.parameters))
return false;
if (returnValues == null) {
if (other.returnValues != null)
return false;
} else if (!returnValues.equals(other.returnValues))
return false;
return true;
}
/**
* Creates a source definition that considers a parameter as tainted
*
* @param index The index of the parameter to consider as tainted
* @param callType The type of call
* @return The newly created source definition
*/
@SuppressWarnings("unchecked")
public static MethodSourceSinkDefinition createParameterSource(int index, CallType callType) {
// For small indices, we have cached shared objects
if (index < 5 && callType == CallType.MethodCall) {
MethodSourceSinkDefinition def = PARAM_OBJ_SOURCE[index];
if (def == null) {
Set[] params = (Set[]) new Set>[index + 1];
params[index] = Collections.singleton(AccessPathTuple.getBlankSourceTuple());
def = new MethodSourceSinkDefinition(null, params, null, callType);
PARAM_OBJ_SOURCE[index] = def;
}
return def;
}
return new MethodSourceSinkDefinition(null,
(Set[]) new Set>[] { Collections.singleton(AccessPathTuple.getBlankSourceTuple()) },
null, callType);
}
/**
* Creates a source definition that considers the return value as tainted
*
* @param callType The type of call
* @return The newly created source definition
*/
public static MethodSourceSinkDefinition createReturnSource(CallType callType) {
return new MethodSourceSinkDefinition(null, null, Collections.singleton(AccessPathTuple.getBlankSourceTuple()),
callType);
}
/**
* Checks whether this definition is equivalent to one of the simple predefined
* ones. If so, it returns the shared predefined object. Otherwise, it returns
* this object.
*
* @return A shared object that is equal to this one if possible, otherwise this
* object
*/
public MethodSourceSinkDefinition simplify() {
MethodSourceSinkDefinition baseObjSource = getBaseObjectSource();
MethodSourceSinkDefinition baseObjSink = getBaseObjectSink();
if (this.equals(baseObjSource))
return baseObjSource;
else if (this.equals(baseObjSink))
return baseObjSink;
else {
for (int i = 0; i < PARAM_OBJ_SOURCE.length; i++) {
MethodSourceSinkDefinition def = createParameterSource(i, getCallType());
if (this.equals(def))
return def;
}
return this;
}
}
@Override
public Set getAllAccessPaths() {
Set aps = new HashSet<>();
if (baseObjects != null && !baseObjects.isEmpty())
aps.addAll(baseObjects);
if (returnValues != null && !returnValues.isEmpty())
aps.addAll(returnValues);
if (parameters != null && parameters.length > 0) {
for (Set paramAPs : parameters) {
if (paramAPs != null && !paramAPs.isEmpty())
aps.addAll(paramAPs);
}
}
return aps;
}
@SuppressWarnings("unchecked")
@Override
public MethodSourceSinkDefinition filter(Collection accessPaths) {
// Filter the base objects
Set filteredBaseObjects = null;
if (baseObjects != null && !baseObjects.isEmpty()) {
filteredBaseObjects = new HashSet<>(baseObjects.size());
for (AccessPathTuple ap : baseObjects)
if (accessPaths.contains(ap))
filteredBaseObjects.add(ap);
}
// Filter the return values
Set filteredReturnValues = null;
if (returnValues != null && !returnValues.isEmpty()) {
filteredReturnValues = new HashSet<>(returnValues.size());
for (AccessPathTuple ap : returnValues)
if (accessPaths.contains(ap))
filteredReturnValues.add(ap);
}
// Filter the parameters
Set[] filteredParameters = null;
if (parameters != null && parameters.length > 0) {
filteredParameters = new Set[parameters.length];
for (int i = 0; i < parameters.length; i++) {
if (parameters[i] != null && !parameters[i].isEmpty()) {
filteredParameters[i] = new HashSet<>();
for (AccessPathTuple ap : parameters[i])
if (accessPaths.contains(ap))
filteredParameters[i].add(ap);
}
}
}
MethodSourceSinkDefinition def = buildNewDefinition(method, filteredBaseObjects, filteredParameters,
filteredReturnValues, callType);
def.setCategory(category);
return def;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy