mirror of
https://github.com/openjdk/jdk.git
synced 2026-01-29 12:38:24 +00:00
8023520: Add missing test for JDK-7118412
The test for JDK-7118412 was dropped from the changeset in a merging accident. Reviewed-by: jjg
This commit is contained in:
parent
64412dad17
commit
febfa82cfb
287
langtools/test/tools/javac/7118412/ShadowingTest.java
Normal file
287
langtools/test/tools/javac/7118412/ShadowingTest.java
Normal file
@ -0,0 +1,287 @@
|
||||
/*
|
||||
* 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
|
||||
* @bug 7118412
|
||||
* @summary Shadowing of type-variables vs. member types
|
||||
*/
|
||||
import java.io.File;
|
||||
import java.io.FileWriter;
|
||||
import java.io.IOException;
|
||||
import java.io.PrintWriter;
|
||||
import java.io.StringWriter;
|
||||
|
||||
public class ShadowingTest {
|
||||
|
||||
// We generate a method "test" that tries to call T.<something,
|
||||
// depending on the value of MethodCall>. This controls whether
|
||||
// "test" is static or not.
|
||||
private enum MethodContext {
|
||||
STATIC("static "),
|
||||
INSTANCE("");
|
||||
|
||||
public final String methodcontext;
|
||||
|
||||
MethodContext(final String methodcontext) {
|
||||
this.methodcontext = methodcontext;
|
||||
}
|
||||
}
|
||||
|
||||
// These control whether or not a type parameter, method type
|
||||
// parameter, or inner class get declared (and in the case of
|
||||
// inner classes, whether it's static or not.
|
||||
|
||||
private enum MethodTypeParameterDecl {
|
||||
NO(""),
|
||||
YES("<T extends Number> ");
|
||||
|
||||
public final String tyvar;
|
||||
|
||||
MethodTypeParameterDecl(final String tyvar) {
|
||||
this.tyvar = tyvar;
|
||||
}
|
||||
}
|
||||
|
||||
private enum InsideDef {
|
||||
NONE(""),
|
||||
STATIC("static class T { public void inner() {} }\n"),
|
||||
INSTANCE("class T { public void inner() {} }\n");
|
||||
|
||||
public final String instancedef;
|
||||
|
||||
InsideDef(final String instancedef) {
|
||||
this.instancedef = instancedef;
|
||||
}
|
||||
}
|
||||
|
||||
private enum TypeParameterDecl {
|
||||
NO(""),
|
||||
YES("<T extends Collection>");
|
||||
|
||||
public final String tyvar;
|
||||
|
||||
TypeParameterDecl(final String tyvar) {
|
||||
this.tyvar = tyvar;
|
||||
}
|
||||
}
|
||||
|
||||
// Represents what method we try to call. This is a way of
|
||||
// checking which T we're seeing.
|
||||
private enum MethodCall {
|
||||
// Method type variables extend Number, so we have intValue
|
||||
METHOD_TYPEVAR("intValue"),
|
||||
// The inner class declaration has a method called "inner"
|
||||
INNER_CLASS("inner"),
|
||||
// The class type variables extend Collection, so we call iterator
|
||||
TYPEVAR("iterator"),
|
||||
// The outer class declaration has a method called "outer"
|
||||
OUTER_CLASS("outer");
|
||||
|
||||
public final String methodcall;
|
||||
|
||||
MethodCall(final String methodcall) {
|
||||
this.methodcall = methodcall;
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
public boolean succeeds(final MethodCall call,
|
||||
final MethodTypeParameterDecl mtyvar,
|
||||
final MethodContext ctx,
|
||||
final InsideDef inside,
|
||||
final TypeParameterDecl tyvar) {
|
||||
switch(call) {
|
||||
// We want to resolve to the method type variable
|
||||
case METHOD_TYPEVAR: switch(mtyvar) {
|
||||
// If the method type variable exists, then T will
|
||||
// resolve to it, and we'll have intValue.
|
||||
case YES: return true;
|
||||
// Otherwise, this cannot succeed.
|
||||
default: return false;
|
||||
}
|
||||
// We want to resolve to the inner class
|
||||
case INNER_CLASS: switch(mtyvar) {
|
||||
// The method type parameter will shadow the inner
|
||||
// class, so there can't be one.
|
||||
case NO: switch(ctx) {
|
||||
// If we're not static, then either one should succeed.
|
||||
case INSTANCE: switch(inside) {
|
||||
case INSTANCE:
|
||||
case STATIC:
|
||||
return true;
|
||||
default: return false;
|
||||
}
|
||||
case STATIC: switch(inside) {
|
||||
// If we are static, and the inner class is
|
||||
// static, then we also succeed, because we
|
||||
// can't see the type variable.
|
||||
case STATIC: return true;
|
||||
case INSTANCE: switch(tyvar) {
|
||||
// If we're calling from a non-static
|
||||
// context, there can't be a class type
|
||||
// variable, because that will shadow the
|
||||
// static inner class definition.
|
||||
case NO: return true;
|
||||
default: return false;
|
||||
}
|
||||
// If the inner class isn't declared, we can't
|
||||
// see it.
|
||||
default: return false;
|
||||
}
|
||||
// Can't get here.
|
||||
default: return false;
|
||||
}
|
||||
default: return false;
|
||||
}
|
||||
// We want to resolve to the class type parameter
|
||||
case TYPEVAR: switch(mtyvar) {
|
||||
// We can't have a method type parameter, as that would
|
||||
// shadow the class type parameter
|
||||
case NO: switch(ctx) {
|
||||
case INSTANCE: switch(inside) {
|
||||
// We have to be in an instance context. If
|
||||
// we're static, we can't see the type
|
||||
// variable.
|
||||
case NONE: switch(tyvar) {
|
||||
// Obviously, the type parameter has to be declared.
|
||||
case YES: return true;
|
||||
default: return false;
|
||||
}
|
||||
default: return false;
|
||||
}
|
||||
default: return false;
|
||||
}
|
||||
default: return false;
|
||||
}
|
||||
// We want to resolve to the outer class
|
||||
case OUTER_CLASS: switch(mtyvar) {
|
||||
case NO: switch(inside) {
|
||||
case NONE: switch(tyvar) {
|
||||
// Basically, nothing else can be declared, or
|
||||
// else we can't see it. Even if our context
|
||||
// is static, the compiler will complain if
|
||||
// non-static T's exist, because they will
|
||||
// shadow the outer class.
|
||||
case NO: return true;
|
||||
default: return false;
|
||||
}
|
||||
default: return false;
|
||||
}
|
||||
default: return false;
|
||||
}
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
private static final File classesdir = new File("7118412");
|
||||
|
||||
private int errors = 0;
|
||||
|
||||
private int dirnum = 0;
|
||||
|
||||
private void doTest(final MethodTypeParameterDecl mtyvar,
|
||||
final TypeParameterDecl tyvar,
|
||||
final InsideDef insidedef, final MethodContext ctx,
|
||||
final MethodCall call)
|
||||
throws IOException {
|
||||
final String content = "import java.util.Collection;\n" +
|
||||
"class Test" + tyvar.tyvar + " {\n" +
|
||||
" " + insidedef.instancedef +
|
||||
" " + ctx.methodcontext + mtyvar.tyvar + "void test(T t) { t." +
|
||||
call.methodcall + "(); }\n" +
|
||||
"}\n" +
|
||||
"class T { void outer() {} }\n";
|
||||
final File dir = new File(classesdir, "" + dirnum);
|
||||
final File Test_java = writeFile(dir, "Test.java", content);
|
||||
dirnum++;
|
||||
if(succeeds(call, mtyvar, ctx, insidedef, tyvar)) {
|
||||
if(!assert_compile_succeed(Test_java))
|
||||
System.err.println("Failed file:\n" + content);
|
||||
}
|
||||
else {
|
||||
if(!assert_compile_fail(Test_java))
|
||||
System.err.println("Failed file:\n" + content);
|
||||
}
|
||||
}
|
||||
|
||||
private void run() throws Exception {
|
||||
classesdir.mkdir();
|
||||
for(MethodTypeParameterDecl mtyvar : MethodTypeParameterDecl.values())
|
||||
for(TypeParameterDecl tyvar : TypeParameterDecl.values())
|
||||
for(InsideDef insidedef : InsideDef.values())
|
||||
for(MethodContext ctx : MethodContext.values())
|
||||
for(MethodCall methodcall : MethodCall.values())
|
||||
doTest(mtyvar, tyvar, insidedef, ctx, methodcall);
|
||||
if (errors != 0)
|
||||
throw new Exception("ShadowingTest test failed with " +
|
||||
errors + " errors.");
|
||||
}
|
||||
|
||||
private boolean assert_compile_fail(final File file) {
|
||||
final String filename = file.getPath();
|
||||
final String[] args = { filename };
|
||||
final StringWriter sw = new StringWriter();
|
||||
final PrintWriter pw = new PrintWriter(sw);
|
||||
final int rc = com.sun.tools.javac.Main.compile(args, pw);
|
||||
pw.close();
|
||||
if (rc == 0) {
|
||||
System.err.println("Compilation of " + file.getName() +
|
||||
" didn't fail as expected.");
|
||||
errors++;
|
||||
return false;
|
||||
} else return true;
|
||||
}
|
||||
|
||||
private boolean assert_compile_succeed(final File file) {
|
||||
final String filename = file.getPath();
|
||||
final String[] args = { filename };
|
||||
final StringWriter sw = new StringWriter();
|
||||
final PrintWriter pw = new PrintWriter(sw);
|
||||
final int rc = com.sun.tools.javac.Main.compile(args, pw);
|
||||
pw.close();
|
||||
if (rc != 0) {
|
||||
System.err.println("Compilation of " + file.getName() +
|
||||
" didn't succeed as expected. Output:");
|
||||
System.err.println(sw.toString());
|
||||
errors++;
|
||||
return false;
|
||||
} else return true;
|
||||
}
|
||||
|
||||
private File writeFile(final File dir,
|
||||
final String path,
|
||||
final String body) throws IOException {
|
||||
final File f = new File(dir, path);
|
||||
f.getParentFile().mkdirs();
|
||||
final FileWriter out = new FileWriter(f);
|
||||
out.write(body);
|
||||
out.close();
|
||||
return f;
|
||||
}
|
||||
|
||||
public static void main(String... args) throws Exception {
|
||||
new ShadowingTest().run();
|
||||
}
|
||||
|
||||
}
|
||||
Loading…
x
Reference in New Issue
Block a user