8210523: runtime/appcds/cacheObject/DifferentHeapSizes.java crash

Reviewed-by: jiangli, ccheung
This commit is contained in:
Ioi Lam 2018-09-12 17:45:22 -07:00
parent cf26c4cc5c
commit 4e74846da8
10 changed files with 65 additions and 35 deletions

@ -282,7 +282,7 @@ public:
CompactHashtable_OopIterator(OopClosure *cl) : _closure(cl) {}
inline void do_value(address base_address, u4 offset) const {
narrowOop v = (narrowOop)offset;
oop obj = HeapShared::decode_with_archived_oop_encoding_mode(v);
oop obj = HeapShared::decode_from_archive(v);
_closure->do_oop(&obj);
}
};

@ -48,7 +48,7 @@ template <class T, class N>
inline oop CompactHashtable<T, N>::decode_entry(CompactHashtable<oop, char>* const t,
u4 offset, const char* name, int len) {
narrowOop v = (narrowOop)offset;
oop string = HeapShared::decode_with_archived_oop_encoding_mode(v);
oop string = HeapShared::decode_from_archive(v);
if (java_lang_String::equals(string, (jchar*)name, len)) {
return string;
}

@ -854,7 +854,7 @@ address FileMapInfo::decode_start_address(CDSFileMapRegion* spc, bool with_curre
if (with_current_oop_encoding_mode) {
return (address)CompressedOops::decode_not_null(offset_of_space(spc));
} else {
return (address)HeapShared::decode_with_archived_oop_encoding_mode(offset_of_space(spc));
return (address)HeapShared::decode_from_archive(offset_of_space(spc));
}
}
@ -880,7 +880,7 @@ MemRegion FileMapInfo::get_heap_regions_range_with_current_oop_encoding_mode() {
CDSFileMapRegion* si = space_at(i);
size_t size = si->_used;
if (size > 0) {
address s = start_address_with_current_oop_encoding_mode(si);
address s = start_address_as_decoded_with_current_oop_encoding_mode(si);
address e = s + size;
if (start > s) {
start = s;
@ -972,21 +972,20 @@ void FileMapInfo::map_heap_regions_impl() {
HeapShared::init_narrow_oop_decoding(narrow_oop_base() + delta, narrow_oop_shift());
CDSFileMapRegion* si = space_at(MetaspaceShared::first_string);
address relocated_strings_bottom = start_address_with_archived_oop_encoding_mode(si);
if (!is_aligned(relocated_strings_bottom + delta, HeapRegion::GrainBytes)) {
address relocated_strings_bottom = start_address_as_decoded_from_archive(si);
if (!is_aligned(relocated_strings_bottom, HeapRegion::GrainBytes)) {
// Align the bottom of the string regions at G1 region boundary. This will avoid
// the situation where the highest open region and the lowest string region sharing
// the same G1 region. Otherwise we will fail to map the open regions.
size_t align = size_t(relocated_strings_bottom) % HeapRegion::GrainBytes;
delta -= align;
assert(is_aligned(relocated_strings_bottom + delta, HeapRegion::GrainBytes), "must be");
log_info(cds)("CDS heap data need to be relocated lower by a further " SIZE_FORMAT
" bytes to be aligned with HeapRegion::GrainBytes", align);
" bytes to " INTX_FORMAT " to be aligned with HeapRegion::GrainBytes", align, delta);
HeapShared::init_narrow_oop_decoding(narrow_oop_base() + delta, narrow_oop_shift());
_heap_pointers_need_patching = true;
relocated_strings_bottom = start_address_as_decoded_from_archive(si);
}
assert(is_aligned(relocated_strings_bottom, HeapRegion::GrainBytes), "must be");
// First, map string regions as closed archive heap regions.
// GC does not write into the regions.
@ -1032,7 +1031,7 @@ bool FileMapInfo::map_heap_data(MemRegion **heap_mem, int first,
si = space_at(i);
size_t size = si->_used;
if (size > 0) {
HeapWord* start = (HeapWord*)start_address_with_archived_oop_encoding_mode(si);
HeapWord* start = (HeapWord*)start_address_as_decoded_from_archive(si);
regions[region_num] = MemRegion(start, size / HeapWordSize);
region_num ++;
log_info(cds)("Trying to map heap data: region[%d] at " INTPTR_FORMAT ", size = " SIZE_FORMAT_W(8) " bytes",
@ -1242,7 +1241,7 @@ char* FileMapInfo::region_addr(int idx) {
if (MetaspaceShared::is_heap_region(idx)) {
assert(DumpSharedSpaces, "The following doesn't work at runtime");
return si->_used > 0 ?
(char*)start_address_with_current_oop_encoding_mode(si) : NULL;
(char*)start_address_as_decoded_with_current_oop_encoding_mode(si) : NULL;
} else {
return si->_addr._base;
}

@ -335,12 +335,12 @@ public:
}
// The starting address of spc, as calculated with CompressedOop::decode_non_null()
address start_address_with_current_oop_encoding_mode(CDSFileMapRegion* spc) {
address start_address_as_decoded_with_current_oop_encoding_mode(CDSFileMapRegion* spc) {
return decode_start_address(spc, true);
}
// The starting address of spc, as calculated with HeapShared::decode_with_archived_oop_encoding_mode()
address start_address_with_archived_oop_encoding_mode(CDSFileMapRegion* spc) {
// The starting address of spc, as calculated with HeapShared::decode_from_archive()
address start_address_as_decoded_from_archive(CDSFileMapRegion* spc) {
return decode_start_address(spc, false);
}

@ -834,7 +834,7 @@ class PatchEmbeddedPointers: public BitMapClosure {
narrowOop* p = _start + offset;
narrowOop v = *p;
assert(!CompressedOops::is_null(v), "null oops should have been filtered out at dump time");
oop o = HeapShared::decode_with_archived_oop_encoding_mode(v);
oop o = HeapShared::decode_from_archive(v);
RawAccess<IS_NOT_NULL>::oop_store(p, o);
return true;
}

@ -137,7 +137,7 @@ class HeapShared: AllStatic {
static size_t build_archived_subgraph_info_records(int num_records);
// Used by decode_with_archived_oop_encoding_mode
// Used by decode_from_archive
static address _narrow_oop_base;
static int _narrow_oop_shift;
@ -194,7 +194,7 @@ class HeapShared: AllStatic {
// than Universe::narrow_oop_{base,shift} -- see FileMapInfo::map_heap_regions_impl.
// To decode them, do not use CompressedOops::decode_not_null. Use this
// function instead.
inline static oop decode_with_archived_oop_encoding_mode(narrowOop v) NOT_CDS_JAVA_HEAP_RETURN_(NULL);
inline static oop decode_from_archive(narrowOop v) NOT_CDS_JAVA_HEAP_RETURN_(NULL);
static void init_narrow_oop_decoding(address base, int shift) NOT_CDS_JAVA_HEAP_RETURN;

@ -30,7 +30,7 @@
#if INCLUDE_CDS_JAVA_HEAP
inline oop HeapShared::decode_with_archived_oop_encoding_mode(narrowOop v) {
inline oop HeapShared::decode_from_archive(narrowOop v) {
assert(!CompressedOops::is_null(v), "narrow oop value can never be zero");
oop result = (oop)(void*)((uintptr_t)_narrow_oop_base + ((uintptr_t)v << _narrow_oop_shift));
assert(check_obj_alignment(result), "address not aligned: " INTPTR_FORMAT, p2i((void*) result));

@ -1944,7 +1944,7 @@ oop MetaspaceShared::materialize_archived_object(narrowOop v) {
assert(archive_heap_region_fixed(),
"must be called after archive heap regions are fixed");
if (!CompressedOops::is_null(v)) {
oop obj = HeapShared::decode_with_archived_oop_encoding_mode(v);
oop obj = HeapShared::decode_from_archive(v);
return G1CollectedHeap::heap()->materialize_archived_object(obj);
}
return NULL;
@ -2021,7 +2021,7 @@ public:
"Archived heap object is not allowed");
assert(MetaspaceShared::open_archive_heap_region_mapped(),
"Open archive heap region is not mapped");
*p = HeapShared::decode_with_archived_oop_encoding_mode(o);
*p = HeapShared::decode_from_archive(o);
}
}

@ -40,6 +40,8 @@ import sun.hotspot.WhiteBox;
import jdk.test.lib.cds.CDSTestUtils;
public class DifferentHeapSizes {
static final String DEDUP = "-XX:+UseStringDeduplication"; // This increases code coverage.
static class Scenario {
int dumpSize; // in MB
int runSizes[]; // in MB
@ -59,7 +61,6 @@ public class DifferentHeapSizes {
};
public static void main(String[] args) throws Exception {
String dedup = "-XX:+UseStringDeduplication"; // This increases code coverage.
JarBuilder.getOrCreateHelloJar();
String appJar = TestCommon.getTestJar("hello.jar");
String appClasses[] = TestCommon.list("Hello");
@ -71,7 +72,7 @@ public class DifferentHeapSizes {
for (int runSize : s.runSizes) {
String runXmx = "-Xmx" + runSize + "m";
CDSTestUtils.Result result = TestCommon.run("-cp", appJar, "-showversion",
"-Xlog:cds", runXmx, dedup, "Hello");
"-Xlog:cds", runXmx, DEDUP, "Hello");
if (runSize < 32768) {
result
.assertNormalExit("Hello World")
@ -80,21 +81,49 @@ public class DifferentHeapSizes {
out.shouldNotContain(CDSTestUtils.MSG_RANGE_ALREADT_IN_USE);
});
} else {
result.assertAbnormalExit("Unable to use shared archive: UseCompressedOops and UseCompressedClassPointers must be on for UseSharedSpaces.");
result.assertAbnormalExit(CDSTestUtils.MSG_COMPRESSION_MUST_BE_USED);
}
}
}
String flag = "HeapBaseMinAddress";
String xxflag = "-XX:" + flag + "=";
String mx = "-Xmx128m";
long base = WhiteBox.getWhiteBox().getSizeTVMFlag(flag).longValue();
TestCommon.dump(appJar, appClasses, mx, xxflag + base);
TestCommon.run("-cp", appJar, "-showversion", "-Xlog:cds", mx, xxflag + (base + 256 * 1024 * 1024), dedup, "Hello")
.assertNormalExit("Hello World")
.assertNormalExit(out -> {
out.shouldNotContain(CDSTestUtils.MSG_RANGE_NOT_WITHIN_HEAP);
out.shouldNotContain(CDSTestUtils.MSG_RANGE_ALREADT_IN_USE);
});
// Test various settings of -XX:HeapBaseMinAddress that would trigger
// "CDS heap data need to be relocated because the desired range ... is outside of the heap"
long default_base = WhiteBox.getWhiteBox().getSizeTVMFlag("HeapBaseMinAddress").longValue();
long M = 1024 * 1024;
long bases[] = new long[] {
/* dump xmx */ /* run xmx */ /* dump base */ /* run base */
128 * M, 128 * M, default_base, default_base + 256L * 1024 * 1024,
128 * M, 16376 * M, 0x0000000119200000L, -1,
};
for (int i = 0; i < bases.length; i += 4) {
String dump_xmx = getXmx(bases[i+0]);
String run_xmx = getXmx(bases[i+1]);
String dump_base = getHeapBaseMinAddress(bases[i+2]);
String run_base = getHeapBaseMinAddress(bases[i+3]);
TestCommon.dump(appJar, appClasses, dump_xmx, dump_base);
TestCommon.run("-cp", appJar, "-showversion", "-Xlog:cds", run_xmx, run_base, DEDUP, "Hello")
.assertNormalExit("Hello World")
.assertNormalExit(out -> {
out.shouldNotContain(CDSTestUtils.MSG_RANGE_NOT_WITHIN_HEAP);
out.shouldNotContain(CDSTestUtils.MSG_RANGE_ALREADT_IN_USE);
});
}
}
static String getXmx(long value) {
if (value < 0) {
return "-showversion"; // This is a harmless command line arg
} else {
return "-Xmx" + (value / 1024 / 1024) + "m";
}
}
static String getHeapBaseMinAddress(long value) {
if (value < 0) {
return "-showversion"; // This is a harmless command line arg
} else {
return "-XX:HeapBaseMinAddress=0x" + Long.toHexString(value);
}
}
}

@ -40,6 +40,8 @@ public class CDSTestUtils {
"UseSharedSpaces: Unable to allocate region, range is not within java heap.";
public static final String MSG_RANGE_ALREADT_IN_USE =
"Unable to allocate region, java heap range is already in use.";
public static final String MSG_COMPRESSION_MUST_BE_USED =
"Unable to use shared archive: UseCompressedOops and UseCompressedClassPointers must be on for UseSharedSpaces.";
public interface Checker {
public void check(OutputAnalyzer output) throws Exception;