Skip to content

Commit d32f99e

Browse files
committedFeb 2, 2022
8279219: [REDO] C2 crash when allocating array of size too large
Reviewed-by: thartmann, neliasso
1 parent 85d839f commit d32f99e

11 files changed

+254
-76
lines changed
 

‎src/hotspot/share/opto/callnode.cpp

+1-48
Original file line numberDiff line numberDiff line change
@@ -1656,6 +1656,7 @@ AllocateNode::AllocateNode(Compile* C, const TypeFunc *atype,
16561656
init_req( KlassNode , klass_node);
16571657
init_req( InitialTest , initial_test);
16581658
init_req( ALength , topnode);
1659+
init_req( ValidLengthTest , topnode);
16591660
C->add_macro_node(this);
16601661
}
16611662

@@ -1682,54 +1683,6 @@ Node *AllocateNode::make_ideal_mark(PhaseGVN *phase, Node* obj, Node* control, N
16821683
return mark_node;
16831684
}
16841685

1685-
//=============================================================================
1686-
Node* AllocateArrayNode::Ideal(PhaseGVN *phase, bool can_reshape) {
1687-
if (remove_dead_region(phase, can_reshape)) return this;
1688-
// Don't bother trying to transform a dead node
1689-
if (in(0) && in(0)->is_top()) return NULL;
1690-
1691-
const Type* type = phase->type(Ideal_length());
1692-
if (type->isa_int() && type->is_int()->_hi < 0) {
1693-
if (can_reshape) {
1694-
PhaseIterGVN *igvn = phase->is_IterGVN();
1695-
// Unreachable fall through path (negative array length),
1696-
// the allocation can only throw so disconnect it.
1697-
Node* proj = proj_out_or_null(TypeFunc::Control);
1698-
Node* catchproj = NULL;
1699-
if (proj != NULL) {
1700-
for (DUIterator_Fast imax, i = proj->fast_outs(imax); i < imax; i++) {
1701-
Node *cn = proj->fast_out(i);
1702-
if (cn->is_Catch()) {
1703-
catchproj = cn->as_Multi()->proj_out_or_null(CatchProjNode::fall_through_index);
1704-
break;
1705-
}
1706-
}
1707-
}
1708-
if (catchproj != NULL && catchproj->outcnt() > 0 &&
1709-
(catchproj->outcnt() > 1 ||
1710-
catchproj->unique_out()->Opcode() != Op_Halt)) {
1711-
assert(catchproj->is_CatchProj(), "must be a CatchProjNode");
1712-
Node* nproj = catchproj->clone();
1713-
igvn->register_new_node_with_optimizer(nproj);
1714-
1715-
Node *frame = new ParmNode( phase->C->start(), TypeFunc::FramePtr );
1716-
frame = phase->transform(frame);
1717-
// Halt & Catch Fire
1718-
Node* halt = new HaltNode(nproj, frame, "unexpected negative array length");
1719-
phase->C->root()->add_req(halt);
1720-
phase->transform(halt);
1721-
1722-
igvn->replace_node(catchproj, phase->C->top());
1723-
return this;
1724-
}
1725-
} else {
1726-
// Can't correct it during regular GVN so register for IGVN
1727-
phase->C->record_for_igvn(this);
1728-
}
1729-
}
1730-
return NULL;
1731-
}
1732-
17331686
// Retrieve the length from the AllocateArrayNode. Narrow the type with a
17341687
// CastII, if appropriate. If we are not allowed to create new nodes, and
17351688
// a CastII is appropriate, return NULL.

‎src/hotspot/share/opto/callnode.hpp

+5-5
Original file line numberDiff line numberDiff line change
@@ -913,6 +913,7 @@ class AllocateNode : public CallNode {
913913
KlassNode, // type (maybe dynamic) of the obj.
914914
InitialTest, // slow-path test (may be constant)
915915
ALength, // array length (or TOP if none)
916+
ValidLengthTest,
916917
ParmLimit
917918
};
918919

@@ -922,6 +923,7 @@ class AllocateNode : public CallNode {
922923
fields[KlassNode] = TypeInstPtr::NOTNULL;
923924
fields[InitialTest] = TypeInt::BOOL;
924925
fields[ALength] = t; // length (can be a bad length)
926+
fields[ValidLengthTest] = TypeInt::BOOL;
925927

926928
const TypeTuple *domain = TypeTuple::make(ParmLimit, fields);
927929

@@ -1016,18 +1018,16 @@ class AllocateNode : public CallNode {
10161018
//
10171019
class AllocateArrayNode : public AllocateNode {
10181020
public:
1019-
AllocateArrayNode(Compile* C, const TypeFunc *atype, Node *ctrl, Node *mem, Node *abio,
1020-
Node* size, Node* klass_node, Node* initial_test,
1021-
Node* count_val
1022-
)
1021+
AllocateArrayNode(Compile* C, const TypeFunc* atype, Node* ctrl, Node* mem, Node* abio, Node* size, Node* klass_node,
1022+
Node* initial_test, Node* count_val, Node* valid_length_test)
10231023
: AllocateNode(C, atype, ctrl, mem, abio, size, klass_node,
10241024
initial_test)
10251025
{
10261026
init_class_id(Class_AllocateArray);
10271027
set_req(AllocateNode::ALength, count_val);
1028+
set_req(AllocateNode::ValidLengthTest, valid_length_test);
10281029
}
10291030
virtual int Opcode() const;
1030-
virtual Node *Ideal(PhaseGVN *phase, bool can_reshape);
10311031

10321032
// Dig the length operand out of a array allocation site.
10331033
Node* Ideal_length() {

‎src/hotspot/share/opto/cfgnode.cpp

+11
Original file line numberDiff line numberDiff line change
@@ -2687,6 +2687,17 @@ const Type* CatchNode::Value(PhaseGVN* phase) const {
26872687
// Rethrows always throw exceptions, never return
26882688
if (call->entry_point() == OptoRuntime::rethrow_stub()) {
26892689
f[CatchProjNode::fall_through_index] = Type::TOP;
2690+
} else if (call->is_AllocateArray()) {
2691+
Node* klass_node = call->in(AllocateNode::KlassNode);
2692+
Node* length = call->in(AllocateNode::ALength);
2693+
const Type* length_type = phase->type(length);
2694+
const Type* klass_type = phase->type(klass_node);
2695+
Node* valid_length_test = call->in(AllocateNode::ValidLengthTest);
2696+
const Type* valid_length_test_t = phase->type(valid_length_test);
2697+
if (length_type == Type::TOP || klass_type == Type::TOP || valid_length_test_t == Type::TOP ||
2698+
valid_length_test_t->is_int()->is_con(0)) {
2699+
f[CatchProjNode::fall_through_index] = Type::TOP;
2700+
}
26902701
} else if( call->req() > TypeFunc::Parms ) {
26912702
const Type *arg0 = phase->type( call->in(TypeFunc::Parms) );
26922703
// Check for null receiver to virtual or interface calls

‎src/hotspot/share/opto/compile.cpp

+20-9
Original file line numberDiff line numberDiff line change
@@ -3812,7 +3812,7 @@ bool Compile::final_graph_reshaping() {
38123812
// 'fall-thru' path, so expected kids is 1 less.
38133813
if (n->is_PCTable() && n->in(0) && n->in(0)->in(0)) {
38143814
if (n->in(0)->in(0)->is_Call()) {
3815-
CallNode *call = n->in(0)->in(0)->as_Call();
3815+
CallNode* call = n->in(0)->in(0)->as_Call();
38163816
if (call->entry_point() == OptoRuntime::rethrow_stub()) {
38173817
required_outcnt--; // Rethrow always has 1 less kid
38183818
} else if (call->req() > TypeFunc::Parms &&
@@ -3821,22 +3821,25 @@ bool Compile::final_graph_reshaping() {
38213821
// detected that the virtual call will always result in a null
38223822
// pointer exception. The fall-through projection of this CatchNode
38233823
// will not be populated.
3824-
Node *arg0 = call->in(TypeFunc::Parms);
3824+
Node* arg0 = call->in(TypeFunc::Parms);
38253825
if (arg0->is_Type() &&
38263826
arg0->as_Type()->type()->higher_equal(TypePtr::NULL_PTR)) {
38273827
required_outcnt--;
38283828
}
3829-
} else if (call->entry_point() == OptoRuntime::new_array_Java() &&
3830-
call->req() > TypeFunc::Parms+1 &&
3831-
call->is_CallStaticJava()) {
3832-
// Check for negative array length. In such case, the optimizer has
3829+
} else if (call->entry_point() == OptoRuntime::new_array_Java() ||
3830+
call->entry_point() == OptoRuntime::new_array_nozero_Java()) {
3831+
// Check for illegal array length. In such case, the optimizer has
38333832
// detected that the allocation attempt will always result in an
38343833
// exception. There is no fall-through projection of this CatchNode .
3835-
Node *arg1 = call->in(TypeFunc::Parms+1);
3836-
if (arg1->is_Type() &&
3837-
arg1->as_Type()->type()->join(TypeInt::POS)->empty()) {
3834+
assert(call->is_CallStaticJava(), "static call expected");
3835+
assert(call->req() == call->jvms()->endoff() + 1, "missing extra input");
3836+
Node* valid_length_test = call->in(call->req()-1);
3837+
call->del_req(call->req()-1);
3838+
if (valid_length_test->find_int_con(1) == 0) {
38383839
required_outcnt--;
38393840
}
3841+
assert(n->outcnt() == required_outcnt, "malformed control flow");
3842+
continue;
38403843
}
38413844
}
38423845
}
@@ -3845,6 +3848,14 @@ bool Compile::final_graph_reshaping() {
38453848
record_method_not_compilable("malformed control flow");
38463849
return true; // Not all targets reachable!
38473850
}
3851+
} else if (n->is_PCTable() && n->in(0) && n->in(0)->in(0) && n->in(0)->in(0)->is_Call()) {
3852+
CallNode* call = n->in(0)->in(0)->as_Call();
3853+
if (call->entry_point() == OptoRuntime::new_array_Java() ||
3854+
call->entry_point() == OptoRuntime::new_array_nozero_Java()) {
3855+
assert(call->is_CallStaticJava(), "static call expected");
3856+
assert(call->req() == call->jvms()->endoff() + 1, "missing extra input");
3857+
call->del_req(call->req()-1); // valid length test useless now
3858+
}
38483859
}
38493860
// Check that I actually visited all kids. Unreached kids
38503861
// must be infinite loops.

‎src/hotspot/share/opto/graphKit.cpp

+13-3
Original file line numberDiff line numberDiff line change
@@ -2728,7 +2728,9 @@ void GraphKit::make_slow_call_ex(Node* call, ciInstanceKlass* ex_klass, bool sep
27282728
// Make a catch node with just two handlers: fall-through and catch-all
27292729
Node* i_o = _gvn.transform( new ProjNode(call, TypeFunc::I_O, separate_io_proj) );
27302730
Node* catc = _gvn.transform( new CatchNode(control(), i_o, 2) );
2731-
Node* norm = _gvn.transform( new CatchProjNode(catc, CatchProjNode::fall_through_index, CatchProjNode::no_handler_bci) );
2731+
Node* norm = new CatchProjNode(catc, CatchProjNode::fall_through_index, CatchProjNode::no_handler_bci);
2732+
_gvn.set_type_bottom(norm);
2733+
C->record_for_igvn(norm);
27322734
Node* excp = _gvn.transform( new CatchProjNode(catc, CatchProjNode::catch_all_index, CatchProjNode::no_handler_bci) );
27332735

27342736
{ PreserveJVMState pjvms(this);
@@ -3969,20 +3971,28 @@ Node* GraphKit::new_array(Node* klass_node, // array klass (maybe variable)
39693971
initial_slow_test = initial_slow_test->as_Bool()->as_int_value(&_gvn);
39703972
}
39713973

3974+
const TypeOopPtr* ary_type = _gvn.type(klass_node)->is_klassptr()->as_instance_type();
3975+
Node* valid_length_test = _gvn.intcon(1);
3976+
if (ary_type->klass()->is_array_klass()) {
3977+
BasicType bt = ary_type->klass()->as_array_klass()->element_type()->basic_type();
3978+
jint max = TypeAryPtr::max_array_length(bt);
3979+
Node* valid_length_cmp = _gvn.transform(new CmpUNode(length, intcon(max)));
3980+
valid_length_test = _gvn.transform(new BoolNode(valid_length_cmp, BoolTest::le));
3981+
}
3982+
39723983
// Create the AllocateArrayNode and its result projections
39733984
AllocateArrayNode* alloc
39743985
= new AllocateArrayNode(C, AllocateArrayNode::alloc_type(TypeInt::INT),
39753986
control(), mem, i_o(),
39763987
size, klass_node,
39773988
initial_slow_test,
3978-
length);
3989+
length, valid_length_test);
39793990

39803991
// Cast to correct type. Note that the klass_node may be constant or not,
39813992
// and in the latter case the actual array type will be inexact also.
39823993
// (This happens via a non-constant argument to inline_native_newArray.)
39833994
// In any case, the value of klass_node provides the desired array type.
39843995
const TypeInt* length_type = _gvn.find_int_type(length);
3985-
const TypeOopPtr* ary_type = _gvn.type(klass_node)->is_klassptr()->as_instance_type();
39863996
if (ary_type->isa_aryptr() && length_type != NULL) {
39873997
// Try to get a better type than POS for the size
39883998
ary_type = ary_type->is_aryptr()->cast_to_size(length_type);

‎src/hotspot/share/opto/macro.cpp

+11-3
Original file line numberDiff line numberDiff line change
@@ -1208,7 +1208,8 @@ void PhaseMacroExpand::expand_allocate_common(
12081208
AllocateNode* alloc, // allocation node to be expanded
12091209
Node* length, // array length for an array allocation
12101210
const TypeFunc* slow_call_type, // Type of slow call
1211-
address slow_call_address // Address of slow call
1211+
address slow_call_address, // Address of slow call
1212+
Node* valid_length_test // whether length is valid or not
12121213
)
12131214
{
12141215
Node* ctrl = alloc->in(TypeFunc::Control);
@@ -1393,6 +1394,12 @@ void PhaseMacroExpand::expand_allocate_common(
13931394
// Copy debug information and adjust JVMState information, then replace
13941395
// allocate node with the call
13951396
call->copy_call_debug_info(&_igvn, alloc);
1397+
// For array allocations, copy the valid length check to the call node so Compile::final_graph_reshaping() can verify
1398+
// that the call has the expected number of CatchProj nodes (in case the allocation always fails and the fallthrough
1399+
// path dies).
1400+
if (valid_length_test != NULL) {
1401+
call->add_req(valid_length_test);
1402+
}
13961403
if (expand_fast_path) {
13971404
call->set_cnt(PROB_UNLIKELY_MAG(4)); // Same effect as RC_UNCOMMON.
13981405
} else {
@@ -1875,11 +1882,12 @@ Node* PhaseMacroExpand::prefetch_allocation(Node* i_o, Node*& needgc_false,
18751882
void PhaseMacroExpand::expand_allocate(AllocateNode *alloc) {
18761883
expand_allocate_common(alloc, NULL,
18771884
OptoRuntime::new_instance_Type(),
1878-
OptoRuntime::new_instance_Java());
1885+
OptoRuntime::new_instance_Java(), NULL);
18791886
}
18801887

18811888
void PhaseMacroExpand::expand_allocate_array(AllocateArrayNode *alloc) {
18821889
Node* length = alloc->in(AllocateNode::ALength);
1890+
Node* valid_length_test = alloc->in(AllocateNode::ValidLengthTest);
18831891
InitializeNode* init = alloc->initialization();
18841892
Node* klass_node = alloc->in(AllocateNode::KlassNode);
18851893
ciKlass* k = _igvn.type(klass_node)->is_klassptr()->klass();
@@ -1894,7 +1902,7 @@ void PhaseMacroExpand::expand_allocate_array(AllocateArrayNode *alloc) {
18941902
}
18951903
expand_allocate_common(alloc, length,
18961904
OptoRuntime::new_array_Type(),
1897-
slow_call_address);
1905+
slow_call_address, valid_length_test);
18981906
}
18991907

19001908
//-------------------mark_eliminated_box----------------------------------

‎src/hotspot/share/opto/macro.hpp

+2-2
Original file line numberDiff line numberDiff line change
@@ -92,8 +92,8 @@ class PhaseMacroExpand : public Phase {
9292
void expand_allocate_common(AllocateNode* alloc,
9393
Node* length,
9494
const TypeFunc* slow_call_type,
95-
address slow_call_address);
96-
void yank_initalize_node(InitializeNode* node);
95+
address slow_call_address,
96+
Node* valid_length_test);
9797
void yank_alloc_node(AllocateNode* alloc);
9898
Node *value_from_mem(Node *mem, Node *ctl, BasicType ft, const Type *ftype, const TypeOopPtr *adr_t, AllocateNode *alloc);
9999
Node *value_from_mem_phi(Node *mem, BasicType ft, const Type *ftype, const TypeOopPtr *adr_t, AllocateNode *alloc, Node_Stack *value_phis, int level);

‎src/hotspot/share/opto/split_if.cpp

+7-6
Original file line numberDiff line numberDiff line change
@@ -128,8 +128,8 @@ bool PhaseIdealLoop::split_up( Node *n, Node *blk1, Node *blk2 ) {
128128
}
129129
} else {
130130
// We might see an Opaque1 from a loop limit check here
131-
assert(use->is_If() || use->is_CMove() || use->Opcode() == Op_Opaque1, "unexpected node type");
132-
Node *use_c = use->is_If() ? use->in(0) : get_ctrl(use);
131+
assert(use->is_If() || use->is_CMove() || use->Opcode() == Op_Opaque1 || use->is_AllocateArray(), "unexpected node type");
132+
Node *use_c = (use->is_If() || use->is_AllocateArray()) ? use->in(0) : get_ctrl(use);
133133
if (use_c == blk1 || use_c == blk2) {
134134
assert(use->is_CMove(), "unexpected node type");
135135
continue;
@@ -166,14 +166,15 @@ bool PhaseIdealLoop::split_up( Node *n, Node *blk1, Node *blk2 ) {
166166
--j;
167167
} else {
168168
// We might see an Opaque1 from a loop limit check here
169-
assert(u->is_If() || u->is_CMove() || u->Opcode() == Op_Opaque1, "unexpected node type");
170-
assert(u->in(1) == bol, "");
169+
assert(u->is_If() || u->is_CMove() || u->Opcode() == Op_Opaque1 || u->is_AllocateArray(), "unexpected node type");
170+
assert(u->is_AllocateArray() || u->in(1) == bol, "");
171+
assert(!u->is_AllocateArray() || u->in(AllocateNode::ValidLengthTest) == bol, "wrong input to AllocateArray");
171172
// Get control block of either the CMove or the If input
172-
Node *u_ctrl = u->is_If() ? u->in(0) : get_ctrl(u);
173+
Node *u_ctrl = (u->is_If() || u->is_AllocateArray()) ? u->in(0) : get_ctrl(u);
173174
assert((u_ctrl != blk1 && u_ctrl != blk2) || u->is_CMove(), "won't converge");
174175
Node *x = bol->clone();
175176
register_new_node(x, u_ctrl);
176-
_igvn.replace_input_of(u, 1, x);
177+
_igvn.replace_input_of(u, u->is_AllocateArray() ? AllocateNode::ValidLengthTest : 1, x);
177178
--j;
178179
}
179180
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,52 @@
1+
/*
2+
* Copyright (c) 2022, Red Hat, Inc. All rights reserved.
3+
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4+
*
5+
* This code is free software; you can redistribute it and/or modify it
6+
* under the terms of the GNU General Public License version 2 only, as
7+
* published by the Free Software Foundation.
8+
*
9+
* This code is distributed in the hope that it will be useful, but WITHOUT
10+
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
11+
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
12+
* version 2 for more details (a copy is included in the LICENSE file that
13+
* accompanied this code).
14+
*
15+
* You should have received a copy of the GNU General Public License version
16+
* 2 along with this work; if not, write to the Free Software Foundation,
17+
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
18+
*
19+
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
20+
* or visit www.oracle.com if you need additional information or have any
21+
* questions.
22+
*/
23+
24+
/**
25+
* @test
26+
* @bug 8279125
27+
* @summary fatal error: no reachable node should have no use
28+
* @requires vm.flavor == "server"
29+
*
30+
* @run main/othervm -XX:-BackgroundCompilation -XX:-DoEscapeAnalysis TestAllocArrayAfterAllocNoUse
31+
*
32+
*/
33+
34+
public class TestAllocArrayAfterAllocNoUse {
35+
private static Object field;
36+
37+
public static void main(String[] args) {
38+
for (int i = 0; i < 20_000; i++) {
39+
test();
40+
}
41+
}
42+
43+
private static void test() {
44+
try {
45+
final TestAllocArrayAfterAllocNoUse o = new TestAllocArrayAfterAllocNoUse();
46+
} catch (Exception e) {
47+
final int[] array = new int[100];
48+
field = array;
49+
}
50+
51+
}
52+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,53 @@
1+
/*
2+
* Copyright (c) 2022, Red Hat, Inc. All rights reserved.
3+
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4+
*
5+
* This code is free software; you can redistribute it and/or modify it
6+
* under the terms of the GNU General Public License version 2 only, as
7+
* published by the Free Software Foundation.
8+
*
9+
* This code is distributed in the hope that it will be useful, but WITHOUT
10+
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
11+
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
12+
* version 2 for more details (a copy is included in the LICENSE file that
13+
* accompanied this code).
14+
*
15+
* You should have received a copy of the GNU General Public License version
16+
* 2 along with this work; if not, write to the Free Software Foundation,
17+
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
18+
*
19+
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
20+
* or visit www.oracle.com if you need additional information or have any
21+
* questions.
22+
*/
23+
24+
/**
25+
* @test
26+
* @bug 8279062
27+
* @summary C2: assert(t->meet(t0) == t) failed: Not monotonic after JDK-8278413
28+
*
29+
* @run main/othervm -XX:-BackgroundCompilation TestCCPAllocateArray
30+
*
31+
*/
32+
33+
public class TestCCPAllocateArray {
34+
public static void main(String[] args) {
35+
for (int i = 0; i < 20_000; i++) {
36+
try {
37+
test();
38+
} catch (OutOfMemoryError e) {
39+
}
40+
length(42);
41+
}
42+
}
43+
44+
private static int[] test() {
45+
int i = 2;
46+
for (; i < 4; i *= 2);
47+
return new int[length(i)];
48+
}
49+
50+
private static int length(int i) {
51+
return i == 4 ? Integer.MAX_VALUE : 0;
52+
}
53+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,79 @@
1+
/*
2+
* Copyright (c) 2022, Red Hat, Inc. All rights reserved.
3+
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4+
*
5+
* This code is free software; you can redistribute it and/or modify it
6+
* under the terms of the GNU General Public License version 2 only, as
7+
* published by the Free Software Foundation.
8+
*
9+
* This code is distributed in the hope that it will be useful, but WITHOUT
10+
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
11+
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
12+
* version 2 for more details (a copy is included in the LICENSE file that
13+
* accompanied this code).
14+
*
15+
* You should have received a copy of the GNU General Public License version
16+
* 2 along with this work; if not, write to the Free Software Foundation,
17+
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
18+
*
19+
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
20+
* or visit www.oracle.com if you need additional information or have any
21+
* questions.
22+
*/
23+
24+
/*
25+
* @test
26+
* bug 8279219
27+
* @summary C2 crash when allocating array of size too large
28+
* @library /test/lib /
29+
* @build sun.hotspot.WhiteBox
30+
* @run driver jdk.test.lib.helpers.ClassFileInstaller sun.hotspot.WhiteBox
31+
* @run main/othervm -ea -Xbootclasspath/a:. -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -XX:-BackgroundCompilation TestFailedAllocationBadGraph
32+
*/
33+
34+
import sun.hotspot.WhiteBox;
35+
import java.lang.reflect.Method;
36+
import compiler.whitebox.CompilerWhiteBoxTest;
37+
38+
public class TestFailedAllocationBadGraph {
39+
private static final WhiteBox WHITE_BOX = WhiteBox.getWhiteBox();
40+
41+
private static long[] array;
42+
private static int field;
43+
private static volatile int barrier;
44+
45+
public static void main(String[] args) throws Exception {
46+
run("test1");
47+
run("test2");
48+
}
49+
50+
private static void run(String method) throws Exception {
51+
Method m = TestFailedAllocationBadGraph.class.getDeclaredMethod(method);
52+
WHITE_BOX.enqueueMethodForCompilation(m, CompilerWhiteBoxTest.COMP_LEVEL_FULL_OPTIMIZATION);
53+
if (!WHITE_BOX.isMethodCompiled(m) || WHITE_BOX.getMethodCompilationLevel(m) != CompilerWhiteBoxTest.COMP_LEVEL_FULL_OPTIMIZATION) {
54+
throw new RuntimeException("should still be compiled");
55+
}
56+
}
57+
58+
private static int test1() {
59+
int length = Integer.MAX_VALUE;
60+
try {
61+
array = new long[length];
62+
} catch (OutOfMemoryError outOfMemoryError) {
63+
barrier = 0x42;
64+
length = field;
65+
}
66+
return length;
67+
}
68+
69+
private static int test2() {
70+
int length = -1;
71+
try {
72+
array = new long[length];
73+
} catch (OutOfMemoryError outOfMemoryError) {
74+
barrier = 0x42;
75+
length = field;
76+
}
77+
return length;
78+
}
79+
}

0 commit comments

Comments
 (0)
Please sign in to comment.