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

soot.dava.toolkits.base.AST.transformations.VoidReturnRemover Maven / Gradle / Ivy

package soot.dava.toolkits.base.AST.transformations;

/*-
 * #%L
 * Soot - a J*va Optimization Framework
 * %%
 * Copyright (C) 2005 Nomair A. Naeem
 * %%
 * 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.Iterator;
import java.util.List;

import soot.SootClass;
import soot.SootMethod;
import soot.VoidType;
import soot.dava.DavaBody;
import soot.dava.internal.AST.ASTMethodNode;
import soot.dava.internal.AST.ASTNode;
import soot.dava.internal.AST.ASTStatementSequenceNode;
import soot.dava.internal.asg.AugmentedStmt;
import soot.jimple.ReturnVoidStmt;
import soot.jimple.Stmt;
import soot.util.Chain;

public class VoidReturnRemover {

  public static void cleanClass(SootClass s) {
    List methods = s.getMethods();
    Iterator it = methods.iterator();
    while (it.hasNext()) {
      removeReturn((SootMethod) it.next());
    }
  }

  private static void removeReturn(SootMethod method) {
    // check if this is a void method
    if (!(method.getReturnType() instanceof VoidType)) {
      return;
    }

    // get the methodnode
    if (!method.hasActiveBody()) {
      return;
    }

    Chain units = ((DavaBody) method.getActiveBody()).getUnits();

    if (units.size() != 1) {
      return;
    }

    ASTNode AST = (ASTNode) units.getFirst();
    if (!(AST instanceof ASTMethodNode)) {
      throw new RuntimeException("Starting node of DavaBody AST is not an ASTMethodNode");
    }

    ASTMethodNode node = (ASTMethodNode) AST;

    // check there is only 1 subBody
    List subBodies = node.get_SubBodies();
    if (subBodies.size() != 1) {
      return;
    }

    List subBody = (List) subBodies.get(0);
    // see if the last of this is a stmtseq node
    if (subBody.size() == 0) {
      // nothing inside subBody
      return;
    }

    // check last node is a ASTStatementSequenceNode
    ASTNode last = (ASTNode) subBody.get(subBody.size() - 1);
    if (!(last instanceof ASTStatementSequenceNode)) {
      return;
    }

    // get last statement
    List stmts = ((ASTStatementSequenceNode) last).getStatements();
    if (stmts.size() == 0) {
      // no stmts inside statement sequence node
      subBody.remove(subBody.size() - 1);
      return;
    }
    AugmentedStmt lastas = (AugmentedStmt) stmts.get(stmts.size() - 1);
    Stmt lastStmt = lastas.get_Stmt();
    if (!(lastStmt instanceof ReturnVoidStmt)) {
      return;
    }

    // we can remove the lastStmt
    stmts.remove(stmts.size() - 1);
    /*
     * we need to check if we have made the size 0 in which case the stmtSeq Node should also be removed
     */
    if (stmts.size() == 0) {
      subBody.remove(subBody.size() - 1);
    }

  }
  // end method
}