8239367: RunThese30M.java failed due to "assert(false) failed: graph should be schedulable"

Reviewed-by: thartmann, vlivanov, neliasso
This commit is contained in:
Roland Westrelin 2020-02-21 15:01:22 +01:00
parent 89448a83d9
commit 1e796eae8f
3 changed files with 61 additions and 80 deletions
src/hotspot/share/opto
test/hotspot/jtreg/compiler/types

@ -1808,6 +1808,8 @@ Compile::AliasType* Compile::find_alias_type(const TypePtr* adr_type, bool no_cr
alias_type(idx)->set_rewritable(false);
if (flat->offset() == in_bytes(Klass::java_mirror_offset()))
alias_type(idx)->set_rewritable(false);
if (flat->offset() == in_bytes(Klass::secondary_super_cache_offset()))
alias_type(idx)->set_rewritable(false);
}
// %%% (We would like to finalize JavaThread::threadObj_offset(),
// but the base pointer type is not distinctive enough to identify

@ -2639,80 +2639,6 @@ static IfNode* gen_subtype_check_compare(Node* ctrl, Node* in1, Node* in2, BoolT
return iff;
}
// Find the memory state for the secondary super type cache load when
// a subtype check is expanded at macro expansion time. That field is
// mutable so should not use immutable memory but
// PartialSubtypeCheckNode that might modify it doesn't produce a new
// memory state so bottom memory is the most accurate memory state to
// hook the load with. This follows the implementation used when the
// subtype check is expanded at parse time.
static Node* find_bottom_mem(Node* ctrl, Compile* C) {
const TypePtr* adr_type = TypeKlassPtr::make(TypePtr::NotNull, C->env()->Object_klass(), Type::OffsetBot);
Node_Stack stack(0);
VectorSet seen(Thread::current()->resource_area());
Node* c = ctrl;
Node* mem = NULL;
uint iter = 0;
do {
iter++;
assert(iter < C->live_nodes(), "infinite loop");
if (c->is_Region()) {
for (DUIterator_Fast imax, i = c->fast_outs(imax); i < imax && mem == NULL; i++) {
Node* u = c->fast_out(i);
if (u->is_Phi() && u->bottom_type() == Type::MEMORY &&
(u->adr_type() == TypePtr::BOTTOM || u->adr_type() == adr_type)) {
mem = u;
}
}
if (mem == NULL) {
if (!seen.test_set(c->_idx)) {
stack.push(c, 2);
c = c->in(1);
} else {
Node* phi = NULL;
uint idx = 0;
for (;;) {
phi = stack.node();
idx = stack.index();
if (idx < phi->req()) {
break;
}
stack.pop();
}
c = phi->in(idx);
stack.set_index(idx+1);
}
}
} else if (c->is_Proj() && c->in(0)->adr_type() == TypePtr::BOTTOM) {
for (DUIterator_Fast imax, i = c->in(0)->fast_outs(imax); i < imax; i++) {
Node* u = c->in(0)->fast_out(i);
if (u->bottom_type() == Type::MEMORY && u->as_Proj()->_is_io_use == c->as_Proj()->_is_io_use) {
assert(mem == NULL, "");
mem = u;
}
}
} else if (c->is_CatchProj() && c->in(0)->in(0)->in(0)->adr_type() == TypePtr::BOTTOM) {
Node* call = c->in(0)->in(0)->in(0);
assert(call->is_Call(), "CatchProj with no call?");
CallProjections projs;
call->as_Call()->extract_projections(&projs, false, false);
if (projs.catchall_memproj == NULL) {
mem = projs.fallthrough_memproj;
} else if (c == projs.fallthrough_catchproj) {
mem = projs.fallthrough_memproj;
} else {
assert(c == projs.catchall_catchproj, "strange control");
mem = projs.catchall_memproj;
}
} else {
assert(!c->is_Start(), "should stop before start");
c = c->in(0);
}
} while (mem == NULL);
return mem;
}
//-------------------------------gen_subtype_check-----------------------------
// Generate a subtyping check. Takes as input the subtype and supertype.
// Returns 2 values: sets the default control() to the true path and returns
@ -2795,12 +2721,12 @@ Node* Phase::gen_subtype_check(Node* subklass, Node* superklass, Node** ctrl, No
// cache which is mutable so can't use immutable memory. Other
// types load from the super-class display table which is immutable.
Node *kmem = C->immutable_memory();
if (might_be_cache) {
assert((C->get_alias_index(TypeKlassPtr::make(TypePtr::NotNull, C->env()->Object_klass(), Type::OffsetBot)) ==
C->get_alias_index(gvn.type(p2)->is_ptr())), "");
if (mem == NULL) {
mem = find_bottom_mem(*ctrl, C);
}
// secondary_super_cache is not immutable but can be treated as such because:
// - no ideal node writes to it in a way that could cause an
// incorrect/missed optimization of the following Load.
// - it's a cache so, worse case, not reading the latest value
// wouldn't cause incorrect execution
if (might_be_cache && mem != NULL) {
kmem = mem->is_MergeMem() ? mem->as_MergeMem()->memory_at(C->get_alias_index(gvn.type(p2)->is_ptr())) : mem;
}
Node *nkls = gvn.transform(LoadKlassNode::make(gvn, NULL, kmem, p2, gvn.type(p2)->is_ptr(), TypeKlassPtr::OBJECT_OR_NULL));

@ -0,0 +1,53 @@
/*
* Copyright (c) 2020, Red Hat, Inc. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
*
* This code is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License version 2 only, as
* published by the Free Software Foundation.
*
* This code is distributed in the hope that it will be useful, but WITHOUT
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
* version 2 for more details (a copy is included in the LICENSE file that
* accompanied this code).
*
* You should have received a copy of the GNU General Public License version
* 2 along with this work; if not, write to the Free Software Foundation,
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
*
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
* or visit www.oracle.com if you need additional information or have any
* questions.
*/
/**
* @test
* @bug 8239367
* @summary Wiring of memory in SubTypeCheck macro node causes graph should be schedulable
*
* @run main/othervm -Xcomp -XX:CompileOnly=TestSubTypeCheckMacroNodeWrongMem::test -XX:-DoEscapeAnalysis TestSubTypeCheckMacroNodeWrongMem
*
*/
public class TestSubTypeCheckMacroNodeWrongMem {
private static int stop = 100;
public static void main(String[] args) {
TestSubTypeCheckMacroNodeWrongMem o = new TestSubTypeCheckMacroNodeWrongMem();
test();
}
private static void test() {
Object o1 = null;
for (int i = 0; i < stop; i++) {
try {
Object o = new TestSubTypeCheckMacroNodeWrongMem();
o1.equals(o);
} catch (NullPointerException npe) {
} catch (Exception e) {
throw new RuntimeException();
}
}
}
}