Merge
This commit is contained in:
commit
7f243a6751
@ -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() {
|
|||||||
// 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()));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1071,7 +1071,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.
|
||||||
@ -1235,7 +1236,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)) {
|
||||||
@ -1257,10 +1258,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;
|
||||||
@ -1289,40 +1291,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:
|
|||||||
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;
|
||||||
}
|
}
|
||||||
@ -2047,10 +2041,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; }
|
||||||
|
|
||||||
@ -2155,7 +2151,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);
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user