Skip to content

Commit b9ae779

Browse files
committedJan 21, 2022
8279675: CDS cannot handle non-existent JAR file in bootclassapth
Reviewed-by: iklam, minqi
1 parent c1e4f3d commit b9ae779

File tree

5 files changed

+122
-15
lines changed

5 files changed

+122
-15
lines changed
 

‎src/hotspot/share/cds/filemap.cpp

+21-1
Original file line numberDiff line numberDiff line change
@@ -767,6 +767,21 @@ int FileMapInfo::num_paths(const char* path) {
767767
return npaths;
768768
}
769769

770+
// Returns true if a path within the paths exists and has non-zero size.
771+
bool FileMapInfo::check_paths_existence(const char* paths) {
772+
ClasspathStream cp_stream(paths);
773+
bool exist = false;
774+
struct stat st;
775+
while (cp_stream.has_next()) {
776+
const char* path = cp_stream.get_next();
777+
if (os::stat(path, &st) == 0 && st.st_size > 0) {
778+
exist = true;
779+
break;
780+
}
781+
}
782+
return exist;
783+
}
784+
770785
GrowableArray<const char*>* FileMapInfo::create_path_array(const char* paths) {
771786
GrowableArray<const char*>* path_array = new GrowableArray<const char*>(10);
772787
JavaThread* current = JavaThread::current();
@@ -850,7 +865,12 @@ bool FileMapInfo::validate_boot_class_paths() {
850865
if (relaxed_check) {
851866
return true; // ok, relaxed check, runtime has extra boot append path entries
852867
} else {
853-
mismatch = true;
868+
ResourceMark rm;
869+
if (check_paths_existence(rp)) {
870+
// If a path exists in the runtime boot paths, it is considered a mismatch
871+
// since there's no boot path specified during dump time.
872+
mismatch = true;
873+
}
854874
}
855875
} else if (dp_len > 0 && rp != NULL) {
856876
int num;

‎src/hotspot/share/cds/filemap.hpp

+1
Original file line numberDiff line numberDiff line change
@@ -554,6 +554,7 @@ class FileMapInfo : public CHeapObj<mtInternal> {
554554
void seek_to_position(size_t pos);
555555
char* skip_first_path_entry(const char* path) NOT_CDS_RETURN_(NULL);
556556
int num_paths(const char* path) NOT_CDS_RETURN_(0);
557+
bool check_paths_existence(const char* paths) NOT_CDS_RETURN_(false);
557558
GrowableArray<const char*>* create_path_array(const char* path) NOT_CDS_RETURN_(NULL);
558559
bool classpath_failure(const char* msg, const char* name) NOT_CDS_RETURN_(false);
559560
bool check_paths(int shared_path_start_idx, int num_paths,

‎test/hotspot/jtreg/runtime/cds/appcds/NonExistClasspath.java

+83-7
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2019, 2020, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2019, 2022, Oracle and/or its affiliates. All rights reserved.
33
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
44
*
55
* This code is free software; you can redistribute it and/or modify it
@@ -40,20 +40,21 @@
4040
import jdk.test.lib.process.OutputAnalyzer;
4141

4242
public class NonExistClasspath {
43+
static final String outDir = CDSTestUtils.getOutputDir();
44+
static final String newFile = "non-exist.jar";
45+
static final String nonExistPath = outDir + File.separator + newFile;
46+
static final String emptyJarPath = outDir + File.separator + "empty.jar";
47+
static final String errorMessage1 = "Unable to use shared archive";
48+
static final String errorMessage2 = "shared class paths mismatch";
49+
4350
public static void main(String[] args) throws Exception {
4451
String appJar = JarBuilder.getOrCreateHelloJar();
4552
doTest(appJar, false);
4653
doTest(appJar, true);
4754
}
4855

4956
static void doTest(String appJar, boolean bootcp) throws Exception {
50-
String outDir = CDSTestUtils.getOutputDir();
51-
String newFile = "non-exist.jar";
52-
String nonExistPath = outDir + File.separator + newFile;
53-
final String errorMessage1 = "Unable to use shared archive";
54-
final String errorMessage2 = "shared class paths mismatch";
5557
final String errorMessage3 = (bootcp ? "BOOT" : "APP") + " classpath mismatch";
56-
5758
(new File(nonExistPath)).delete();
5859

5960
String classPath = nonExistPath + File.pathSeparator + appJar;
@@ -100,6 +101,81 @@ static void doTest(String appJar, boolean bootcp) throws Exception {
100101
"-Xlog:class+path=trace",
101102
"Hello"))
102103
.assertAbnormalExit(errorMessage1, errorMessage2, errorMessage3);
104+
105+
if (bootcp) {
106+
doMoreBCPTests(appJar, errorMessage3);
107+
}
108+
}
109+
110+
static void doMoreBCPTests(String appJar, String errorMessage3) throws Exception {
111+
112+
// Dump an archive with non-existent boot class path.
113+
(new File(nonExistPath)).delete();
114+
TestCommon.testDump("foobar", TestCommon.list("Hello"), make_args(true, nonExistPath, "-cp", appJar));
115+
116+
// Run with non-existent boot class path, test should pass.
117+
TestCommon.run(make_args(true,
118+
nonExistPath,
119+
"-cp", appJar,
120+
"-Xlog:class+path=trace",
121+
"Hello"))
122+
.assertNormalExit();
123+
124+
// Run with existent boot class path, test should fail.
125+
TestCommon.run(make_args(true,
126+
appJar,
127+
"-cp", appJar,
128+
"-Xlog:class+path=trace",
129+
"Hello"))
130+
.assertAbnormalExit(errorMessage1, errorMessage2, errorMessage3);
131+
132+
// Dump an archive with existent boot class path.
133+
TestCommon.testDump("foobar", TestCommon.list("Hello"), make_args(true, appJar));
134+
135+
// Run with non-existent boot class path, test should fail.
136+
TestCommon.run(make_args(true,
137+
nonExistPath,
138+
"-Xlog:class+path=trace",
139+
"Hello"))
140+
.assertAbnormalExit(errorMessage1, errorMessage2, errorMessage3);
141+
142+
// Run with existent boot class path, test should pass.
143+
TestCommon.run(make_args(true,
144+
appJar,
145+
"-Xlog:class+path=trace",
146+
"Hello"))
147+
.assertNormalExit();
148+
149+
// Test with empty jar file.
150+
(new File(emptyJarPath)).delete();
151+
(new File(emptyJarPath)).createNewFile();
152+
153+
// Dump an archive with an empty jar in the boot class path.
154+
TestCommon.testDump("foobar", TestCommon.list("Hello"), make_args(true, emptyJarPath, "-cp", appJar));
155+
156+
// Run with an empty jar in boot class path, test should pass.
157+
TestCommon.run(make_args(true,
158+
emptyJarPath,
159+
"-cp", appJar,
160+
"-Xlog:class+path=trace",
161+
"Hello"))
162+
.assertNormalExit();
163+
164+
// Run with non-existent boot class path, test should pass.
165+
TestCommon.run(make_args(true,
166+
nonExistPath,
167+
"-cp", appJar,
168+
"-Xlog:class+path=trace",
169+
"Hello"))
170+
.assertNormalExit();
171+
172+
// Run with existent boot class path, test should fail.
173+
TestCommon.run(make_args(true,
174+
appJar,
175+
"-cp", appJar,
176+
"-Xlog:class+path=trace",
177+
"Hello"))
178+
.assertAbnormalExit(errorMessage1, errorMessage2, errorMessage3);
103179
}
104180

105181
static String[] make_args(boolean bootcp, String cp, String... suffix) {

‎test/hotspot/jtreg/runtime/cds/appcds/PrintSharedArchiveAndExit.java

+6-6
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2014, 2019, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2014, 2022, Oracle and/or its affiliates. All rights reserved.
33
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
44
*
55
* This code is free software; you can redistribute it and/or modify it
@@ -87,16 +87,16 @@ public static void main(String[] args) throws Exception {
8787
TestCommon.run("-cp", cp, "-XX:+PrintSharedArchiveAndExit", "Hello")
8888
.ifNoMappingFailure(output -> check(output, 0, true, lastCheckMsg));
8989

90-
log("Execution with simple errors -- with 'simple' errors like missing or modified\n" +
91-
"JAR files, the VM should try to continue to print the remaining information.\n" +
92-
"Use an invalid Boot CP -- all the JAR paths should be checked");
90+
log("Non-existent boot cp should be ignored, test should pass.");
9391
TestCommon.run(
9492
"-cp", cp,
9593
"-Xbootclasspath/a:foo.jar",
9694
"-XX:+PrintSharedArchiveAndExit")
97-
.ifNoMappingFailure(output -> check(output, 1, true, lastCheckMsg, "[BOOT classpath mismatch, "));
95+
.ifNoMappingFailure(output -> check(output, 0, true, lastCheckMsg));
9896

99-
log("Use an App CP shorter than the one at dump time -- all the JAR paths should be checked");
97+
log("Execution with simple errors -- with 'simple' errors like missing or modified\n" +
98+
"JAR files, the VM should try to continue to print the remaining information.\n" +
99+
"Use an App CP shorter than the one at dump time -- all the JAR paths should be checked");
100100
TestCommon.run(
101101
"-cp", ".",
102102
"-XX:+PrintSharedArchiveAndExit")

‎test/hotspot/jtreg/runtime/cds/appcds/jigsaw/modulepath/OptimizeModuleHandlingTest.java

+11-1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2020, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2020, 2022, Oracle and/or its affiliates. All rights reserved.
33
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
44
*
55
* This code is free software; you can redistribute it and/or modify it
@@ -219,6 +219,16 @@ public static void runWithModulePath(String... extraRuntimeArgs) throws Exceptio
219219
"-Xbootclasspath/a:", ".",
220220
"--module-path", libsDir.toString(),
221221
MAIN_CLASS)
222+
.assertAbnormalExit(out -> {
223+
out.shouldNotContain(CLASS_FOUND_MESSAGE)
224+
.shouldContain(OPTIMIZE_DISABLED) // mapping info
225+
.shouldContain("Error: Could not find or load main class .");
226+
});
227+
tty("11. run with CDS on, --module-path, with -Xbootclasspath/a:.");
228+
TestCommon.run("-Xlog:cds",
229+
"-Xbootclasspath/a:.",
230+
"--module-path", libsDir.toString(),
231+
MAIN_CLASS)
222232
.assertAbnormalExit(out -> {
223233
out.shouldNotContain(CLASS_FOUND_MESSAGE)
224234
.shouldContain(OPTIMIZE_DISABLED) // mapping info

0 commit comments

Comments
 (0)
Please sign in to comment.