Skip to content

Commit

Permalink
improve Index[View].getKnownUsers()
Browse files Browse the repository at this point in the history
Previously, the `getKnownUsers()` method only considered a class
as used in another class when it occured in the list of class
references in the other class's constant pool.

With this commit, a class is also considered as used in another class
when it occurs in the other class's signature (superclass type,
superinterface types, type parameters), in the signatures of
the other class's methods (return type, parameter types, exception
types, type parameters), or in the types of the other class's fields
and record components.
  • Loading branch information
Ladicek committed Apr 25, 2024
1 parent af12bff commit 84e7350
Show file tree
Hide file tree
Showing 3 changed files with 221 additions and 15 deletions.
39 changes: 33 additions & 6 deletions core/src/main/java/org/jboss/jandex/IndexView.java
Original file line number Diff line number Diff line change
Expand Up @@ -452,17 +452,35 @@ default ModuleInfo getModuleByName(String moduleName) {
}

/**
* Obtains a list of classes that use the specified class. In other words, a list of classes that include
* a reference to the specified class in their constant pool.
* Returns a list of classes in this index that use the specified class. For one class
* to <em>use</em> another class, the other class has to:
* <ul>
* <li>occur in the signature of the class (that is, in the superclass type,
* in the superinterface types, or in the type parameters), or</li>
* <li>occur in the signature of any of the class's methods (that is, in the return type,
* in the parameter types, in the exception types, or in the type parameters), or</li>
* <li>occur in the type of any of the class's fields or record components, or</li>
* <li>occur in the list of class references in the constant pool, as described
* by the JLS and JVMS.</li>
* </ul>
*
* @param className the name of the class to look for
* @return a non-null list of classes that use the specified class
*/
Collection<ClassInfo> getKnownUsers(DotName className);

/**
* Obtains a list of classes that use the specified class. In other words, a list of classes that include
* a reference to the specified class in their constant pool.
* Returns a list of classes in this index that use the specified class. For one class
* to <em>use</em> another class, the other class has to:
* <ul>
* <li>occur in the signature of the class (that is, in the superclass type,
* in the superinterface types, or in the type parameters), or</li>
* <li>occur in the signature of any of the class's methods (that is, in the return type,
* in the parameter types, in the exception types, or in the type parameters), or</li>
* <li>occur in the type of any of the class's fields or record components, or</li>
* <li>occur in the list of class references in the constant pool, as described
* by the JLS and JVMS.</li>
* </ul>
*
* @param className the name of the class to look for
* @return a non-null list of classes that use the specified class
Expand All @@ -472,8 +490,17 @@ default Collection<ClassInfo> getKnownUsers(String className) {
}

/**
* Obtains a list of classes that use the specified class. In other words, a list of classes that include
* a reference to the specified class in their constant pool.
* Returns a list of classes in this index that use the specified class. For one class
* to <em>use</em> another class, the other class has to:
* <ul>
* <li>occur in the signature of the class (that is, in the superclass type,
* in the superinterface types, or in the type parameters), or</li>
* <li>occur in the signature of any of the class's methods (that is, in the return type,
* in the parameter types, in the exception types, or in the type parameters), or</li>
* <li>occur in the type of any of the class's fields or record components, or</li>
* <li>occur in the list of class references in the constant pool of the class,
* as described by the JLS and JVMS.</li>
* </ul>
*
* @param clazz the class to look for
* @return a non-null list of classes that use the specified class
Expand Down
87 changes: 78 additions & 9 deletions core/src/main/java/org/jboss/jandex/Indexer.java
Original file line number Diff line number Diff line change
Expand Up @@ -35,6 +35,7 @@
import java.util.HashMap;
import java.util.HashSet;
import java.util.IdentityHashMap;
import java.util.LinkedHashSet;
import java.util.List;
import java.util.Locale;
import java.util.Map;
Expand Down Expand Up @@ -407,7 +408,7 @@ void returnConstantAnnoAttributes(byte[] attributes) {
private Map<DotName, List<ClassInfo>> implementors;
private Map<DotName, ClassInfo> classes;
private Map<DotName, ModuleInfo> modules;
private Map<DotName, List<ClassInfo>> users;
private Map<DotName, Set<ClassInfo>> users; // must be a linked set for reproducibility
private NameTable names;
private GenericSignatureParser signatureParser;
private final TmpObjects tmpObjects = new TmpObjects();
Expand All @@ -432,7 +433,7 @@ private void initIndexMaps() {
modules = new HashMap<DotName, ModuleInfo>();

if (users == null)
users = new HashMap<DotName, List<ClassInfo>>();
users = new HashMap<DotName, Set<ClassInfo>>();

if (names == null)
names = new NameTable();
Expand Down Expand Up @@ -1082,6 +1083,7 @@ private void resolveTypeAnnotations() {
}

private void resolveUsers() throws IOException {
// class references in constant pool
int poolSize = constantPoolSize;
byte[] pool = constantPool;
int[] offsets = constantPoolOffsets;
Expand All @@ -1091,16 +1093,79 @@ private void resolveUsers() throws IOException {
if (pool[offset] == CONSTANT_CLASS) {
int nameIndex = (pool[++offset] & 0xFF) << 8 | (pool[++offset] & 0xFF);
DotName usedClass = names.convertToName(decodeUtf8Entry(nameIndex), '/');
List<ClassInfo> usersOfClass = users.get(usedClass);
if (usersOfClass == null) {
usersOfClass = new ArrayList<ClassInfo>();
users.put(usedClass, usersOfClass);
}
usersOfClass.add(this.currentClass);
recordUsedClass(usedClass);
}
}

// class declaration
for (TypeVariable typeParameter : currentClass.typeParameters()) {
recordUsedType(typeParameter);
}
recordUsedType(currentClass.superClassType());
for (Type interfaceType : currentClass.interfaceTypes()) {
recordUsedType(interfaceType);
}
// field declarations
for (FieldInfo field : fields) {
recordUsedType(field.type());
}
// method declarations (ignoring receiver types, they are always the current class)
for (MethodInfo method : methods) {
for (TypeVariable typeParameter : method.typeParameters()) {
recordUsedType(typeParameter);
}
recordUsedType(method.returnType());
for (Type parameterType : method.parameterTypes()) {
recordUsedType(parameterType);
}
for (Type exceptionType : method.exceptions()) {
recordUsedType(exceptionType);
}
}
// record component declarations
for (RecordComponentInfo recordComponent : recordComponents) {
recordUsedType(recordComponent.type());
}
}

private void recordUsedType(Type type) {
if (type == null) {
return;
}

switch (type.kind()) {
case CLASS:
recordUsedClass(type.asClassType().name());
break;
case PARAMETERIZED_TYPE:
recordUsedClass(type.asParameterizedType().name());
for (Type typeArgument : type.asParameterizedType().arguments()) {
recordUsedType(typeArgument);
}
break;
case ARRAY:
recordUsedType(type.asArrayType().elementType());
break;
case WILDCARD_TYPE:
recordUsedType(type.asWildcardType().bound());
break;
case TYPE_VARIABLE:
for (Type bound : type.asTypeVariable().boundArray()) {
recordUsedType(bound);
}
break;
}
}

private void recordUsedClass(DotName usedClass) {
Set<ClassInfo> usersOfClass = users.get(usedClass);
if (usersOfClass == null) {
usersOfClass = new LinkedHashSet<>();
users.put(usedClass, usersOfClass);
}
usersOfClass.add(this.currentClass);
}

private void updateTypeTargets() {
for (Map.Entry<AnnotationTarget, List<TypeAnnotationState>> entry : typeAnnotations.entrySet()) {
AnnotationTarget key = entry.getKey();
Expand Down Expand Up @@ -2557,7 +2622,11 @@ public Index complete() {
propagateTypeVariables();

try {
return new Index(masterAnnotations, subclasses, subinterfaces, implementors, classes, modules, users);
Map<DotName, List<ClassInfo>> userLists = new HashMap<>();
for (Map.Entry<DotName, Set<ClassInfo>> entry : users.entrySet()) {
userLists.put(entry.getKey(), new ArrayList<>(entry.getValue()));
}
return new Index(masterAnnotations, subclasses, subinterfaces, implementors, classes, modules, userLists);
} finally {
masterAnnotations = null;
subclasses = null;
Expand Down
110 changes: 110 additions & 0 deletions core/src/test/java/org/jboss/jandex/test/KnownUsersTest.java
Original file line number Diff line number Diff line change
@@ -0,0 +1,110 @@
package org.jboss.jandex.test;

import static org.junit.jupiter.api.Assertions.assertFalse;
import static org.junit.jupiter.api.Assertions.assertNotNull;
import static org.junit.jupiter.api.Assertions.fail;

import java.io.File;
import java.io.IOException;
import java.nio.file.Path;
import java.nio.file.Paths;
import java.util.ArrayList;
import java.util.Collection;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.Queue;
import java.util.Set;

import org.jboss.jandex.ClassInfo;
import org.jboss.jandex.Index;
import org.jboss.jandex.test.util.IndexingUtil;
import org.junit.jupiter.api.Test;

public class KnownUsersTest {
static class SuperClass<T> {
}

interface ImplementedInterface1<T> {
}

interface ImplementedInterface2<T> {
}

static class TestClass<T extends Number> extends SuperClass<CharSequence>
implements ImplementedInterface1<T>, ImplementedInterface2<RuntimeException> {
int i;

Map<String, List<Integer>> m;

TestClass(StringBuilder str) {
m = new HashMap<>();
m.put("foo", new ArrayList<>());
}

<U extends T, V extends Collection<?>, W extends Exception> U bar(Set<? extends Long> s, Queue<? super Double> q)
throws IllegalArgumentException, IllegalStateException, W {
// `toString()` to force a class reference to `File` into the constant pool
Paths.get("").toFile().toString();
return null;
}

static class NestedClass {
}

class InnerClass {
}
}

@Test
public void test() throws IOException {
Index index = Index.of(SuperClass.class, ImplementedInterface1.class, ImplementedInterface2.class, TestClass.class);
doTest(index);
doTest(IndexingUtil.roundtrip(index));
}

private void doTest(Index index) {
// from class signature
assertKnownUsers(index, SuperClass.class);
assertKnownUsers(index, ImplementedInterface1.class);
assertKnownUsers(index, ImplementedInterface2.class);
assertKnownUsers(index, Number.class);
assertKnownUsers(index, CharSequence.class);
assertKnownUsers(index, RuntimeException.class);
// from field types
assertKnownUsers(index, String.class);
assertKnownUsers(index, Integer.class);
// from method signatures
assertKnownUsers(index, StringBuilder.class);
assertKnownUsers(index, Collection.class);
assertKnownUsers(index, Exception.class);
assertKnownUsers(index, Set.class);
assertKnownUsers(index, Long.class);
assertKnownUsers(index, Queue.class);
assertKnownUsers(index, Double.class);
assertKnownUsers(index, IllegalArgumentException.class);
assertKnownUsers(index, IllegalStateException.class);
// from method bodies (class references in the constant pool)
assertKnownUsers(index, HashMap.class);
assertKnownUsers(index, ArrayList.class);
assertKnownUsers(index, Paths.class);
assertKnownUsers(index, Path.class);
assertKnownUsers(index, File.class);
// member classes (class references in the constant pool)
assertKnownUsers(index, TestClass.NestedClass.class);
assertKnownUsers(index, TestClass.InnerClass.class);
}

private void assertKnownUsers(Index index, Class<?> clazz) {
Collection<ClassInfo> knownUsers = index.getKnownUsers(clazz);

assertNotNull(knownUsers);
assertFalse(knownUsers.isEmpty());
for (ClassInfo knownUser : knownUsers) {
if (TestClass.class.getName().equals(knownUser.name().toString())) {
return;
}
}
fail("Expected " + TestClass.class.getName() + " to be a known user of " + clazz.getName());
}
}

0 comments on commit 84e7350

Please sign in to comment.