1 /*
  2  * Copyright (c) 2022, 2025, 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 8266670 8293626 8297271
 27  * @summary Basic tests of AccessFlag
 28  * @run junit BasicAccessFlagTest
 29  */
 30 
 31 import java.lang.classfile.ClassFile;
 32 import java.lang.reflect.AccessFlag;
 33 import java.lang.reflect.ClassFileFormatVersion;
 34 import java.lang.reflect.Field;
 35 import java.lang.reflect.Modifier;
 36 import java.util.EnumSet;
 37 import java.util.Map;
 38 import java.util.LinkedHashMap;
 39 import java.util.HashSet;
 40 import java.util.Set;
 41 
 42 import org.junit.Test;
 43 import org.junit.jupiter.api.Assertions;
 44 
 45 import static org.junit.jupiter.api.Assertions.*;
 46 
 47 public class BasicAccessFlagTest {
 48 
 49     /*
 50      * Verify sourceModifier() == true access flags have a
 51      * corresponding constant in java.lang.reflect.Modifier.
 52      */
 53     @Test
 54     public void testSourceModifiers() throws Exception {
 55         Class<?> modifierClass = Modifier.class;
 56 
 57         for(AccessFlag accessFlag : AccessFlag.values()) {
 58             if (accessFlag.sourceModifier()) {
 59                 // Check for consistency
 60                 Field f = modifierClass.getField(accessFlag.name());
 61                 assertEquals(f.getInt(null), accessFlag.mask(), accessFlag + " mask");
 62             }
 63         }
 64     }
 65 
 66     // The mask values of the enum constants must be non-decreasing;
 67     // in other words stay the same (for colliding mask values) or go
 68     // up.
 69     @Test
 70     public void testMaskOrdering() {
 71         AccessFlag[] values = AccessFlag.values();
 72         for (int i = 1; i < values.length; i++) {
 73             AccessFlag left  = values[i-1];
 74             AccessFlag right = values[i];
 75             assertTrue(left.mask() <= right.mask(), () -> left
 76                     + "has a greater mask than "
 77                     + right);
 78         }
 79     }
 80 
 81     // Test that if access flags have a matching mask, their locations
 82     // are disjoint.
 83     @Test
 84     public void testDisjoint() {
 85         // First build the mask -> access flags map...
 86         Map<Integer, Set<AccessFlag>> maskToFlags = new LinkedHashMap<>();
 87 
 88         for (var accessFlag : AccessFlag.values()) {
 89             Integer mask = accessFlag.mask();
 90             Set<AccessFlag> flags = maskToFlags.get(mask);
 91 
 92             if (flags == null) {
 93                 flags = new HashSet<>();
 94                 flags.add(accessFlag);
 95                 maskToFlags.put(mask, flags);
 96             } else {
 97                 flags.add(accessFlag);
 98             }
 99         }
100 
101         // ...then test for disjointness
102         for (var entry : maskToFlags.entrySet()) {
103             var value = entry.getValue();
104             if (value.size() == 0) {
105                 throw new AssertionError("Bad flag set " + entry);
106             } else if (value.size() == 1) {
107                 // Need at least two flags to be non-disjointness to
108                 // be possible
109                 continue;
110             }
111 
112             Set<AccessFlag.Location> locations = new HashSet<>();
113             for (var accessFlag : value) {
114                 for (var location : accessFlag.locations()) {
115                     boolean added = locations.add(location);
116                     if (!added) {
117                         reportError(location, accessFlag,
118                                     entry.getKey(), value);
119                     }
120                 }
121             }
122         }
123     }
124 
125     private static void reportError(AccessFlag.Location location,
126                                     AccessFlag accessFlag,
127                                     Integer mask, Set<AccessFlag> value) {
128         System.err.println("Location " + location +
129                            " from " + accessFlag +
130                            " already present for 0x" +
131                            Integer.toHexString(mask) + ": " + value);
132         throw new RuntimeException();
133     }
134 
135     // For each access flag, make sure it is recognized on every kind
136     // of location it can apply to
137     @Test
138     public void testMaskToAccessFlagsPositive() {
139         for (var accessFlag : AccessFlag.values()) {
140             Set<AccessFlag> expectedSet = EnumSet.of(accessFlag);
141             for (var location : accessFlag.locations()) {
142                 Set<AccessFlag> computedSet =
143                     AccessFlag.maskToAccessFlags(accessFlag.mask(), location);
144                 if (!expectedSet.equals(computedSet)) {
145                     throw new RuntimeException("Bad set computation on " +
146                                                accessFlag + ", " + location);
147                 }
148             }
149             for (var cffv : ClassFileFormatVersion.values()) {
150                 for (var location : accessFlag.locations(cffv)) {
151                     Set<AccessFlag> computedSet =
152                             AccessFlag.maskToAccessFlags(accessFlag.mask(), location, cffv);
153                     if (!expectedSet.equals(computedSet)) {
154                         throw new RuntimeException("Bad set computation on " +
155                                 accessFlag + ", " + location);
156                     }
157                 }
158             }
159         }
160         assertEquals(Set.of(AccessFlag.STRICT), AccessFlag.maskToAccessFlags(Modifier.STRICT, AccessFlag.Location.METHOD, ClassFileFormatVersion.RELEASE_8));
161     }
162 
163     @Test
164     public void testMaskToAccessFlagsNegative() {
165         assertThrows(IllegalArgumentException.class, () -> AccessFlag.maskToAccessFlags(Modifier.STRICT, AccessFlag.Location.METHOD));
166         assertThrows(IllegalArgumentException.class, () -> AccessFlag.maskToAccessFlags(Modifier.STRICT, AccessFlag.Location.METHOD, ClassFileFormatVersion.RELEASE_17));
167         assertThrows(IllegalArgumentException.class, () -> AccessFlag.maskToAccessFlags(Modifier.STRICT, AccessFlag.Location.METHOD, ClassFileFormatVersion.RELEASE_1));
168         assertThrows(IllegalArgumentException.class, () -> AccessFlag.maskToAccessFlags(Modifier.PRIVATE, AccessFlag.Location.CLASS));
169         assertThrows(IllegalArgumentException.class, () -> AccessFlag.maskToAccessFlags(ClassFile.ACC_MODULE, AccessFlag.Location.CLASS, ClassFileFormatVersion.RELEASE_8));
170         assertThrows(IllegalArgumentException.class, () -> AccessFlag.maskToAccessFlags(ClassFile.ACC_ANNOTATION, AccessFlag.Location.CLASS, ClassFileFormatVersion.RELEASE_4));
171         assertThrows(IllegalArgumentException.class, () -> AccessFlag.maskToAccessFlags(ClassFile.ACC_ENUM, AccessFlag.Location.FIELD, ClassFileFormatVersion.RELEASE_4));
172         assertThrows(IllegalArgumentException.class, () -> AccessFlag.maskToAccessFlags(ClassFile.ACC_SYNTHETIC, AccessFlag.Location.INNER_CLASS, ClassFileFormatVersion.RELEASE_4));
173         assertThrows(IllegalArgumentException.class, () -> AccessFlag.maskToAccessFlags(ClassFile.ACC_PUBLIC, AccessFlag.Location.INNER_CLASS, ClassFileFormatVersion.RELEASE_0));
174         assertThrows(IllegalArgumentException.class, () -> AccessFlag.maskToAccessFlags(ClassFile.ACC_MANDATED, AccessFlag.Location.METHOD_PARAMETER, ClassFileFormatVersion.RELEASE_7));
175         assertThrows(IllegalArgumentException.class, () -> AccessFlag.maskToAccessFlags(ClassFile.ACC_MANDATED, AccessFlag.Location.MODULE, ClassFileFormatVersion.RELEASE_7));
176     }
177 
178     @Test
179     public void testLocationsNullHandling() {
180         for (var flag : AccessFlag.values()) {
181             assertThrows(NullPointerException.class, () -> flag.locations(null));
182         }
183 
184         for (var location : AccessFlag.Location.values()) {
185             assertThrows(NullPointerException.class, () -> location.flags(null));
186         }
187 
188         for (var location : AccessFlag.Location.values()) {
189             try {
190                 location.flags(null);
191                 throw new RuntimeException("Did not get NPE on " + location + ".flags(null)");
192             } catch (NullPointerException npe ) {
193                 ; // Expected
194             }
195             try {
196                 location.flagsMask(null);
197                 throw new RuntimeException("Did not get NPE on " + location + ".flagsMask(null)");
198             } catch (NullPointerException npe ) {
199                 ; // Expected
200             }
201         }
202     }
203 }