All Downloads are FREE. Search and download functionalities are using the official Maven repository.

soot.jimple.infoflow.android.entryPointCreators.AndroidEntryPointCreator Maven / Gradle / Ivy

There is a newer version: 2.14.1
Show newest version
/*******************************************************************************
 * Copyright (c) 2012 Secure Software Engineering Group at EC SPRIDE.
 * All rights reserved. This program and the accompanying materials
 * are made available under the terms of the GNU Lesser Public License v2.1
 * which accompanies this distribution, and is available at
 * http://www.gnu.org/licenses/old-licenses/gpl-2.0.html
 * 
 * Contributors: Christian Fritz, Steven Arzt, Siegfried Rasthofer, Eric
 * Bodden, and others.
 ******************************************************************************/
package soot.jimple.infoflow.android.entryPointCreators;

import java.util.ArrayList;
import java.util.Collection;
import java.util.Collections;
import java.util.HashMap;
import java.util.HashSet;
import java.util.Iterator;
import java.util.List;
import java.util.Map;
import java.util.Set;

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import soot.Body;
import soot.Local;
import soot.Modifier;
import soot.RefType;
import soot.Scene;
import soot.SootClass;
import soot.SootField;
import soot.SootMethod;
import soot.Type;
import soot.Unit;
import soot.UnitPatchingChain;
import soot.Value;
import soot.jimple.AssignStmt;
import soot.jimple.IfStmt;
import soot.jimple.InvokeStmt;
import soot.jimple.Jimple;
import soot.jimple.NopStmt;
import soot.jimple.NullConstant;
import soot.jimple.Stmt;
import soot.jimple.infoflow.android.entryPointCreators.AndroidEntryPointUtils.ComponentType;
import soot.jimple.infoflow.android.entryPointCreators.components.AbstractComponentEntryPointCreator;
import soot.jimple.infoflow.android.entryPointCreators.components.ActivityEntryPointCreator;
import soot.jimple.infoflow.android.entryPointCreators.components.BroadcastReceiverEntryPointCreator;
import soot.jimple.infoflow.android.entryPointCreators.components.ComponentEntryPointCollection;
import soot.jimple.infoflow.android.entryPointCreators.components.ContentProviderEntryPointCreator;
import soot.jimple.infoflow.android.entryPointCreators.components.FragmentEntryPointCreator;
import soot.jimple.infoflow.android.entryPointCreators.components.ServiceConnectionEntryPointCreator;
import soot.jimple.infoflow.android.entryPointCreators.components.ServiceEntryPointCreator;
import soot.jimple.infoflow.android.manifest.IAndroidApplication;
import soot.jimple.infoflow.android.manifest.IManifestHandler;
import soot.jimple.infoflow.cfg.LibraryClassPatcher;
import soot.jimple.infoflow.data.SootMethodAndClass;
import soot.jimple.infoflow.entryPointCreators.IEntryPointCreator;
import soot.jimple.infoflow.entryPointCreators.SimulatedCodeElementTag;
import soot.jimple.infoflow.typing.TypeUtils;
import soot.jimple.infoflow.util.SootMethodRepresentationParser;
import soot.jimple.infoflow.util.SystemClassHandler;
import soot.jimple.toolkits.scalar.NopEliminator;
import soot.options.Options;
import soot.util.HashMultiMap;
import soot.util.MultiMap;

/**
 * class which creates a dummy main method with the entry points according to
 * the Android lifecycles
 * 
 * based on:
 * http://developer.android.com/reference/android/app/Activity.html#ActivityLifecycle
 * and http://developer.android.com/reference/android/app/Service.html and
 * http://developer.android.com/reference/android/content/BroadcastReceiver.html#ReceiverLifecycle
 * and
 * http://developer.android.com/reference/android/content/BroadcastReceiver.html
 * and https://developer.android.com/reference/android/app/Fragment.html
 * 
 * @author Christian, Steven Arzt
 * 
 */
public class AndroidEntryPointCreator extends AbstractAndroidEntryPointCreator implements IEntryPointCreator {

	private final Logger logger = LoggerFactory.getLogger(getClass());

	private static final boolean DEBUG = false;

	protected MultiMap callbackFunctions = new HashMultiMap<>();;

	private SootClass applicationClass = null;
	private Local applicationLocal = null;

	private MultiMap activityLifecycleCallbacks = new HashMultiMap<>();
	private MultiMap applicationCallbackClasses = new HashMultiMap<>();
	private Map callbackClassToField = new HashMap<>();

	private MultiMap fragmentClasses = null;
	private final ComponentEntryPointCollection componentToInfo = new ComponentEntryPointCollection();

	private Collection components;

	private MultiMap javascriptInterfaceStmts;

	/**
	 * Creates a new instance of the {@link AndroidEntryPointCreator} class and
	 * registers a list of classes to be automatically scanned for Android lifecycle
	 * methods
	 * 
	 * @param components The list of classes to be automatically scanned for Android
	 *                   lifecycle methods
	 */
	public AndroidEntryPointCreator(IManifestHandler manifest, Collection components) {
		super(manifest);
		this.components = components;
		this.overwriteDummyMainMethod = true;
	}

	@Override
	protected SootMethod createDummyMainInternal() {
		// Make sure that we don't have any leftover state
		// from previous runs
		reset();

		logger.info(String.format("Creating Android entry point for %d components...", components.size()));

		// For some weird reason unknown to anyone except the flying spaghetti
		// monster, the onCreate() methods of content providers run even before
		// the application object's onCreate() is called.
		{
			boolean hasContentProviders = false;
			NopStmt beforeContentProvidersStmt = Jimple.v().newNopStmt();
			body.getUnits().add(beforeContentProvidersStmt);
			for (SootClass currentClass : components) {
				if (entryPointUtils.getComponentType(currentClass) == ComponentType.ContentProvider) {
					// Create an instance of the content provider
					Local localVal = generateClassConstructor(currentClass);
					if (localVal == null)
						continue;
					localVarsForClasses.put(currentClass, localVal);

					// Conditionally call the onCreate method
					NopStmt thenStmt = Jimple.v().newNopStmt();
					createIfStmt(thenStmt);
					searchAndBuildMethod(AndroidEntryPointConstants.CONTENTPROVIDER_ONCREATE, currentClass, localVal);
					body.getUnits().add(thenStmt);
					hasContentProviders = true;
				}
			}
			// Jump back to the beginning of this section to overapproximate the
			// order in which the methods are called
			if (hasContentProviders)
				createIfStmt(beforeContentProvidersStmt);
		}

		// If we have an implementation of android.app.Application, this needs
		// special treatment
		initializeApplicationClass();

		// If we have an application, we need to start it in the very beginning
		if (applicationClass != null) {
			// Create the application
			applicationLocal = generateClassConstructor(applicationClass);
			localVarsForClasses.put(applicationClass, applicationLocal);
			if (applicationLocal != null) {
				localVarsForClasses.put(applicationClass, applicationLocal);

				boolean hasApplicationCallbacks = applicationCallbackClasses != null
						&& !applicationCallbackClasses.isEmpty();
				boolean hasActivityLifecycleCallbacks = activityLifecycleCallbacks != null
						&& !activityLifecycleCallbacks.isEmpty();

				// Create instances of all application callback classes
				if (hasApplicationCallbacks || hasActivityLifecycleCallbacks) {
					NopStmt beforeCbCons = Jimple.v().newNopStmt();
					body.getUnits().add(beforeCbCons);

					if (hasApplicationCallbacks)
						createClassInstances(applicationCallbackClasses.keySet());
					if (hasActivityLifecycleCallbacks) {
						createClassInstances(activityLifecycleCallbacks.keySet());

						// Assign the instance to the field
						for (SootClass sc : activityLifecycleCallbacks.keySet()) {
							SootField fld = callbackClassToField.get(sc);
							Local lc = localVarsForClasses.get(sc);
							if (sc != null && lc != null)
								body.getUnits()
										.add(Jimple.v().newAssignStmt(Jimple.v().newStaticFieldRef(fld.makeRef()), lc));
						}
					}

					// Jump back to overapproximate the order in which the
					// constructors are called
					createIfStmt(beforeCbCons);
				}

				// Call the onCreate() method
				searchAndBuildMethod(AndroidEntryPointConstants.APPLICATION_ONCREATE, applicationClass,
						applicationLocal);

				//////////////
				// Initializes the ApplicationHolder static field with the
				////////////// singleton application
				// instance created above
				// (Used by the Activity::getApplication patched in
				////////////// LibraryClassPatcher)
				SootClass scApplicationHolder = LibraryClassPatcher.createOrGetApplicationHolder();
				body.getUnits()
						.add(Jimple.v().newAssignStmt(
								Jimple.v()
										.newStaticFieldRef(scApplicationHolder.getFieldByName("application").makeRef()),
								applicationLocal));
				//////////////
			}
		}

		// prepare outer loop:
		NopStmt outerStartStmt = Jimple.v().newNopStmt();
		body.getUnits().add(outerStartStmt);

		// We need to create methods for all fragments, because they can be used by
		// multiple activities
		Map fragmentToMainMethod = new HashMap<>();
		for (SootClass parentActivity : fragmentClasses.keySet()) {
			Set fragments = fragmentClasses.get(parentActivity);
			for (SootClass fragment : fragments) {
				FragmentEntryPointCreator entryPointCreator = new FragmentEntryPointCreator(fragment, applicationClass,
						this.manifest);
				entryPointCreator.setDummyClassName(mainMethod.getDeclaringClass().getName());
				entryPointCreator.setCallbacks(callbackFunctions.get(fragment));

				SootMethod fragmentMethod = entryPointCreator.createDummyMain();
				fragmentToMainMethod.put(fragment, fragmentMethod);
				componentToInfo.put(fragment, fragmentMethod);
			}
		}

		for (SootClass currentClass : components) {
			currentClass.setApplicationClass();

			// Get the callbacks and component type of the current component
			ComponentType componentType = entryPointUtils.getComponentType(currentClass);

			// Before-class marker
			Stmt beforeComponentStmt = Jimple.v().newNopStmt();
			Stmt afterComponentStmt = Jimple.v().newNopStmt();
			body.getUnits().add(beforeComponentStmt);

			// Generate the lifecycles for the different kinds of Android
			// classes
			AbstractComponentEntryPointCreator componentCreator = null;
			switch (componentType) {
			case Activity:
				Map curActivityToFragmentMethod = new HashMap<>();
				if (fragmentClasses != null) {
					Set fragments = fragmentClasses.get(currentClass);
					if (fragments != null && !fragments.isEmpty()) {
						for (SootClass fragment : fragments)
							curActivityToFragmentMethod.put(fragment, fragmentToMainMethod.get(fragment));
					}
				}
				componentCreator = new ActivityEntryPointCreator(currentClass, applicationClass,
						activityLifecycleCallbacks, callbackClassToField, curActivityToFragmentMethod, this.manifest);
				break;
			case Service:
			case GCMBaseIntentService:
			case GCMListenerService:
			case HostApduService:
				componentCreator = new ServiceEntryPointCreator(currentClass, applicationClass, this.manifest);
				break;
			case ServiceConnection:
				componentCreator = new ServiceConnectionEntryPointCreator(currentClass, applicationClass,
						this.manifest);
				break;
			case BroadcastReceiver:
				componentCreator = new BroadcastReceiverEntryPointCreator(currentClass, applicationClass,
						this.manifest);
				break;
			case ContentProvider:
				componentCreator = new ContentProviderEntryPointCreator(currentClass, applicationClass, this.manifest);
				break;
			default:
				componentCreator = null;
				break;
			}

			// We may skip the complete component
			createIfStmt(afterComponentStmt);

			// Create a call to the component's lifecycle method
			if (componentCreator != null) {
				componentCreator.setDummyClassName(mainMethod.getDeclaringClass().getName());
				componentCreator.setCallbacks(callbackFunctions.get(currentClass));
				SootMethod lifecycleMethod = componentCreator.createDummyMain();
				componentToInfo.put(currentClass, componentCreator.getComponentInfo());

				// dummyMain(component, intent)
				if (shouldAddLifecycleCall(currentClass)) {
					body.getUnits()
							.add(Jimple.v().newInvokeStmt(Jimple.v().newStaticInvokeExpr(lifecycleMethod.makeRef(),
									Collections.singletonList(NullConstant.v()))));
				}
			}

			// Jump back to the front of the component
			createIfStmt(beforeComponentStmt);
			body.getUnits().add(afterComponentStmt);
		}

		// Add conditional calls to the application callback methods
		if (applicationLocal != null) {
			Unit beforeAppCallbacks = Jimple.v().newNopStmt();
			body.getUnits().add(beforeAppCallbacks);
			addApplicationCallbackMethods();
			createIfStmt(beforeAppCallbacks);
		}
		createJavascriptCallbacks();

		createIfStmt(outerStartStmt);

		// Add a call to application.onTerminate()
		if (applicationLocal != null)
			searchAndBuildMethod(AndroidEntryPointConstants.APPLICATION_ONTERMINATE, applicationClass,
					applicationLocal);

		body.getUnits().add(Jimple.v().newReturnVoidStmt());

		// Optimize and check the generated main method
		NopEliminator.v().transform(body);
		eliminateSelfLoops();
		eliminateFallthroughIfs(body);

		if (DEBUG || Options.v().validate())
			mainMethod.getActiveBody().validate();

		return mainMethod;
	}

	private void createJavascriptCallbacks() {
		Jimple j = Jimple.v();
		for (SootMethod m : javascriptInterfaceStmts.keySet()) {
			Set statements = javascriptInterfaceStmts.get(m);
			for (Stmt s : statements) {
				UnitPatchingChain units = m.retrieveActiveBody().getUnits();
				SootField f = null;
				Value arg = s.getInvokeExpr().getArg(0);
				DummyMainFieldElementTag dm = (DummyMainFieldElementTag) s.getTag(DummyMainFieldElementTag.TAG_NAME);
				if (dm != null) {
					f = mainMethod.getDeclaringClass().getFieldByNameUnsafe(dm.getFieldName());
				}
				if (f == null) {
					//create field
					f = createField(arg.getType(), "jsInterface");
					AssignStmt assign = j.newAssignStmt(j.newStaticFieldRef(f.makeRef()), arg);
					assign.addTag(SimulatedCodeElementTag.TAG);
					s.addTag(new DummyMainFieldElementTag(f.getName()));
					units.insertAfter(assign, s);
				}

				Local l = j.newLocal(f.getName(), f.getType());
				body.getLocals().add(l);
				Stmt assignF = j.newAssignStmt(l, j.newStaticFieldRef(f.makeRef()));
				body.getUnits().add(assignF);
				SootClass cbtype = ((RefType) f.getType()).getSootClass();

				for (SootClass c : TypeUtils.getAllDerivedClasses(cbtype)) {
					for (SootMethod cbm : c.getMethods()) {
						if (AndroidEntryPointUtils.isCallableFromJS(cbm)) {
							List args = new ArrayList<>();
							for (Type t : cbm.getParameterTypes())
								args.add(getSimpleDefaultValue(t));
							InvokeStmt st = j.newInvokeStmt(j.newVirtualInvokeExpr(l, cbm.makeRef(), args));
							body.getUnits().add(st);
						}
					}

				}
				createIfStmt(assignF);

			}
		}
	}

	/**
	 * Checks whether a lifecycle call should be added to the given SootClass,
	 * designed to be overridden by different implementations
	 *
	 * @param clazz the SootClass in question
	 * @return True if a lifecycle call should be added in
	 *         {@link #createDummyMainInternal()}
	 */
	protected boolean shouldAddLifecycleCall(SootClass clazz) {
		return true;
	}

	/**
	 * Find the application class and its callbacks
	 */
	private void initializeApplicationClass() {

		IAndroidApplication app = manifest.getApplication();
		if (app != null) {
			String applicationName = app.getName();
			// We can only look for callbacks if we have an application class
			if (applicationName == null || applicationName.isEmpty())
				return;

			// Find the application class
			for (SootClass currentClass : components) {
				// Is this the application class?
				if (entryPointUtils.isApplicationClass(currentClass)
						&& currentClass.getName().equals(applicationName)) {
					if (applicationClass != null && currentClass != applicationClass)
						throw new RuntimeException("Multiple application classes in app");
					applicationClass = currentClass;
					break;
				}
			}
		}

		// We can only look for callbacks if we have an application class
		if (applicationClass == null)
			return;

		// Look into the application class' callbacks
		SootClass scActCallbacks = Scene.v()
				.getSootClassUnsafe(AndroidEntryPointConstants.ACTIVITYLIFECYCLECALLBACKSINTERFACE);
		Collection callbacks = callbackFunctions.get(applicationClass);
		if (callbacks != null) {
			for (SootMethod smCallback : callbacks) {
				if (smCallback != null) {
					// Is this a special callback class? We have callbacks that model activity
					// lifecycle events and ones that model generic events (e.g., low memory)
					if (scActCallbacks != null && Scene.v().getOrMakeFastHierarchy()
							.canStoreType(smCallback.getDeclaringClass().getType(), scActCallbacks.getType()))
						activityLifecycleCallbacks.put(smCallback.getDeclaringClass(), smCallback.getSignature());
					else
						applicationCallbackClasses.put(smCallback.getDeclaringClass(), smCallback.getSignature());
				}
			}
		}

		// Create fields for the activity lifecycle classes
		for (SootClass callbackClass : activityLifecycleCallbacks.keySet()) {
			String baseName = callbackClass.getName();
			if (baseName.contains("."))
				baseName = baseName.substring(baseName.lastIndexOf(".") + 1);

			// Generate a fresh field name
			SootField fld = createField(RefType.v(callbackClass), baseName);
			callbackClassToField.put(callbackClass, fld);
		}
	}

	protected SootField createField(Type type, String baseName) {
		SootClass dummyMainClass = mainMethod.getDeclaringClass();
		int idx = 0;
		String fieldName = baseName;
		while (dummyMainClass.declaresFieldByName(fieldName)) {
			fieldName = baseName + "_" + idx;
			idx++;
		}
		SootField fld = Scene.v().makeSootField(fieldName, type, Modifier.PRIVATE | Modifier.STATIC);
		mainMethod.getDeclaringClass().addField(fld);
		return fld;
	}

	/**
	 * Removes if statements that jump to the fall-through successor
	 * 
	 * @param body The body from which to remove unnecessary if statements
	 */
	private void eliminateFallthroughIfs(Body body) {
		boolean changed = false;
		do {
			changed = false;
			IfStmt ifs = null;
			Iterator unitIt = body.getUnits().snapshotIterator();
			while (unitIt.hasNext()) {
				Unit u = unitIt.next();
				if (ifs != null && ifs.getTarget() == u) {
					body.getUnits().remove(ifs);
					changed = true;
				}
				ifs = null;
				if (u instanceof IfStmt)
					ifs = (IfStmt) u;
			}
		} while (changed);
	}

	/**
	 * Adds calls to the callback methods defined in the application class
	 * 
	 * @param applicationClass The class in which the user-defined application is
	 *                         implemented
	 * @param applicationLocal The local containing the instance of the user-defined
	 *                         application
	 */
	private void addApplicationCallbackMethods() {
		if (!this.callbackFunctions.containsKey(applicationClass))
			return;

		// Do not try to generate calls to methods in non-concrete classes
		if (applicationClass.isAbstract())
			return;
		if (applicationClass.isPhantom()) {
			logger.warn("Skipping possible application callbacks in " + "phantom class %s", applicationClass);
			return;
		}

		List lifecycleMethods = AndroidEntryPointConstants.getApplicationLifecycleMethods();
		for (SootClass sc : applicationCallbackClasses.keySet())
			for (String methodSig : applicationCallbackClasses.get(sc)) {
				SootMethodAndClass methodAndClass = SootMethodRepresentationParser.v().parseSootMethodString(methodSig);
				String subSig = methodAndClass.getSubSignature();
				SootMethod method = findMethod(Scene.v().getSootClass(sc.getName()), subSig);

				// We do not consider lifecycle methods which are directly
				// inserted at their respective positions
				if (sc == applicationClass && lifecycleMethods.contains(subSig))
					continue;

				// If this is an activity lifecycle method, we skip it as well
				// TODO: can be removed once we filter it in general
				if (activityLifecycleCallbacks.containsKey(sc))
					if (lifecycleMethods.contains(subSig))
						continue;

				// If we found no implementation or if the implementation we found is in a
				// system class, we skip it. Note that null methods may happen since all
				// callback interfaces for application callbacks are registered under the name
				// of the application class.
				if (method == null)
					continue;
				if (SystemClassHandler.v().isClassInSystemPackage(method.getDeclaringClass().getName()))
					continue;

				// Get the local instance of the target class
				Local local = this.localVarsForClasses.get(sc);
				if (local == null) {
					logger.warn(String.format("Could not create call to application callback %s. Local was null.",
							method.getSignature()));
					continue;
				}

				// Add a conditional call to the method
				NopStmt thenStmt = Jimple.v().newNopStmt();
				createIfStmt(thenStmt);
				buildMethodCall(method, local);
				body.getUnits().add(thenStmt);
			}
	}

	@Override
	public Collection getRequiredClasses() {
		Set requiredClasses = new HashSet(components.size());
		for (SootClass sc : components)
			requiredClasses.add(sc.getName());
		return requiredClasses;
	}

	public void setFragments(MultiMap fragments) {
		fragmentClasses = fragments;
	}

	@Override
	public Collection getAdditionalMethods() {
		return componentToInfo.getLifecycleMethods();
	}

	@Override
	public Collection getAdditionalFields() {
		return componentToInfo.getAdditionalFields();
	}

	public ComponentEntryPointCollection getComponentToEntryPointInfo() {
		return componentToInfo;
	}

	/**
	 * Sets the list of callback functions to be integrated into the Android
	 * lifecycle
	 * 
	 * @param callbackFunctions The list of callback functions to be integrated into
	 *                          the Android lifecycle. This is a mapping from the
	 *                          Android element class (activity, service, etc.) to
	 *                          the list of callback methods for that element.
	 */
	public void setCallbackFunctions(MultiMap callbackFunctions) {
		this.callbackFunctions = callbackFunctions;
	}

	/**
	 * Returns the list of callback functions of the Android lifecycle.
	 * 
	 * @return callbackFunctions The list of callback functions of the Android
	 *         lifecycle. This is a mapping from the Android element class
	 *         (activity, service, etc.) to the list of callback methods for that
	 *         element.
	 */
	public MultiMap getCallbackFunctions() {
		return callbackFunctions;
	}

	@Override
	public void reset() {
		super.reset();

		// Get rid of the generated component methods
		for (SootMethod sm : getAdditionalMethods()) {
			if (sm.isDeclared())
				sm.getDeclaringClass().removeMethod(sm);
		}
		for (SootField sf : getAdditionalFields()) {
			if (sf.isDeclared())
				sf.getDeclaringClass().removeField(sf);
		}

		// Get rid of the generated fields
		for (SootField fld : callbackClassToField.values()) {
			if (fld.isDeclared())
				fld.getDeclaringClass().removeField(fld);
		}

		componentToInfo.clear();
		callbackClassToField.clear();
	}

	/**
	 * Sets the Android components for which a dummy main method shall be created
	 * 
	 * @param components The Android components for which a dummy main method shall
	 *                   be created
	 */
	public void setComponents(Collection components) {
		this.components = components;
	}

	/**
	 * Removes all methods that have been generated by this entry point creator
	 * 
	 * @param removeClass True if the generated class shall also be removed. False
	 *                    to only remove the methods, but keep the class
	 */
	public void removeGeneratedMethods(boolean removeClass) {
		// Remove the dummy main method itself
		final SootClass mainClass = mainMethod.getDeclaringClass();
		if (removeClass)
			Scene.v().removeClass(mainClass);
		else
			mainClass.removeMethod(mainMethod);

		// Remove the additional methods
		for (SootMethod sm : getAdditionalMethods()) {
			if (sm.isDeclared()) {
				final SootClass declaringClass = sm.getDeclaringClass();
				if (declaringClass.isInScene())
					declaringClass.removeMethod(sm);
			}
		}
	}

	public void setJavaScriptInterfaces(MultiMap javascriptInterfaceStmts) {
		this.javascriptInterfaceStmts = javascriptInterfaceStmts;
	}

}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy