/****************************************************************************** * Copyright (c) 2002 - 2006 IBM Corporation. * All rights reserved. This program and the accompanying materials * are made available under the terms of the Eclipse Public License v1.0 * which accompanies this distribution, and is available at * http://www.eclipse.org/legal/epl-v10.html * * Contributors: * IBM Corporation - initial API and implementation *****************************************************************************/ package com.ibm.wala.cast.js.client; import java.io.IOException; import java.util.Collections; import java.util.Set; import java.util.jar.JarFile; import com.ibm.wala.cast.ipa.callgraph.CAstAnalysisScope; import com.ibm.wala.cast.ir.ssa.AstIRFactory; import com.ibm.wala.cast.js.callgraph.fieldbased.FieldBasedCallGraphBuilder; import com.ibm.wala.cast.js.callgraph.fieldbased.OptimisticCallgraphBuilder; import com.ibm.wala.cast.js.callgraph.fieldbased.PessimisticCallGraphBuilder; import com.ibm.wala.cast.js.callgraph.fieldbased.flowgraph.vertices.ObjectVertex; import com.ibm.wala.cast.js.client.impl.ZeroCFABuilderFactory; import com.ibm.wala.cast.js.ipa.callgraph.JSAnalysisOptions; import com.ibm.wala.cast.js.ipa.callgraph.JSCallGraph; import com.ibm.wala.cast.js.ipa.callgraph.JSCallGraphUtil; import com.ibm.wala.cast.js.ipa.callgraph.JavaScriptEntryPoints; import com.ibm.wala.cast.js.loader.JavaScriptLoader; import com.ibm.wala.cast.js.loader.JavaScriptLoaderFactory; import com.ibm.wala.cast.js.translator.JavaScriptTranslatorFactory; import com.ibm.wala.cast.js.types.JavaScriptTypes; import com.ibm.wala.classLoader.Module; import com.ibm.wala.classLoader.SourceModule; import com.ibm.wala.client.AbstractAnalysisEngine; import com.ibm.wala.ipa.callgraph.AnalysisCacheImpl; import com.ibm.wala.ipa.callgraph.AnalysisOptions; import com.ibm.wala.ipa.callgraph.AnalysisScope; import com.ibm.wala.ipa.callgraph.CallGraph; import com.ibm.wala.ipa.callgraph.CallGraphBuilder; import com.ibm.wala.ipa.callgraph.CallGraphBuilderCancelException; import com.ibm.wala.ipa.callgraph.Entrypoint; import com.ibm.wala.ipa.callgraph.IAnalysisCacheView; import com.ibm.wala.ipa.callgraph.propagation.InstanceKey; import com.ibm.wala.ipa.callgraph.propagation.PointerAnalysis; import com.ibm.wala.ipa.cha.ClassHierarchyException; import com.ibm.wala.ipa.cha.IClassHierarchy; import com.ibm.wala.ipa.cha.SeqClassHierarchyFactory; import com.ibm.wala.util.CancelException; import com.ibm.wala.util.MonitorUtil.IProgressMonitor; import com.ibm.wala.util.collections.HashSetFactory; import com.ibm.wala.util.collections.Pair; import com.ibm.wala.util.debug.Assertions; public abstract class JavaScriptAnalysisEngine extends AbstractAnalysisEngine { protected JavaScriptLoaderFactory loaderFactory; protected JavaScriptTranslatorFactory translatorFactory; @Override public void buildAnalysisScope() { try { loaderFactory = new JavaScriptLoaderFactory(translatorFactory); SourceModule[] files = moduleFiles.toArray(new SourceModule[moduleFiles.size()]); scope = new CAstAnalysisScope(files, loaderFactory, Collections.singleton(JavaScriptLoader.JS)); } catch (IOException e) { Assertions.UNREACHABLE(e.toString()); } } @Override public IClassHierarchy buildClassHierarchy() { try { return setClassHierarchy(SeqClassHierarchyFactory.make(getScope(), loaderFactory, JavaScriptLoader.JS)); } catch (ClassHierarchyException e) { Assertions.UNREACHABLE(e.toString()); return null; } } public void setTranslatorFactory(JavaScriptTranslatorFactory factory) { this.translatorFactory = factory; } @Override public void setJ2SELibraries(JarFile[] libs) { Assertions.UNREACHABLE("Illegal to call setJ2SELibraries"); } @Override public void setJ2SELibraries(Module[] libs) { Assertions.UNREACHABLE("Illegal to call setJ2SELibraries"); } @Override protected Iterable makeDefaultEntrypoints(AnalysisScope scope, IClassHierarchy cha) { return new JavaScriptEntryPoints(cha, cha.getLoader(JavaScriptTypes.jsLoader)); } @Override public IAnalysisCacheView makeDefaultCache() { return new AnalysisCacheImpl(AstIRFactory.makeDefaultFactory()); } @Override public JSAnalysisOptions getDefaultOptions(Iterable roots) { final JSAnalysisOptions options = new JSAnalysisOptions(scope, roots); options.setUseConstantSpecificKeys(true); options.setUseStacksForLexicalScoping(true); return options; } public static class FieldBasedJavaScriptAnalysisEngine extends JavaScriptAnalysisEngine { public enum BuilderType { PESSIMISTIC, OPTIMISTIC, REFLECTIVE } private BuilderType builderType = BuilderType.OPTIMISTIC; /** * @return the builderType */ public BuilderType getBuilderType() { return builderType; } /** * @param builderType the builderType to set */ public void setBuilderType(BuilderType builderType) { this.builderType = builderType; } @Override public JSAnalysisOptions getDefaultOptions(Iterable roots) { return JSCallGraphUtil.makeOptions(scope, getClassHierarchy(), roots); } @Override protected CallGraphBuilder getCallGraphBuilder(final IClassHierarchy cha, AnalysisOptions options, final IAnalysisCacheView cache) { Set roots = HashSetFactory.make(); for(Entrypoint e : options.getEntrypoints()) { roots.add(e); } if (builderType.equals(BuilderType.OPTIMISTIC)) { ((JSAnalysisOptions)options).setHandleCallApply(false); } final FieldBasedCallGraphBuilder builder = builderType.equals(BuilderType.PESSIMISTIC)? new PessimisticCallGraphBuilder(getClassHierarchy(), options, makeDefaultCache(), true) : new OptimisticCallgraphBuilder(getClassHierarchy(), options, makeDefaultCache(), true); return new CallGraphBuilder() { private PointerAnalysis ptr; @Override public CallGraph makeCallGraph(AnalysisOptions options, IProgressMonitor monitor) throws IllegalArgumentException, CallGraphBuilderCancelException { Pair> dat; try { dat = builder.buildCallGraph(options.getEntrypoints(), monitor); } catch (CancelException e) { throw CallGraphBuilderCancelException.createCallGraphBuilderCancelException(e, null, null); } ptr = dat.snd; return dat.fst; } @Override public PointerAnalysis getPointerAnalysis() { return ptr; } @Override public IAnalysisCacheView getAnalysisCache() { return cache; } @Override public IClassHierarchy getClassHierarchy() { return cha; } }; } } public static class PropagationJavaScriptAnalysisEngine extends JavaScriptAnalysisEngine { @Override protected CallGraphBuilder getCallGraphBuilder(IClassHierarchy cha, AnalysisOptions options, IAnalysisCacheView cache) { return new ZeroCFABuilderFactory().make((JSAnalysisOptions) options, cache, cha, scope, false); } } }