diff --git a/src/share/classes/com/sun/tools/javac/code/Symbol.java b/src/share/classes/com/sun/tools/javac/code/Symbol.java index f9ef7271e28e6b1c78b34659665a7756dcf8a939..3d5b0d0391490bfc061dc3b90d393e16b45176b1 100644 --- a/src/share/classes/com/sun/tools/javac/code/Symbol.java +++ b/src/share/classes/com/sun/tools/javac/code/Symbol.java @@ -237,7 +237,7 @@ public abstract class Symbol implements Element { } /** Has this symbol an empty name? This includes anonymous - * inner classses. + * inner classes. */ public boolean isAnonymous() { return name.isEmpty(); diff --git a/src/share/classes/com/sun/tools/javac/code/Symtab.java b/src/share/classes/com/sun/tools/javac/code/Symtab.java index 351780d8ae3906521f54f9956f1fe0610d6679d7..12918fd46d987a14a9da8ff1ccce3cb0c92dc70e 100644 --- a/src/share/classes/com/sun/tools/javac/code/Symtab.java +++ b/src/share/classes/com/sun/tools/javac/code/Symtab.java @@ -148,6 +148,7 @@ public class Symtab { public final Type listType; public final Type collectionsType; public final Type comparableType; + public final Type comparatorType; public final Type arraysType; public final Type iterableType; public final Type iteratorType; @@ -502,6 +503,7 @@ public class Symtab { listType = enterClass("java.util.List"); collectionsType = enterClass("java.util.Collections"); comparableType = enterClass("java.lang.Comparable"); + comparatorType = enterClass("java.util.Comparator"); arraysType = enterClass("java.util.Arrays"); iterableType = target.hasIterable() ? enterClass("java.lang.Iterable") diff --git a/src/share/classes/com/sun/tools/javac/comp/Attr.java b/src/share/classes/com/sun/tools/javac/comp/Attr.java index e32457d38c87f4ab83668ce354405951d2dbffe6..e6f9e8e0e9f65ab2c0c349206b6dad85498f28c9 100644 --- a/src/share/classes/com/sun/tools/javac/comp/Attr.java +++ b/src/share/classes/com/sun/tools/javac/comp/Attr.java @@ -4016,7 +4016,7 @@ public class Attr extends JCTree.Visitor { attribClassBody(env, c); chk.checkDeprecatedAnnotation(env.tree.pos(), c); - chk.checkClassOverrideEqualsAndHash(env.tree.pos(), c); + chk.checkClassOverrideEqualsAndHashIfNeeded(env.tree.pos(), c); } finally { env.info.returnResult = prevReturnRes; log.useSource(prev); diff --git a/src/share/classes/com/sun/tools/javac/comp/Check.java b/src/share/classes/com/sun/tools/javac/comp/Check.java index 02a6bd77d3924c6aed194ccafa1b997ac4d0cf0d..9af4e5654eb108c2521b33773b84139e83618f57 100644 --- a/src/share/classes/com/sun/tools/javac/comp/Check.java +++ b/src/share/classes/com/sun/tools/javac/comp/Check.java @@ -1972,14 +1972,32 @@ public class Check { } }; - public void checkClassOverrideEqualsAndHash(DiagnosticPosition pos, + public void checkClassOverrideEqualsAndHashIfNeeded(DiagnosticPosition pos, + ClassSymbol someClass) { + /* At present, annotations cannot possibly have a method that is override + * equivalent with Object.equals(Object) but in any case the condition is + * fine for completeness. + */ + if (someClass == (ClassSymbol)syms.objectType.tsym || + someClass.isInterface() || someClass.isEnum() || + (someClass.flags() & ANNOTATION) != 0 || + (someClass.flags() & ABSTRACT) != 0) return; + //anonymous inner classes implementing interfaces need especial treatment + if (someClass.isAnonymous()) { + List interfaces = types.interfaces(someClass.type); + if (interfaces != null && !interfaces.isEmpty() && + interfaces.head.tsym == syms.comparatorType.tsym) return; + } + checkClassOverrideEqualsAndHash(pos, someClass); + } + + private void checkClassOverrideEqualsAndHash(DiagnosticPosition pos, ClassSymbol someClass) { if (lint.isEnabled(LintCategory.OVERRIDES)) { MethodSymbol equalsAtObject = (MethodSymbol)syms.objectType .tsym.members().lookup(names.equals).sym; MethodSymbol hashCodeAtObject = (MethodSymbol)syms.objectType .tsym.members().lookup(names.hashCode).sym; - boolean overridesEquals = types.implementation(equalsAtObject, someClass, false, equalsHasCodeFilter).owner == someClass; boolean overridesHashCode = types.implementation(hashCodeAtObject, @@ -1987,7 +2005,7 @@ public class Check { if (overridesEquals && !overridesHashCode) { log.warning(LintCategory.OVERRIDES, pos, - "override.equals.but.not.hashcode", someClass.fullname); + "override.equals.but.not.hashcode", someClass); } } } diff --git a/src/share/classes/com/sun/tools/javac/resources/compiler.properties b/src/share/classes/com/sun/tools/javac/resources/compiler.properties index 20de921fae9ee5e2909d23fc813591b01ba9d721..6904d15946d60a51b8c27be7eb5ae8a921cc8a1f 100644 --- a/src/share/classes/com/sun/tools/javac/resources/compiler.properties +++ b/src/share/classes/com/sun/tools/javac/resources/compiler.properties @@ -2077,7 +2077,7 @@ compiler.warn.override.unchecked.thrown=\ {0}\n\ overridden method does not throw {1} -# 0: class name +# 0: symbol compiler.warn.override.equals.but.not.hashcode=\ Class {0} overrides equals, but neither it nor any superclass overrides hashCode method diff --git a/test/tools/javac/6563143/EqualsHashCodeWarningTest.java b/test/tools/javac/6563143/EqualsHashCodeWarningTest.java new file mode 100644 index 0000000000000000000000000000000000000000..2c30c46ed8e41e9d252df5e2ec3300539cba0702 --- /dev/null +++ b/test/tools/javac/6563143/EqualsHashCodeWarningTest.java @@ -0,0 +1,71 @@ +/* + * @test /nodynamiccopyright/ + * @bug 6563143 8008436 8009138 + * @summary javac should issue a warning for overriding equals without hashCode + * @summary javac should not issue a warning for overriding equals without hasCode + * @summary javac, equals-hashCode warning tuning + * if hashCode has been overriden by a superclass + * @compile/ref=EqualsHashCodeWarningTest.out -Xlint:overrides -XDrawDiagnostics EqualsHashCodeWarningTest.java + */ + +import java.util.Comparator; + +public class EqualsHashCodeWarningTest { + @Override + public boolean equals(Object o) { + return o == this; + } + + @Override + public int hashCode() { + return 0; + } + + public Comparator m() { + return new Comparator() { + @Override + public boolean equals(Object o) {return true;} + + @Override + public int compare(Object o1, Object o2) { + return 0; + } + }; + } +} + +class SubClass extends EqualsHashCodeWarningTest { + @Override + public boolean equals(Object o) { + return true; + } +} + +@SuppressWarnings("overrides") +class DontWarnMe { + @Override + public boolean equals(Object o) { + return true; + } +} + +class DoWarnMe { + @Override + public boolean equals(Object o) { + return o == this; + } +} + +abstract class IamAbstractGetMeOutOfHere { + public boolean equals(Object o){return true;} +} + +interface I { + public boolean equals(Object o); +} + +enum E { + A, B +} + +@interface anno {} diff --git a/test/tools/javac/6563143/EqualsHashCodeWarningTest.out b/test/tools/javac/6563143/EqualsHashCodeWarningTest.out new file mode 100644 index 0000000000000000000000000000000000000000..51f21b09d6fa62acaf2501ebcbc126648ca7d6b4 --- /dev/null +++ b/test/tools/javac/6563143/EqualsHashCodeWarningTest.out @@ -0,0 +1,2 @@ +EqualsHashCodeWarningTest.java:52:1: compiler.warn.override.equals.but.not.hashcode: DoWarnMe +1 warning diff --git a/test/tools/javac/6563143/OverridesEqualsButNotHashCodeTest.java b/test/tools/javac/6563143/OverridesEqualsButNotHashCodeTest.java deleted file mode 100644 index 07d92f4ee3aae2d1bb6e2176b29cf40b6d79518a..0000000000000000000000000000000000000000 --- a/test/tools/javac/6563143/OverridesEqualsButNotHashCodeTest.java +++ /dev/null @@ -1,42 +0,0 @@ -/* - * @test /nodynamiccopyright/ - * @bug 6563143 8008436 - * @summary javac should issue a warning for overriding equals without hashCode - * @summary javac should not issue a warning for overriding equals without hasCode - * if hashCode has been overriden by a superclass - * @compile/ref=OverridesEqualsButNotHashCodeTest.out -Xlint:overrides -XDrawDiagnostics OverridesEqualsButNotHashCodeTest.java - */ - -public class OverridesEqualsButNotHashCodeTest { - @Override - public boolean equals(Object o) { - return o == this; - } - - @Override - public int hashCode() { - return 0; - } -} - -class SubClass extends OverridesEqualsButNotHashCodeTest { - @Override - public boolean equals(Object o) { - return o == this; - } -} - -@SuppressWarnings("overrides") -class NoWarning { - @Override - public boolean equals(Object o) { - return o == this; - } -} - -class DoWarnMe { - @Override - public boolean equals(Object o) { - return o == this; - } -} diff --git a/test/tools/javac/6563143/OverridesEqualsButNotHashCodeTest.out b/test/tools/javac/6563143/OverridesEqualsButNotHashCodeTest.out deleted file mode 100644 index 9d3825a0584c42b11c8b464f952c6acffc8ad520..0000000000000000000000000000000000000000 --- a/test/tools/javac/6563143/OverridesEqualsButNotHashCodeTest.out +++ /dev/null @@ -1,2 +0,0 @@ -OverridesEqualsButNotHashCodeTest.java:37:1: compiler.warn.override.equals.but.not.hashcode: DoWarnMe -1 warning