From e42d2ed266a4d6f5f3d4f06446632612014ec533 Mon Sep 17 00:00:00 2001 From: Martin Kouba Date: Mon, 13 Jan 2025 11:04:45 +0100 Subject: [PATCH] ArC: optimize "List all" injection points - only collect InjectionPoint metadata if any of the resolved beans is dependent --- .../io/quarkus/arc/processor/BuiltinBean.java | 59 ++++++++++++------- 1 file changed, 39 insertions(+), 20 deletions(-) diff --git a/independent-projects/arc/processor/src/main/java/io/quarkus/arc/processor/BuiltinBean.java b/independent-projects/arc/processor/src/main/java/io/quarkus/arc/processor/BuiltinBean.java index 20dd2f1bfe44e..64f53e7acd632 100644 --- a/independent-projects/arc/processor/src/main/java/io/quarkus/arc/processor/BuiltinBean.java +++ b/independent-projects/arc/processor/src/main/java/io/quarkus/arc/processor/BuiltinBean.java @@ -10,6 +10,7 @@ import java.util.function.Consumer; import java.util.function.Predicate; import java.util.function.Supplier; +import java.util.stream.Collectors; import jakarta.enterprise.inject.spi.DefinitionException; import jakarta.enterprise.inject.spi.InjectionPoint; @@ -407,31 +408,49 @@ private static void generateListBytecode(GeneratorContext ctx) { requiredType = Types.getTypeHandle(mc, type); usesInstanceHandle = mc.load(false); } - ResultHandle qualifiers = BeanGenerator.collectInjectionPointQualifiers( ctx.beanDeployment, ctx.constructor, ctx.injectionPoint, ctx.annotationLiterals); - ResultHandle annotationsHandle = BeanGenerator.collectInjectionPointAnnotations( - ctx.beanDeployment, - ctx.constructor, ctx.injectionPoint, ctx.annotationLiterals, ctx.injectionPointAnnotationsPredicate); - ResultHandle javaMemberHandle = BeanGenerator.getJavaMemberHandle(ctx.constructor, ctx.injectionPoint, - ctx.reflectionRegistration); + + // Note that we only collect the injection point metadata if needed, i.e. if any of the resolved beans is dependent, + // and requires InjectionPoint metadata + Set beans = ctx.beanDeployment.beanResolver.resolveBeans( + type.name().equals(DotNames.INSTANCE_HANDLE) ? type.asParameterizedType().arguments().get(0) : type, + ctx.injectionPoint.getRequiredQualifiers().stream().filter(a -> !a.name().equals(DotNames.ALL)) + .collect(Collectors.toSet())); + boolean collectMetadata = beans.stream() + .anyMatch(b -> BuiltinScope.DEPENDENT.isDeclaredBy(b) && b.requiresInjectionPointMetadata()); + + ResultHandle annotationsHandle; + ResultHandle javaMemberHandle; ResultHandle beanHandle; - switch (ctx.targetInfo.kind()) { - case OBSERVER: - // For observers the first argument is always the declaring bean - beanHandle = ctx.constructor.invokeInterfaceMethod( - MethodDescriptors.SUPPLIER_GET, ctx.constructor.getMethodParam(0)); - break; - case BEAN: - beanHandle = ctx.constructor.getThis(); - break; - case INVOKER: - beanHandle = loadInvokerTargetBean(ctx.targetInfo.asInvoker(), ctx.constructor); - break; - default: - throw new IllegalStateException("Unsupported target info: " + ctx.targetInfo); + if (collectMetadata) { + annotationsHandle = BeanGenerator.collectInjectionPointAnnotations( + ctx.beanDeployment, + ctx.constructor, ctx.injectionPoint, ctx.annotationLiterals, ctx.injectionPointAnnotationsPredicate); + javaMemberHandle = BeanGenerator.getJavaMemberHandle(ctx.constructor, ctx.injectionPoint, + ctx.reflectionRegistration); + switch (ctx.targetInfo.kind()) { + case OBSERVER: + // For observers the first argument is always the declaring bean + beanHandle = ctx.constructor.invokeInterfaceMethod( + MethodDescriptors.SUPPLIER_GET, ctx.constructor.getMethodParam(0)); + break; + case BEAN: + beanHandle = ctx.constructor.getThis(); + break; + case INVOKER: + beanHandle = loadInvokerTargetBean(ctx.targetInfo.asInvoker(), ctx.constructor); + break; + default: + throw new IllegalStateException("Unsupported target info: " + ctx.targetInfo); + } + } else { + annotationsHandle = ctx.constructor.loadNull(); + javaMemberHandle = ctx.constructor.loadNull(); + beanHandle = ctx.constructor.loadNull(); } + ResultHandle listProvider = ctx.constructor.newInstance( MethodDescriptor.ofConstructor(ListProvider.class, java.lang.reflect.Type.class, java.lang.reflect.Type.class, Set.class,