|
1 | 1 | /*
|
2 |
| - * Copyright (c) 1997, 2021, Oracle and/or its affiliates. All rights reserved. |
| 2 | + * Copyright (c) 1997, 2022, Oracle and/or its affiliates. All rights reserved. |
3 | 3 | * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
4 | 4 | *
|
5 | 5 | * This code is free software; you can redistribute it and/or modify it
|
@@ -105,7 +105,7 @@ RegisterMap::RegisterMap(const RegisterMap* map) {
|
105 | 105 | DEBUG_ONLY(_skip_missing = map->_skip_missing;)
|
106 | 106 |
|
107 | 107 | // only the original RegisterMap's handle lives long enough for StackWalker; this is bound to cause trouble with nested continuations.
|
108 |
| - _chunk = map->_chunk; // stackChunkHandle(Thread::current(), map->_chunk(), map->_chunk.not_null()); // |
| 108 | + _chunk = map->_chunk; // stackChunkHandle(Thread::current(), map->_chunk(), map->_chunk.not_null()); // |
109 | 109 |
|
110 | 110 | pd_initialize_from(map);
|
111 | 111 | if (update_map()) {
|
@@ -911,17 +911,18 @@ oop frame::interpreter_callee_receiver(Symbol* signature) {
|
911 | 911 |
|
912 | 912 | template <bool relative>
|
913 | 913 | void frame::oops_interpreted_do(OopClosure* f, const RegisterMap* map, bool query_oop_map_cache) const {
|
914 |
| - Thread *thread = Thread::current(); |
915 |
| - methodHandle m (thread, interpreter_frame_method()); |
| 914 | + Thread* current = Thread::current(); |
| 915 | + methodHandle m(current, interpreter_frame_method()); |
916 | 916 | jint bci = interpreter_frame_bci();
|
917 | 917 |
|
| 918 | + ResourceMark rm; |
918 | 919 | InterpreterOopMap mask;
|
919 | 920 | if (query_oop_map_cache) {
|
920 | 921 | m->mask_for(bci, &mask);
|
921 | 922 | } else {
|
922 | 923 | OopMapCache::compute_one_oop_map(m, bci, &mask);
|
923 | 924 | }
|
924 |
| - |
| 925 | + |
925 | 926 | oops_interpreted_do0<relative>(f, map, m, bci, mask);
|
926 | 927 | }
|
927 | 928 |
|
@@ -1178,8 +1179,8 @@ bool frame::is_deoptimized_frame() const {
|
1178 | 1179 | }
|
1179 | 1180 |
|
1180 | 1181 | /* This method only checks if the frame is deoptimized
|
1181 |
| - * as in return address being patched. |
1182 |
| - * It doesn't care if the OP that we return to is a |
| 1182 | + * as in return address being patched. |
| 1183 | + * It doesn't care if the OP that we return to is a |
1183 | 1184 | * deopt instruction */
|
1184 | 1185 | /*if (_cb != NULL && _cb->is_nmethod()) {
|
1185 | 1186 | return NativeDeoptInstruction::is_deopt_at(_pc);
|
@@ -1294,18 +1295,40 @@ extern "C" bool dbg_is_safe(const void* p, intptr_t errvalue);
|
1294 | 1295 |
|
1295 | 1296 | class FrameValuesOopClosure: public OopClosure, public DerivedOopClosure {
|
1296 | 1297 | private:
|
1297 |
| - FrameValues& _values; |
1298 |
| - int _frame_no; |
| 1298 | + GrowableArray<oop*>* _oops; |
| 1299 | + GrowableArray<narrowOop*>* _narrow_oops; |
| 1300 | + NoSafepointVerifier nsv; |
1299 | 1301 | public:
|
1300 |
| - FrameValuesOopClosure(FrameValues& values, int frame_no) : _values(values), _frame_no(frame_no) {} |
1301 |
| - virtual void do_oop(oop* p) { |
1302 |
| - bool good = *p == nullptr || (dbg_is_safe(*p, -1) && dbg_is_safe((*p)->klass(), -1) && oopDesc::is_oop_or_null(*p)); |
1303 |
| - _values.describe(_frame_no, (intptr_t*)p, err_msg("oop%s for #%d", good ? "" : " (BAD)", _frame_no)); |
| 1302 | + FrameValuesOopClosure() { |
| 1303 | + _oops = new (ResourceObj::C_HEAP, mtThread) GrowableArray<oop*>(100, mtThread); |
| 1304 | + _narrow_oops = new (ResourceObj::C_HEAP, mtThread) GrowableArray<narrowOop*>(100, mtThread); |
| 1305 | + } |
| 1306 | + ~FrameValuesOopClosure() { |
| 1307 | + delete _oops; |
| 1308 | + delete _narrow_oops; |
1304 | 1309 | }
|
1305 |
| - virtual void do_oop(narrowOop* p) { _values.describe(_frame_no, (intptr_t*)p, err_msg("narrow oop for #%d", _frame_no)); } |
1306 |
| - virtual void do_derived_oop(oop* base, derived_pointer* derived) { |
| 1310 | + void describe(FrameValues& values, int frame_no) { |
| 1311 | + for (int i = 0; i < _oops->length(); i++) { |
| 1312 | + oop* p = _oops->at(i); |
| 1313 | + bool good = *p == nullptr || (dbg_is_safe(*p, -1) && dbg_is_safe((*p)->klass(), -1) && oopDesc::is_oop_or_null(*p)); |
| 1314 | + values.describe(frame_no, (intptr_t*)p, err_msg("oop%s for #%d", good ? "" : " (BAD)", frame_no)); |
| 1315 | + } |
| 1316 | + for (int i = 0; i < _narrow_oops->length(); i++) { |
| 1317 | + narrowOop* p = _narrow_oops->at(i); |
| 1318 | + values.describe(frame_no, (intptr_t*)p, err_msg("narrow oop for #%d", frame_no)); |
| 1319 | + } |
| 1320 | + } |
| 1321 | +#if 0 |
| 1322 | + virtual void do_derived_oop(oop* base, derived_pointer* derived) { |
1307 | 1323 | _values.describe(_frame_no, (intptr_t*)derived, err_msg("derived pointer (base: " INTPTR_FORMAT ") for #%d", p2i(base), _frame_no));
|
1308 | 1324 | }
|
| 1325 | +#endif |
| 1326 | + virtual void do_oop(oop* p) { _oops->push(p); } |
| 1327 | + virtual void do_oop(narrowOop* p) { _narrow_oops->push(p); } |
| 1328 | + virtual void do_derived_oop(oop* base, derived_pointer* derived) { |
| 1329 | + ShouldNotReachHere(); // ??? |
| 1330 | + // _values.describe(_frame_no, (intptr_t*)derived, err_msg("derived pointer (base: " INTPTR_FORMAT ") for #%d", p2i(base), _frame_no)); |
| 1331 | + } |
1309 | 1332 | };
|
1310 | 1333 |
|
1311 | 1334 | class FrameValuesOopMapClosure: public OopMapClosure {
|
@@ -1369,7 +1392,7 @@ void frame::describe(FrameValues& values, int frame_no, const RegisterMap* reg_m
|
1369 | 1392 | values.describe(-1, info_address,
|
1370 | 1393 | FormatBuffer<1024>("#%d method %s @ %d", frame_no, m->name_and_sig_as_C_string(), bci), 3);
|
1371 | 1394 | if (desc != NULL) {
|
1372 |
| - values.describe(-1, info_address, err_msg("- %s codelet: %s", |
| 1395 | + values.describe(-1, info_address, err_msg("- %s codelet: %s", |
1373 | 1396 | desc->bytecode() >= 0 ? Bytecodes::name(desc->bytecode()) : "",
|
1374 | 1397 | desc->description() != NULL ? desc->description() : "?"), 2);
|
1375 | 1398 | }
|
@@ -1408,8 +1431,9 @@ void frame::describe(FrameValues& values, int frame_no, const RegisterMap* reg_m
|
1408 | 1431 | }
|
1409 | 1432 |
|
1410 | 1433 | if (reg_map != NULL) {
|
1411 |
| - FrameValuesOopClosure oopsFn(values, frame_no); |
| 1434 | + FrameValuesOopClosure oopsFn; |
1412 | 1435 | oops_do(&oopsFn, NULL, &oopsFn, reg_map);
|
| 1436 | + oopsFn.describe(values, frame_no); |
1413 | 1437 | }
|
1414 | 1438 | } else if (is_entry_frame()) {
|
1415 | 1439 | // For now just label the frame
|
@@ -1497,8 +1521,9 @@ void frame::describe(FrameValues& values, int frame_no, const RegisterMap* reg_m
|
1497 | 1521 | }
|
1498 | 1522 | }
|
1499 | 1523 |
|
1500 |
| - FrameValuesOopClosure oopsFn(values, frame_no); |
| 1524 | + FrameValuesOopClosure oopsFn; |
1501 | 1525 | oops_do(&oopsFn, NULL, &oopsFn, reg_map);
|
| 1526 | + oopsFn.describe(values, frame_no); |
1502 | 1527 |
|
1503 | 1528 | if (oop_map() != NULL) {
|
1504 | 1529 | FrameValuesOopMapClosure valuesFn(this, reg_map, values, frame_no);
|
@@ -1603,7 +1628,7 @@ void FrameValues::print_on(JavaThread* thread, outputStream* st) {
|
1603 | 1628 | while (!thread->is_in_full_stack((address)v1)) v1 = _values.at(--max_index).location;
|
1604 | 1629 | }
|
1605 | 1630 | }
|
1606 |
| - |
| 1631 | + |
1607 | 1632 | print_on(st, min_index, max_index, v0, v1);
|
1608 | 1633 | }
|
1609 | 1634 |
|
@@ -1641,7 +1666,7 @@ void FrameValues::print_on(outputStream* st, int min_index, int max_index, intpt
|
1641 | 1666 | st->print_cr(" %s %s %s", spacer, spacer, fv.description);
|
1642 | 1667 | } else {
|
1643 | 1668 | if (relative
|
1644 |
| - && *fv.location != 0 && *fv.location > -100 && *fv.location < 100 |
| 1669 | + && *fv.location != 0 && *fv.location > -100 && *fv.location < 100 |
1645 | 1670 | && (strncmp(fv.description, "interpreter_frame_", 18) == 0 || strstr(fv.description, " method "))) {
|
1646 | 1671 | st->print_cr(" " INTPTR_FORMAT ": %18d %s", p2i(fv.location), (int)*fv.location, fv.description);
|
1647 | 1672 | } else {
|
|
0 commit comments