/* * Copyright 2013 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ package com.google.gwt.core.ext.soyc.coderef; import com.google.gwt.core.ext.soyc.impl.DependencyRecorder; import com.google.gwt.dev.jjs.ast.JDeclaredType; import com.google.gwt.dev.jjs.ast.JMethod; import com.google.gwt.dev.jjs.ast.JProgram; import com.google.gwt.thirdparty.guava.common.collect.Maps; import java.io.OutputStream; import java.util.Map; /** * Builds the model for the (new) soyc through reading method dependencies. The code model is in a * mapping from fully qualified class names (eg. com.google.gwt.MyClass, see * {@link com.google.gwt.dev.jjs.ast.JDeclaredType#getName()}) to class descriptors. Where * each class descriptor has methods and their dependents. * */ public class DependencyGraphRecorder extends DependencyRecorder { private Map<String, ClassDescriptor> codeModel = Maps.newTreeMap(); private int nextUniqueId = 0; private JProgram jProgram; public DependencyGraphRecorder(OutputStream out, JProgram jProgram) { super(out); this.jProgram = jProgram; } protected int nextPointerId() { return ++nextUniqueId; } /** * Returns the code model that maps fully qualified class names (eg. com.google.gwt.MyClass, see * {@link com.google.gwt.dev.jjs.ast.JDeclaredType#getName()}) to class descriptors. */ public Map<String, ClassDescriptor> getCodeModel() { return this.codeModel; } @Override public void startDependencyGraph(String name, String extendz) { super.startDependencyGraph(name, extendz); } @Override protected void printMethodDependencyBetween(JMethod curMethod, JMethod depMethod) { super.printMethodDependencyBetween(curMethod, depMethod); methodDescriptorFrom(curMethod).addDependant(methodDescriptorFrom(depMethod)); } protected String signatureFor(JMethod method) { JMethod instanceMethod = jProgram.instanceMethodForStaticImpl(method); if (instanceMethod == null) { return method.getSignature(); } return instanceMethod.getSignature(); } public MethodDescriptor methodDescriptorFrom(JMethod method) { MethodDescriptor mth = classDescriptorFrom(method.getEnclosingType()) .methodFrom(method, signatureFor(method)); if (!isValid(mth.getUniqueId())) { mth.setUniqueId(nextPointerId()); } return mth; } protected boolean isValid(int n) { return n > 0; } /** * Returns a class descriptor from a JDeclaredType. If the class descriptor is not in the code * model, it will be added. */ public ClassDescriptor classDescriptorFrom(JDeclaredType classType) { // JDeclaredType.getName returns the fully qualified name ClassDescriptor classDescriptor = codeModel.get(classType.getName()); if (classDescriptor == null) { classDescriptor = ClassDescriptor.from(classType); codeModel.put(classType.getName(), classDescriptor); } return classDescriptor; } }