Skip to content

Commit a5080ef

Browse files
committedOct 5, 2021
8272564: Incorrect attribution of method invocations of Object methods on interfaces
Reviewed-by: jlaskey, mcimadamore, vromero
1 parent a914ee7 commit a5080ef

File tree

6 files changed

+180
-4
lines changed

6 files changed

+180
-4
lines changed
 

‎src/jdk.compiler/share/classes/com/sun/tools/javac/comp/Attr.java

+1-1
Original file line numberDiff line numberDiff line change
@@ -2590,7 +2590,7 @@ public void visitApply(JCMethodInvocation tree) {
25902590
//where
25912591
Type adjustMethodReturnType(Symbol msym, Type qualifierType, Name methodName, List<Type> argtypes, Type restype) {
25922592
if (msym != null &&
2593-
msym.owner == syms.objectType.tsym &&
2593+
(msym.owner == syms.objectType.tsym || msym.owner.isInterface()) &&
25942594
methodName == names.getClass &&
25952595
argtypes.isEmpty()) {
25962596
// as a special case, x.getClass() has type Class<? extends |X|>

‎src/jdk.compiler/share/classes/com/sun/tools/javac/comp/Resolve.java

+16-2
Original file line numberDiff line numberDiff line change
@@ -469,7 +469,7 @@ private boolean notOverriddenIn(Type site, Symbol sym) {
469469
return true;
470470
else {
471471
Symbol s2 = ((MethodSymbol)sym).implementation(site.tsym, types, true);
472-
return (s2 == null || s2 == sym || sym.owner == s2.owner ||
472+
return (s2 == null || s2 == sym || sym.owner == s2.owner || (sym.owner.isInterface() && s2.owner == syms.objectType.tsym) ||
473473
!types.isSubSignature(types.memberType(site, s2), types.memberType(site, sym)));
474474
}
475475
}
@@ -1854,7 +1854,8 @@ private Symbol findMethod(Env<AttrContext> env,
18541854
List<Type>[] itypes = (List<Type>[])new List[] { List.<Type>nil(), List.<Type>nil() };
18551855

18561856
InterfaceLookupPhase iphase = InterfaceLookupPhase.ABSTRACT_OK;
1857-
for (TypeSymbol s : superclasses(intype)) {
1857+
boolean isInterface = site.tsym.isInterface();
1858+
for (TypeSymbol s : isInterface ? List.of(intype.tsym) : superclasses(intype)) {
18581859
bestSoFar = findMethodInScope(env, site, name, argtypes, typeargtypes,
18591860
s.members(), bestSoFar, allowBoxing, useVarargs, true);
18601861
if (name == names.init) return bestSoFar;
@@ -1892,6 +1893,19 @@ private Symbol findMethod(Env<AttrContext> env,
18921893
}
18931894
}
18941895
}
1896+
if (isInterface && bestSoFar.kind.isResolutionError()) {
1897+
bestSoFar = findMethodInScope(env, site, name, argtypes, typeargtypes,
1898+
syms.objectType.tsym.members(), bestSoFar, allowBoxing, useVarargs, true);
1899+
if (bestSoFar.kind.isValid()) {
1900+
Symbol baseSymbol = bestSoFar;
1901+
bestSoFar = new MethodSymbol(bestSoFar.flags_field, bestSoFar.name, bestSoFar.type, intype.tsym) {
1902+
@Override
1903+
public Symbol baseSymbol() {
1904+
return baseSymbol;
1905+
}
1906+
};
1907+
}
1908+
}
18951909
return bestSoFar;
18961910
}
18971911

‎test/langtools/tools/javac/api/TestGetElementReference.java

+1-1
Original file line numberDiff line numberDiff line change
@@ -23,7 +23,7 @@
2323

2424
/*
2525
* @test
26-
* @bug 8012929 8243074 8266281
26+
* @bug 8012929 8243074 8266281 8272564
2727
* @summary Trees.getElement should work not only for declaration trees, but also for use-trees
2828
* @modules jdk.compiler
2929
* @build TestGetElementReference

‎test/langtools/tools/javac/api/TestGetElementReferenceData.java

+8
Original file line numberDiff line numberDiff line change
@@ -37,6 +37,10 @@ private static void test() {
3737
target(TestGetElementReferenceData :: test/*getElement:METHOD:test.nested.TestGetElementReferenceData.test()*/);
3838
Object/*getElement:CLASS:java.lang.Object*/ o = null;
3939
if (o/*getElement:LOCAL_VARIABLE:o*/ instanceof String/*getElement:CLASS:java.lang.String*/ str/*getElement:BINDING_VARIABLE:str*/) ;
40+
I i = null;
41+
i.toString/*getElement:METHOD:test.nested.TestGetElementReferenceData.I.toString()*/();
42+
J j = null;
43+
j.toString/*getElement:METHOD:test.nested.TestGetElementReferenceData.I.toString()*/();
4044
}
4145
private static void target(Runnable r) { r.run(); }
4246
public static class Base {
@@ -50,4 +54,8 @@ public static class TypeParam<TT/*getElement:TYPE_PARAMETER:TT*/> {
5054
@Target(ElementType.TYPE_USE)
5155
@interface TypeAnnotation {
5256
}
57+
interface I {
58+
public String toString();
59+
}
60+
interface J extends I {}
5361
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,75 @@
1+
/*
2+
* Copyright (c) 2010, 2015, 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 8272564
27+
* @summary Verify accessibility of Object-based methods inherited from super interfaces.
28+
* @modules jdk.compiler
29+
*/
30+
31+
import com.sun.source.tree.CompilationUnitTree;
32+
import com.sun.source.tree.Scope;
33+
import com.sun.source.util.JavacTask;
34+
import com.sun.source.util.TreePath;
35+
import com.sun.source.util.Trees;
36+
import java.net.URI;
37+
import java.util.Arrays;
38+
import javax.lang.model.element.Element;
39+
import javax.lang.model.element.TypeElement;
40+
import javax.lang.model.type.DeclaredType;
41+
import javax.tools.JavaCompiler;
42+
import javax.tools.JavaFileObject;
43+
import javax.tools.SimpleJavaFileObject;
44+
import javax.tools.ToolProvider;
45+
46+
public class TestIsAccessible {
47+
public static void main(String[] args) throws Exception {
48+
final JavaCompiler tool = ToolProvider.getSystemJavaCompiler();
49+
assert tool != null;
50+
final JavacTask ct = (JavacTask)tool.getTask(null, null, null, null, null, Arrays.asList(new MyFileObject()));
51+
52+
CompilationUnitTree cut = ct.parse().iterator().next();
53+
TreePath tp = new TreePath(new TreePath(cut), cut.getTypeDecls().get(0));
54+
Scope s = Trees.instance(ct).getScope(tp);
55+
TypeElement name = ct.getElements().getTypeElement("javax.lang.model.element.Name");
56+
Trees trees = Trees.instance(ct);
57+
58+
if (trees.isAccessible(s, name)) {
59+
for (Element member : ct.getElements().getAllMembers(name)) {
60+
if (!trees.isAccessible(s, member, (DeclaredType) name.asType())) {
61+
throw new IllegalStateException("Inaccessible Name member: " + member);
62+
}
63+
}
64+
}
65+
}
66+
67+
static class MyFileObject extends SimpleJavaFileObject {
68+
public MyFileObject() {
69+
super(URI.create("myfo:/Test.java"), JavaFileObject.Kind.SOURCE);
70+
}
71+
public CharSequence getCharContent(boolean ignoreEncodingErrors) {
72+
return "public class Test { }";
73+
}
74+
}
75+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,79 @@
1+
/*
2+
* Copyright (c) 2021, 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 8272564
27+
* @summary Correct resolution of toString() (and other similar calls) on interfaces
28+
* @modules jdk.jdeps/com.sun.tools.classfile
29+
* @compile NoObjectToString.java
30+
* @run main NoObjectToString
31+
*/
32+
33+
import java.io.*;
34+
import com.sun.tools.classfile.*;
35+
import com.sun.tools.classfile.ConstantPool.CONSTANT_Methodref_info;
36+
37+
public class NoObjectToString {
38+
public static void main(String... args) throws Exception {
39+
NoObjectToString c = new NoObjectToString();
40+
c.run(args);
41+
}
42+
43+
void run(String... args) throws Exception {
44+
//Verify there are no references to Object.toString() in a Test:
45+
InputStream in = NoObjectToString.class.getResourceAsStream("NoObjectToString$Test.class");
46+
try {
47+
ClassFile cf = ClassFile.read(in);
48+
for (ConstantPool.CPInfo cpinfo: cf.constant_pool.entries()) {
49+
if (cpinfo.getTag() == ConstantPool.CONSTANT_Methodref) {
50+
CONSTANT_Methodref_info ref = (CONSTANT_Methodref_info) cpinfo;
51+
String methodDesc = ref.getClassInfo().getName() + "." + ref.getNameAndTypeInfo().getName() + ":" + ref.getNameAndTypeInfo().getType();
52+
53+
if ("java/lang/Object.toString:()Ljava/lang/String;".equals(methodDesc)) {
54+
throw new AssertionError("Found call to j.l.Object.toString");
55+
}
56+
}
57+
}
58+
} catch (ConstantPoolException ignore) {
59+
throw new AssertionError(ignore);
60+
} finally {
61+
in.close();
62+
}
63+
}
64+
65+
class Test {
66+
void test(I i, J j, K k) {
67+
i.toString();
68+
j.toString();
69+
k.toString();
70+
}
71+
}
72+
73+
interface I {
74+
public String toString();
75+
}
76+
interface J extends I {}
77+
interface K {}
78+
79+
}

0 commit comments

Comments
 (0)
Please sign in to comment.