From 5dbfae0176dd675c0b73704eb9c3db4c8288ea7f Mon Sep 17 00:00:00 2001 From: Christian Hagedorn Date: Tue, 17 Nov 2020 09:16:06 +0000 Subject: [PATCH] 8255058: C1: assert(is_virtual()) failed: type check Reviewed-by: neliasso, kvn --- src/hotspot/share/c1/c1_LinearScan.cpp | 5 +- .../c1/TestPinnedConstantExceptionEdge.java | 71 +++++++++++++++++++ 2 files changed, 73 insertions(+), 3 deletions(-) create mode 100644 test/hotspot/jtreg/compiler/c1/TestPinnedConstantExceptionEdge.java diff --git a/src/hotspot/share/c1/c1_LinearScan.cpp b/src/hotspot/share/c1/c1_LinearScan.cpp index 782fa7ada03..1d49366cc1a 100644 --- a/src/hotspot/share/c1/c1_LinearScan.cpp +++ b/src/hotspot/share/c1/c1_LinearScan.cpp @@ -1942,15 +1942,14 @@ void LinearScan::resolve_exception_edge(XHandler* handler, int throwing_op_id, i move_resolver.set_multiple_reads_allowed(); Constant* con = from_value->as_Constant(); - if (con != NULL && !con->is_pinned()) { - // unpinned constants may have no register, so add mapping from constant to interval + if (con != NULL && (!con->is_pinned() || con->operand()->is_constant())) { + // Need a mapping from constant to interval if unpinned (may have no register) or if the operand is a constant (no register). move_resolver.add_mapping(LIR_OprFact::value_type(con->type()), to_interval); } else { // search split child at the throwing op_id Interval* from_interval = interval_at_op_id(from_value->operand()->vreg_number(), throwing_op_id); move_resolver.add_mapping(from_interval, to_interval); } - } else { // no phi function, so use reg_num also for from_interval // search split child at the throwing op_id diff --git a/test/hotspot/jtreg/compiler/c1/TestPinnedConstantExceptionEdge.java b/test/hotspot/jtreg/compiler/c1/TestPinnedConstantExceptionEdge.java new file mode 100644 index 00000000000..18f145eb7fa --- /dev/null +++ b/test/hotspot/jtreg/compiler/c1/TestPinnedConstantExceptionEdge.java @@ -0,0 +1,71 @@ +/* + * Copyright (c) 2020, Oracle and/or its affiliates. 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 8255058 + * @summary Add check in LinearScan::resolve_exception_edge for pinned constant that is not virtual which cannot be used to find an interval which + resulted in an assertion error. + * @run main/othervm -Xcomp -XX:TieredStopAtLevel=1 -XX:CompileCommand=dontinline,compiler.c1.TestPinnedConstantExceptionEdge::dontInline + * -XX:CompileCommand=compileonly,compiler.c1.TestPinnedConstantExceptionEdge::* compiler.c1.TestPinnedConstantExceptionEdge + */ +package compiler.c1; + +public class TestPinnedConstantExceptionEdge { + + public static long iFld = 0; + public static boolean b1; + public static boolean b2; + + public static void test() { + int x = 5; + int y = 11; + for (int i = 1; i < 8; i++) { + for (int j = 1; j < 2; ++j) { + if (b1) { + try { + y = (x / x); + y = (500 / i); + y = (-214 / i); + } catch (ArithmeticException a_e) {} + // Recursion too deep in UseCountComputer::uses_do and therefore constant 1 is pinned. + iFld += (b1 ? 1 : 0) + (b2 ? 1 : 0) + 5 + 7 + 6 + 5 + y + + dontInline(7) + dontInline(5) + 8 + 8 + 9 + + dontInline(3) + dontInline(3) + dontInline(4) + + dontInline(dontInline(5)) + dontInline(2); + return; + } + } + } + } + + // Not inlined + public static int dontInline(int a) { + return 0; + } + + public static void main(String[] strArr) { + test(); + } +} +