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

org.codehaus.janino.Descriptor Maven / Gradle / Ivy

Go to download

The "JANINO" implementation of the "commons-compiler" API: Super-small, super-fast, independent from the JDK's "tools.jar".

There is a newer version: 3.1.12
Show newest version

/*
 * Janino - An embedded Java[TM] compiler
 *
 * Copyright (c) 2001-2010, Arno Unkrig
 * All rights reserved.
 *
 * Redistribution and use in source and binary forms, with or without modification, are permitted provided that the
 * following conditions are met:
 *
 *    1. Redistributions of source code must retain the above copyright notice, this list of conditions and the
 *       following disclaimer.
 *    2. Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the
 *       following disclaimer in the documentation and/or other materials provided with the distribution.
 *    3. The name of the author may not be used to endorse or promote products derived from this software without
 *       specific prior written permission.
 *
 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
 * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL
 * THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
 * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
 * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
 * POSSIBILITY OF SUCH DAMAGE.
 */

package org.codehaus.janino;

import java.util.Collections;
import java.util.HashMap;
import java.util.Map;

/**
 * Helper class that defines useful methods for handling "field descriptors"
 * (JVMS 4.3.2) and "method descriptors" (JVMS 4.3.3).

* Typical descriptors are: *

    *
  • I Integer *
  • [I Array of integer *
  • Lpkg1/pkg2/Cls; Class *
  • Lpkg1/pkg2/Outer$Inner; Member class *
*/ @SuppressWarnings({ "rawtypes", "unchecked" }) public final class Descriptor { private Descriptor() {} /** @return Whether this {@link Descriptor} describes a reference (i.e. non-primitive) type */ public static boolean isReference(String d) { return d.length() > 1; } /** * @return Whether this {@link Descriptor} describes a class or an interface (and not an array or a primitive type) */ public static boolean isClassOrInterfaceReference(String d) { return d.charAt(0) == 'L'; } /** @return Whether this {@link Descriptor} describes an array type */ public static boolean isArrayReference(String d) { return d.charAt(0) == '['; } /** * @return The descriptor of the component of the array type {@code d} * @throws JaninoRuntimeException {@code d} does not describe an array type */ public static String getComponentDescriptor(String d) { if (d.charAt(0) != '[') { throw new JaninoRuntimeException( "Cannot determine component descriptor from non-array descriptor \"" + d + "\"" ); } return d.substring(1); } /** * @return The number of slots (1 or two) that a value of the type described by {@code d} occupies on the operand * stack or in the local variable array, or 0 iff {@code d} describes the type VOID */ public static short size(String d) { if (d.equals(Descriptor.VOID)) return 0; if (Descriptor.hasSize1(d)) return 1; if (Descriptor.hasSize2(d)) return 2; throw new JaninoRuntimeException("No size defined for type \"" + Descriptor.toString(d) + "\""); } /** @return {@code true} iff {@code d} describes a primitive type except LONG and DOUBLE, or a reference type */ public static boolean hasSize1(String d) { if (d.length() == 1) return "BCFISZ".indexOf(d) != -1; return Descriptor.isReference(d); } /** @return {@code true} iff {@code d} LONG or DOUBLE */ public static boolean hasSize2(String d) { return d.equals(Descriptor.LONG) || d.equals(Descriptor.DOUBLE); } /** * Pretty-prints the given descriptor. * * @param d A valid field or method descriptor */ public static String toString(String d) { int idx = 0; StringBuilder sb = new StringBuilder(); if (d.charAt(0) == '(') { ++idx; sb.append("("); while (idx < d.length() && d.charAt(idx) != ')') { if (idx != 1) sb.append(", "); idx = Descriptor.toString(d, idx, sb); } if (idx >= d.length()) throw new JaninoRuntimeException("Invalid descriptor \"" + d + "\""); sb.append(") => "); ++idx; } Descriptor.toString(d, idx, sb); return sb.toString(); } private static int toString(String d, int idx, StringBuilder sb) { int dimensions = 0; while (idx < d.length() && d.charAt(idx) == '[') { ++dimensions; ++idx; } if (idx >= d.length()) throw new JaninoRuntimeException("Invalid descriptor \"" + d + "\""); switch (d.charAt(idx)) { case 'L': { int idx2 = d.indexOf(';', idx); if (idx2 == -1) throw new JaninoRuntimeException("Invalid descriptor \"" + d + "\""); sb.append(d.substring(idx + 1, idx2).replace('/', '.')); idx = idx2; } break; case 'V': sb.append("void"); break; case 'B': sb.append("byte"); break; case 'C': sb.append("char"); break; case 'D': sb.append("double"); break; case 'F': sb.append("float"); break; case 'I': sb.append("int"); break; case 'J': sb.append("long"); break; case 'S': sb.append("short"); break; case 'Z': sb.append("boolean"); break; default: throw new JaninoRuntimeException("Invalid descriptor \"" + d + "\""); } for (; dimensions > 0; --dimensions) sb.append("[]"); return idx + 1; } /** Converts a class name as defined by "Class.getName()" into a descriptor. */ public static String fromClassName(String className) { String res = (String) Descriptor.CLASS_NAME_TO_DESCRIPTOR.get(className); if (res != null) { return res; } if (className.startsWith("[")) return className.replace('.', '/'); return 'L' + className.replace('.', '/') + ';'; } /** * Convert a class name in the "internal form" as described in JVMS 4.2 into a descriptor. *

* Also implement the encoding of array types as described in JVMS 4.4.1. */ public static String fromInternalForm(String internalForm) { if (internalForm.charAt(0) == '[') return internalForm; return 'L' + internalForm + ';'; } /** Converts a field descriptor into a class name as defined by {@link Class#getName()}. */ public static String toClassName(String d) { String res = (String) Descriptor.DESCRIPTOR_TO_CLASSNAME.get(d); if (res != null) { return res; } char firstChar = d.charAt(0); if (firstChar == 'L' && d.endsWith(";")) { // Class or interface -- convert "Ljava/lang/String;" to "java.lang.String". return d.substring(1, d.length() - 1).replace('/', '.'); } if (firstChar == '[') { // Array type -- convert "[Ljava/lang/String;" to "[Ljava.lang.String;". return d.replace('/', '.'); } throw new JaninoRuntimeException("(Invalid field descriptor \"" + d + "\")"); } /** Converts a descriptor into the "internal form" as defined by JVMS 4.2. */ public static String toInternalForm(String d) { if (d.charAt(0) != 'L') { throw new JaninoRuntimeException( "Attempt to convert non-class descriptor \"" + d + "\" into internal form" ); } return d.substring(1, d.length() - 1); } /** @return Whether {@code d} describes a primitive type or VOID */ public static boolean isPrimitive(String d) { return d.length() == 1 && "VBCDFIJSZ".indexOf(d.charAt(0)) != -1; } /** @return Whether {@code d} describes a primitive type except BOOLEAN and VOID */ public static boolean isPrimitiveNumeric(String d) { return d.length() == 1 && "BDFIJSC".indexOf(d.charAt(0)) != -1; } /** * Returns the package name of a class or interface reference descriptor, * or null if the class or interface is declared in the * default package. */ public static String getPackageName(String d) { if (d.charAt(0) != 'L') { throw new JaninoRuntimeException("Attempt to get package name of non-class descriptor \"" + d + "\""); } int idx = d.lastIndexOf('/'); return idx == -1 ? null : d.substring(1, idx).replace('/', '.'); } /** Checks whether two reference types are declared in the same package. */ public static boolean areInSamePackage(String d1, String d2) { String packageName1 = Descriptor.getPackageName(d1); String packageName2 = Descriptor.getPackageName(d2); return packageName1 == null ? packageName2 == null : packageName1.equals(packageName2); } /** The field descriptor for the type VOID. */ public static final String VOID = "V"; // Primitive types. /** The field descriptor for the primitive type BYTE. */ public static final String BYTE = "B"; /** The field descriptor for the primitive type CHAR. */ public static final String CHAR = "C"; /** The field descriptor for the primitive type DOUBLE. */ public static final String DOUBLE = "D"; /** The field descriptor for the primitive type FLOAT. */ public static final String FLOAT = "F"; /** The field descriptor for the primitive type INT. */ public static final String INT = "I"; /** The field descriptor for the primitive type LONG. */ public static final String LONG = "J"; /** The field descriptor for the primitive type SHORT. */ public static final String SHORT = "S"; /** The field descriptor for the primitive type BOOLEAN. */ public static final String BOOLEAN = "Z"; // Annotations. /** The field descriptor for the annotation {@link java.lang.Override}. */ public static final String JAVA_LANG_OVERRIDE = "Ljava/lang/Override;"; // Classes. /** The field descriptor for the class {@link java.lang.AssertionError}. */ public static final String JAVA_LANG_ASSERTIONERROR = "Ljava/lang/AssertionError;"; /** The field descriptor for the class {@link java.lang.Boolean}. */ public static final String JAVA_LANG_BOOLEAN = "Ljava/lang/Boolean;"; /** The field descriptor for the class {@link java.lang.Byte}. */ public static final String JAVA_LANG_BYTE = "Ljava/lang/Byte;"; /** The field descriptor for the class {@link java.lang.Character}. */ public static final String JAVA_LANG_CHARACTER = "Ljava/lang/Character;"; /** The field descriptor for the class {@link java.lang.Class}. */ public static final String JAVA_LANG_CLASS = "Ljava/lang/Class;"; /** The field descriptor for the class {@link java.lang.Double}. */ public static final String JAVA_LANG_DOUBLE = "Ljava/lang/Double;"; /** The field descriptor for the class {@link java.lang.Exception}. */ public static final String JAVA_LANG_EXCEPTION = "Ljava/lang/Exception;"; /** The field descriptor for the class {@link java.lang.Error}. */ public static final String JAVA_LANG_ERROR = "Ljava/lang/Error;"; /** The field descriptor for the class {@link java.lang.Float}. */ public static final String JAVA_LANG_FLOAT = "Ljava/lang/Float;"; /** The field descriptor for the class {@link java.lang.Integer}. */ public static final String JAVA_LANG_INTEGER = "Ljava/lang/Integer;"; /** The field descriptor for the class {@link java.lang.Long}. */ public static final String JAVA_LANG_LONG = "Ljava/lang/Long;"; /** The field descriptor for the class {@link java.lang.Object}. */ public static final String JAVA_LANG_OBJECT = "Ljava/lang/Object;"; /** The field descriptor for the class {@link java.lang.RuntimeException}. */ public static final String JAVA_LANG_RUNTIMEEXCEPTION = "Ljava/lang/RuntimeException;"; /** The field descriptor for the class {@link java.lang.Short}. */ public static final String JAVA_LANG_SHORT = "Ljava/lang/Short;"; /** The field descriptor for the class {@link java.lang.String}. */ public static final String JAVA_LANG_STRING = "Ljava/lang/String;"; /** The field descriptor for the class {@link java.lang.StringBuilder}. */ public static final String JAVA_LANG_STRINGBUILDER = "Ljava/lang/StringBuilder;"; // Since 1.5! /** The field descriptor for the class {@link java.lang.Throwable}. */ public static final String JAVA_LANG_THROWABLE = "Ljava/lang/Throwable;"; // Interfaces. /** The field descriptor for the interface {@link java.io.Serializable}. */ public static final String JAVA_IO_SERIALIZABLE = "Ljava/io/Serializable;"; /** The field descriptor for the interface {@link java.lang.Cloneable}. */ public static final String JAVA_LANG_CLONEABLE = "Ljava/lang/Cloneable;"; /** The field descriptor for the interface {@link java.lang.Iterable}. */ public static final String JAVA_LANG_ITERABLE = "Ljava/lang/Iterable;"; /** The field descriptor for the interface {@link java.util.Iterator}. */ public static final String JAVA_UTIL_ITERATOR = "Ljava/util/Iterator;"; private static final Map DESCRIPTOR_TO_CLASSNAME; static { Map m = new HashMap(); m.put(Descriptor.VOID, "void"); // Primitive types. m.put(Descriptor.BYTE, "byte"); m.put(Descriptor.CHAR, "char"); m.put(Descriptor.DOUBLE, "double"); m.put(Descriptor.FLOAT, "float"); m.put(Descriptor.INT, "int"); m.put(Descriptor.LONG, "long"); m.put(Descriptor.SHORT, "short"); m.put(Descriptor.BOOLEAN, "boolean"); // Annotations. m.put(Descriptor.JAVA_LANG_OVERRIDE, "java.lang.Override"); // Classes. m.put(Descriptor.JAVA_LANG_ASSERTIONERROR, "java.lang.AssertionError"); m.put(Descriptor.JAVA_LANG_BOOLEAN, "java.lang.Boolean"); m.put(Descriptor.JAVA_LANG_BYTE, "java.lang.Byte"); m.put(Descriptor.JAVA_LANG_CHARACTER, "java.lang.Character"); m.put(Descriptor.JAVA_LANG_CLASS, "java.lang.Class"); m.put(Descriptor.JAVA_LANG_DOUBLE, "java.lang.Double"); m.put(Descriptor.JAVA_LANG_EXCEPTION, "java.lang.Exception"); m.put(Descriptor.JAVA_LANG_ERROR, "java.lang.Error"); m.put(Descriptor.JAVA_LANG_FLOAT, "java.lang.Float"); m.put(Descriptor.JAVA_LANG_INTEGER, "java.lang.Integer"); m.put(Descriptor.JAVA_LANG_LONG, "java.lang.Long"); m.put(Descriptor.JAVA_LANG_OBJECT, "java.lang.Object"); m.put(Descriptor.JAVA_LANG_RUNTIMEEXCEPTION, "java.lang.RuntimeException"); m.put(Descriptor.JAVA_LANG_SHORT, "java.lang.Short"); m.put(Descriptor.JAVA_LANG_STRING, "java.lang.String"); m.put(Descriptor.JAVA_LANG_STRINGBUILDER, "java.lang.StringBuilder"); m.put(Descriptor.JAVA_LANG_THROWABLE, "java.lang.Throwable"); // Interfaces. m.put(Descriptor.JAVA_IO_SERIALIZABLE, "java.io.Serializable"); m.put(Descriptor.JAVA_LANG_CLONEABLE, "java.lang.Cloneable"); m.put(Descriptor.JAVA_LANG_ITERABLE, "java.lang.Iterable"); m.put(Descriptor.JAVA_UTIL_ITERATOR, "java.util.Iterator"); DESCRIPTOR_TO_CLASSNAME = Collections.unmodifiableMap(m); } private static final Map CLASS_NAME_TO_DESCRIPTOR; static { Map m = new HashMap(); for (Map.Entry e : Descriptor.DESCRIPTOR_TO_CLASSNAME.entrySet()) { m.put(e.getValue(), e.getKey()); } CLASS_NAME_TO_DESCRIPTOR = Collections.unmodifiableMap(m); } }





© 2015 - 2024 Weber Informatics LLC | Privacy Policy