Skip to content
Permalink

Comparing changes

Choose two branches to see what’s changed or to start a new pull request. If you need to, you can also or learn more about diff comparisons.

Open a pull request

Create a new pull request by comparing changes across two branches. If you need to, you can also . Learn more about diff comparisons here.
base repository: openjdk/jdk
Failed to load repositories. Confirm that selected base ref is valid, then try again.
Loading
base: d072c40f
Choose a base ref
...
head repository: openjdk/jdk
Failed to load repositories. Confirm that selected head ref is valid, then try again.
Loading
compare: 584edbae
Choose a head ref

Commits on Jun 11, 2023

  1. 8240567: MethodTooLargeException thrown while creating a jlink image

    Co-authored-by: Christoph Schwentker <siedlerkiller@gmail.com>
    koppor and Siedlerchr committed Jun 11, 2023
    Copy the full SHA
    7c48693 View commit details

Commits on Jun 12, 2023

  1. Remove comments

    koppor committed Jun 12, 2023
    Copy the full SHA
    23bbc0c View commit details
  2. Copy the full SHA
    9d4142c View commit details

Commits on Jun 14, 2023

  1. Copy the full SHA
    e158a2c View commit details

Commits on Jun 16, 2023

  1. Copy the full SHA
    e973295 View commit details

Commits on Jun 23, 2023

  1. Copy the full SHA
    a084ca5 View commit details

Commits on Jun 29, 2023

  1. Fix threshold

    koppor authored Jun 29, 2023
    Copy the full SHA
    b61c1dc View commit details

Commits on Jul 1, 2023

  1. Add parameterization for the number of moduleDescriptors per helper m…

    …ethod (was magic number 50 / 75 before)
    koppor committed Jul 1, 2023
    Copy the full SHA
    a380441 View commit details
  2. Fix nit (indent)

    koppor committed Jul 1, 2023
    Copy the full SHA
    4898ba8 View commit details
  3. Add description

    koppor committed Jul 1, 2023
    Copy the full SHA
    edd85c9 View commit details
  4. Merge remote-tracking branch 'upstream/master' into fix-8240567

    # Conflicts:
    #	src/jdk.jlink/share/classes/jdk/tools/jlink/internal/plugins/SystemModulesPlugin.java
    koppor committed Jul 1, 2023
    Copy the full SHA
    a6c9c98 View commit details
  5. Apply suggestions from code review

    Co-authored-by: Mandy Chung <mandy.chung@oracle.com>
    koppor and mlchung authored Jul 1, 2023
    Copy the full SHA
    4704349 View commit details
  6. No final for class parameters

    koppor committed Jul 1, 2023
    Copy the full SHA
    b2051c3 View commit details
  7. Reuse helper array list

    koppor committed Jul 1, 2023
    Copy the full SHA
    912f34c View commit details
  8. Copy the full SHA
    fe5be2f View commit details
  9. Copy the full SHA
    5b3d712 View commit details
  10. Fix test

    koppor committed Jul 1, 2023
    Copy the full SHA
    cdc9aee View commit details

Commits on Jul 3, 2023

  1. Copy the full SHA
    b31a017 View commit details
  2. Copy the full SHA
    6abca9f View commit details
  3. Copy the full SHA
    9835a64 View commit details
  4. Refine comment

    Co-authored-by: Mandy Chung <mandy.chung@oracle.com>
    koppor and mlchung committed Jul 3, 2023
    Copy the full SHA
    ae1bbe7 View commit details

Commits on Jul 4, 2023

  1. Update src/jdk.jlink/share/classes/jdk/tools/jlink/internal/plugins/S…

    …ystemModulesPlugin.java
    
    Co-authored-by: Mandy Chung <mandy.chung@oracle.com>
    koppor and mlchung authored Jul 4, 2023
    Copy the full SHA
    b150a28 View commit details
  2. Update src/jdk.jlink/share/classes/jdk/tools/jlink/internal/plugins/S…

    …ystemModulesPlugin.java
    
    Co-authored-by: Mandy Chung <mandy.chung@oracle.com>
    koppor and mlchung authored Jul 4, 2023
    Copy the full SHA
    142b555 View commit details
  3. Update src/jdk.jlink/share/classes/jdk/tools/jlink/internal/plugins/S…

    …ystemModulesPlugin.java
    
    Co-authored-by: Mandy Chung <mandy.chung@oracle.com>
    koppor and mlchung authored Jul 4, 2023
    Copy the full SHA
    f50ec77 View commit details
  4. Copy the full SHA
    b2b9036 View commit details
  5. Copy the full SHA
    bbbbb0e View commit details
  6. Copy the full SHA
    15d7448 View commit details

Commits on Jul 5, 2023

  1. Update test/jdk/tools/jlink/JLink100Modules.java

    Co-authored-by: Mandy Chung <mandy.chung@oracle.com>
    koppor and mlchung authored Jul 5, 2023
    Copy the full SHA
    8d02911 View commit details
  2. Reformat

    koppor committed Jul 5, 2023
    Copy the full SHA
    13797d2 View commit details
  3. Copy the full SHA
    e112f03 View commit details
  4. Copy the full SHA
    8494517 View commit details

Commits on Jul 6, 2023

  1. Copy the full SHA
    584edba View commit details
Original file line number Diff line number Diff line change
@@ -120,6 +120,8 @@ public final class SystemModulesPlugin extends AbstractPlugin {
ClassDesc.ofInternalName(SYSTEM_MODULES_MAP_CLASSNAME);
private static final MethodTypeDesc MTD_StringArray = MethodTypeDesc.of(CD_String.arrayType());
private static final MethodTypeDesc MTD_SystemModules = MethodTypeDesc.of(CD_SYSTEM_MODULES);

private int moduleDescriptorsPerMethod = 75;
private boolean enabled;

public SystemModulesPlugin() {
@@ -142,7 +144,14 @@ public boolean hasArguments() {
public void configure(Map<String, String> config) {
String arg = config.get(getName());
if (arg != null) {
throw new IllegalArgumentException(getName() + ": " + arg);
String[] split = arg.split("=");
if (split.length != 2) {
throw new IllegalArgumentException(getName() + ": " + arg);
}
if (split[0].equals("batch-size")) {
throw new IllegalArgumentException(getName() + ": " + arg);
}
this.moduleDescriptorsPerMethod = Integer.parseInt(split[1]);
}
}

@@ -318,7 +327,7 @@ private String genSystemModulesClass(List<ModuleInfo> moduleInfos,
String className,
ResourcePoolBuilder out) {
SystemModulesClassGenerator generator
= new SystemModulesClassGenerator(className, moduleInfos);
= new SystemModulesClassGenerator(className, moduleInfos, moduleDescriptorsPerMethod);
byte[] bytes = generator.genClassBytes(cf);
String rn = "/java.base/" + className + ".class";
ResourcePoolEntry e = ResourcePoolEntry.create(rn, bytes);
@@ -533,28 +542,33 @@ static class SystemModulesClassGenerator {

private static final int MAX_LOCAL_VARS = 256;

private final int BUILDER_VAR = 0;
private final int MD_VAR = 1; // variable for ModuleDescriptor
private final int MT_VAR = 1; // variable for ModuleTarget
private final int MH_VAR = 1; // variable for ModuleHashes
private int nextLocalVar = 2; // index to next local variable
private final int DEDUP_LIST_VAR = 2;
private final int BUILDER_VAR = 3;
private int nextLocalVar = 4; // index to next local variable

// name of class to generate
private final ClassDesc classDesc;

// list of all ModuleInfos
private final List<ModuleInfo> moduleInfos;

private final int moduleDescriptorsPerMethod;

// A builder to create one single Set instance for a given set of
// names or modifiers to reduce the footprint
// e.g. target modules of qualified exports
private final DedupSetBuilder dedupSetBuilder
= new DedupSetBuilder(this::getNextLocalVar);

public SystemModulesClassGenerator(String className,
List<ModuleInfo> moduleInfos) {
List<ModuleInfo> moduleInfos,
int moduleDescriptorsPerMethod) {
this.classDesc = ClassDesc.ofInternalName(className);
this.moduleInfos = moduleInfos;
this.moduleDescriptorsPerMethod = moduleDescriptorsPerMethod;
moduleInfos.forEach(mi -> dedups(mi.descriptor()));
}

@@ -680,25 +694,146 @@ private void genIncubatorModules(ClassBuilder clb) {
* Generate bytecode for moduleDescriptors method
*/
private void genModuleDescriptorsMethod(ClassBuilder clb) {
if (moduleInfos.size() <= moduleDescriptorsPerMethod) {
clb.withMethodBody(
"moduleDescriptors",
MTD_ModuleDescriptorArray,
ACC_PUBLIC,
cob -> {
cob.constantInstruction(moduleInfos.size())
.anewarray(CD_MODULE_DESCRIPTOR)
.astore(MD_VAR);

for (int index = 0; index < moduleInfos.size(); index++) {
ModuleInfo minfo = moduleInfos.get(index);
new ModuleDescriptorBuilder(cob,
minfo.descriptor(),
minfo.packages(),
index).build();
}
cob.aload(MD_VAR)
.areturn();
});
return;
}


// Split the module descriptors be created by multiple helper methods.
// Each helper method "subi" creates the maximum N number of module descriptors
// mi, m{i+1} ...
// to avoid exceeding the 64kb limit of method length. Then it will call
// "sub{i+1}" to creates the next batch of module descriptors m{i+n}, m{i+n+1}...
// and so on. During the construction of the module descriptors, the string sets and
// modifier sets are deduplicated (see SystemModulesClassGenerator.DedupSetBuilder)
// and cached in the locals. These locals are saved in an array list so
// that the helper method can restore the local variables that may be
// referenced by the bytecode generated for creating module descriptors.
// Pseudo code looks like this:
//
// void subi(ModuleDescriptor[] mdescs, ArrayList<Object> localvars) {
// // assign localvars to local variables
// var l3 = localvars.get(0);
// var l4 = localvars.get(1);
// :
// // fill mdescs[i] to mdescs[i+n-1]
// mdescs[i] = ...
// mdescs[i+1] = ...
// :
// // save new local variables added
// localvars.add(lx)
// localvars.add(l{x+1})
// :
// sub{i+i}(mdescs, localvars);
// }

List<List<ModuleInfo>> splitModuleInfos = new ArrayList<>();
List<ModuleInfo> currentModuleInfos = null;
for (int index = 0; index < moduleInfos.size(); index++) {
if (index % moduleDescriptorsPerMethod == 0) {
currentModuleInfos = new ArrayList<>();
splitModuleInfos.add(currentModuleInfos);
}
currentModuleInfos.add(moduleInfos.get(index));
}

String helperMethodNamePrefix = "sub";
ClassDesc arrayListClassDesc = ClassDesc.ofInternalName("java/util/ArrayList");

clb.withMethodBody(
"moduleDescriptors",
MTD_ModuleDescriptorArray,
ACC_PUBLIC,
cob -> {
cob.constantInstruction(moduleInfos.size())
.anewarray(CD_MODULE_DESCRIPTOR)
.dup()
.astore(MD_VAR);

for (int index = 0; index < moduleInfos.size(); index++) {
ModuleInfo minfo = moduleInfos.get(index);
new ModuleDescriptorBuilder(cob,
minfo.descriptor(),
minfo.packages(),
index).build();
}
cob.aload(MD_VAR)
cob.new_(arrayListClassDesc)
.dup()
.constantInstruction(moduleInfos.size())
.invokespecial(arrayListClassDesc, INIT_NAME, MethodTypeDesc.of(CD_void, CD_int))
.astore(DEDUP_LIST_VAR);
cob.aload(0)
.aload(MD_VAR)
.aload(DEDUP_LIST_VAR)
.invokevirtual(
this.classDesc,
helperMethodNamePrefix + "0",
MethodTypeDesc.of(CD_void, CD_MODULE_DESCRIPTOR.arrayType(), arrayListClassDesc)
)
.areturn();
});

int dedupVarStart = nextLocalVar;
for (int n = 0, count = 0; n < splitModuleInfos.size(); count += splitModuleInfos.get(n).size(), n++) {
int index = n; // the index of which ModuleInfo being processed in the current batch
int start = count; // the start index to the return ModuleDescriptor array for the current batch
int curDedupVar = nextLocalVar;
clb.withMethodBody(
helperMethodNamePrefix + index,
MethodTypeDesc.of(CD_void, CD_MODULE_DESCRIPTOR.arrayType(), arrayListClassDesc),
ACC_PUBLIC,
cob -> {
if (curDedupVar > dedupVarStart) {
for (int i = dedupVarStart; i < curDedupVar; i++) {
cob.aload(DEDUP_LIST_VAR)
.constantInstruction(i - dedupVarStart)
.invokevirtual(arrayListClassDesc, "get", MethodTypeDesc.of(CD_Object, CD_int))
.astore(i);
}
}

List<ModuleInfo> currentBatch = splitModuleInfos.get(index);
for (int j = 0; j < currentBatch.size(); j++) {
ModuleInfo minfo = currentBatch.get(j);
new ModuleDescriptorBuilder(cob,
minfo.descriptor(),
minfo.packages(),
start + j).build();
}

if (index < splitModuleInfos.size() - 1) {
if (nextLocalVar > curDedupVar) {
for (int i = curDedupVar; i < nextLocalVar; i++) {
cob.aload(DEDUP_LIST_VAR)
.aload(i)
.invokevirtual(arrayListClassDesc, "add", MethodTypeDesc.of(CD_boolean, CD_Object))
.pop();
}
}
cob.aload(0)
.aload(MD_VAR)
.aload(DEDUP_LIST_VAR)
.invokevirtual(
this.classDesc,
helperMethodNamePrefix + (index+1),
MethodTypeDesc.of(CD_void, CD_MODULE_DESCRIPTOR.arrayType(), arrayListClassDesc)
);
}

cob.return_();
});
}
}

/**
Original file line number Diff line number Diff line change
@@ -147,13 +147,16 @@ generate-jli-classes.usage=\
\ correctness add ignore-version=true\n\
\ to override this.

system-modules.argument=retainModuleTarget

system-modules.description=Fast loading of module descriptors (always enabled)
system-modules.argument=batch-size=<N> sets the batch size of module descriptors\n\
\ to avoid exceeding the method length limit. The default\n\
\ batch size is 75.

system-modules.usage=\
\ --system-modules retainModuleTarget\n\
\ Fast loading of module descriptors (always enabled)
\ --system-modules [batch-size=<N>]\n\
\ The batch size specifies the maximum number of modules\n\
\ be handled in one method to workaround if the generated\n\
\ bytecode exceeds the method size limit. The default\n\
\ batch size is 75.

onoff.argument=<on|off>

128 changes: 128 additions & 0 deletions test/jdk/tools/jlink/JLink100Modules.java
Original file line number Diff line number Diff line change
@@ -0,0 +1,128 @@
/*
* Copyright (c) 2022, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
*
* This code is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License version 2 only, as
* published by the Free Software Foundation.
*
* This code is distributed in the hope that it will be useful, but WITHOUT
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
* version 2 for more details (a copy is included in the LICENSE file that
* accompanied this code).
*
* You should have received a copy of the GNU General Public License version
* 2 along with this work; if not, write to the Free Software Foundation,
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
*
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
* or visit www.oracle.com if you need additional information or have any
* questions.
*/

import java.nio.file.Files;
import java.nio.file.Path;
import java.nio.file.Paths;
import java.util.Arrays;
import java.util.StringJoiner;
import java.util.spi.ToolProvider;

import tests.JImageGenerator;

/*
* @test
* @summary Make sure that 100 modules can be linked using jlink.
* @bug 8240567
* @library ../lib
* @modules java.base/jdk.internal.jimage
* jdk.jdeps/com.sun.tools.classfile
* jdk.jlink/jdk.tools.jlink.internal
* jdk.jlink/jdk.tools.jlink.plugin
* jdk.jlink/jdk.tools.jmod
* jdk.jlink/jdk.tools.jimage
* jdk.compiler
* @build tests.*
* @run main/othervm -Xmx1g -Xlog:init=debug -XX:+UnlockDiagnosticVMOptions -XX:+BytecodeVerificationLocal JLink100Modules
*/
public class JLink100Modules {
private static final ToolProvider JAVAC_TOOL = ToolProvider.findFirst("javac")
.orElseThrow(() -> new RuntimeException("javac tool not found"));

static void report(String command, String[] args) {
System.out.println(command + " " + String.join(" ", Arrays.asList(args)));
}

static void javac(String[] args) {
report("javac", args);
JAVAC_TOOL.run(System.out, System.err, args);
}

public static void main(String[] args) throws Exception {
Path src = Paths.get("bug8240567");

StringJoiner mainModuleInfoContent = new StringJoiner(";\n requires ", "module bug8240567x {\n requires ", ";\n}");

for (int i = 0; i < 1_000; i++) {
String name = "module" + i + "x";
Path moduleDir = Files.createDirectories(src.resolve(name));

StringBuilder moduleInfoContent = new StringBuilder("module ");
moduleInfoContent.append(name).append(" {\n");
if (i != 0) {
moduleInfoContent.append(" requires module0x;\n");
}
moduleInfoContent.append("}\n");
Files.writeString(moduleDir.resolve("module-info.java"), moduleInfoContent.toString());

mainModuleInfoContent.add(name);
}

// create module reading the generated modules
Path mainModulePath = src.resolve("bug8240567x");
Files.createDirectories(mainModulePath);
Path mainModuleInfo = mainModulePath.resolve("module-info.java");
Files.writeString(mainModuleInfo, mainModuleInfoContent.toString());

Path mainClassDir = mainModulePath.resolve("testpackage");
Files.createDirectories(mainClassDir);

Files.writeString(mainClassDir.resolve("JLink100ModulesTest.java"), """
package testpackage;
public class JLink100ModulesTest {
public static void main(String[] args) throws Exception {
System.out.println("JLink100ModulesTest started.");
}
}
""");

String out = src.resolve("out").toString();
javac(new String[]{
"-d", out,
"--module-source-path", src.toString(),
"--module", "bug8240567x"
});

JImageGenerator.getJLinkTask()
.modulePath(out)
.output(src.resolve("out-jlink"))
.addMods("bug8240567x")
.call()
.assertSuccess();

Path binDir = src.resolve("out-jlink").resolve("bin").toAbsolutePath();
Path bin = binDir.resolve("java");

ProcessBuilder processBuilder = new ProcessBuilder(bin.toString(),
"-XX:+UnlockDiagnosticVMOptions",
"-XX:+BytecodeVerificationLocal",
"-m", "bug8240567x/testpackage.JLink100ModulesTest");
processBuilder.inheritIO();
processBuilder.directory(binDir.toFile());
Process process = processBuilder.start();
int exitCode = process.waitFor();
if (exitCode != 0)
throw new AssertionError("JLink100ModulesTest failed to launch");
}
}