Skip to content

Commit 124ba45

Browse files
committedJun 6, 2022
8287101: CDS should check for file truncation for all regions
Reviewed-by: iklam, coleenp
1 parent 0e06bf3 commit 124ba45

File tree

3 files changed

+44
-8
lines changed

3 files changed

+44
-8
lines changed
 

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

+4-6
Original file line numberDiff line numberDiff line change
@@ -1370,12 +1370,10 @@ bool FileMapInfo::init_from_file(int fd) {
13701370

13711371
_file_offset = header()->header_size(); // accounts for the size of _base_archive_name
13721372

1373-
if (is_static()) {
1374-
// just checking the last region is sufficient since the archive is written
1375-
// in sequential order
1376-
size_t len = os::lseek(fd, 0, SEEK_END);
1377-
FileMapRegion* si = space_at(MetaspaceShared::last_valid_region);
1378-
// The last space might be empty
1373+
size_t len = os::lseek(fd, 0, SEEK_END);
1374+
1375+
for (int i = 0; i <= MetaspaceShared::last_valid_region; i++) {
1376+
FileMapRegion* si = space_at(i);
13791377
if (si->file_offset() > len || len - si->file_offset() < si->used()) {
13801378
fail_continue("The shared archive file has been truncated.");
13811379
return false;

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

+11-2
Original file line numberDiff line numberDiff line change
@@ -76,7 +76,7 @@ public static String startNewArchive(String testName) {
7676
return newArchiveName;
7777
}
7878

79-
public static void testAndCheck(String[] execArgs) throws Exception {
79+
public static void testAndCheck(String[] execArgs, String... expectedMessages) throws Exception {
8080
OutputAnalyzer output = shareAuto ? TestCommon.execAuto(execArgs) : TestCommon.execCommon(execArgs);
8181
String stdtxt = output.getOutput();
8282
System.out.println("Note: this test may fail in very rare occasions due to CRC32 checksum collision");
@@ -88,6 +88,9 @@ public static void testAndCheck(String[] execArgs) throws Exception {
8888
output.shouldNotContain("A fatal error has been detected by the Java Runtime Environment");
8989
}
9090
}
91+
for (int i = 0; i < expectedMessages.length; i++) {
92+
output.shouldContain(expectedMessages[i]);
93+
}
9194
for (String message : matchMessages) {
9295
if (stdtxt.contains(message)) {
9396
// match any to return
@@ -236,11 +239,17 @@ public static void main(String... args) throws Exception {
236239
testAndCheck(verifyExecArgs);
237240

238241
// delete bytes in data section forward
239-
System.out.println("\n6. Delete bytes at beginning of data section, should fail\n");
242+
System.out.println("\n6a. Delete bytes at beginning of data section, should fail\n");
240243
String deleteBytes = startNewArchive("delete-bytes");
241244
CDSArchiveUtils.deleteBytesAtRandomPositionAfterHeader(orgJsaFile, deleteBytes, 4096 /*bytes*/);
242245
testAndCheck(verifyExecArgs);
243246

247+
// delete bytes at the end
248+
System.out.println("\n6b. Delete bytes at the end, should fail\n");
249+
deleteBytes = startNewArchive("delete-bytes-end");
250+
CDSArchiveUtils.deleteBytesAtTheEnd(orgJsaFile, deleteBytes);
251+
testAndCheck(verifyExecArgs, "The shared archive file has been truncated.");
252+
244253
// modify contents in random area
245254
System.out.println("\n7. modify Content in random areas, should fail\n");
246255
String randomAreas = startNewArchive("random-areas");

‎test/lib/jdk/test/lib/cds/CDSArchiveUtils.java

+29
Original file line numberDiff line numberDiff line change
@@ -461,6 +461,35 @@ public static File deleteBytesAtRandomPositionAfterHeader(File orgFile, String n
461461
return dstFile;
462462
}
463463

464+
// returns the size of the last region with used bytes > 0.
465+
private static long getLastUsedRegionSize(File jsaFile) throws Exception {
466+
int i = num_regions - 1;
467+
long regionSize = 0;
468+
while (i >= 0) {
469+
regionSize = usedRegionSizeAligned(jsaFile, i);
470+
if (regionSize > 0) {
471+
break;
472+
}
473+
i--;
474+
}
475+
return regionSize;
476+
}
477+
478+
// delete last regions's used bytes at the end, so new file will be smaller than the original
479+
public static File deleteBytesAtTheEnd(File orgFile, String newFileName) throws Exception {
480+
long offset = fileHeaderSize(orgFile);
481+
long bytesToDelete = getLastUsedRegionSize(orgFile);
482+
File dstFile = new File(newFileName);
483+
try (FileChannel inputChannel = new FileInputStream(orgFile).getChannel();
484+
FileChannel outputChannel = new FileOutputStream(dstFile).getChannel()) {
485+
long orgSize = inputChannel.size();
486+
transferFrom(inputChannel, outputChannel, 0, offset);
487+
inputChannel.position(offset);
488+
transferFrom(inputChannel, outputChannel, offset, orgSize - bytesToDelete);
489+
}
490+
return dstFile;
491+
}
492+
464493
// used region size
465494
public static long usedRegionSizeAligned(File archiveFile, int region) throws Exception {
466495
long offset = spOffset + cdsFileMapRegionSize * region + spUsedOffset;

0 commit comments

Comments
 (0)
Please sign in to comment.