test/tools/javac/7079713/TestCircularClassfile.java

changeset 0
959103a6100f
child 2525
2eb010b6cb22
equal deleted inserted replaced
-1:000000000000 0:959103a6100f
1 /*
2 * Copyright (c) 2011, Oracle and/or its affiliates. All rights reserved.
3 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4 *
5 * This code is free software; you can redistribute it and/or modify it
6 * under the terms of the GNU General Public License version 2 only, as
7 * published by the Free Software Foundation.
8 *
9 * This code is distributed in the hope that it will be useful, but WITHOUT
10 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
11 * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
12 * version 2 for more details (a copy is included in the LICENSE file that
13 * accompanied this code).
14 *
15 * You should have received a copy of the GNU General Public License version
16 * 2 along with this work; if not, write to the Free Software Foundation,
17 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
18 *
19 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
20 * or visit www.oracle.com if you need additional information or have any
21 * questions.
22 */
23
24 /*
25 * @test
26 * @bug 7079713
27 * @summary javac hangs when compiling a class that references a cyclically inherited class
28 * @run main TestCircularClassfile
29 */
30
31 import java.io.*;
32 import java.net.URI;
33 import java.util.Arrays;
34 import javax.tools.JavaCompiler;
35 import javax.tools.JavaFileObject;
36 import javax.tools.SimpleJavaFileObject;
37 import javax.tools.StandardJavaFileManager;
38 import javax.tools.StandardLocation;
39 import javax.tools.ToolProvider;
40
41 import com.sun.source.util.JavacTask;
42
43 public class TestCircularClassfile {
44
45 enum SourceKind {
46 A_EXTENDS_B("class B {} class A extends B { void m() {} }"),
47 B_EXTENDS_A("class A { void m() {} } class B extends A {}");
48
49 String sourceStr;
50
51 private SourceKind(String sourceStr) {
52 this.sourceStr = sourceStr;
53 }
54
55 SimpleJavaFileObject getSource() {
56 return new SimpleJavaFileObject(URI.create("myfo:/Test.java"), JavaFileObject.Kind.SOURCE) {
57 @Override
58 public CharSequence getCharContent(boolean ignoreEncodingErrors) throws IOException {
59 return sourceStr;
60 }
61 };
62 }
63 }
64
65 enum TestKind {
66 REPLACE_A("A.class"),
67 REPLACE_B("B.class");
68
69 String targetClass;
70
71 private TestKind(String targetClass) {
72 this.targetClass = targetClass;
73 }
74 }
75
76 enum ClientKind {
77 METHOD_CALL1("A a = null; a.m();"),
78 METHOD_CALL2("B b = null; b.m();"),
79 CONSTR_CALL1("new A();"),
80 CONSTR_CALL2("new B();"),
81 ASSIGN1("A a = null; B b = a;"),
82 ASSIGN2("B b = null; A a = b;");
83
84 String mainMethod;
85
86 private ClientKind(String mainMethod) {
87 this.mainMethod = mainMethod;
88 }
89
90 SimpleJavaFileObject getSource() {
91 return new SimpleJavaFileObject(URI.create("myfo:/Test.java"), JavaFileObject.Kind.SOURCE) {
92 @Override
93 public CharSequence getCharContent(boolean ignoreEncodingErrors) throws IOException {
94 return "class Test { public static void main(String[] args) { #M } }"
95 .replace("#M", mainMethod);
96 }
97 };
98 }
99 }
100
101 public static void main(String... args) throws Exception {
102 JavaCompiler comp = ToolProvider.getSystemJavaCompiler();
103 StandardJavaFileManager fm = comp.getStandardFileManager(null, null, null);
104 int count = 0;
105 for (SourceKind sk1 : SourceKind.values()) {
106 for (SourceKind sk2 : SourceKind.values()) {
107 for (TestKind tk : TestKind.values()) {
108 for (ClientKind ck : ClientKind.values()) {
109 new TestCircularClassfile("sub_"+count++, sk1, sk2, tk, ck).check(comp, fm);
110 }
111 }
112 }
113 }
114 }
115
116 static String workDir = System.getProperty("user.dir");
117
118 String destPath;
119 SourceKind sk1;
120 SourceKind sk2;
121 TestKind tk;
122 ClientKind ck;
123
124 TestCircularClassfile(String destPath, SourceKind sk1, SourceKind sk2, TestKind tk, ClientKind ck) {
125 this.destPath = destPath;
126 this.sk1 = sk1;
127 this.sk2 = sk2;
128 this.tk = tk;
129 this.ck = ck;
130 }
131
132 void check(JavaCompiler comp, StandardJavaFileManager fm) throws Exception {
133 //step 1: compile first source code in the test subfolder
134 File destDir = new File(workDir, destPath); destDir.mkdir();
135 //output dir must be set explicitly as we are sharing the fm (see bug 7026941)
136 fm.setLocation(javax.tools.StandardLocation.CLASS_OUTPUT, Arrays.asList(destDir));
137 JavacTask ct = (JavacTask)comp.getTask(null, fm, null,
138 null, null, Arrays.asList(sk1.getSource()));
139 ct.generate();
140
141 //step 2: compile second source code in a temp folder
142 File tmpDir = new File(destDir, "tmp"); tmpDir.mkdir();
143 //output dir must be set explicitly as we are sharing the fm (see bug 7026941)
144 fm.setLocation(javax.tools.StandardLocation.CLASS_OUTPUT, Arrays.asList(tmpDir));
145 ct = (JavacTask)comp.getTask(null, fm, null,
146 null, null, Arrays.asList(sk2.getSource()));
147 ct.generate();
148
149 //step 3: move a classfile from the temp folder to the test subfolder
150 File fileToMove = new File(tmpDir, tk.targetClass);
151 File target = new File(destDir, tk.targetClass);
152 target.delete();
153 boolean success = fileToMove.renameTo(target);
154
155 if (!success) {
156 throw new AssertionError("error when moving file " + tk.targetClass);
157 }
158
159 //step 4: compile the client class against the classes in the test subfolder
160 //input/output dir must be set explicitly as we are sharing the fm (see bug 7026941)
161 fm.setLocation(StandardLocation.CLASS_OUTPUT, Arrays.asList(destDir));
162 fm.setLocation(StandardLocation.CLASS_PATH, Arrays.asList(destDir));
163 ct = (JavacTask)comp.getTask(null, fm, null,
164 null, null, Arrays.asList(ck.getSource()));
165
166 ct.generate();
167 }
168 }

mercurial