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

com.google.gwt.dev.jjs.impl.JavaAstVerifier Maven / Gradle / Ivy

The newest version!
/*
 * Copyright 2014 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.dev.jjs.impl;

import com.google.gwt.dev.jjs.ast.Context;
import com.google.gwt.dev.jjs.ast.JClassType;
import com.google.gwt.dev.jjs.ast.JDeclaredType;
import com.google.gwt.dev.jjs.ast.JField;
import com.google.gwt.dev.jjs.ast.JFieldRef;
import com.google.gwt.dev.jjs.ast.JInterfaceType;
import com.google.gwt.dev.jjs.ast.JMethod;
import com.google.gwt.dev.jjs.ast.JMethodCall;
import com.google.gwt.dev.jjs.ast.JNode;
import com.google.gwt.dev.jjs.ast.JProgram;
import com.google.gwt.dev.jjs.ast.JThisRef;
import com.google.gwt.dev.jjs.ast.JVisitor;
import com.google.gwt.dev.jjs.ast.js.JsniFieldRef;
import com.google.gwt.dev.jjs.ast.js.JsniMethodRef;
import com.google.gwt.thirdparty.guava.common.collect.HashMultimap;
import com.google.gwt.thirdparty.guava.common.collect.Lists;
import com.google.gwt.thirdparty.guava.common.collect.Multimap;
import com.google.gwt.thirdparty.guava.common.collect.Sets;

import java.util.List;
import java.util.Set;

/**
 * Verifies that all the references from AST nodes to AST nodes are reachable from the
 * top of the AST.
 * 

* The purpose fo this pass is to verify the consistency of the AST after a specific pass has * run. */ public class JavaAstVerifier extends JVisitor { private Multimap membersByType = HashMultimap.create(); private Set seenTypeNames = Sets.newHashSet(); private Multimap seenMethodsByType = HashMultimap.create(); private Multimap seenFieldsByType = HashMultimap.create(); private JProgram program; JavaAstVerifier(JProgram program) { this.program = program; for (JDeclaredType type :program.getDeclaredTypes()) { membersByType.putAll(type, type.getMethods()); membersByType.putAll(type, type.getFields()); } } /** * Throws an assertion error if the AST for a program is not consistent. */ public static void assertProgramIsConsistent(JProgram program) { if (JavaAstVerifier.class.desiredAssertionStatus()) { new JavaAstVerifier(program).accept(program); } } public static void assertCorrectOverriddenOrder(JProgram program, JMethod method) { // The order of in the overriden set is most specific to least. List seenMethods = Lists.newArrayList(method); JMethod lastMethod = method; for (JMethod overriden : method.getOverriddenMethods()) { for (JMethod seenMethod : seenMethods) { assert !program.typeOracle.isSubType( seenMethod.getEnclosingType(), overriden.getEnclosingType()) : "Superclass method '" + seenMethod.getQualifiedName() + "' appeared before subclass method '" + overriden.getQualifiedName() + "' in '" + method.getQualifiedName() + "' overridden list"; } assert overriden.getEnclosingType() instanceof JInterfaceType || lastMethod.getEnclosingType() instanceof JClassType : "Class method '" + overriden.getQualifiedName() + "' appeared before after interface method '" + lastMethod.getQualifiedName() + "' in '" + method.getQualifiedName() + "' overridden list"; } } public static void assertCorrectOverridingOrder(JProgram program, JMethod method) { // The order of in the overriden set is most specific to least. List seenMethods = Lists.newArrayList(method); for (JMethod overriden : method.getOverridingMethods()) { for (JMethod seenMethod : seenMethods) { assert !program.typeOracle.isSubType( overriden.getEnclosingType(), seenMethod.getEnclosingType()) : "Subclass method '" + seenMethod.getQualifiedName() + "' appeared before superclass method '" + overriden.getQualifiedName() + "' in '" + method.getQualifiedName() + "' overriding list"; } } } @Override public void endVisit(JClassType x, Context ctx) { assertNotSeenBefore(x); assertJsoCorrectness(x); } @Override public void endVisit(JField x, Context ctx) { JDeclaredType enclosingType = x.getEnclosingType(); String fieldName = x.getName(); assert !seenFieldsByType.containsEntry(enclosingType, fieldName) : "Field " + x + " is duplicated."; seenFieldsByType.put(enclosingType, fieldName); } @Override public void endVisit(JFieldRef x, Context ctx) { assertReferencedFieldIsInAst(x); } @Override public void endVisit(JInterfaceType x, Context ctx) { assertNotSeenBefore(x); } JMethod currentMethod; @Override public boolean visit(JMethod x, Context ctx) { assert currentMethod == null; currentMethod = x; return true; } @Override public void endVisit(JMethod x, Context ctx) { JDeclaredType enclosingType = x.getEnclosingType(); String methodSignature = x.getSignature(); assert !seenMethodsByType.containsEntry(enclosingType, methodSignature) : "Method " + x + " is duplicated."; seenMethodsByType.put(enclosingType, methodSignature); assertCorrectOverriddenOrder(program, x); assertCorrectOverridingOrder(program, x); assert currentMethod == x; currentMethod = null; } @Override public void endVisit(JMethodCall x, Context ctx) { assertCalledMethodIsInAst(x); } public void endVisit(JThisRef x, Context ctx) { assert !currentMethod.isStatic() || currentMethod.isConstructor() : "JThisRef found in static method " + currentMethod; } @Override public void endVisit(JsniFieldRef x, Context ctx) { assertReferencedFieldIsInAst(x); } @Override public void endVisit(JsniMethodRef x, Context ctx) { assertCalledMethodIsInAst(x); } private void assertCalledMethodIsInAst(JMethodCall x) { if (x.getTarget() == JMethod.NULL_METHOD) { return; } assert membersByType.containsEntry(x.getTarget().getEnclosingType(), x.getTarget()) : "Method " + x.getTarget() + " is called but is not part of the AST"; JMethod staticImpl = program.getStaticImpl(x.getTarget()); assert staticImpl == null || membersByType.containsEntry(staticImpl.getEnclosingType(), staticImpl) : "Method " + staticImpl + " is the static implementation of " + x.getTarget() + " but is not part of the AST"; } private void assertReferencedFieldIsInAst(JFieldRef x) { if (x.getField() == JField.NULL_FIELD) { return; } assert membersByType.containsEntry(x.getField().getEnclosingType(), x.getField()) : "Field " + x.getTarget() + " is referenced but is not part of the AST"; } private void assertJsoCorrectness(JClassType x) { boolean isJSOorSubclassOfJSO = false; for (JClassType current = x; current != null; current = current.getSuperClass()) { if (current.getName().equals(JProgram.JAVASCRIPTOBJECT)) { isJSOorSubclassOfJSO = true; break; } } assert isJSOorSubclassOfJSO == x.isJsoType() : x.isJsoType() ? "Type " + x.getName() + " is considered a Jso but is not subclass of " + JProgram.JAVASCRIPTOBJECT : "Type " + x.getName() + " is subclass of " + JProgram.JAVASCRIPTOBJECT + " but is not " + "considered a Jso"; } private void assertNotSeenBefore(JDeclaredType type) { assert !seenTypeNames.contains(type.getName()) : "Found two types with same name " + type.getName(); seenTypeNames.add(type.getName()); } }





© 2015 - 2024 Weber Informatics LLC | Privacy Policy