File: improve-attribute-equals.patch

package info (click to toggle)
checker-framework-java 3.2.0%2Bds-4
  • links: PTS, VCS
  • area: main
  • in suites: forky, sid
  • size: 23,212 kB
  • sloc: java: 146,011; xml: 839; sh: 518; makefile: 404; perl: 26
file content (201 lines) | stat: -rw-r--r-- 9,363 bytes parent folder | download
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
From 66ba31a59cd369db0bd70fc000948281ab0fb334 Mon Sep 17 00:00:00 2001
From: Michael Ernst <mernst@cs.washington.edu>
Date: Wed, 16 Dec 2020 09:28:15 -0800
Subject: [PATCH] Improve attributeEquals (#4026)
Origin: upstream, https://github.com/typetools/checker-framework/commit/66ba31a59cd369db0bd70fc000948281ab0fb334
Bug-Debian: https://bugs.debian.org/cgi-bin/bugreport.cgi?bug=1128165
Bug-Ubuntu: https://bugs.launchpad.net/debian/+source/checker-framework-java/+bug/2142432

---
 .../javacutil/TypeAnnotationUtils.java        | 96 +++++++++++++++----
 1 file changed, 78 insertions(+), 18 deletions(-)

--- a/javacutil/src/main/java/org/checkerframework/javacutil/TypeAnnotationUtils.java
+++ b/javacutil/src/main/java/org/checkerframework/javacutil/TypeAnnotationUtils.java
@@ -1,7 +1,6 @@
 package org.checkerframework.javacutil;
 
 import com.sun.tools.javac.code.Attribute;
-import com.sun.tools.javac.code.Attribute.Constant;
 import com.sun.tools.javac.code.Attribute.TypeCompound;
 import com.sun.tools.javac.code.Symbol;
 import com.sun.tools.javac.code.Symbol.MethodSymbol;
@@ -12,7 +11,6 @@
 import com.sun.tools.javac.util.List;
 import com.sun.tools.javac.util.Name;
 import com.sun.tools.javac.util.Pair;
-import java.nio.ByteBuffer;
 import java.util.Arrays;
 import java.util.Iterator;
 import java.util.Map;
@@ -24,10 +22,12 @@
 import javax.lang.model.element.ExecutableElement;
 import javax.lang.model.element.VariableElement;
 import javax.lang.model.type.ArrayType;
+import javax.lang.model.type.DeclaredType;
 import javax.lang.model.type.TypeKind;
 import javax.lang.model.type.TypeMirror;
 import javax.lang.model.util.Elements;
 import javax.lang.model.util.Types;
+import org.checkerframework.checker.nullness.qual.NonNull;
 
 /**
  * A collection of helper methods related to type annotation handling.
@@ -46,12 +46,13 @@
      *
      * @param list the input list of TypeCompounds
      * @param tc the TypeCompound to find
+     * @param types type utilities
      * @return true, iff a TypeCompound equal to tc is contained in list
      */
     public static boolean isTypeCompoundContained(
             List<TypeCompound> list, TypeCompound tc, Types types) {
         for (Attribute.TypeCompound rawat : list) {
-            if (typeCompoundEquals(rawat, tc)) {
+            if (typeCompoundEquals(rawat, tc, types)) {
                 return true;
             }
         }
@@ -63,14 +64,15 @@
      *
      * @param tc1 the first TypeCompound to compare
      * @param tc2 the second TypeCompound to compare
+     * @param types type utilities
      * @return true if the TypeCompounds represent the same compound element value
      */
-    private static boolean typeCompoundEquals(TypeCompound tc1, TypeCompound tc2) {
+    private static boolean typeCompoundEquals(TypeCompound tc1, TypeCompound tc2, Types types) {
         // For the first conjunct, both of these forms fail in some cases:
         //   tc1.type == tc2.type
         //   types.isSameType(tc1.type, tc2.type)
         return contentEquals(tc1.type.tsym.name, tc2.type.tsym.name)
-                && typeCompoundValuesEquals(tc1.values, tc2.values)
+                && typeCompoundValuesEquals(tc1.values, tc2.values, types)
                 && isSameTAPositionExceptTreePos(tc1.position, tc2.position);
     }
 
@@ -81,12 +83,17 @@
      * @param n2 the second Name to compare
      * @return true if the two names represent the same string
      */
+    @SuppressWarnings(
+            "interning:unnecessary.equals" // Name is interned within a single instance of javac,
+    // but call equals anyway out of paranoia.
+    )
     private static boolean contentEquals(Name n1, Name n2) {
-        // Views of underlying bytes, not copies as with Name#contentEquals
-        ByteBuffer b1 = ByteBuffer.wrap(n1.getByteArray(), n1.getByteOffset(), n1.getByteLength());
-        ByteBuffer b2 = ByteBuffer.wrap(n2.getByteArray(), n2.getByteOffset(), n2.getByteLength());
-
-        return b1.equals(b2);
+        if (n1.getClass() == n2.getClass()) {
+            return n1.equals(n2);
+        } else {
+            // Slightly less efficient because it makes a copy.
+            return n1.contentEquals(n2);
+        }
     }
 
     /**
@@ -95,13 +102,15 @@
      *
      * @param values1 the first {@code values} field
      * @param values2 the second {@code values} field
+     * @param types type utilities
      * @return true if the two {@code values} fields represent the same name-to-value mapping, in
      *     the same order
      */
     @SuppressWarnings("InvalidParam") // Error Prone tries to be clever, but it is not
     private static boolean typeCompoundValuesEquals(
             List<Pair<MethodSymbol, Attribute>> values1,
-            List<Pair<MethodSymbol, Attribute>> values2) {
+            List<Pair<MethodSymbol, Attribute>> values2,
+            Types types) {
         if (values1.size() != values2.size()) {
             return false;
         }
@@ -111,7 +120,7 @@
                 iter1.hasNext(); ) {
             Pair<MethodSymbol, Attribute> pair1 = iter1.next();
             Pair<MethodSymbol, Attribute> pair2 = iter2.next();
-            if (!(pair1.fst.equals(pair2.fst) && attributeEquals(pair1.snd, pair2.snd))) {
+            if (!(pair1.fst.equals(pair2.fst) && attributeEquals(pair1.snd, pair2.snd, types))) {
                 return false;
             }
         }
@@ -119,18 +128,69 @@
     }
 
     /**
-     * Compares two attributes. Is more lenient for constants than {@code equals}, which is
-     * reference equality.
+     * Compares two attributes. Is more lenient for constants than {@code Attribute.equals}, which
+     * is reference equality.
      *
      * @param a1 the first attribute to compare
      * @param a2 the second attribute to compare
+     * @param types type utilities
      * @return true if the two attributes are the same
      */
-    private static boolean attributeEquals(Attribute a1, Attribute a2) {
-        if (a1 instanceof Constant && a2 instanceof Constant) {
-            Object v1 = ((Constant) a1).getValue();
-            Object v2 = ((Constant) a1).getValue();
+    private static boolean attributeEquals(Attribute a1, Attribute a2, Types types) {
+        if (a1 instanceof Attribute.Array && a2 instanceof Attribute.Array) {
+            List<Attribute> list1 = ((Attribute.Array) a1).getValue();
+            List<Attribute> list2 = ((Attribute.Array) a2).getValue();
+            if (list1.size() != list2.size()) {
+                return false;
+            }
+            // This requires the array elements to be in the same order.  Is that the right thing?
+            for (int i = 0; i < list1.size(); i++) {
+                if (!attributeEquals(list1.get(i), list2.get(i), types)) {
+                    return false;
+                }
+            }
+            return true;
+        } else if (a1 instanceof Attribute.Class && a2 instanceof Attribute.Class) {
+            Type t1 = ((Attribute.Class) a1).getValue();
+            Type t2 = ((Attribute.Class) a2).getValue();
+            return types.isSameType(t1, t2);
+        } else if (a1 instanceof Attribute.Constant && a2 instanceof Attribute.Constant) {
+            Object v1 = ((Attribute.Constant) a1).getValue();
+            Object v2 = ((Attribute.Constant) a2).getValue();
             return v1.equals(v2);
+        } else if (a1 instanceof Attribute.Compound && a2 instanceof Attribute.Compound) {
+            // The annotation value is another annotation.  `a1` and `a2` implement
+            // AnnotationMirror.
+            DeclaredType t1 = ((Attribute.Compound) a1).getAnnotationType();
+            DeclaredType t2 = ((Attribute.Compound) a2).getAnnotationType();
+            if (!types.isSameType(t1, t2)) {
+                return false;
+            }
+            Map<Symbol.MethodSymbol, Attribute> map1 = ((Attribute.Compound) a1).getElementValues();
+            Map<Symbol.MethodSymbol, Attribute> map2 = ((Attribute.Compound) a2).getElementValues();
+            // Is this test, which uses equals() for the keys, too strict?
+            if (!map1.keySet().equals(map2.keySet())) {
+                return false;
+            }
+            for (Symbol.MethodSymbol key : map1.keySet()) {
+                Attribute attr1 = map1.get(key);
+                @SuppressWarnings(
+                        "nullness:assignment.type.incompatible") // same keys in map1 & map2
+                @NonNull Attribute attr2 = map2.get(key);
+                if (!attributeEquals(attr1, attr2, types)) {
+                    return false;
+                }
+            }
+            return true;
+        } else if (a1 instanceof Attribute.Enum && a2 instanceof Attribute.Enum) {
+            Symbol.VarSymbol s1 = ((Attribute.Enum) a1).getValue();
+            Symbol.VarSymbol s2 = ((Attribute.Enum) a2).getValue();
+            // VarSymbol.equals() is reference equality.
+            return s1.equals(s2) || s1.toString().equals(s2.toString());
+        } else if (a1 instanceof Attribute.Error && a2 instanceof Attribute.Error) {
+            String s1 = ((Attribute.Error) a1).getValue();
+            String s2 = ((Attribute.Error) a2).getValue();
+            return s1.equals(s2);
         } else {
             return a1.equals(a2);
         }