From 5f1c71948b79413d912a8409d29751f703452cba Mon Sep 17 00:00:00 2001 From: vromero Date: Thu, 7 Mar 2013 10:04:28 +0000 Subject: [PATCH] 8009138: javac, equals-hashCode warning tuning Reviewed-by: mcimadamore --- .../com/sun/tools/javac/code/Symbol.java | 2 +- .../com/sun/tools/javac/code/Symtab.java | 2 + .../com/sun/tools/javac/comp/Attr.java | 2 +- .../com/sun/tools/javac/comp/Check.java | 24 ++++++- .../tools/javac/resources/compiler.properties | 2 +- .../6563143/EqualsHashCodeWarningTest.java | 71 +++++++++++++++++++ .../6563143/EqualsHashCodeWarningTest.out | 2 + .../OverridesEqualsButNotHashCodeTest.java | 42 ----------- .../OverridesEqualsButNotHashCodeTest.out | 2 - 9 files changed, 99 insertions(+), 50 deletions(-) create mode 100644 test/tools/javac/6563143/EqualsHashCodeWarningTest.java create mode 100644 test/tools/javac/6563143/EqualsHashCodeWarningTest.out delete mode 100644 test/tools/javac/6563143/OverridesEqualsButNotHashCodeTest.java delete mode 100644 test/tools/javac/6563143/OverridesEqualsButNotHashCodeTest.out 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 f9ef7271..3d5b0d03 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 351780d8..12918fd4 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 e32457d3..e6f9e8e0 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 02a6bd77..9af4e565 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 20de921f..6904d159 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 00000000..2c30c46e --- /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 00000000..51f21b09 --- /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 07d92f4e..00000000 --- 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 9d3825a0..00000000 --- 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 -- GitLab