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

soot.JastAddInitialResolver Maven / Gradle / Ivy

package soot;

/*-
 * #%L
 * Soot - a J*va Optimization Framework
 * %%
 * Copyright (C) 2008 Eric Bodden
 * Copyright (C) 2008 Torbjorn Ekman
 * %%
 * 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.Collection;
import java.util.HashMap;
import java.util.HashSet;
import java.util.List;
import java.util.Map;

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

import soot.JastAddJ.BodyDecl;
import soot.JastAddJ.CompilationUnit;
import soot.JastAddJ.ConstructorDecl;
import soot.JastAddJ.MethodDecl;
import soot.JastAddJ.Program;
import soot.JastAddJ.TypeDecl;
import soot.javaToJimple.IInitialResolver;

/**
 * An {@link IInitialResolver} for the JastAdd frontend.
 *
 * @author Torbjorn Ekman
 * @author Eric Bodden
 */
public class JastAddInitialResolver implements IInitialResolver {
  private static final Logger logger = LoggerFactory.getLogger(JastAddInitialResolver.class);

  public JastAddInitialResolver(soot.Singletons.Global g) {
  }

  public static JastAddInitialResolver v() {
    return soot.G.v().soot_JastAddInitialResolver();
  }

  protected Map classNameToCU = new HashMap();

  public void formAst(String fullPath, List locations, String className) {
    Program program = SootResolver.v().getProgram();
    CompilationUnit u = program.getCachedOrLoadCompilationUnit(fullPath);
    if (u != null && !u.isResolved) {
      u.isResolved = true;
      java.util.ArrayList errors = new java.util.ArrayList();
      u.errorCheck(errors);
      if (!errors.isEmpty()) {
        for (soot.JastAddJ.Problem p : errors) {
          logger.debug("" + p);
        }
        // die
        throw new CompilationDeathException(CompilationDeathException.COMPILATION_ABORTED,
            "there were errors during parsing and/or type checking (JastAdd frontend)");
      }
      u.transformation();
      u.jimplify1phase1();
      u.jimplify1phase2();
      HashSet types = new HashSet();
      for (TypeDecl typeDecl : u.getTypeDecls()) {
        collectTypeDecl(typeDecl, types);
      }
      if (types.isEmpty()) {
        classNameToCU.put(className, u);
      } else {
        for (SootClass sc : types) {
          classNameToCU.put(sc.getName(), u);
        }
      }
    }
  }

  @SuppressWarnings("unchecked")
  private void collectTypeDecl(TypeDecl typeDecl, HashSet types) {
    types.add(typeDecl.getSootClassDecl());
    for (TypeDecl nestedType : (Collection) typeDecl.nestedTypes()) {
      collectTypeDecl(nestedType, types);
    }
  }

  @SuppressWarnings("unchecked")
  private TypeDecl findNestedTypeDecl(TypeDecl typeDecl, SootClass sc) {
    if (typeDecl.sootClass() == sc) {
      return typeDecl;
    }
    for (TypeDecl nestedType : (Collection) typeDecl.nestedTypes()) {
      TypeDecl t = findNestedTypeDecl(nestedType, sc);
      if (t != null) {
        return t;
      }
    }
    return null;
  }

  public Dependencies resolveFromJavaFile(SootClass sootclass) {
    CompilationUnit u = classNameToCU.get(sootclass.getName());

    if (u == null) {
      throw new RuntimeException("Error: couldn't find class: " + sootclass.getName() + " are the packages set properly?");
    }

    HashSet types = new HashSet();
    for (TypeDecl typeDecl : u.getTypeDecls()) {
      collectTypeDecl(typeDecl, types);
    }
    Dependencies deps = new Dependencies();
    u.collectTypesToHierarchy(deps.typesToHierarchy);
    u.collectTypesToSignatures(deps.typesToSignature);

    for (SootClass sc : types) {
      for (SootMethod m : sc.getMethods()) {
        m.setSource(new MethodSource() {
          public Body getBody(SootMethod m, String phaseName) {
            SootClass sc = m.getDeclaringClass();
            CompilationUnit u = classNameToCU.get(sc.getName());
            for (TypeDecl typeDecl : u.getTypeDecls()) {
              typeDecl = findNestedTypeDecl(typeDecl, sc);
              if (typeDecl != null) {
                if (typeDecl.clinit == m) {
                  typeDecl.jimplify2clinit();
                  return m.getActiveBody();
                }
                for (BodyDecl bodyDecl : typeDecl.getBodyDecls()) {
                  if (bodyDecl instanceof MethodDecl) {
                    MethodDecl methodDecl = (MethodDecl) bodyDecl;
                    if (m.equals(methodDecl.sootMethod)) {
                      methodDecl.jimplify2();
                      return m.getActiveBody();
                    }
                  } else if (bodyDecl instanceof ConstructorDecl) {
                    ConstructorDecl constrDecl = (ConstructorDecl) bodyDecl;
                    if (m.equals(constrDecl.sootMethod)) {
                      constrDecl.jimplify2();
                      return m.getActiveBody();
                    }
                  }
                }

              }
            }
            throw new RuntimeException(
                "Could not find body for " + m.getSignature() + " in " + m.getDeclaringClass().getName());
          }
        });
      }
    }
    return deps;
  }

}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy