Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

8240567: MethodTooLargeException thrown while creating a jlink image #14408

Closed
wants to merge 32 commits into from
Closed
Changes from 3 commits
Commits
Show all changes
32 commits
Select commit Hold shift + click to select a range
7c48693
8240567: MethodTooLargeException thrown while creating a jlink image
koppor Jun 11, 2023
23bbc0c
Remove comments
koppor Jun 12, 2023
9d4142c
Remove "final" at "enabled" - not part of the point of this PR
koppor Jun 12, 2023
e158a2c
Merge remote-tracking branch 'upstream/master' into fix-8240567
koppor Jun 14, 2023
e973295
Merge remote-tracking branch 'upstream/master' into fix-8240567
koppor Jun 16, 2023
a084ca5
Merge branch 'openjdk:master' into fix-8240567
koppor Jun 23, 2023
b61c1dc
Fix threshold
koppor Jun 29, 2023
a380441
Add parameterization for the number of moduleDescriptors per helper m…
koppor Jul 1, 2023
4898ba8
Fix nit (indent)
koppor Jul 1, 2023
edd85c9
Add description
koppor Jul 1, 2023
a6c9c98
Merge remote-tracking branch 'upstream/master' into fix-8240567
koppor Jul 1, 2023
4704349
Apply suggestions from code review
koppor Jul 1, 2023
b2051c3
No final for class parameters
koppor Jul 1, 2023
912f34c
Reuse helper array list
koppor Jul 1, 2023
fe5be2f
Try to execute java(.exe) of build runtime
koppor Jul 1, 2023
5b3d712
Fix ArrayList initialization and off-by-one errors
koppor Jul 1, 2023
cdc9aee
Fix test
koppor Jul 1, 2023
b31a017
Merge remote-tracking branch 'upstream/master' into fix-8240567
koppor Jul 3, 2023
6abca9f
Reposition DEDUP_LIST_VAR and BUILDER_VAR
koppor Jul 3, 2023
9835a64
Add --system-modules [batch-size=<N>]
koppor Jul 3, 2023
ae1bbe7
Refine comment
koppor Jul 3, 2023
b150a28
Update src/jdk.jlink/share/classes/jdk/tools/jlink/internal/plugins/S…
koppor Jul 4, 2023
142b555
Update src/jdk.jlink/share/classes/jdk/tools/jlink/internal/plugins/S…
koppor Jul 4, 2023
f50ec77
Update src/jdk.jlink/share/classes/jdk/tools/jlink/internal/plugins/S…
koppor Jul 4, 2023
b2b9036
Replace non-final wrapper by local variable
koppor Jul 4, 2023
bbbbb0e
Merge remote-tracking branch 'upstream/master' into fix-8240567
koppor Jul 4, 2023
15d7448
Refine test to fill DedupSetBuilder
koppor Jul 4, 2023
8d02911
Update test/jdk/tools/jlink/JLink100Modules.java
koppor Jul 5, 2023
13797d2
Reformat
koppor Jul 5, 2023
e112f03
Merge remote-tracking branch 'upstream/master' into fix-8240567
koppor Jul 5, 2023
8494517
Revert (non-working) DedupSetBuilder filling
koppor Jul 5, 2023
584edba
Merge remote-tracking branch 'upstream/master' into fix-8240567
koppor Jul 6, 2023
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Original file line number Diff line number Diff line change
@@ -118,10 +118,15 @@ public final class SystemModulesPlugin extends AbstractPlugin {
ClassDesc.ofInternalName("jdk/internal/module/SystemModules");
private static final ClassDesc CD_SYSTEM_MODULES_MAP =
ClassDesc.ofInternalName(SYSTEM_MODULES_MAP_CLASSNAME);
private final int moduleDescriptorsPerMethod;
private boolean enabled;

public SystemModulesPlugin() {
this(75);
}
public SystemModulesPlugin(int moduleDescriptorsPerMethod) {
super("system-modules");
this.moduleDescriptorsPerMethod = moduleDescriptorsPerMethod;
this.enabled = true;
}

@@ -316,7 +321,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);
@@ -531,16 +536,20 @@ static class SystemModulesClassGenerator {
// 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()));
}

@@ -666,7 +675,7 @@ private void genIncubatorModules(ClassBuilder clb) {
* Generate bytecode for moduleDescriptors method
*/
private void genModuleDescriptorsMethod(ClassBuilder clb) {
if (moduleInfos.size() <= 75) {
if (moduleInfos.size() <= moduleDescriptorsPerMethod) {
clb.withMethodBody(
"moduleDescriptors",
MethodTypeDesc.of(CD_MODULE_DESCRIPTOR.arrayType()),
@@ -678,20 +687,34 @@ private void genModuleDescriptorsMethod(ClassBuilder clb) {
for (int index = 0; index < moduleInfos.size(); index++) {
ModuleInfo minfo = moduleInfos.get(index);
new ModuleDescriptorBuilder(cob,
minfo.descriptor(),
minfo.packages(),
index).build();
minfo.descriptor(),
minfo.packages(),
index).build();
}
cob.aload(MD_VAR)
.areturn();
});
return;
}

// Instead of adding the module descriptor calls in this method,
// helper methods are created in separate helper methods.
// Let m1, m2, ..., mn be the calls to the module descriptor builder
// Here, n is larger than moduleDescriptorsPerMethod, which will hit the 64kb limit of method length.
// thus, we create f1, f2, ... where
// - f1 calls m1, ... m_{moduleDescriptorsPerMethod-1},
// - f2 calls m_{moduleDescriptorsPerMethod}, ... m_{2xmoduleDescriptorsPerMethod-1},
// - etc.
//
// Inside m, the class SystemModulesClassGenerator.DedupSetBuilder is used.
// That class creates sets caching variables. It stores its caches in the local storage.
// Since the local storage is destroyed across each method, but the sets are needed across methods,
// the local variables are passed on to the next helper method using a list of these variables.

List<List<ModuleInfo>> splitModuleInfos = new ArrayList<>();
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Please add the comments to describe what this does. Pseudo code may also help. It'd be helpful to explain the reason why the Sub method stores and restores what local variables.

I also wonder if you consider moduleDescriptors does this:

    ArrayList<Object> localVars = new ArrayList<>();
    moduleDescriptorsSub0(moduleInfos, localVars);
    .... // add new local variables to localVars
    moduleDescriptorsSub1(moduleInfos, localVars);
    .... // add new local variables to localVars
    :
    :

The same localVars array list is used and just add the new local variables defined from each batch (no need to create a new ArrayList and stores local variables every time)

Copy link
Contributor Author

@koppor koppor Jul 1, 2023

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I assume the "old" comments were too detailed. I removed them at JabRef/jdk@23bbc0c (#2) to have the code reviewable. I can readd some of them if that helps.

I also added a compressed description of the idea at edd85c9 (#14408). Update here, in the PR one simply needs to scroll up and sees the new comment.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Have you considered passing the same ArrayList for saving and restoring local variables? Currently each method creates one new array list and save the local variables from firstVariableForDedup to nextLocalVar, but the local variables from firstVariableForDedup are already added to the array list created by the previous method.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Ah, I see your commit. I will check it out.

List<ModuleInfo> currentModuleInfos = null;
for (int index = 0; index < moduleInfos.size(); index++) {
if (index % 75 == 0) {
if (index % moduleDescriptorsPerMethod == 0) {
currentModuleInfos = new ArrayList<>();
splitModuleInfos.add(currentModuleInfos);
}
@@ -748,9 +771,9 @@ private void genModuleDescriptorsMethod(ClassBuilder clb) {
for (int j = 0; j < moduleInfosPackage.size(); j++) {
ModuleInfo minfo = moduleInfosPackage.get(j);
new ModuleDescriptorBuilder(cob,
minfo.descriptor(),
minfo.packages(),
globalCount[0]).build();
minfo.descriptor(),
minfo.packages(),
globalCount[0]).build();
globalCount[0]++;
}