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

org.jruby.util.CodegenUtils Maven / Gradle / Ivy

/*
 * CodegenUtils.java
 *
 * Created on January 31, 2007, 11:54 AM
 *
 * To change this template, choose Tools | Template Manager
 * and open the template in the editor.
 */

package org.jruby.util;

import org.objectweb.asm.AnnotationVisitor;
import org.objectweb.asm.Type;

import javax.lang.model.element.Name;
import java.util.Arrays;
import java.util.HashMap;
import java.util.Map;

/**
 *
 * @author headius
 */
public class CodegenUtils {
    /**
     * Creates a dotted class name from a path/package name
     */
    public static String c(String p) {
        return p.replace('/', '.');
    }

    /**
     * Creates a class path name, from a Class.
     */
    public static String p(Class n) {
        return n.getName().replace('.','/');
    }

    /**
     * Creates a class identifier of form Labc/abc;, from a Class.
     */
    public static String ci(Class n) {
        if (n.isArray()) {
            n = n.getComponentType();
            if (n.isPrimitive()) {
                if (n == Byte.TYPE) {
                    return "[B";
                } else if (n == Boolean.TYPE) {
                    return "[Z";
                } else if (n == Short.TYPE) {
                    return "[S";
                } else if (n == Character.TYPE) {
                    return "[C";
                } else if (n == Integer.TYPE) {
                    return "[I";
                } else if (n == Float.TYPE) {
                    return "[F";
                } else if (n == Double.TYPE) {
                    return "[D";
                } else if (n == Long.TYPE) {
                    return "[J";
                } else {
                    throw new RuntimeException("Unrecognized type in compiler: " + n.getName());
                }
            } else {
                return '[' + ci(n);
            }
        } else {
            if (n.isPrimitive()) {
                if (n == Byte.TYPE) {
                    return "B";
                } else if (n == Boolean.TYPE) {
                    return "Z";
                } else if (n == Short.TYPE) {
                    return "S";
                } else if (n == Character.TYPE) {
                    return "C";
                } else if (n == Integer.TYPE) {
                    return "I";
                } else if (n == Float.TYPE) {
                    return "F";
                } else if (n == Double.TYPE) {
                    return "D";
                } else if (n == Long.TYPE) {
                    return "J";
                } else if (n == Void.TYPE) {
                    return "V";
                } else {
                    throw new RuntimeException("Unrecognized type in compiler: " + n.getName());
                }
            } else {
                return 'L' + p(n) + ';';
            }
        }
    }

    /**
     * Creates a human-readable representation, from a Class.
     */
    public static String human(Class n) {
        return n.getCanonicalName();
    }

    public static String humanShort(Class n) {
        return n.getSimpleName();
    }

    /**
     * Create a method signature from the given param types and return values
     */
    public static String sig(Class retval, Class... params) {
        return sigParams(new StringBuilder(), null, params).append( ci(retval) ).toString();
    }

    public static String sig(Class[] retvalParams) {
        Class[] justParams = new Class[retvalParams.length - 1];
        System.arraycopy(retvalParams, 1, justParams, 0, justParams.length);
        return sigParams(new StringBuilder(), null, justParams).append( ci(retvalParams[0]) ).toString();
    }

    public static String sig(Class retval, String descriptor, Class... params) {
        return sigParams(new StringBuilder(), descriptor, params).append( ci(retval) ).toString();
    }

    @Deprecated // not used
    public static String sigParams(Class... params) {
        return sigParams(new StringBuilder(), null, params).toString();
    }

    @Deprecated // not used
    public static String sigParams(String descriptor, Class... params) {
        return sigParams(new StringBuilder(), descriptor, params).toString();
    }

    private static StringBuilder sigParams(final StringBuilder str,
        final String descriptor, final Class... params) {
        str.append('(');
        if ( descriptor != null ) str.append(descriptor);

        for (int i = 0; i < params.length; i++) {
            str.append( ci( params[i] ) );
        }

        return str.append(')');
    }

    public static String pretty(Class retval, Class... params) {
        return prettyParams(params) + human(retval);
    }

    public static CharSequence prettyParams(final Class... types) {
        return prettyParams(new StringBuilder(), types);
    }

    public static StringBuilder prettyParams(final StringBuilder str, final Class... types) {
        final int len1 = types.length - 1;
        str.append('(');

        for ( int i = 0; i <= len1; i++ ) {
            str.append( human( types[i] ) );
            if ( i < len1 ) str.append(',');
        }

        return str.append(')');
    }

    public static String prettyShortParams(final Class... types) {
        final int len1 = types.length - 1;

        StringBuilder signature = new StringBuilder();
        signature.append('(');

        for (int i = 0; i <= len1; i++) {
            signature.append( humanShort(types[i]) );
            if ( i < len1 ) signature.append(',');
        }

        signature.append(')');
        return signature.toString();
    }

    public static Class[] params(Class... classes) {
        return classes;
    }

    public static Class[] params(Class cls, int times) {
        Class[] classes = new Class[times];
        Arrays.fill(classes, cls);
        return classes;
    }

    public static Class[] params(Class cls1, Class clsFill, int times) {
        Class[] classes = new Class[times + 1];
        Arrays.fill(classes, 1, 1 + times, clsFill);
        classes[0] = cls1;
        return classes;
    }

    public static Class[] params(Class cls1, Class cls2, Class clsFill, int times) {
        Class[] classes = new Class[times + 2];
        Arrays.fill(classes, 2, 2 + times, clsFill);
        classes[0] = cls1;
        classes[1] = cls2;
        return classes;
    }

    public static Class[] params(Class cls1, Class cls2, Class cls3, Class clsFill, int times) {
        Class[] classes = new Class[times + 3];
        Arrays.fill(classes, 3, 3 + times, clsFill);
        classes[0] = cls1;
        classes[1] = cls2;
        classes[2] = cls3;
        return classes;
    }

    public static Class[] params(Class cls1, Class cls2, Class cls3, Class cls4, Class clsFill, int times) {
        Class[] classes = new Class[times + 4];
        Arrays.fill(classes, 4, 4 + times, clsFill);
        classes[0] = cls1;
        classes[1] = cls2;
        classes[2] = cls3;
        classes[3] = cls4;
        return classes;
    }

    public static Class[] params(Class cls1, Class clsFill, int times, Class clsTail) {
        Class[] classes = new Class[times + 2];
        Arrays.fill(classes, 1, 1 + times, clsFill);
        classes[0] = cls1;
        classes[times + 1] = clsTail;
        return classes;
    }

    public static Class[] params(Class cls1, Class cls2, Class clsFill, int times, Class clsTail) {
        Class[] classes = new Class[times + 3];
        Arrays.fill(classes, 2, 2 + times, clsFill);
        classes[0] = cls1;
        classes[1] = cls2;
        classes[times + 2] = clsTail;
        return classes;
    }

    public static Class[] params(Class cls1, Class cls2, Class cls3, Class clsFill, int times, Class clsTail) {
        Class[] classes = new Class[times + 4];
        Arrays.fill(classes, 3, 3 + times, clsFill);
        classes[0] = cls1;
        classes[1] = cls2;
        classes[2] = cls3;
        classes[times + 3] = clsTail;
        return classes;
    }

    public static Class[] params(Class cls1, Class cls2, Class cls3, Class cls4, Class clsFill, int times, Class clsTail) {
        Class[] classes = new Class[times + 5];
        Arrays.fill(classes, 4, 4 + times, clsFill);
        classes[0] = cls1;
        classes[1] = cls2;
        classes[2] = cls3;
        classes[3] = cls4;
        classes[times + 4] = clsTail;
        return classes;
    }

    public static Class[] params(Class cls1, Class[] clsFills, int times) {
        Class[] classes = new Class[clsFills.length * times + 1];
        classes[0] = cls1;
        for (int i = 0; i < times; i++) {
            int base = i * clsFills.length + 1;
            for (int j = 0; j < clsFills.length; j++) {
                classes[base + j] = clsFills[j];
            }
        }
        return classes;
    }

    public static String getAnnotatedBindingClassName(Name javaMethodName, CharSequence typeName, boolean isStatic, int required, int optional, boolean multi, boolean framed) {
        return getAnnotatedBindingClassName(javaMethodName.toString(), typeName, isStatic, required, optional, multi, framed);
    }

    public static String getAnnotatedBindingClassName(String javaMethodName, CharSequence typeName, boolean isStatic, int required, int optional, boolean multi, boolean framed) {
        if (multi) {
            return new StringBuilder(typeName.length() + 8 + 3 + javaMethodName.length())
                    .append(typeName)
                    .append("$INVOKER").append(isStatic ? "$s$" : "$i$")
                    .append(javaMethodName)
                    .toString();
        }
        return new StringBuilder(typeName.length() + 8 + 3 + 4 + javaMethodName.length())
                .append(typeName)
                .append("$INVOKER").append(isStatic ? "$s$" : "$i$")
                .append(required).append('$').append(optional).append('$')
                .append(javaMethodName)
                .toString();
    }

    public static void visitAnnotationFields(AnnotationVisitor visitor, Map fields) {
        try {
            for (Map.Entry fieldEntry : fields.entrySet()) {
                Object value = fieldEntry.getValue();
                String key = fieldEntry.getKey();

                if (value instanceof Map) {
                    @SuppressWarnings("unchecked")
                    Map> nestedAnnotationMap = (Map>) value;

                    for (Map.Entry> nestedAnnotation : nestedAnnotationMap.entrySet()) {
                        AnnotationVisitor annotationV;

                        annotationV = visitor.visitAnnotation(key, Type.getType(nestedAnnotation.getKey()).getDescriptor());
                        visitAnnotationFields(annotationV, nestedAnnotation.getValue());
                        annotationV.visitEnd();
                    }
                } else if (value.getClass().isArray()) {
                    Object[] values = (Object[]) value;

                    AnnotationVisitor arrayV = visitor.visitArray(key);
                    for (int i = 0; i < values.length; i++) {
                        Map map = new HashMap();
                        map.put(null, values[i]);
                        visitAnnotationFields(arrayV, map);
                    }
                    arrayV.visitEnd();
                } else if (value.getClass().isEnum()) {
                    visitor.visitEnum(key, ci(value.getClass()), value.toString());
                } else if (value instanceof Class) {
                    visitor.visit(key, Type.getType((Class) value));
                } else {
                    visitor.visit(key, value);
                }
            }
        } catch (ClassCastException e) {
            throw new InvalidAnnotationDescriptorException("Fields "
                + fields
                + " did not match annotation format.  See CodegenUtils#visitAnnotationFields for format",
                e);
        }
    }

    public static Class getBoxType(Class type) {
        if (type == int.class) {
            return Integer.class;
        } else if (type == byte.class) {
            return Byte.class;
        } else if (type == short.class) {
            return Short.class;
        } else if (type == char.class) {
            return Character.class;
        } else if (type == long.class) {
            return Long.class;
        } else if (type == float.class) {
            return Float.class;
        } else if (type == double.class) {
            return Double.class;
        } else if (type == boolean.class) {
            return Boolean.class;
        } else if (type == void.class) {
            return Void.class;
        } else {
            throw new RuntimeException("Not a native type: " + type);
        }
    }

    public static class InvalidAnnotationDescriptorException extends RuntimeException {
        public InvalidAnnotationDescriptorException() {
        }

        public InvalidAnnotationDescriptorException(String s) {
            super(s);
        }

        public InvalidAnnotationDescriptorException(String s, Throwable throwable) {
            super(s, throwable);
        }

        public InvalidAnnotationDescriptorException(Throwable throwable) {
            super(throwable);
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy