8335110: Fix instruction name and API spec inconsistencies in CodeBuilder
Reviewed-by: asotona
This commit is contained in:
parent
8a664a4c35
commit
f7af4504a8
@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 2022, Oracle and/or its affiliates. All rights reserved.
|
* Copyright (c) 2022, 2024, Oracle and/or its affiliates. All rights reserved.
|
||||||
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
||||||
*
|
*
|
||||||
* This code is free software; you can redistribute it and/or modify it
|
* This code is free software; you can redistribute it and/or modify it
|
||||||
@ -97,6 +97,27 @@ import jdk.internal.javac.PreviewFeature;
|
|||||||
* #with(ClassFileElement)} or concretely by calling the various {@code withXxx}
|
* #with(ClassFileElement)} or concretely by calling the various {@code withXxx}
|
||||||
* methods.
|
* methods.
|
||||||
*
|
*
|
||||||
|
* <h2>Instruction Factories</h2>
|
||||||
|
* {@code CodeBuilder} provides convenience methods to create instructions (See
|
||||||
|
* JVMS {@jvms 6.5} Instructions) by their mnemonic, taking necessary operands.
|
||||||
|
* <ul>
|
||||||
|
* <li>Instructions that encode their operands in their opcode, such as {@code
|
||||||
|
* aload_<n>}, share their factories with their generic version like {@link
|
||||||
|
* #aload aload}. Note that some constant instructions, such as {@link #iconst_1
|
||||||
|
* iconst_1}, do not have generic versions, and thus have their own factories.
|
||||||
|
* <li>Instructions that accept wide operands, such as {@code ldc2_w} or {@code
|
||||||
|
* wide}, share their factories with their regular version like {@link #ldc}. Note
|
||||||
|
* that {@link #goto_w goto_w} has its own factory to avoid {@linkplain
|
||||||
|
* ClassFile.ShortJumpsOption short jumps}.
|
||||||
|
* <li>The {@code goto}, {@code instanceof}, {@code new}, and {@code return}
|
||||||
|
* instructions' factories are named {@link #goto_ goto_}, {@link #instanceOf
|
||||||
|
* instanceOf}, {@link #new_ new_}, and {@link #return_() return_} respectively,
|
||||||
|
* due to clashes with keywords in the Java programming language.
|
||||||
|
* <li>Factories are not provided for instructions {@code jsr}, {@code jsr_w},
|
||||||
|
* {@code ret}, and {@code wide ret}, which cannot appear in class files with
|
||||||
|
* major version {@value ClassFile#JAVA_7_VERSION} or higher. (JVMS {@jvms 4.9.1})
|
||||||
|
* </ul>
|
||||||
|
*
|
||||||
* @see CodeTransform
|
* @see CodeTransform
|
||||||
*
|
*
|
||||||
* @since 22
|
* @since 22
|
||||||
@ -130,7 +151,7 @@ public sealed interface CodeBuilder
|
|||||||
/**
|
/**
|
||||||
* {@return the local variable slot associated with the receiver}.
|
* {@return the local variable slot associated with the receiver}.
|
||||||
*
|
*
|
||||||
* @throws IllegalStateException if this is not a static method
|
* @throws IllegalStateException if this is a static method
|
||||||
*/
|
*/
|
||||||
int receiverSlot();
|
int receiverSlot();
|
||||||
|
|
||||||
@ -699,7 +720,7 @@ public sealed interface CodeBuilder
|
|||||||
* @return this builder
|
* @return this builder
|
||||||
*/
|
*/
|
||||||
default CodeBuilder exceptionCatch(Label start, Label end, Label handler, ClassEntry catchType) {
|
default CodeBuilder exceptionCatch(Label start, Label end, Label handler, ClassEntry catchType) {
|
||||||
return with(ExceptionCatch.of(handler, start, end, Optional.of(catchType)));
|
return with(ExceptionCatch.of(handler, start, end, Optional.ofNullable(catchType)));
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -837,6 +858,10 @@ public sealed interface CodeBuilder
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Generate an instruction to load a reference from a local variable
|
* Generate an instruction to load a reference from a local variable
|
||||||
|
*
|
||||||
|
* <p>This may also generate {@code aload_<N>} and
|
||||||
|
* {@code wide aload} instructions.
|
||||||
|
*
|
||||||
* @param slot the local variable slot
|
* @param slot the local variable slot
|
||||||
* @return this builder
|
* @return this builder
|
||||||
*/
|
*/
|
||||||
@ -881,6 +906,10 @@ public sealed interface CodeBuilder
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Generate an instruction to store a reference into a local variable
|
* Generate an instruction to store a reference into a local variable
|
||||||
|
*
|
||||||
|
* <p>This may also generate {@code astore_<N>} and
|
||||||
|
* {@code wide astore} instructions.
|
||||||
|
*
|
||||||
* @param slot the local variable slot
|
* @param slot the local variable slot
|
||||||
* @return this builder
|
* @return this builder
|
||||||
*/
|
*/
|
||||||
@ -1046,6 +1075,10 @@ public sealed interface CodeBuilder
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Generate an instruction to load a double from a local variable
|
* Generate an instruction to load a double from a local variable
|
||||||
|
*
|
||||||
|
* <p>This may also generate {@code dload_<N>} and
|
||||||
|
* {@code wide dload} instructions.
|
||||||
|
*
|
||||||
* @param slot the local variable slot
|
* @param slot the local variable slot
|
||||||
* @return this builder
|
* @return this builder
|
||||||
*/
|
*/
|
||||||
@ -1087,6 +1120,10 @@ public sealed interface CodeBuilder
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Generate an instruction to store a double into a local variable
|
* Generate an instruction to store a double into a local variable
|
||||||
|
*
|
||||||
|
* <p>This may also generate {@code dstore_<N>} and
|
||||||
|
* {@code wide dstore} instructions.
|
||||||
|
*
|
||||||
* @param slot the local variable slot
|
* @param slot the local variable slot
|
||||||
* @return this builder
|
* @return this builder
|
||||||
*/
|
*/
|
||||||
@ -1250,6 +1287,10 @@ public sealed interface CodeBuilder
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Generate an instruction to load a float from a local variable
|
* Generate an instruction to load a float from a local variable
|
||||||
|
*
|
||||||
|
* <p>This may also generate {@code fload_<N>} and
|
||||||
|
* {@code wide fload} instructions.
|
||||||
|
*
|
||||||
* @param slot the local variable slot
|
* @param slot the local variable slot
|
||||||
* @return this builder
|
* @return this builder
|
||||||
*/
|
*/
|
||||||
@ -1291,6 +1332,10 @@ public sealed interface CodeBuilder
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Generate an instruction to store a float into a local variable
|
* Generate an instruction to store a float into a local variable
|
||||||
|
*
|
||||||
|
* <p>This may also generate {@code fstore_<N>} and
|
||||||
|
* {@code wide fstore} instructions.
|
||||||
|
*
|
||||||
* @param slot the local variable slot
|
* @param slot the local variable slot
|
||||||
* @return this builder
|
* @return this builder
|
||||||
*/
|
*/
|
||||||
@ -1350,6 +1395,15 @@ public sealed interface CodeBuilder
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Generate an instruction to branch always
|
* Generate an instruction to branch always
|
||||||
|
*
|
||||||
|
* <p>This may also generate {@code goto_w} instructions if the {@link
|
||||||
|
* ClassFile.ShortJumpsOption#FIX_SHORT_JUMPS FIX_SHORT_JUMPS} option
|
||||||
|
* is set.
|
||||||
|
*
|
||||||
|
* @apiNote The instruction's name is {@code goto}, which coincides with a
|
||||||
|
* reserved keyword of the Java programming language, thus this method is
|
||||||
|
* named with an extra {@code _} suffix instead.
|
||||||
|
*
|
||||||
* @param target the branch target
|
* @param target the branch target
|
||||||
* @return this builder
|
* @return this builder
|
||||||
*/
|
*/
|
||||||
@ -1587,7 +1641,7 @@ public sealed interface CodeBuilder
|
|||||||
* @param target the branch target
|
* @param target the branch target
|
||||||
* @return this builder
|
* @return this builder
|
||||||
*/
|
*/
|
||||||
default CodeBuilder if_nonnull(Label target) {
|
default CodeBuilder ifnonnull(Label target) {
|
||||||
return branch(Opcode.IFNONNULL, target);
|
return branch(Opcode.IFNONNULL, target);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1596,7 +1650,7 @@ public sealed interface CodeBuilder
|
|||||||
* @param target the branch target
|
* @param target the branch target
|
||||||
* @return this builder
|
* @return this builder
|
||||||
*/
|
*/
|
||||||
default CodeBuilder if_null(Label target) {
|
default CodeBuilder ifnull(Label target) {
|
||||||
return branch(Opcode.IFNULL, target);
|
return branch(Opcode.IFNULL, target);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1666,6 +1720,10 @@ public sealed interface CodeBuilder
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Generate an instruction to load an int from a local variable
|
* Generate an instruction to load an int from a local variable
|
||||||
|
*
|
||||||
|
* <p>This may also generate {@code iload_<N>} and
|
||||||
|
* {@code wide iload} instructions.
|
||||||
|
*
|
||||||
* @param slot the local variable slot
|
* @param slot the local variable slot
|
||||||
* @return this builder
|
* @return this builder
|
||||||
*/
|
*/
|
||||||
@ -1691,6 +1749,11 @@ public sealed interface CodeBuilder
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Generate an instruction to determine if an object is of the given type
|
* Generate an instruction to determine if an object is of the given type
|
||||||
|
*
|
||||||
|
* @apiNote The instruction's name is {@code instanceof}, which coincides with a
|
||||||
|
* reserved keyword of the Java programming language, thus this method is
|
||||||
|
* named with camel case instead.
|
||||||
|
*
|
||||||
* @param target the target type
|
* @param target the target type
|
||||||
* @return this builder
|
* @return this builder
|
||||||
* @since 23
|
* @since 23
|
||||||
@ -1701,6 +1764,11 @@ public sealed interface CodeBuilder
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Generate an instruction to determine if an object is of the given type
|
* Generate an instruction to determine if an object is of the given type
|
||||||
|
*
|
||||||
|
* @apiNote The instruction's name is {@code instanceof}, which coincides with a
|
||||||
|
* reserved keyword of the Java programming language, thus this method is
|
||||||
|
* named with camel case instead.
|
||||||
|
*
|
||||||
* @param target the target type
|
* @param target the target type
|
||||||
* @return this builder
|
* @return this builder
|
||||||
* @throws IllegalArgumentException if {@code target} represents a primitive type
|
* @throws IllegalArgumentException if {@code target} represents a primitive type
|
||||||
@ -1910,6 +1978,10 @@ public sealed interface CodeBuilder
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Generate an instruction to store an int into a local variable
|
* Generate an instruction to store an int into a local variable
|
||||||
|
*
|
||||||
|
* <p>This may also generate {@code istore_<N>} and
|
||||||
|
* {@code wide istore} instructions.
|
||||||
|
*
|
||||||
* @param slot the local variable slot
|
* @param slot the local variable slot
|
||||||
* @return this builder
|
* @return this builder
|
||||||
*/
|
*/
|
||||||
@ -2033,6 +2105,12 @@ public sealed interface CodeBuilder
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Generate an instruction pushing an item from the run-time constant pool onto the operand stack
|
* Generate an instruction pushing an item from the run-time constant pool onto the operand stack
|
||||||
|
*
|
||||||
|
* <p>This may also generate {@code ldc_w} and {@code ldc2_w} instructions.
|
||||||
|
*
|
||||||
|
* @apiNote {@link #loadConstant(ConstantDesc) loadConstant} generates more optimal instructions
|
||||||
|
* and should be used for general constants if an {@code ldc} instruction is not strictly required.
|
||||||
|
*
|
||||||
* @param value the constant value
|
* @param value the constant value
|
||||||
* @return this builder
|
* @return this builder
|
||||||
*/
|
*/
|
||||||
@ -2042,6 +2120,9 @@ public sealed interface CodeBuilder
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Generate an instruction pushing an item from the run-time constant pool onto the operand stack
|
* Generate an instruction pushing an item from the run-time constant pool onto the operand stack
|
||||||
|
*
|
||||||
|
* <p>This may also generate {@code ldc_w} and {@code ldc2_w} instructions.
|
||||||
|
*
|
||||||
* @param entry the constant value
|
* @param entry the constant value
|
||||||
* @return this builder
|
* @return this builder
|
||||||
*/
|
*/
|
||||||
@ -2062,6 +2143,10 @@ public sealed interface CodeBuilder
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Generate an instruction to load a long from a local variable
|
* Generate an instruction to load a long from a local variable
|
||||||
|
*
|
||||||
|
* <p>This may also generate {@code lload_<N>} and
|
||||||
|
* {@code wide lload} instructions.
|
||||||
|
*
|
||||||
* @param slot the local variable slot
|
* @param slot the local variable slot
|
||||||
* @return this builder
|
* @return this builder
|
||||||
*/
|
*/
|
||||||
@ -2127,6 +2212,10 @@ public sealed interface CodeBuilder
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Generate an instruction to store a long into a local variable
|
* Generate an instruction to store a long into a local variable
|
||||||
|
*
|
||||||
|
* <p>This may also generate {@code lstore_<N>} and
|
||||||
|
* {@code wide lstore} instructions.
|
||||||
|
*
|
||||||
* @param slot the local variable slot
|
* @param slot the local variable slot
|
||||||
* @return this builder
|
* @return this builder
|
||||||
*/
|
*/
|
||||||
@ -2197,6 +2286,11 @@ public sealed interface CodeBuilder
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Generate an instruction to create a new object
|
* Generate an instruction to create a new object
|
||||||
|
*
|
||||||
|
* @apiNote The instruction's name is {@code new}, which coincides with a
|
||||||
|
* reserved keyword of the Java programming language, thus this method is
|
||||||
|
* named with an extra {@code _} suffix instead.
|
||||||
|
*
|
||||||
* @param clazz the new class type
|
* @param clazz the new class type
|
||||||
* @return this builder
|
* @return this builder
|
||||||
*/
|
*/
|
||||||
@ -2206,6 +2300,11 @@ public sealed interface CodeBuilder
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Generate an instruction to create a new object
|
* Generate an instruction to create a new object
|
||||||
|
*
|
||||||
|
* @apiNote The instruction's name is {@code new}, which coincides with a
|
||||||
|
* reserved keyword of the Java programming language, thus this method is
|
||||||
|
* named with an extra {@code _} suffix instead.
|
||||||
|
*
|
||||||
* @param clazz the new class type
|
* @param clazz the new class type
|
||||||
* @return this builder
|
* @return this builder
|
||||||
* @throws IllegalArgumentException if {@code clazz} represents a primitive type
|
* @throws IllegalArgumentException if {@code clazz} represents a primitive type
|
||||||
@ -2283,6 +2382,11 @@ public sealed interface CodeBuilder
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Generate an instruction to return void from the method
|
* Generate an instruction to return void from the method
|
||||||
|
*
|
||||||
|
* @apiNote The instruction's name is {@code return}, which coincides with a
|
||||||
|
* reserved keyword of the Java programming language, thus this method is
|
||||||
|
* named with an extra {@code _} suffix instead.
|
||||||
|
*
|
||||||
* @return this builder
|
* @return this builder
|
||||||
*/
|
*/
|
||||||
default CodeBuilder return_() {
|
default CodeBuilder return_() {
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 2022, Oracle and/or its affiliates. All rights reserved.
|
* Copyright (c) 2022, 2024, Oracle and/or its affiliates. All rights reserved.
|
||||||
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
||||||
*
|
*
|
||||||
* This code is free software; you can redistribute it and/or modify it
|
* This code is free software; you can redistribute it and/or modify it
|
||||||
@ -58,9 +58,9 @@ public sealed interface CodeAttribute extends Attribute<CodeAttribute>, CodeMode
|
|||||||
byte[] codeArray();
|
byte[] codeArray();
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* {@return the position of the {@code Label} in the {@code codeArray}
|
* {@return the position of the {@code label} in the {@link #codeArray codeArray}}
|
||||||
* or -1 if the {@code Label} does not point to the {@code codeArray}}
|
|
||||||
* @param label a marker for a position within this {@code CodeAttribute}
|
* @param label a marker for a position within this {@code CodeAttribute}
|
||||||
|
* @throws IllegalArgumentException if the {@code label} is not from this attribute
|
||||||
*/
|
*/
|
||||||
int labelToBci(Label label);
|
int labelToBci(Label label);
|
||||||
}
|
}
|
||||||
|
@ -422,7 +422,7 @@ public class SwitchBootstraps {
|
|||||||
cb.pop();
|
cb.pop();
|
||||||
cb.aload(SELECTOR_OBJ);
|
cb.aload(SELECTOR_OBJ);
|
||||||
Label nonNullLabel = cb.newLabel();
|
Label nonNullLabel = cb.newLabel();
|
||||||
cb.if_nonnull(nonNullLabel);
|
cb.ifnonnull(nonNullLabel);
|
||||||
cb.iconst_m1();
|
cb.iconst_m1();
|
||||||
cb.ireturn();
|
cb.ireturn();
|
||||||
cb.labelBinding(nonNullLabel);
|
cb.labelBinding(nonNullLabel);
|
||||||
|
@ -51,7 +51,7 @@ public final class LabelImpl
|
|||||||
private final LabelContext labelContext;
|
private final LabelContext labelContext;
|
||||||
private int bci;
|
private int bci;
|
||||||
|
|
||||||
public LabelImpl(LabelContext labelContext, int bci) {
|
public LabelImpl(LabelContext labelContext, int bci) {
|
||||||
this.labelContext = Objects.requireNonNull(labelContext);
|
this.labelContext = Objects.requireNonNull(labelContext);
|
||||||
this.bci = bci;
|
this.bci = bci;
|
||||||
}
|
}
|
||||||
|
@ -461,7 +461,7 @@ final class EventInstrumentation {
|
|||||||
catchAllHandler.dup();
|
catchAllHandler.dup();
|
||||||
// stack: [ex] [EW] [EW]
|
// stack: [ex] [EW] [EW]
|
||||||
Label rethrow = catchAllHandler.newLabel();
|
Label rethrow = catchAllHandler.newLabel();
|
||||||
catchAllHandler.if_null(rethrow);
|
catchAllHandler.ifnull(rethrow);
|
||||||
// stack: [ex] [EW]
|
// stack: [ex] [EW]
|
||||||
catchAllHandler.dup();
|
catchAllHandler.dup();
|
||||||
// stack: [ex] [EW] [EW]
|
// stack: [ex] [EW] [EW]
|
||||||
@ -486,7 +486,7 @@ final class EventInstrumentation {
|
|||||||
Label fail = codeBuilder.newLabel();
|
Label fail = codeBuilder.newLabel();
|
||||||
if (guardEventConfiguration) {
|
if (guardEventConfiguration) {
|
||||||
getEventConfiguration(codeBuilder);
|
getEventConfiguration(codeBuilder);
|
||||||
codeBuilder.if_null(fail);
|
codeBuilder.ifnull(fail);
|
||||||
}
|
}
|
||||||
// if (!eventConfiguration.shouldCommit(duration) goto fail;
|
// if (!eventConfiguration.shouldCommit(duration) goto fail;
|
||||||
getEventConfiguration(codeBuilder);
|
getEventConfiguration(codeBuilder);
|
||||||
@ -525,7 +525,7 @@ final class EventInstrumentation {
|
|||||||
if (guardEventConfiguration) {
|
if (guardEventConfiguration) {
|
||||||
// if (eventConfiguration == null) goto fail;
|
// if (eventConfiguration == null) goto fail;
|
||||||
getEventConfiguration(codeBuilder);
|
getEventConfiguration(codeBuilder);
|
||||||
codeBuilder.if_null(fail);
|
codeBuilder.ifnull(fail);
|
||||||
}
|
}
|
||||||
// return eventConfiguration.shouldCommit(duration);
|
// return eventConfiguration.shouldCommit(duration);
|
||||||
getEventConfiguration(codeBuilder);
|
getEventConfiguration(codeBuilder);
|
||||||
@ -738,7 +738,7 @@ final class EventInstrumentation {
|
|||||||
Label nullLabel = codeBuilder.newLabel();
|
Label nullLabel = codeBuilder.newLabel();
|
||||||
if (guardEventConfiguration) {
|
if (guardEventConfiguration) {
|
||||||
getEventConfiguration(codeBuilder);
|
getEventConfiguration(codeBuilder);
|
||||||
codeBuilder.if_null(nullLabel);
|
codeBuilder.ifnull(nullLabel);
|
||||||
}
|
}
|
||||||
getEventConfiguration(codeBuilder);
|
getEventConfiguration(codeBuilder);
|
||||||
invokevirtual(codeBuilder, TYPE_EVENT_CONFIGURATION, METHOD_IS_ENABLED);
|
invokevirtual(codeBuilder, TYPE_EVENT_CONFIGURATION, METHOD_IS_ENABLED);
|
||||||
|
@ -257,8 +257,8 @@ class RebuildingTransformation {
|
|||||||
case IF_ICMPLE -> cob.if_icmple(target);
|
case IF_ICMPLE -> cob.if_icmple(target);
|
||||||
case IF_ICMPLT -> cob.if_icmplt(target);
|
case IF_ICMPLT -> cob.if_icmplt(target);
|
||||||
case IF_ICMPNE -> cob.if_icmpne(target);
|
case IF_ICMPNE -> cob.if_icmpne(target);
|
||||||
case IFNONNULL -> cob.if_nonnull(target);
|
case IFNONNULL -> cob.ifnonnull(target);
|
||||||
case IFNULL -> cob.if_null(target);
|
case IFNULL -> cob.ifnull(target);
|
||||||
case IFEQ -> cob.ifeq(target);
|
case IFEQ -> cob.ifeq(target);
|
||||||
case IFGE -> cob.ifge(target);
|
case IFGE -> cob.ifge(target);
|
||||||
case IFGT -> cob.ifgt(target);
|
case IFGT -> cob.ifgt(target);
|
||||||
|
Loading…
Reference in New Issue
Block a user