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

org.codehaus.groovy.control.OptimizerVisitor Maven / Gradle / Ivy

There is a newer version: 5.0.0-alpha-11
Show newest version
/*
 *  Licensed to the Apache Software Foundation (ASF) under one
 *  or more contributor license agreements.  See the NOTICE file
 *  distributed with this work for additional information
 *  regarding copyright ownership.  The ASF licenses this file
 *  to you 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 org.codehaus.groovy.control;

import org.codehaus.groovy.ast.ClassCodeExpressionTransformer;
import org.codehaus.groovy.ast.ClassNode;
import org.codehaus.groovy.ast.FieldNode;
import org.codehaus.groovy.ast.expr.ClosureExpression;
import org.codehaus.groovy.ast.expr.ConstantExpression;
import org.codehaus.groovy.ast.expr.Expression;
import org.objectweb.asm.Opcodes;

import java.util.ArrayList;
import java.util.HashMap;
import java.util.LinkedList;
import java.util.List;
import java.util.Map;

import static org.codehaus.groovy.ast.ClassHelper.isPrimitiveType;
import static org.codehaus.groovy.ast.ClassHelper.isGroovyObjectType;

/**
 * Visitor to produce several optimizations:
 * 
    *
  • to replace numbered constants with references to static fields
  • *
  • remove superfluous references to GroovyObject interface
  • *
*/ public class OptimizerVisitor extends ClassCodeExpressionTransformer { private ClassNode currentClass; private SourceUnit source; // TODO make @CS lookup smarter so that we don't need both these maps private final Map const2Objects = new HashMap(); private final Map const2Prims = new HashMap(); private int index; private final List missingFields = new LinkedList(); public OptimizerVisitor(CompilationUnit cu) { } public void visitClass(ClassNode node, SourceUnit source) { this.currentClass = node; this.source = source; const2Objects.clear(); const2Prims.clear(); missingFields.clear(); index = 0; super.visitClass(node); addMissingFields(); pruneUnneededGroovyObjectInterface(node); } private void pruneUnneededGroovyObjectInterface(ClassNode node) { ClassNode superClass = node.getSuperClass(); boolean isSuperGroovy = superClass.isDerivedFromGroovyObject(); if (isSuperGroovy) { ClassNode[] interfaces = node.getInterfaces(); boolean needsFix = false; for (ClassNode classNode : interfaces) { if (isGroovyObjectType(classNode)) { needsFix = true; break; } } if (needsFix) { List newInterfaces = new ArrayList(interfaces.length); for (ClassNode classNode : interfaces) { if (!isGroovyObjectType(classNode)) { newInterfaces.add(classNode); } } node.setInterfaces(newInterfaces.toArray(ClassNode.EMPTY_ARRAY)); } } } private void addMissingFields() { for (FieldNode f : missingFields) { currentClass.addField(f); } } private void setConstField(ConstantExpression constantExpression) { final Object n = constantExpression.getValue(); if (!(n instanceof Number)) return; if (n instanceof Integer || n instanceof Double) return; if (n instanceof Long && (0L == (Long) n || 1L == (Long) n)) return; // LCONST_0, LCONST_1 boolean isPrimitive = isPrimitiveType(constantExpression.getType()); FieldNode field = isPrimitive ? const2Prims.get(n) : const2Objects.get(n); if (field != null) { constantExpression.setConstantName(field.getName()); return; } String name; do { name = "$const$" + index++; } while (currentClass.getDeclaredField(name) != null); // TODO consider moving initcode to and remaking field final field = new FieldNode(name, Opcodes.ACC_PRIVATE | Opcodes.ACC_STATIC | Opcodes.ACC_SYNTHETIC, constantExpression.getType(), currentClass, constantExpression); field.setSynthetic(true); missingFields.add(field); constantExpression.setConstantName(field.getName()); if (isPrimitive) { const2Prims.put(n, field); } else { const2Objects.put(n, field); } } @Override public Expression transform(Expression exp) { if (exp == null) return null; if (!currentClass.isInterface() && exp.getClass() == ConstantExpression.class) { setConstField((ConstantExpression) exp); } return exp.transformExpression(this); } @Override protected SourceUnit getSourceUnit() { return source; } @Override public void visitClosureExpression(ClosureExpression expression) { /* * GROOVY-3339 - do nothing - so that numbers don't get replaced by cached constants in closure classes */ } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy