提交 474da996 编写于 作者: R roland

8035841: assert(dp_src->tag() == dp_dst->tag()) failed: should be same tags 1...

8035841: assert(dp_src->tag() == dp_dst->tag()) failed: should be same tags 1 != 0 at ciMethodData.cpp:90
Summary: concurrent update of traps with construction of ciMethodData
Reviewed-by: kvn, twisti
上级 f51328c3
...@@ -87,8 +87,9 @@ void ciMethodData::load_extra_data() { ...@@ -87,8 +87,9 @@ void ciMethodData::load_extra_data() {
DataLayout* dp_dst = extra_data_base(); DataLayout* dp_dst = extra_data_base();
for (;; dp_src = MethodData::next_extra(dp_src), dp_dst = MethodData::next_extra(dp_dst)) { for (;; dp_src = MethodData::next_extra(dp_src), dp_dst = MethodData::next_extra(dp_dst)) {
assert(dp_src < end_src, "moved past end of extra data"); assert(dp_src < end_src, "moved past end of extra data");
assert(dp_src->tag() == dp_dst->tag(), err_msg("should be same tags %d != %d", dp_src->tag(), dp_dst->tag())); // New traps in the MDO can be added as we translate the copy so
switch(dp_src->tag()) { // look at the entries in the copy.
switch(dp_dst->tag()) {
case DataLayout::speculative_trap_data_tag: { case DataLayout::speculative_trap_data_tag: {
ciSpeculativeTrapData* data_dst = new ciSpeculativeTrapData(dp_dst); ciSpeculativeTrapData* data_dst = new ciSpeculativeTrapData(dp_dst);
SpeculativeTrapData* data_src = new SpeculativeTrapData(dp_src); SpeculativeTrapData* data_src = new SpeculativeTrapData(dp_src);
...@@ -102,7 +103,7 @@ void ciMethodData::load_extra_data() { ...@@ -102,7 +103,7 @@ void ciMethodData::load_extra_data() {
// An empty slot or ArgInfoData entry marks the end of the trap data // An empty slot or ArgInfoData entry marks the end of the trap data
return; return;
default: default:
fatal(err_msg("bad tag = %d", dp_src->tag())); fatal(err_msg("bad tag = %d", dp_dst->tag()));
} }
} }
} }
......
...@@ -1066,7 +1066,8 @@ void MethodData::post_initialize(BytecodeStream* stream) { ...@@ -1066,7 +1066,8 @@ void MethodData::post_initialize(BytecodeStream* stream) {
} }
// Initialize the MethodData* corresponding to a given method. // Initialize the MethodData* corresponding to a given method.
MethodData::MethodData(methodHandle method, int size, TRAPS) { MethodData::MethodData(methodHandle method, int size, TRAPS)
: _extra_data_lock(Monitor::leaf, "MDO extra data lock") {
No_Safepoint_Verifier no_safepoint; // init function atomic wrt GC No_Safepoint_Verifier no_safepoint; // init function atomic wrt GC
ResourceMark rm; ResourceMark rm;
// Set the method back-pointer. // Set the method back-pointer.
...@@ -1230,7 +1231,7 @@ DataLayout* MethodData::next_extra(DataLayout* dp) { ...@@ -1230,7 +1231,7 @@ DataLayout* MethodData::next_extra(DataLayout* dp) {
return (DataLayout*)((address)dp + DataLayout::compute_size_in_bytes(nb_cells)); return (DataLayout*)((address)dp + DataLayout::compute_size_in_bytes(nb_cells));
} }
ProfileData* MethodData::bci_to_extra_data_helper(int bci, Method* m, DataLayout*& dp) { ProfileData* MethodData::bci_to_extra_data_helper(int bci, Method* m, DataLayout*& dp, bool concurrent) {
DataLayout* end = extra_data_limit(); DataLayout* end = extra_data_limit();
for (;; dp = next_extra(dp)) { for (;; dp = next_extra(dp)) {
...@@ -1252,10 +1253,11 @@ ProfileData* MethodData::bci_to_extra_data_helper(int bci, Method* m, DataLayout ...@@ -1252,10 +1253,11 @@ ProfileData* MethodData::bci_to_extra_data_helper(int bci, Method* m, DataLayout
if (m != NULL) { if (m != NULL) {
SpeculativeTrapData* data = new SpeculativeTrapData(dp); SpeculativeTrapData* data = new SpeculativeTrapData(dp);
// data->method() may be null in case of a concurrent // data->method() may be null in case of a concurrent
// allocation. Assume it's for the same method and use that // allocation. Maybe it's for the same method. Try to use that
// entry in that case. // entry in that case.
if (dp->bci() == bci) { if (dp->bci() == bci) {
if (data->method() == NULL) { if (data->method() == NULL) {
assert(concurrent, "impossible because no concurrent allocation");
return NULL; return NULL;
} else if (data->method() == m) { } else if (data->method() == m) {
return data; return data;
...@@ -1284,40 +1286,40 @@ ProfileData* MethodData::bci_to_extra_data(int bci, Method* m, bool create_if_mi ...@@ -1284,40 +1286,40 @@ ProfileData* MethodData::bci_to_extra_data(int bci, Method* m, bool create_if_mi
// Allocation in the extra data space has to be atomic because not // Allocation in the extra data space has to be atomic because not
// all entries have the same size and non atomic concurrent // all entries have the same size and non atomic concurrent
// allocation would result in a corrupted extra data space. // allocation would result in a corrupted extra data space.
while (true) { ProfileData* result = bci_to_extra_data_helper(bci, m, dp, true);
ProfileData* result = bci_to_extra_data_helper(bci, m, dp); if (result != NULL) {
if (result != NULL) { return result;
}
if (create_if_missing && dp < end) {
MutexLocker ml(&_extra_data_lock);
// Check again now that we have the lock. Another thread may
// have added extra data entries.
ProfileData* result = bci_to_extra_data_helper(bci, m, dp, false);
if (result != NULL || dp >= end) {
return result; return result;
} }
if (create_if_missing && dp < end) { assert(dp->tag() == DataLayout::no_tag || (dp->tag() == DataLayout::speculative_trap_data_tag && m != NULL), "should be free");
assert(dp->tag() == DataLayout::no_tag || (dp->tag() == DataLayout::speculative_trap_data_tag && m != NULL), "should be free"); assert(next_extra(dp)->tag() == DataLayout::no_tag || next_extra(dp)->tag() == DataLayout::arg_info_data_tag, "should be free or arg info");
assert(next_extra(dp)->tag() == DataLayout::no_tag || next_extra(dp)->tag() == DataLayout::arg_info_data_tag, "should be free or arg info"); u1 tag = m == NULL ? DataLayout::bit_data_tag : DataLayout::speculative_trap_data_tag;
u1 tag = m == NULL ? DataLayout::bit_data_tag : DataLayout::speculative_trap_data_tag; // SpeculativeTrapData is 2 slots. Make sure we have room.
// SpeculativeTrapData is 2 slots. Make sure we have room. if (m != NULL && next_extra(dp)->tag() != DataLayout::no_tag) {
if (m != NULL && next_extra(dp)->tag() != DataLayout::no_tag) { return NULL;
return NULL; }
} DataLayout temp;
DataLayout temp; temp.initialize(tag, bci, 0);
temp.initialize(tag, bci, 0);
// May have been set concurrently dp->set_header(temp.header());
if (dp->header() != temp.header() && !dp->atomic_set_header(temp.header())) { assert(dp->tag() == tag, "sane");
// Allocation failure because of concurrent allocation. Try assert(dp->bci() == bci, "no concurrent allocation");
// again. if (tag == DataLayout::bit_data_tag) {
continue; return new BitData(dp);
} } else {
assert(dp->tag() == tag, "sane"); SpeculativeTrapData* data = new SpeculativeTrapData(dp);
assert(dp->bci() == bci, "no concurrent allocation"); data->set_method(m);
if (tag == DataLayout::bit_data_tag) { return data;
return new BitData(dp);
} else {
// If being allocated concurrently, one trap may be lost
SpeculativeTrapData* data = new SpeculativeTrapData(dp);
data->set_method(m);
return data;
}
} }
return NULL;
} }
return NULL; return NULL;
} }
......
...@@ -190,12 +190,6 @@ public: ...@@ -190,12 +190,6 @@ public:
void set_header(intptr_t value) { void set_header(intptr_t value) {
_header._bits = value; _header._bits = value;
} }
bool atomic_set_header(intptr_t value) {
if (Atomic::cmpxchg_ptr(value, (volatile intptr_t*)&_header._bits, 0) == 0) {
return true;
}
return false;
}
intptr_t header() { intptr_t header() {
return _header._bits; return _header._bits;
} }
...@@ -1856,10 +1850,12 @@ private: ...@@ -1856,10 +1850,12 @@ private:
// Cached hint for bci_to_dp and bci_to_data // Cached hint for bci_to_dp and bci_to_data
int _hint_di; int _hint_di;
Mutex _extra_data_lock;
MethodData(methodHandle method, int size, TRAPS); MethodData(methodHandle method, int size, TRAPS);
public: public:
static MethodData* allocate(ClassLoaderData* loader_data, methodHandle method, TRAPS); static MethodData* allocate(ClassLoaderData* loader_data, methodHandle method, TRAPS);
MethodData() {}; // For ciMethodData MethodData() : _extra_data_lock(Monitor::leaf, "MDO extra data lock") {}; // For ciMethodData
bool is_methodData() const volatile { return true; } bool is_methodData() const volatile { return true; }
...@@ -1964,7 +1960,7 @@ private: ...@@ -1964,7 +1960,7 @@ private:
// What is the index of the first data entry? // What is the index of the first data entry?
int first_di() const { return 0; } int first_di() const { return 0; }
ProfileData* bci_to_extra_data_helper(int bci, Method* m, DataLayout*& dp); ProfileData* bci_to_extra_data_helper(int bci, Method* m, DataLayout*& dp, bool concurrent);
// Find or create an extra ProfileData: // Find or create an extra ProfileData:
ProfileData* bci_to_extra_data(int bci, Method* m, bool create_if_missing); ProfileData* bci_to_extra_data(int bci, Method* m, bool create_if_missing);
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册