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

soot.jimple.toolkits.base.PartialConstructorFolder Maven / Gradle / Ivy

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

/*-
 * #%L
 * Soot - a J*va Optimization Framework
 * %%
 * Copyright (C) 2004 Jennifer Lhotak
 * %%
 * 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 java.util.ArrayList;
import java.util.Iterator;
import java.util.List;
import java.util.Map;

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

import soot.Body;
import soot.BodyTransformer;
import soot.Local;
import soot.Type;
import soot.Unit;
import soot.Value;
import soot.jimple.AssignStmt;
import soot.jimple.InvokeExpr;
import soot.jimple.InvokeStmt;
import soot.jimple.Jimple;
import soot.jimple.JimpleBody;
import soot.jimple.NewExpr;
import soot.jimple.SpecialInvokeExpr;
import soot.options.Options;
import soot.tagkit.SourceLnPosTag;
import soot.toolkits.scalar.LocalUses;
import soot.toolkits.scalar.UnitValueBoxPair;
import soot.util.Chain;

public class PartialConstructorFolder extends BodyTransformer {
  private static final Logger logger = LoggerFactory.getLogger(PartialConstructorFolder.class);
  // public JimpleConstructorFolder( Singletons.Global g ) {}
  // public static JimpleConstructorFolder v() { return G.v().JimpleConstructorFolder(); }

  private List types;

  public void setTypes(List t) {
    this.types = t;
  }

  public List getTypes() {
    return types;
  }

  /**
   * This method pushes all newExpr down to be the stmt directly before every invoke of the init only if they are in the
   * types list
   */
  @Override
  public void internalTransform(Body b, String phaseName, Map options) {
    JimpleBody body = (JimpleBody) b;

    if (Options.v().verbose()) {
      logger.debug("[" + body.getMethod().getName() + "] Folding Jimple constructors...");
    }

    final Chain units = body.getUnits();
    List stmtList = new ArrayList(units);

    LocalUses localUses = LocalUses.Factory.newLocalUses(body);

    Iterator nextStmtIt = stmtList.iterator();
    nextStmtIt.next(); // start ahead one

    /* fold in NewExpr's with specialinvoke's */
    for (final Unit u : stmtList) {
      if (u instanceof AssignStmt) {
        final AssignStmt as = (AssignStmt) u;

        /* this should be generalized to ArrayRefs */
        // only deal with stmts that are an local = newExpr
        final Value lhs = as.getLeftOp();
        if (!(lhs instanceof Local)) {
          continue;
        }

        final Value rhs = as.getRightOp();
        if (!(rhs instanceof NewExpr)) {
          continue;
        }

        // check if very next statement is invoke -->
        // this indicates there is no control flow between
        // new and invoke and should do nothing
        if (nextStmtIt.hasNext()) {
          Unit next = nextStmtIt.next();
          if (next instanceof InvokeStmt) {
            InvokeExpr ie = ((InvokeStmt) next).getInvokeExpr();
            if (ie instanceof SpecialInvokeExpr) {
              SpecialInvokeExpr invokeExpr = (SpecialInvokeExpr) ie;
              if (invokeExpr.getBase() == lhs) {
                break;
              }
            }
          }
        }

        // check if new is in the types list - only process these
        if (!types.contains(((NewExpr) rhs).getType())) {
          continue;
        }

        boolean madeNewInvokeExpr = false;
        for (UnitValueBoxPair uvb : localUses.getUsesOf(u)) {
          Unit use = uvb.unit;
          if (use instanceof InvokeStmt) {
            InvokeExpr ie = ((InvokeStmt) use).getInvokeExpr();
            if (!(ie instanceof SpecialInvokeExpr) || lhs != ((SpecialInvokeExpr) ie).getBase()) {
              continue;
            }

            // make a new one here
            AssignStmt constructStmt = Jimple.v().newAssignStmt(lhs, rhs);
            constructStmt.setRightOp(Jimple.v().newNewExpr(((NewExpr) rhs).getBaseType()));
            madeNewInvokeExpr = true;

            // redirect jumps
            use.redirectJumpsToThisTo(constructStmt);
            // insert new one here
            units.insertBefore(constructStmt, use);

            constructStmt.addTag(u.getTag(SourceLnPosTag.NAME));
          }
        }
        if (madeNewInvokeExpr) {
          units.remove(u);
        }
      }
    }
  }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy