diff --git a/src/java.base/share/classes/java/lang/invoke/LambdaMetafactory.java b/src/java.base/share/classes/java/lang/invoke/LambdaMetafactory.java index b9bfd64432c..7976960139e 100644 --- a/src/java.base/share/classes/java/lang/invoke/LambdaMetafactory.java +++ b/src/java.base/share/classes/java/lang/invoke/LambdaMetafactory.java @@ -263,8 +263,12 @@ public final class LambdaMetafactory { * methods from {@code Object}. * * @param caller Represents a lookup context with the accessibility - * privileges of the caller. When used with {@code invokedynamic}, - * this is stacked automatically by the VM. + * privileges of the caller. Specifically, the lookup context + * must have + * private access + * privileges. + * When used with {@code invokedynamic}, this is stacked + * automatically by the VM. * @param invokedName The name of the method to implement. When used with * {@code invokedynamic}, this is provided by the * {@code NameAndType} of the {@code InvokeDynamic} @@ -294,7 +298,8 @@ public final class LambdaMetafactory { * instances of the interface named by {@code invokedType} * @throws LambdaConversionException If any of the linkage invariants * described {@link LambdaMetafactory above} - * are violated + * are violated, or the lookup context + * does not have private access privileges. */ public static CallSite metafactory(MethodHandles.Lookup caller, String invokedName, @@ -404,8 +409,12 @@ public final class LambdaMetafactory { * * * @param caller Represents a lookup context with the accessibility - * privileges of the caller. When used with {@code invokedynamic}, - * this is stacked automatically by the VM. + * privileges of the caller. Specifically, the lookup context + * must have + * private access + * privileges. + * When used with {@code invokedynamic}, this is stacked + * automatically by the VM. * @param invokedName The name of the method to implement. When used with * {@code invokedynamic}, this is provided by the * {@code NameAndType} of the {@code InvokeDynamic} @@ -429,7 +438,8 @@ public final class LambdaMetafactory { * instances of the interface named by {@code invokedType} * @throws LambdaConversionException If any of the linkage invariants * described {@link LambdaMetafactory above} - * are violated + * are violated, or the lookup context + * does not have private access privileges. */ public static CallSite altMetafactory(MethodHandles.Lookup caller, String invokedName, diff --git a/src/java.base/share/classes/java/lang/invoke/StringConcatFactory.java b/src/java.base/share/classes/java/lang/invoke/StringConcatFactory.java index 72feb9f1ca0..a5bf7f85d21 100644 --- a/src/java.base/share/classes/java/lang/invoke/StringConcatFactory.java +++ b/src/java.base/share/classes/java/lang/invoke/StringConcatFactory.java @@ -385,9 +385,16 @@ public final class StringConcatFactory { * invoked, it returns the result of String concatenation, taking all * function arguments passed to the linkage method as inputs for * concatenation. The target signature is given by {@code concatType}. - * The arguments are concatenated as per requirements stated in JLS 15.18.1 - * "String Concatenation Operator +". Notably, the inputs are converted as - * per JLS 5.1.11 "String Conversion", and combined from left to right. + * For a target accepting: + *
Assume the linkage arguments are as follows: * @@ -404,8 +411,12 @@ public final class StringConcatFactory { * * * @param lookup Represents a lookup context with the accessibility - * privileges of the caller. When used with {@code - * invokedynamic}, this is stacked automatically by the VM. + * privileges of the caller. Specifically, the lookup + * context must have + * private access + * privileges. + * When used with {@code invokedynamic}, this is stacked + * automatically by the VM. * @param name The name of the method to implement. This name is * arbitrary, and has no meaning for this linkage method. * When used with {@code invokedynamic}, this is provided by @@ -422,7 +433,8 @@ public final class StringConcatFactory { * concatenation, with dynamic concatenation arguments described by the given * {@code concatType}. * @throws StringConcatException If any of the linkage invariants described - * here are violated. + * here are violated, or the lookup context + * does not have private access privileges. * @throws NullPointerException If any of the incoming arguments is null. * This will never happen when a bootstrap method * is called with invokedynamic. @@ -452,10 +464,17 @@ public final class StringConcatFactory { * invoked, it returns the result of String concatenation, taking all * function arguments and constants passed to the linkage method as inputs for * concatenation. The target signature is given by {@code concatType}, and - * does not include constants. The arguments are concatenated as per requirements - * stated in JLS 15.18.1 "String Concatenation Operator +". Notably, the inputs - * are converted as per JLS 5.1.11 "String Conversion", and combined from left - * to right. + * does not include constants. + * For a target accepting: + *
The concatenation recipe is a String description for the way to * construct a concatenated String from the arguments and constants. The @@ -502,9 +521,12 @@ public final class StringConcatFactory { * * * @param lookup Represents a lookup context with the accessibility - * privileges of the caller. When used with {@code - * invokedynamic}, this is stacked automatically by the - * VM. + * privileges of the caller. Specifically, the lookup + * context must have + * private access + * privileges. + * When used with {@code invokedynamic}, this is stacked + * automatically by the VM. * @param name The name of the method to implement. This name is * arbitrary, and has no meaning for this linkage method. * When used with {@code invokedynamic}, this is provided @@ -524,7 +546,8 @@ public final class StringConcatFactory { * concatenation, with dynamic concatenation arguments described by the given * {@code concatType}. * @throws StringConcatException If any of the linkage invariants described - * here are violated. + * here are violated, or the lookup context + * does not have private access privileges. * @throws NullPointerException If any of the incoming arguments is null, or * any constant in {@code recipe} is null. * This will never happen when a bootstrap method diff --git a/test/jdk/java/lang/String/concat/StringConcatFactoryInvariants.java b/test/jdk/java/lang/String/concat/StringConcatFactoryInvariants.java index 3dff1c2f5ec..bc43d9144fb 100644 --- a/test/jdk/java/lang/String/concat/StringConcatFactoryInvariants.java +++ b/test/jdk/java/lang/String/concat/StringConcatFactoryInvariants.java @@ -282,6 +282,30 @@ public class StringConcatFactoryInvariants { // Advanced factory: public Lookup is rejected fail("Passing public Lookup", () -> StringConcatFactory.makeConcatWithConstants(MethodHandles.publicLookup(), methodName, mtEmpty, recipeEmpty)); + + // Zero inputs + { + MethodType zero = MethodType.methodType(String.class); + CallSite cs = StringConcatFactory.makeConcat(lookup, methodName, zero); + test("", (String) cs.getTarget().invokeExact()); + + cs = StringConcatFactory.makeConcatWithConstants(lookup, methodName, zero, ""); + test("", (String) cs.getTarget().invokeExact()); + } + + // One input + { + MethodType zero = MethodType.methodType(String.class); + MethodType one = MethodType.methodType(String.class, String.class); + CallSite cs = StringConcatFactory.makeConcat(lookup, methodName, one); + test("A", (String) cs.getTarget().invokeExact("A")); + + cs = StringConcatFactory.makeConcatWithConstants(lookup, methodName, one, "\1"); + test("A", (String) cs.getTarget().invokeExact("A")); + + cs = StringConcatFactory.makeConcatWithConstants(lookup, methodName, zero, "\2", "A"); + test("A", (String) cs.getTarget().invokeExact()); + } } public static void ok(String msg, Callable runnable) {