|
1 | 1 | /*
|
2 |
| - * Copyright (c) 2000, 2020, Oracle and/or its affiliates. All rights reserved. |
| 2 | + * Copyright (c) 2000, 2021, 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
|
@@ -554,6 +554,57 @@ void LateInlineVirtualCallGenerator::do_late_inline() {
|
554 | 554 | CallGenerator::do_late_inline_helper();
|
555 | 555 | }
|
556 | 556 |
|
| 557 | +static bool has_non_debug_usages(Node* n) { |
| 558 | + for (DUIterator_Fast imax, i = n->fast_outs(imax); i < imax; i++) { |
| 559 | + Node* m = n->fast_out(i); |
| 560 | + if (!m->is_SafePoint() |
| 561 | + || (m->is_Call() && m->as_Call()->has_non_debug_use(n))) { |
| 562 | + return true; |
| 563 | + } |
| 564 | + } |
| 565 | + return false; |
| 566 | +} |
| 567 | + |
| 568 | +// delay box in runtime, treat box as a scalarized object |
| 569 | +static void scalarize_debug_usages(CallNode* call, Node* resproj) { |
| 570 | + GraphKit kit(call->jvms()); |
| 571 | + PhaseGVN& gvn = kit.gvn(); |
| 572 | + |
| 573 | + ProjNode* res = resproj->as_Proj(); |
| 574 | + ciInstanceKlass* klass = call->as_CallStaticJava()->method()->holder(); |
| 575 | + int n_fields = klass->nof_nonstatic_fields(); |
| 576 | + assert(n_fields == 1, "the klass must be an auto-boxing klass"); |
| 577 | + |
| 578 | + for (DUIterator_Last imin, i = res->last_outs(imin); i >= imin;) { |
| 579 | + SafePointNode* sfpt = res->last_out(i)->as_SafePoint(); |
| 580 | + uint first_ind = sfpt->req() - sfpt->jvms()->scloff(); |
| 581 | + Node* sobj = new SafePointScalarObjectNode(gvn.type(res)->isa_oopptr(), |
| 582 | +#ifdef ASSERT |
| 583 | + call, |
| 584 | +#endif // ASSERT |
| 585 | + first_ind, n_fields, true); |
| 586 | + sobj->init_req(0, kit.root()); |
| 587 | + sfpt->add_req(call->in(TypeFunc::Parms)); |
| 588 | + sobj = gvn.transform(sobj); |
| 589 | + JVMState* jvms = sfpt->jvms(); |
| 590 | + jvms->set_endoff(sfpt->req()); |
| 591 | + int start = jvms->debug_start(); |
| 592 | + int end = jvms->debug_end(); |
| 593 | + int num_edges = sfpt->replace_edges_in_range(res, sobj, start, end, &gvn); |
| 594 | + i -= num_edges; |
| 595 | + } |
| 596 | + |
| 597 | + assert(res->outcnt() == 0, "the box must have no use after replace"); |
| 598 | + |
| 599 | +#ifndef PRODUCT |
| 600 | + if (PrintEliminateAllocations) { |
| 601 | + tty->print("++++ Eliminated: %d ", call->_idx); |
| 602 | + call->as_CallStaticJava()->method()->print_short_name(tty); |
| 603 | + tty->cr(); |
| 604 | + } |
| 605 | +#endif |
| 606 | +} |
| 607 | + |
557 | 608 | void CallGenerator::do_late_inline_helper() {
|
558 | 609 | assert(is_late_inline(), "only late inline allowed");
|
559 | 610 |
|
@@ -603,10 +654,23 @@ void CallGenerator::do_late_inline_helper() {
|
603 | 654 | C->remove_macro_node(call);
|
604 | 655 | }
|
605 | 656 |
|
606 |
| - bool result_not_used = (callprojs.resproj == NULL || callprojs.resproj->outcnt() == 0); |
607 |
| - if (is_pure_call() && result_not_used) { |
| 657 | + bool result_not_used = false; |
| 658 | + |
| 659 | + if (is_pure_call()) { |
| 660 | + if (is_boxing_late_inline() && callprojs.resproj != nullptr) { |
| 661 | + // replace box node to scalar node only in case it is directly referenced by debug info |
| 662 | + assert(call->as_CallStaticJava()->is_boxing_method(), "sanity"); |
| 663 | + if (!has_non_debug_usages(callprojs.resproj)) { |
| 664 | + scalarize_debug_usages(call, callprojs.resproj); |
| 665 | + } |
| 666 | + } |
| 667 | + |
608 | 668 | // The call is marked as pure (no important side effects), but result isn't used.
|
609 | 669 | // It's safe to remove the call.
|
| 670 | + result_not_used = (callprojs.resproj == NULL || callprojs.resproj->outcnt() == 0); |
| 671 | + } |
| 672 | + |
| 673 | + if (result_not_used) { |
610 | 674 | GraphKit kit(call->jvms());
|
611 | 675 | kit.replace_call(call, C->top(), true);
|
612 | 676 | } else {
|
@@ -741,6 +805,8 @@ class LateInlineBoxingCallGenerator : public LateInlineCallGenerator {
|
741 | 805 | return new_jvms;
|
742 | 806 | }
|
743 | 807 |
|
| 808 | + virtual bool is_boxing_late_inline() const { return true; } |
| 809 | + |
744 | 810 | virtual CallGenerator* with_call_node(CallNode* call) {
|
745 | 811 | LateInlineBoxingCallGenerator* cg = new LateInlineBoxingCallGenerator(method(), _inline_cg);
|
746 | 812 | cg->set_call_node(call->as_CallStaticJava());
|
|
0 commit comments