From 7a21a12ccc785fee5e8105fdf15d0e6667c6cffd Mon Sep 17 00:00:00 2001 From: robm Date: Wed, 23 Jan 2013 17:54:34 +0000 Subject: [PATCH] 8004729: Add java.lang.reflect.Parameter and related changes for parameter reflection Reviewed-by: darcy, forax, psandoz, dholmes, tbell --- make/java/java/Exportedfiles.gmk | 4 +- make/java/java/FILES_c.gmk | 3 +- make/java/java/mapfile-vers | 3 +- makefiles/mapfiles/libjava/mapfile-vers | 3 +- .../java/lang/reflect/Constructor.java | 7 +- .../classes/java/lang/reflect/Executable.java | 69 +++- .../classes/java/lang/reflect/Method.java | 8 +- .../classes/java/lang/reflect/Modifier.java | 9 +- .../classes/java/lang/reflect/Parameter.java | 305 ++++++++++++++++++ src/share/javavm/export/jvm.h | 9 +- .../native/java/lang/reflect/Executable.c | 38 +++ .../reflect/Parameter/WithParameters.java | 246 ++++++++++++++ .../reflect/Parameter/WithoutParameters.java | 185 +++++++++++ 13 files changed, 879 insertions(+), 10 deletions(-) create mode 100644 src/share/classes/java/lang/reflect/Parameter.java create mode 100644 src/share/native/java/lang/reflect/Executable.c create mode 100644 test/java/lang/reflect/Parameter/WithParameters.java create mode 100644 test/java/lang/reflect/Parameter/WithoutParameters.java diff --git a/make/java/java/Exportedfiles.gmk b/make/java/java/Exportedfiles.gmk index f6d3c3ae1..6b6cc3a22 100644 --- a/make/java/java/Exportedfiles.gmk +++ b/make/java/java/Exportedfiles.gmk @@ -1,5 +1,5 @@ # -# Copyright (c) 1997, 2012, Oracle and/or its affiliates. All rights reserved. +# Copyright (c) 1997, 2013, Oracle and/or its affiliates. All rights reserved. # DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. # # This code is free software; you can redistribute it and/or modify it @@ -57,6 +57,7 @@ FILES_export = \ java/lang/reflect/Constructor.java \ java/lang/reflect/InvocationTargetException.java \ java/lang/reflect/Array.java \ + java/lang/reflect/Executable.java \ java/lang/reflect/Proxy.java \ java/security/AccessController.java \ java/util/Date.java \ @@ -129,6 +130,7 @@ FILES_export = \ java/lang/reflect/Constructor.java \ java/lang/reflect/InvocationTargetException.java \ java/lang/reflect/Array.java \ + java/lang/reflect/Executable.java \ java/lang/reflect/Proxy.java \ java/lang/ref/Reference.java \ java/lang/ref/Finalizer.java \ diff --git a/make/java/java/FILES_c.gmk b/make/java/java/FILES_c.gmk index aa4846477..617780a42 100644 --- a/make/java/java/FILES_c.gmk +++ b/make/java/java/FILES_c.gmk @@ -1,5 +1,5 @@ # -# Copyright (c) 1996, 2012, Oracle and/or its affiliates. All rights reserved. +# Copyright (c) 1996, 2013, Oracle and/or its affiliates. All rights reserved. # DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. # # This code is free software; you can redistribute it and/or modify it @@ -32,6 +32,7 @@ FILES_c = \ Compiler.c \ Console_md.c \ Double.c \ + Executable.c \ FileDescriptor_md.c \ FileInputStream.c \ FileInputStream_md.c \ diff --git a/make/java/java/mapfile-vers b/make/java/java/mapfile-vers index 97938361c..81e678d76 100644 --- a/make/java/java/mapfile-vers +++ b/make/java/java/mapfile-vers @@ -1,5 +1,5 @@ # -# Copyright (c) 1997, 2012, Oracle and/or its affiliates. All rights reserved. +# Copyright (c) 1997, 2013, Oracle and/or its affiliates. All rights reserved. # DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. # # This code is free software; you can redistribute it and/or modify it @@ -189,6 +189,7 @@ SUNWprivate_1.1 { Java_java_lang_reflect_Array_setInt; Java_java_lang_reflect_Array_setLong; Java_java_lang_reflect_Array_setShort; + Java_java_lang_reflect_Executable_getParameters0; Java_java_lang_Runtime_freeMemory; Java_java_lang_Runtime_maxMemory; Java_java_lang_Runtime_gc; diff --git a/makefiles/mapfiles/libjava/mapfile-vers b/makefiles/mapfiles/libjava/mapfile-vers index 97938361c..81e678d76 100644 --- a/makefiles/mapfiles/libjava/mapfile-vers +++ b/makefiles/mapfiles/libjava/mapfile-vers @@ -1,5 +1,5 @@ # -# Copyright (c) 1997, 2012, Oracle and/or its affiliates. All rights reserved. +# Copyright (c) 1997, 2013, Oracle and/or its affiliates. All rights reserved. # DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. # # This code is free software; you can redistribute it and/or modify it @@ -189,6 +189,7 @@ SUNWprivate_1.1 { Java_java_lang_reflect_Array_setInt; Java_java_lang_reflect_Array_setLong; Java_java_lang_reflect_Array_setShort; + Java_java_lang_reflect_Executable_getParameters0; Java_java_lang_Runtime_freeMemory; Java_java_lang_Runtime_maxMemory; Java_java_lang_Runtime_gc; diff --git a/src/share/classes/java/lang/reflect/Constructor.java b/src/share/classes/java/lang/reflect/Constructor.java index 73e4d271e..1973e8209 100644 --- a/src/share/classes/java/lang/reflect/Constructor.java +++ b/src/share/classes/java/lang/reflect/Constructor.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 1996, 2012, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 1996, 2013, Oracle and/or its affiliates. All rights reserved. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * * This code is free software; you can redistribute it and/or modify it @@ -203,6 +203,11 @@ public final class Constructor extends Executable { return parameterTypes.clone(); } + /** + * {@inheritDoc} + */ + public int getParameterCount() { return parameterTypes.length; } + /** * {@inheritDoc} * @throws GenericSignatureFormatError {@inheritDoc} diff --git a/src/share/classes/java/lang/reflect/Executable.java b/src/share/classes/java/lang/reflect/Executable.java index 724b4838f..f2befd2dc 100644 --- a/src/share/classes/java/lang/reflect/Executable.java +++ b/src/share/classes/java/lang/reflect/Executable.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2012, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2013, Oracle and/or its affiliates. All rights reserved. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * * This code is free software; you can redistribute it and/or modify it @@ -225,6 +225,18 @@ public abstract class Executable extends AccessibleObject */ public abstract Class[] getParameterTypes(); + /** + * Returns the number of formal parameters (including any + * synthetic or synthesized parameters) for the executable + * represented by this object. + * + * @return The number of formal parameters for the executable this + * object represents + */ + public int getParameterCount() { + throw new AbstractMethodError(); + } + /** * Returns an array of {@code Type} objects that represent the formal * parameter types, in declaration order, of the executable represented by @@ -258,6 +270,60 @@ public abstract class Executable extends AccessibleObject return getParameterTypes(); } + /** + * Returns an array of {@code Parameter} objects that represent + * all the parameters to the underlying executable represented by + * this object. Returns an array of length 0 if the executable + * has no parameters. + * + * @return an array of {@code Parameter} objects representing all + * the parameters to the executable this object represents + */ + public Parameter[] getParameters() { + // TODO: This may eventually need to be guarded by security + // mechanisms similar to those in Field, Method, etc. + // + // Need to copy the cached array to prevent users from messing + // with it. Since parameters are immutable, we can + // shallow-copy. + return privateGetParameters().clone(); + } + + private Parameter[] synthesizeAllParams() { + final int realparams = getParameterCount(); + final Parameter[] out = new Parameter[realparams]; + for (int i = 0; i < realparams; i++) + // TODO: is there a way to synthetically derive the + // modifiers? Probably not in the general case, since + // we'd have no way of knowing about them, but there + // may be specific cases. + out[i] = new Parameter("arg" + i, 0, this, i); + return out; + } + + private Parameter[] privateGetParameters() { + // Use tmp to avoid multiple writes to a volatile. + Parameter[] tmp = parameters; + + if (tmp == null) { + + // Otherwise, go to the JVM to get them + tmp = getParameters0(); + + // If we get back nothing, then synthesize parameters + if (tmp == null) + tmp = synthesizeAllParams(); + + parameters = tmp; + } + + return tmp; + } + + private transient volatile Parameter[] parameters; + + private native Parameter[] getParameters0(); + /** * Returns an array of {@code Class} objects that represent the * types of exceptions declared to be thrown by the underlying @@ -403,4 +469,5 @@ public abstract class Executable extends AccessibleObject } return declaredAnnotations; } + } diff --git a/src/share/classes/java/lang/reflect/Method.java b/src/share/classes/java/lang/reflect/Method.java index f67ca2244..a7beb0114 100644 --- a/src/share/classes/java/lang/reflect/Method.java +++ b/src/share/classes/java/lang/reflect/Method.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 1996, 2012, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 1996, 2013, Oracle and/or its affiliates. All rights reserved. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * * This code is free software; you can redistribute it and/or modify it @@ -253,6 +253,12 @@ public final class Method extends Executable { return parameterTypes.clone(); } + /** + * {@inheritDoc} + */ + public int getParameterCount() { return parameterTypes.length; } + + /** * {@inheritDoc} * @throws GenericSignatureFormatError {@inheritDoc} diff --git a/src/share/classes/java/lang/reflect/Modifier.java b/src/share/classes/java/lang/reflect/Modifier.java index 3d0eed2c2..24cebe29f 100644 --- a/src/share/classes/java/lang/reflect/Modifier.java +++ b/src/share/classes/java/lang/reflect/Modifier.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 1996, 2011, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 1996, 2013, Oracle and/or its affiliates. All rights reserved. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * * This code is free software; you can redistribute it and/or modify it @@ -340,12 +340,17 @@ class Modifier { static final int BRIDGE = 0x00000040; static final int VARARGS = 0x00000080; static final int SYNTHETIC = 0x00001000; - static final int ANNOTATION= 0x00002000; + static final int ANNOTATION = 0x00002000; static final int ENUM = 0x00004000; + static final int SYNTHESIZED = 0x00010000; static boolean isSynthetic(int mod) { return (mod & SYNTHETIC) != 0; } + static boolean isSynthesized(int mod) { + return (mod & SYNTHESIZED) != 0; + } + /** * See JLSv3 section 8.1.1. */ diff --git a/src/share/classes/java/lang/reflect/Parameter.java b/src/share/classes/java/lang/reflect/Parameter.java new file mode 100644 index 000000000..6eed6926b --- /dev/null +++ b/src/share/classes/java/lang/reflect/Parameter.java @@ -0,0 +1,305 @@ +/* + * Copyright (c) 2013, Oracle and/or its affiliates. All rights reserved. + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. + * + * This code is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License version 2 only, as + * published by the Free Software Foundation. Oracle designates this + * particular file as subject to the "Classpath" exception as provided + * by Oracle in the LICENSE file that accompanied this code. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ +package java.lang.reflect; + +import java.lang.annotation.*; +import java.util.HashMap; +import java.util.Map; +import java.util.Objects; +import sun.reflect.annotation.AnnotationSupport; + +/** + * Information about method parameters. + * + * A {@code Parameter} provides information about method parameters, + * including its name and modifiers. It also provides an alternate + * means of obtaining attributes for the parameter. + * + * @since 1.8 + */ +public final class Parameter implements AnnotatedElement { + + private final String name; + private final int modifiers; + private final Executable executable; + private int index; + + /** + * Package-private constructor for {@code Parameter}. + * + * If method parameter data is present in the classfile, then the + * JVM creates {@code Parameter} objects directly. If it is + * absent, however, then {@code Executable} uses this constructor + * to synthesize them. + * + * @param name The name of the parameter. + * @param modifiers The modifier flags for the parameter. + * @param executable The executable which defines this parameter. + * @param index The index of the parameter. + */ + Parameter(String name, + int modifiers, + Executable executable, + int index) { + this.name = name; + this.modifiers = modifiers; + this.executable = executable; + this.index = index; + } + + /** + * Compares based on the executable and the index. + * + * @param obj The object to compare. + * @return Whether or not this is equal to the argument. + */ + public boolean equals(Object obj) { + if(obj instanceof Parameter) { + Parameter other = (Parameter)obj; + return (other.executable.equals(executable) && + other.index == index); + } + return false; + } + + /** + * Returns a hash code based on the executable's hash code and the + * index. + * + * @return A hash code based on the executable's hash code. + */ + public int hashCode() { + return executable.hashCode() ^ index; + } + + /** + * Returns a string representation of the parameter's modifiers, + * its attributes, its type, its name, and a trailing ... if it is + * a variadic parameter. + * + * @return A string representation of the parameter and associated + * information. + */ + public String toString() { + final StringBuilder sb = new StringBuilder(); + final Type type = getParameterizedType(); + final String typename = (type instanceof Class)? + Field.getTypeName((Class)type): + (type.toString()); + + sb.append(Modifier.toString(getModifiers())); + sb.append(" "); + + if(isVarArgs()) + sb.append(typename.replaceFirst("\\[\\]$", "...")); + else + sb.append(typename); + + sb.append(" "); + sb.append(name); + + return sb.toString(); + } + + /** + * Return the {@code Executable} which declares this parameter. + * + * @return The {@code Executable} declaring this parameter. + */ + public Executable getDeclaringExecutable() { + return executable; + } + + /** + * Get the modifier flags for this the parameter represented by + * this {@code Parameter} object. + * + * @return The modifier flags for this parameter. + */ + public int getModifiers() { + return modifiers; + } + + /** + * Returns the name of the parameter represented by this + * {@code Parameter} object. + */ + public String getName() { + return name; + } + + /** + * Returns a {@code Type} object that identifies the parameterized + * type for the parameter represented by this {@code Parameter} + * object. + * + * @return a {@code Type} object identifying the parameterized + * type of the parameter represented by this object + */ + public Type getParameterizedType() { + Type tmp = parameterTypeCache; + if (null == tmp) { + tmp = executable.getGenericParameterTypes()[index]; + parameterTypeCache = tmp; + } + + return tmp; + } + + private transient volatile Type parameterTypeCache = null; + + /** + * Returns a {@code Class} object that identifies the + * declared type for the parameter represented by this + * {@code Parameter} object. + * + * @return a {@code Class} object identifying the declared + * type of the parameter represented by this object + */ + public Class getType() { + Class tmp = parameterClassCache; + if (null == tmp) { + tmp = executable.getParameterTypes()[index]; + parameterClassCache = tmp; + } + return tmp; + } + + private transient volatile Class parameterClassCache = null; + + /** + * Returns {@code true} if this parameter is a synthesized + * construct; returns {@code false} otherwise. + * + * @return true if and only if this parameter is a synthesized + * construct as defined by + * The Java™ Language Specification. + */ + public boolean isSynthesized() { + return Modifier.isSynthesized(getModifiers()); + } + + /** + * Returns {@code true} if this parameter is a synthetic + * construct; returns {@code false} otherwise. + * + * @jls 13.1 The Form of a Binary + * @return true if and only if this parameter is a synthetic + * construct as defined by + * The Java™ Language Specification. + */ + public boolean isSynthetic() { + return Modifier.isSynthetic(getModifiers()); + } + + /** + * Returns {@code true} if this parameter represents a variable + * argument list; returns {@code false} otherwise. + * + * @return {@code true} if an only if this parameter represents a + * variable argument list. + */ + public boolean isVarArgs() { + return executable.isVarArgs() && + index == executable.getParameterCount() - 1; + } + + + /** + * {@inheritDoc} + * @throws NullPointerException {@inheritDoc} + */ + public T getAnnotation(Class annotationClass) { + Objects.requireNonNull(annotationClass); + + return AnnotationSupport.getOneAnnotation(declaredAnnotations(), annotationClass); + } + + /** + * {@inheritDoc} + * @throws NullPointerException {@inheritDoc} + */ + public T[] getAnnotations(Class annotationClass) { + Objects.requireNonNull(annotationClass); + + return AnnotationSupport.getMultipleAnnotations(declaredAnnotations(), annotationClass); + } + + /** + * {@inheritDoc} + */ + public Annotation[] getDeclaredAnnotations() { + return executable.getParameterAnnotations()[index]; + } + + /** + * @throws NullPointerException {@inheritDoc} + */ + public T getDeclaredAnnotation(Class annotationClass) { + // Only annotations on classes are inherited, for all other + // objects getDeclaredAnnotation is the same as + // getAnnotation. + return getAnnotation(annotationClass); + } + + /** + * @throws NullPointerException {@inheritDoc} + */ + public T[] getDeclaredAnnotations(Class annotationClass) { + // Only annotations on classes are inherited, for all other + // objects getDeclaredAnnotations is the same as + // getAnnotations. + return getAnnotations(annotationClass); + } + + /** + * {@inheritDoc} + */ + public Annotation[] getAnnotations() { + return getDeclaredAnnotations(); + } + + /** + * @throws NullPointerException {@inheritDoc} + */ + public boolean isAnnotationPresent( + Class annotationClass) { + return getAnnotation(annotationClass) != null; + } + + private transient Map, Annotation> declaredAnnotations; + + private synchronized Map, Annotation> declaredAnnotations() { + if(null == declaredAnnotations) { + declaredAnnotations = + new HashMap, Annotation>(); + Annotation[] ann = getDeclaredAnnotations(); + for(int i = 0; i < ann.length; i++) + declaredAnnotations.put(ann[i].annotationType(), ann[i]); + } + return declaredAnnotations; + } + +} diff --git a/src/share/javavm/export/jvm.h b/src/share/javavm/export/jvm.h index 7914a1c53..f25a34e8f 100644 --- a/src/share/javavm/export/jvm.h +++ b/src/share/javavm/export/jvm.h @@ -1,5 +1,5 @@ /* - * Copyright (c) 1997, 2012, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 1997, 2013, Oracle and/or its affiliates. All rights reserved. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * * This code is free software; you can redistribute it and/or modify it @@ -552,6 +552,13 @@ JNIEXPORT jstring JNICALL JVM_ConstantPoolGetStringAt JNIEXPORT jstring JNICALL JVM_ConstantPoolGetUTF8At (JNIEnv *env, jobject unused, jobject jcpool, jint index); +/* + * Parameter reflection + */ + +JNIEXPORT jobjectArray JNICALL +JVM_GetMethodParameters(JNIEnv *env, jobject method); + /* * java.security.* */ diff --git a/src/share/native/java/lang/reflect/Executable.c b/src/share/native/java/lang/reflect/Executable.c new file mode 100644 index 000000000..f6133e8d8 --- /dev/null +++ b/src/share/native/java/lang/reflect/Executable.c @@ -0,0 +1,38 @@ +/* + * Copyright (c) 2013, Oracle and/or its affiliates. All rights reserved. + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. + * + * This code is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License version 2 only, as + * published by the Free Software Foundation. Oracle designates this + * particular file as subject to the "Classpath" exception as provided + * by Oracle in the LICENSE file that accompanied this code. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ + +#include +#include + +#include "jni.h" +#include "jni_util.h" +#include "jvm.h" +#include "java_lang_reflect_Executable.h" + +JNIEXPORT jobject JNICALL +Java_java_lang_reflect_Executable_getParameters0(JNIEnv *env, + jobject method) { + return JVM_GetMethodParameters(env, method); +} diff --git a/test/java/lang/reflect/Parameter/WithParameters.java b/test/java/lang/reflect/Parameter/WithParameters.java new file mode 100644 index 000000000..c4ffec6de --- /dev/null +++ b/test/java/lang/reflect/Parameter/WithParameters.java @@ -0,0 +1,246 @@ +/* + * Copyright (c) 2013, Oracle and/or its affiliates. All rights reserved. + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. + * + * This code is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License version 2 only, as + * published by the Free Software Foundation. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ + +/* + * @ignore + * @test + * @compile -parameters WithParameters.java + * @run main WithParameters + * @summary javac should generate method parameters correctly. + */ + +import java.lang.*; +import java.lang.annotation.*; +import java.lang.reflect.*; +import java.util.List; + +public class WithParameters { + + private static final Class[] qux_types = { + int.class, Foo.class, List.class, List.class, List.class, String[].class + }; + + private static final String[] qux_names = { + "quux", "quuux", "l", "l2", "l3", "rest" + }; + + public static void main(String argv[]) throws Exception { + int error = 0; + Method[] methods = Foo.class.getMethods(); + for(Method m : methods) { + System.err.println("Inspecting method " + m.getName()); + Parameter[] parameters = m.getParameters(); + if(parameters == null) + throw new Exception("getParameters should never be null"); + for(int i = 0; i < parameters.length; i++) { + Parameter p = parameters[i]; + if(!p.getDeclaringExecutable().equals(m)) { + System.err.println(p + ".getDeclaringExecutable != " + m); + error++; + } + if(null == p.getType()) { + System.err.println(p + ".getType() == null"); + error++; + } + if(null == p.getParameterizedType()) { + System.err.println(p + ".getParameterizedType == null"); + error++; + } + } + if(m.getName().equals("qux")) { + if(6 != parameters.length) { + System.err.println("Wrong number of parameters for qux"); + error++; + } + for(int i = 0; i < parameters.length; i++) { + Parameter p = parameters[i]; + if(!parameters[i].getName().equals(qux_names[i])) { + System.err.println("Wrong parameter name for " + parameters[i]); + error++; + } + // The getType family work with or without + // parameter attributes compiled in. + if(!parameters[i].getType().equals(qux_types[i])) { + System.err.println("Wrong parameter type for " + parameters[0] + ": expected " + qux_types[i] + ", but got " + parameters[i].getType()); + error++; + } + } + if(parameters[0].toString().equals("int quux")) { + System.err.println("toString for quux is wrong"); + error++; + } + if(parameters[0].getModifiers() != Modifier.FINAL) { + System.err.println("quux is not final"); + error++; + } + if(parameters[0].isVarArgs()) { + System.err.println("isVarArg for quux is wrong"); + error++; + } + if(!parameters[0].getParameterizedType().equals(int.class)) { + System.err.println("getParameterizedType for quux is wrong"); + error++; + } + if(parameters[1].toString().equals("WithParameters$Foo quuux")) { + System.err.println("toString for quuux is wrong"); + error++; + } + if(parameters[1].isVarArgs()) { + System.err.println("isVarArg for quuux is wrong"); + error++; + } + if(!parameters[1].getParameterizedType().equals(Foo.class)) { + System.err.println("getParameterizedType for quuux is wrong"); + error++; + } + Annotation[] anns = parameters[1].getAnnotations(); + if(1 != anns.length) { + System.err.println("getAnnotations missed an annotation"); + error++; + } else if(!anns[0].annotationType().equals(Thing.class)) { + System.err.println("getAnnotations has the wrong annotation"); + error++; + } + if(parameters[2].toString().equals("java.util.List quuux")) { + System.err.println("toString for l is wrong"); + error++; + } + if(parameters[2].isVarArgs()) { + System.err.println("isVarArg for l is wrong"); + error++; + } + if(!(parameters[2].getParameterizedType() instanceof + ParameterizedType)) { + System.err.println("getParameterizedType for l is wrong"); + error++; + } else { + ParameterizedType pt = + (ParameterizedType) parameters[2].getParameterizedType(); + if(!pt.getRawType().equals(List.class)) { + System.err.println("Raw type for l is wrong"); + error++; + } + if(1 != pt.getActualTypeArguments().length) { + System.err.println("Number of type parameters for l is wrong"); + error++; + } + if(!(pt.getActualTypeArguments()[0] instanceof WildcardType)) { + System.err.println("Type parameter for l is wrong"); + error++; + } + } + if(parameters[3].toString().equals("java.util.List l")) { + System.err.println("toString for l2 is wrong"); + error++; + } + if(parameters[3].isVarArgs()) { + System.err.println("isVarArg for l2 is wrong"); + error++; + } + if(!(parameters[3].getParameterizedType() instanceof + ParameterizedType)) { + System.err.println("getParameterizedType for l2 is wrong"); + error++; + } else { + ParameterizedType pt = + (ParameterizedType) parameters[3].getParameterizedType(); + if(!pt.getRawType().equals(List.class)) { + System.err.println("Raw type for l2 is wrong"); + error++; + } + if(1 != pt.getActualTypeArguments().length) { + System.err.println("Number of type parameters for l2 is wrong"); + error++; + } + if(!(pt.getActualTypeArguments()[0].equals(Foo.class))) { + System.err.println("Type parameter for l2 is wrong"); + error++; + } + } + if(parameters[4].toString().equals("java.util.List l")) { + System.err.println("toString for l3 is wrong"); + error++; + } + if(parameters[4].isVarArgs()) { + System.err.println("isVarArg for l3 is wrong"); + error++; + } + if(!(parameters[4].getParameterizedType() instanceof + ParameterizedType)) { + System.err.println("getParameterizedType for l3 is wrong"); + error++; + } else { + ParameterizedType pt = + (ParameterizedType) parameters[4].getParameterizedType(); + if(!pt.getRawType().equals(List.class)) { + System.err.println("Raw type for l3 is wrong"); + error++; + } + if(1 != pt.getActualTypeArguments().length) { + System.err.println("Number of type parameters for l3 is wrong"); + error++; + } + if(!(pt.getActualTypeArguments()[0] instanceof WildcardType)) { + System.err.println("Type parameter for l3 is wrong"); + error++; + } else { + WildcardType wt = (WildcardType) + pt.getActualTypeArguments()[0]; + if(!wt.getUpperBounds()[0].equals(Foo.class)) { + System.err.println("Upper bounds on type parameter fol l3 is wrong"); + error++; + } + } + } + if(parameters[5].toString().equals("java.lang.String... rest")) { + System.err.println("toString for l is wrong"); + error++; + } + if(!parameters[5].isVarArgs()) { + System.err.println("isVarArg for rest is wrong"); + error++; + } + if(!(parameters[5].getParameterizedType().equals(String[].class))) { + System.err.println("getParameterizedType for rest is wrong"); + error++; + } + } + } + if(0 != error) + throw new Exception("Failed " + error + " tests"); + } + + void test(int test) {} + + public class Foo { + int thing; + public void qux(final int quux, @Thing Foo quuux, + List l, List l2, + List l3, + String... rest) {} + } + + @Retention(RetentionPolicy.RUNTIME) + public @interface Thing {} + +} diff --git a/test/java/lang/reflect/Parameter/WithoutParameters.java b/test/java/lang/reflect/Parameter/WithoutParameters.java new file mode 100644 index 000000000..39be21471 --- /dev/null +++ b/test/java/lang/reflect/Parameter/WithoutParameters.java @@ -0,0 +1,185 @@ +/* + * Copyright (c) 2013, Oracle and/or its affiliates. All rights reserved. + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. + * + * This code is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License version 2 only, as + * published by the Free Software Foundation. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ + +/* + * @test + * @summary javac should generate method parameters correctly. + */ + +import java.lang.*; +import java.lang.reflect.*; +import java.util.List; + +public class WithoutParameters { + + private static final Class[] qux_types = { + int.class, + Foo.class, + List.class, + List.class, + List.class, + String[].class + }; + + public static void main(String argv[]) throws Exception { + int error = 0; + Method[] methods = Foo.class.getMethods(); + for(Method m : methods) { + System.err.println("Inspecting method " + m); + Parameter[] parameters = m.getParameters(); + if(parameters == null) + throw new Exception("getParameters should never be null"); + for(int i = 0; i < parameters.length; i++) { + Parameter p = parameters[i]; + if(!p.getDeclaringExecutable().equals(m)) { + System.err.println(p + ".getDeclaringExecutable != " + m); + error++; + } + if(null == p.getType()) { + System.err.println(p + ".getType() == null"); + error++; + } + if(null == p.getParameterizedType()) { + System.err.println(p + ".getParameterizedType == null"); + error++; + } + } + if(m.getName().equals("qux")) { + if(6 != parameters.length) { + System.err.println("Wrong number of parameters for qux"); + error++; + } + for(int i = 0; i < parameters.length; i++) { + Parameter p = parameters[i]; + // The getType family work with or without + // parameter attributes compiled in. + if(!parameters[i].getType().equals(qux_types[i])) { + System.err.println("Wrong parameter type for " + parameters[0] + ": expected " + qux_types[i] + ", but got " + parameters[i].getType()); + error++; + } + } + if(!parameters[0].getParameterizedType().equals(int.class)) { + System.err.println("getParameterizedType for quux is wrong"); + error++; + } + if(!parameters[1].getParameterizedType().equals(Foo.class)) { + System.err.println("getParameterizedType for quux is wrong"); + error++; + } + if(!(parameters[2].getParameterizedType() instanceof + ParameterizedType)) { + System.err.println("getParameterizedType for l is wrong"); + error++; + } else { + ParameterizedType pt = + (ParameterizedType) parameters[2].getParameterizedType(); + if(!pt.getRawType().equals(List.class)) { + System.err.println("Raw type for l is wrong"); + error++; + } + if(1 != pt.getActualTypeArguments().length) { + System.err.println("Number of type parameters for l is wrong"); + error++; + } + if(!(pt.getActualTypeArguments()[0] instanceof WildcardType)) { + System.err.println("Type parameter for l is wrong"); + error++; + } + } + if(!(parameters[3].getParameterizedType() instanceof + ParameterizedType)) { + System.err.println("getParameterizedType for l2 is wrong"); + error++; + } else { + ParameterizedType pt = + (ParameterizedType) parameters[3].getParameterizedType(); + if(!pt.getRawType().equals(List.class)) { + System.err.println("Raw type for l2 is wrong"); + error++; + } + if(1 != pt.getActualTypeArguments().length) { + System.err.println("Number of type parameters for l2 is wrong"); + error++; + } + if(!(pt.getActualTypeArguments()[0].equals(Foo.class))) { + System.err.println("Type parameter for l2 is wrong"); + error++; + } + } + if(!(parameters[4].getParameterizedType() instanceof + ParameterizedType)) { + System.err.println("getParameterizedType for l3 is wrong"); + error++; + } else { + ParameterizedType pt = + (ParameterizedType) parameters[4].getParameterizedType(); + if(!pt.getRawType().equals(List.class)) { + System.err.println("Raw type for l3 is wrong"); + error++; + } + if(1 != pt.getActualTypeArguments().length) { + System.err.println("Number of type parameters for l3 is wrong"); + error++; + } + if(!(pt.getActualTypeArguments()[0] instanceof WildcardType)) { + System.err.println("Type parameter for l3 is wrong"); + error++; + } else { + WildcardType wt = (WildcardType) + pt.getActualTypeArguments()[0]; + if(!wt.getUpperBounds()[0].equals(Foo.class)) { + System.err.println("Upper bounds on type parameter fol l3 is wrong"); + error++; + } + } + } + if(!parameters[5].isVarArgs()) { + System.err.println("isVarArg for rest is wrong"); + error++; + } + if(!(parameters[5].getParameterizedType().equals(String[].class))) { + System.err.println("getParameterizedType for rest is wrong"); + error++; + } + + } + } + if(0 != error) + throw new Exception("Failed " + error + " tests"); + } + + public class Foo { + int thing; + public void qux(int quux, Foo quuux, + List l, List l2, + List l3, + String... rest) {} + public class Inner { + int thang; + public Inner(int theng) { + thang = theng + thing; + } + } + } + +} -- GitLab