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

soot.toolkits.exceptions.AbstractThrowAnalysis Maven / Gradle / Ivy

There is a newer version: 4.6.0
Show newest version
package soot.toolkits.exceptions;

import java.util.Set;
import java.util.stream.Collectors;

/*-
 * #%L
 * Soot - a J*va Optimization Framework
 * %%
 * Copyright (C) 2004 John Jorgensen
 * %%
 * This program is free software: you can redistribute it and/or modify
 * it under the terms of the GNU Lesser General Public License as
 * published by the Free Software Foundation, either version 2.1 of the
 * License, or (at your option) any later version.
 * 
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Lesser Public License for more details.
 * 
 * You should have received a copy of the GNU General Lesser Public
 * License along with this program.  If not, see
 * .
 * #L%
 */

import soot.AnySubType;
import soot.Local;
import soot.NullType;
import soot.RefType;
import soot.SootMethod;
import soot.Type;
import soot.Unit;
import soot.UnknownType;
import soot.Value;
import soot.baf.ThrowInst;
import soot.grimp.NewInvokeExpr;
import soot.jimple.DefinitionStmt;
import soot.jimple.NewExpr;
import soot.jimple.ThrowStmt;

/**
 * Abstract class implementing parts of the {@link ThrowAnalysis} interface which may be common to multiple concrete
 * ThrowAnalysis classes. AbstractThrowAnalysis provides straightforward implementations of
 * {@link mightThrowExplicitly(ThrowInst)} and {@link mightThrowExplicitly(ThrowStmt)}, since concrete implementations of
 * ThrowAnalysis seem likely to differ mainly in their treatment of implicit exceptions.
 */
public abstract class AbstractThrowAnalysis implements ThrowAnalysis {

  abstract public ThrowableSet mightThrow(Unit u);

  @Override
  public ThrowableSet mightThrowExplicitly(ThrowInst t) {
    // Deducing the type at the top of the Baf stack is beyond me, so...
    return ThrowableSet.Manager.v().ALL_THROWABLES;
  }

  @Override
  public ThrowableSet mightThrowExplicitly(ThrowStmt t) {
    return mightThrowExplicitly(t, null);
  }

  public ThrowableSet mightThrowExplicitly(ThrowStmt t, SootMethod sm) {
    Value thrownExpression = t.getOp();
    Type thrownType = thrownExpression.getType();
    if (thrownType == null || thrownType instanceof UnknownType) {
      // We can't identify the type of thrownExpression, so...
      return ThrowableSet.Manager.v().ALL_THROWABLES;
    } else if (thrownType instanceof NullType) {
      ThrowableSet result = ThrowableSet.Manager.v().EMPTY;
      result = result.add(ThrowableSet.Manager.v().NULL_POINTER_EXCEPTION);
      return result;
    } else if (!(thrownType instanceof RefType)) {
      throw new IllegalStateException("UnitThrowAnalysis StmtSwitch: type of throw argument is not a RefType!");
    } else {
      ThrowableSet result = ThrowableSet.Manager.v().EMPTY;
      if (thrownExpression instanceof NewInvokeExpr) {
        // In this case, we know the exact type of the
        // argument exception.
        result = result.add((RefType) thrownType);
      } else {
        RefType preciseType = null;
        // If there is only one allocation site, we know the type as well
        if (thrownExpression instanceof Local && sm != null) {
          Set types = sm.getActiveBody().getUnits().stream().filter(u -> u instanceof DefinitionStmt)
              .map(u -> (DefinitionStmt) u).filter(d -> d.getLeftOp() == thrownExpression).map(d -> d.getRightOp())
              .filter(o -> o instanceof NewExpr).map(o -> (NewExpr) o).map(n -> n.getType())
              .filter(r -> r instanceof RefType).map(r -> (RefType) r).collect(Collectors.toSet());
          if (types.size() == 1) {
            preciseType = types.iterator().next();
          }
        }

        if (preciseType == null) {
          result = result.add(AnySubType.v((RefType) thrownType));
        } else {
          result = result.add(preciseType);
        }
      }
      return result;
    }
  }

  abstract public ThrowableSet mightThrowImplicitly(ThrowInst t);

  abstract public ThrowableSet mightThrowImplicitly(ThrowStmt t);
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy