From f03df8356b2cbb87ebbece3fca9afa68ca630de1 Mon Sep 17 00:00:00 2001 From: Leonid Mesnik Date: Thu, 13 Dec 2018 14:16:21 -0800 Subject: [PATCH 001/101] 8215369: Jcstress pollute /var/tmp with temporary files Reviewed-by: iignatyev, mseledtsov --- .../applications/jcstress/JcstressRunner.java | 18 +++++++++--------- 1 file changed, 9 insertions(+), 9 deletions(-) diff --git a/test/hotspot/jtreg/applications/jcstress/JcstressRunner.java b/test/hotspot/jtreg/applications/jcstress/JcstressRunner.java index eaebb4cbdd5..e5d3df3dc0f 100644 --- a/test/hotspot/jtreg/applications/jcstress/JcstressRunner.java +++ b/test/hotspot/jtreg/applications/jcstress/JcstressRunner.java @@ -89,6 +89,11 @@ public class JcstressRunner { private static String[] getCmd(String[] args) { List extraFlags = new ArrayList<>(); + // java.io.tmpdir is set for both harness and forked VM so temporary files + // created like this File.createTempFile("jcstress", "stdout"); + // don't pollute temporary directories + extraFlags.add("-Djava.io.tmpdir=" + System.getProperty("user.dir")); + // add jar with jcstress tests and harness to CP extraFlags.add("-cp"); extraFlags.add(System.getProperty("java.class.path") @@ -97,16 +102,11 @@ public class JcstressRunner { extraFlags.add(MAIN_CLASS); - String[] javaOpts = Utils.getTestJavaOpts(); - // disable flags auto-detection - if (0 == javaOpts.length) { + extraFlags.add("--jvmArgs"); + extraFlags.add("-Djava.io.tmpdir=" + System.getProperty("user.dir")); + for (String jvmArg : Utils.getTestJavaOpts()) { extraFlags.add("--jvmArgs"); - extraFlags.add(""); - } else { - for (String jvmArg : Utils.getTestJavaOpts()) { - extraFlags.add("--jvmArgs"); - extraFlags.add(jvmArg); - } + extraFlags.add(jvmArg); } String[] result = new String[extraFlags.size() + args.length]; From 165c644cf8219c5d27d7ae2befe42c4ab85c22f8 Mon Sep 17 00:00:00 2001 From: Erik Gahlin Date: Thu, 13 Dec 2018 23:25:00 +0100 Subject: [PATCH 002/101] 8215362: JFR GTest JfrTestNetworkUtilization fails Reviewed-by: mgronlun --- .../gtest/jfr/test_networkUtilization.cpp | 18 ++--- .../runtime/TestNetworkUtilizationEvent.java | 78 +++++++++---------- 2 files changed, 47 insertions(+), 49 deletions(-) diff --git a/test/hotspot/gtest/jfr/test_networkUtilization.cpp b/test/hotspot/gtest/jfr/test_networkUtilization.cpp index b654365d2f9..19d6a6e2c27 100644 --- a/test/hotspot/gtest/jfr/test_networkUtilization.cpp +++ b/test/hotspot/gtest/jfr/test_networkUtilization.cpp @@ -258,7 +258,7 @@ TEST_VM_F(JfrTestNetworkUtilization, RequestFunctionBasic) { JfrNetworkUtilization::send_events(); ASSERT_EQ(1u, MockEventNetworkUtilization::committed.size()); MockEventNetworkUtilization& e = MockEventNetworkUtilization::committed[0]; - EXPECT_EQ(5, e.readRate); + EXPECT_EQ(40, e.readRate); EXPECT_EQ(0, e.writeRate); EXPECT_STREQ("eth0", e.iface.c_str()); } @@ -282,16 +282,16 @@ TEST_VM_F(JfrTestNetworkUtilization, RequestFunctionMultiple) { const MockEventNetworkUtilization& eth1_event = MockEventNetworkUtilization::get_committed("eth1"); const MockEventNetworkUtilization& ppp0_event = MockEventNetworkUtilization::get_committed("ppp0"); - EXPECT_EQ(5, eth0_event.readRate); + EXPECT_EQ(40, eth0_event.readRate); EXPECT_EQ(0, eth0_event.writeRate); EXPECT_STREQ("eth0", eth0_event.iface.c_str()); - EXPECT_EQ(50, eth1_event.readRate); + EXPECT_EQ(400, eth1_event.readRate); EXPECT_EQ(0, eth1_event.writeRate); EXPECT_STREQ("eth1", eth1_event.iface.c_str()); EXPECT_EQ(0, ppp0_event.readRate); - EXPECT_EQ(25, ppp0_event.writeRate); + EXPECT_EQ(200, ppp0_event.writeRate); EXPECT_STREQ("ppp0", ppp0_event.iface.c_str()); } @@ -310,11 +310,11 @@ TEST_VM_F(JfrTestNetworkUtilization, InterfaceRemoved) { const MockEventNetworkUtilization& eth0_event = MockEventNetworkUtilization::get_committed("eth0"); const MockEventNetworkUtilization& eth1_event = MockEventNetworkUtilization::get_committed("eth1"); - EXPECT_EQ(5, eth0_event.readRate); + EXPECT_EQ(40, eth0_event.readRate); EXPECT_EQ(0, eth0_event.writeRate); EXPECT_STREQ("eth0", eth0_event.iface.c_str()); - EXPECT_EQ(10, eth1_event.readRate); + EXPECT_EQ(80, eth1_event.readRate); EXPECT_EQ(0, eth1_event.writeRate); EXPECT_STREQ("eth1", eth1_event.iface.c_str()); @@ -327,7 +327,7 @@ TEST_VM_F(JfrTestNetworkUtilization, InterfaceRemoved) { ASSERT_EQ(1u, MockEventNetworkUtilization::committed.size()); const MockEventNetworkUtilization& eth1_event_v2 = MockEventNetworkUtilization::get_committed("eth1"); - EXPECT_EQ(5, eth1_event_v2.readRate); + EXPECT_EQ(40, eth1_event_v2.readRate); EXPECT_EQ(0, eth1_event_v2.writeRate); EXPECT_STREQ("eth1", eth1_event_v2.iface.c_str()); } @@ -343,7 +343,7 @@ TEST_VM_F(JfrTestNetworkUtilization, InterfaceReset) { JfrNetworkUtilization::send_events(); ASSERT_EQ(1u, MockEventNetworkUtilization::committed.size()); const MockEventNetworkUtilization& event = MockEventNetworkUtilization::committed[0]; - EXPECT_EQ(5, event.readRate); + EXPECT_EQ(40, event.readRate); EXPECT_EQ(0, event.writeRate); EXPECT_STREQ("eth0", event.iface.c_str()); @@ -360,7 +360,7 @@ TEST_VM_F(JfrTestNetworkUtilization, InterfaceReset) { JfrNetworkUtilization::send_events(); ASSERT_EQ(1u, MockEventNetworkUtilization::committed.size()); const MockEventNetworkUtilization& event_v2 = MockEventNetworkUtilization::committed[0]; - EXPECT_EQ(5, event_v2.readRate); + EXPECT_EQ(40, event_v2.readRate); EXPECT_EQ(0, event_v2.writeRate); EXPECT_STREQ("eth0", event_v2.iface.c_str()); } diff --git a/test/jdk/jdk/jfr/event/runtime/TestNetworkUtilizationEvent.java b/test/jdk/jdk/jfr/event/runtime/TestNetworkUtilizationEvent.java index 857ce41acb4..73dba80b91d 100644 --- a/test/jdk/jdk/jfr/event/runtime/TestNetworkUtilizationEvent.java +++ b/test/jdk/jdk/jfr/event/runtime/TestNetworkUtilizationEvent.java @@ -25,6 +25,13 @@ package jdk.jfr.event.runtime; +import java.net.DatagramPacket; +import java.net.DatagramSocket; +import java.net.InetAddress; +import java.util.HashSet; +import java.util.List; +import java.util.Set; + import jdk.jfr.Recording; import jdk.jfr.consumer.RecordedEvent; import jdk.test.lib.Asserts; @@ -32,17 +39,6 @@ import jdk.test.lib.Platform; import jdk.test.lib.jfr.EventNames; import jdk.test.lib.jfr.Events; -import java.net.DatagramPacket; -import java.net.DatagramSocket; -import java.net.InetAddress; -import java.time.Duration; -import java.time.Instant; -import java.util.List; -import java.util.Map; - -import static java.util.stream.Collectors.averagingLong; -import static java.util.stream.Collectors.groupingBy; - /** * @test * @key jfr @@ -56,21 +52,17 @@ public class TestNetworkUtilizationEvent { private static final long packetSendCount = 100; public static void main(String[] args) throws Throwable { - testSimple(); - } - static void testSimple() throws Throwable { - - Instant start = Instant.now(); Recording recording = new Recording(); - recording.enable(EventNames.NetworkUtilization); + recording.enable(EventNames.NetworkUtilization).with("period", "endChunk"); recording.start(); DatagramSocket socket = new DatagramSocket(); String msg = "hello!"; byte[] buf = msg.getBytes(); - - // Send a few packets both to the loopback address as well to an external + forceEndChunk(); + // Send a few packets both to the loopback address as well to an + // external DatagramPacket packet = new DatagramPacket(buf, buf.length, InetAddress.getLoopbackAddress(), 12345); for (int i = 0; i < packetSendCount; ++i) { socket.send(packet); @@ -79,32 +71,38 @@ public class TestNetworkUtilizationEvent { for (int i = 0; i < packetSendCount; ++i) { socket.send(packet); } - - // Now there should have been traffic on at least two different interfaces + forceEndChunk(); + socket.close(); + // Now there should have been traffic on at least two different + // interfaces recording.stop(); - Duration runtime = Duration.between(start, Instant.now()); + + Set networkInterfaces = new HashSet<>(); List events = Events.fromRecording(recording); - - // Calculate the average write rate for each interface - Map writeRates = events.stream() - .collect(groupingBy(e -> Events.assertField(e, "networkInterface").getValue(), - averagingLong(e -> Events.assertField(e, "writeRate").getValue()))); - - // Our test packets should have generated at least this much traffic per second - long expectedTraffic = (buf.length * packetSendCount) / Math.max(1, runtime.toSeconds()); - - // Count the number of interfaces that have seen at least our test traffic - long interfacesWithTraffic = writeRates.values().stream() - .filter(d -> d >= expectedTraffic) - .count(); + Events.hasEvents(events); + for (RecordedEvent event : events) { + System.out.println(event); + Events.assertField(event, "writeRate").atLeast(0L).atMost(1000L * Integer.MAX_VALUE); + Events.assertField(event, "readRate").atLeast(0L).atMost(1000L * Integer.MAX_VALUE); + Events.assertField(event, "networkInterface").notNull(); + if (event.getLong("writeRate") > 0) { + networkInterfaces.add(event.getString("networkInterface")); + } + } if (Platform.isWindows() || Platform.isSolaris()) { - // Windows and Solaris do not track statistics for the loopback interface - Asserts.assertGreaterThanOrEqual(writeRates.size(), 1); - Asserts.assertGreaterThanOrEqual(interfacesWithTraffic, Long.valueOf(1)); + // Windows and Solaris do not track statistics for the loopback + // interface + Asserts.assertGreaterThanOrEqual(networkInterfaces.size(), 1); } else { - Asserts.assertGreaterThanOrEqual(writeRates.size(), 2); - Asserts.assertGreaterThanOrEqual(interfacesWithTraffic, Long.valueOf(2)); + Asserts.assertGreaterThanOrEqual(networkInterfaces.size(), 2); } } + + private static void forceEndChunk() { + try(Recording r = new Recording()) { + r.start(); + r.stop(); + } + } } From cdf49edb29d2b2e6ab920dbb59649beacf19ff68 Mon Sep 17 00:00:00 2001 From: Jesper Wilhelmsson Date: Fri, 14 Dec 2018 06:39:14 +0100 Subject: [PATCH 003/101] Added tag jdk-12+24 for changeset 7d4397b43fa3 --- .hgtags | 1 + 1 file changed, 1 insertion(+) diff --git a/.hgtags b/.hgtags index 479d7b40b59..f2d1a58c394 100644 --- a/.hgtags +++ b/.hgtags @@ -526,3 +526,4 @@ dc1f9dec2018a37fedba47d8a2aedef99faaec64 jdk-12+19 f8fb0c86f2b3d24294d39c5685a628e1beb14ba7 jdk-12+21 732bec44c89e8b93a38296bf690f97b7230c5b6d jdk-12+22 eef755718cb24813031a842bbfc716a6cea18e9a jdk-12+23 +7d4397b43fa305806160785a4c7210600d59581a jdk-12+24 From 78669ad35ec26f34690acacf3bc1a2bd45f46530 Mon Sep 17 00:00:00 2001 From: Claes Redestad Date: Fri, 14 Dec 2018 12:02:39 +0100 Subject: [PATCH 004/101] 8215380: Backout accidental change to String::length Reviewed-by: darcy, smarks --- src/java.base/share/classes/java/lang/String.java | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/java.base/share/classes/java/lang/String.java b/src/java.base/share/classes/java/lang/String.java index 9251e6c0895..be9f846341b 100644 --- a/src/java.base/share/classes/java/lang/String.java +++ b/src/java.base/share/classes/java/lang/String.java @@ -664,7 +664,7 @@ public final class String * object. */ public int length() { - return isLatin1() ? value.length : value.length >> UTF16; + return value.length >> coder(); } /** From 7f21b8e73b525d2dc0a14e13c8e8525342897ac4 Mon Sep 17 00:00:00 2001 From: Aleksey Shipilev Date: Fri, 14 Dec 2018 12:56:57 +0100 Subject: [PATCH 005/101] 8215353: x86_32 build failures after JDK-8214751 (X86: Support for VNNI Instructions) Reviewed-by: thartmann, kvn --- src/hotspot/cpu/x86/x86_32.ad | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/src/hotspot/cpu/x86/x86_32.ad b/src/hotspot/cpu/x86/x86_32.ad index 1c7d0f4440c..87ccfb6ee19 100644 --- a/src/hotspot/cpu/x86/x86_32.ad +++ b/src/hotspot/cpu/x86/x86_32.ad @@ -7760,9 +7760,9 @@ instruct mulAddS2I_rReg(rRegI dst, rRegI src1, rRegI src2, rRegI src3, eFlagsReg match(Set dst (MulAddS2I (Binary dst src1) (Binary src2 src3))); effect(KILL cr, KILL src2); - expand %{ mulI_rReg(dst, src1, cr); - mulI_rReg(src2, src3, cr); - addI_rReg(dst, src2, cr); %} + expand %{ mulI_eReg(dst, src1, cr); + mulI_eReg(src2, src3, cr); + addI_eReg(dst, src2, cr); %} %} // Multiply Register Int to Long From 80acd4f3996dbc5d2d7940bcb8da8d317ec4d2e7 Mon Sep 17 00:00:00 2001 From: Aleksey Shipilev Date: Fri, 14 Dec 2018 12:56:59 +0100 Subject: [PATCH 006/101] 8215374: 32-bit build failures after JDK-8181143 (Introduce diagnostic flag to abort VM on too long VM operations) Reviewed-by: roland --- src/hotspot/share/runtime/vmThread.cpp | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/hotspot/share/runtime/vmThread.cpp b/src/hotspot/share/runtime/vmThread.cpp index c0667686f38..c2de946a0e6 100644 --- a/src/hotspot/share/runtime/vmThread.cpp +++ b/src/hotspot/share/runtime/vmThread.cpp @@ -206,7 +206,7 @@ void VMOperationTimeoutTask::task() { if (is_armed()) { jlong delay = (os::javaTimeMillis() - _arm_time); if (delay > AbortVMOnVMOperationTimeoutDelay) { - fatal("VM operation took too long: " SIZE_FORMAT " ms (timeout: " SIZE_FORMAT " ms)", + fatal("VM operation took too long: " JLONG_FORMAT " ms (timeout: " INTX_FORMAT " ms)", delay, AbortVMOnVMOperationTimeoutDelay); } } From dd6344fc769e588a4205070d33933db12bcc3e39 Mon Sep 17 00:00:00 2001 From: Yang Zhang Date: Wed, 28 Nov 2018 16:22:03 +0800 Subject: [PATCH 007/101] 8213134: AArch64: vector shift failed with MaxVectorSize=8 Add vshiftcnt instructions for vector64 and add vsra/vsrl instructions to AArch64 backend. To detect shift failures, MaxVectorSize options are added to jtreg test cases. Reviewed-by: aph, kvn --- src/hotspot/cpu/aarch64/aarch64.ad | 271 ++++++++++++++---- .../compiler/c2/cr6340864/TestByteVect.java | 3 + .../compiler/c2/cr6340864/TestDoubleVect.java | 3 + .../compiler/c2/cr6340864/TestFloatVect.java | 3 + .../compiler/c2/cr6340864/TestIntVect.java | 3 + .../compiler/c2/cr6340864/TestLongVect.java | 3 + .../compiler/c2/cr6340864/TestShortVect.java | 3 + .../jtreg/compiler/codegen/TestCharVect2.java | 3 + 8 files changed, 241 insertions(+), 51 deletions(-) diff --git a/src/hotspot/cpu/aarch64/aarch64.ad b/src/hotspot/cpu/aarch64/aarch64.ad index 9ba919947b8..ed575070c48 100644 --- a/src/hotspot/cpu/aarch64/aarch64.ad +++ b/src/hotspot/cpu/aarch64/aarch64.ad @@ -2133,7 +2133,12 @@ const uint Matcher::vector_ideal_reg(int len) { } const uint Matcher::vector_shift_count_ideal_reg(int size) { - return Op_VecX; + switch(size) { + case 8: return Op_VecD; + case 16: return Op_VecX; + } + ShouldNotReachHere(); + return 0; } // AES support not yet implemented @@ -16524,32 +16529,32 @@ instruct vxor16B(vecX dst, vecX src1, vecX src2) %} // ------------------------------ Shift --------------------------------------- - -instruct vshiftcntL(vecX dst, iRegIorL2I cnt) %{ +instruct vshiftcnt8B(vecD dst, iRegIorL2I cnt) %{ + predicate(n->as_Vector()->length_in_bytes() == 8); match(Set dst (LShiftCntV cnt)); - format %{ "dup $dst, $cnt\t# shift count (vecX)" %} - ins_encode %{ - __ dup(as_FloatRegister($dst$$reg), __ T16B, as_Register($cnt$$reg)); - %} - ins_pipe(vdup_reg_reg128); -%} - -// Right shifts on aarch64 SIMD are implemented as left shift by -ve amount -instruct vshiftcntR(vecX dst, iRegIorL2I cnt) %{ match(Set dst (RShiftCntV cnt)); - format %{ "dup $dst, $cnt\t# shift count (vecX)\n\tneg $dst, $dst\t T16B" %} + format %{ "dup $dst, $cnt\t# shift count vector (8B)" %} + ins_encode %{ + __ dup(as_FloatRegister($dst$$reg), __ T8B, as_Register($cnt$$reg)); + %} + ins_pipe(vdup_reg_reg64); +%} + +instruct vshiftcnt16B(vecX dst, iRegIorL2I cnt) %{ + predicate(n->as_Vector()->length_in_bytes() == 16); + match(Set dst (LShiftCntV cnt)); + match(Set dst (RShiftCntV cnt)); + format %{ "dup $dst, $cnt\t# shift count vector (16B)" %} ins_encode %{ __ dup(as_FloatRegister($dst$$reg), __ T16B, as_Register($cnt$$reg)); - __ negr(as_FloatRegister($dst$$reg), __ T16B, as_FloatRegister($dst$$reg)); %} ins_pipe(vdup_reg_reg128); %} -instruct vsll8B(vecD dst, vecD src, vecX shift) %{ +instruct vsll8B(vecD dst, vecD src, vecD shift) %{ predicate(n->as_Vector()->length() == 4 || n->as_Vector()->length() == 8); match(Set dst (LShiftVB src shift)); - match(Set dst (RShiftVB src shift)); ins_cost(INSN_COST); format %{ "sshl $dst,$src,$shift\t# vector (8B)" %} ins_encode %{ @@ -16563,7 +16568,6 @@ instruct vsll8B(vecD dst, vecD src, vecX shift) %{ instruct vsll16B(vecX dst, vecX src, vecX shift) %{ predicate(n->as_Vector()->length() == 16); match(Set dst (LShiftVB src shift)); - match(Set dst (RShiftVB src shift)); ins_cost(INSN_COST); format %{ "sshl $dst,$src,$shift\t# vector (16B)" %} ins_encode %{ @@ -16574,29 +16578,93 @@ instruct vsll16B(vecX dst, vecX src, vecX shift) %{ ins_pipe(vshift128); %} -instruct vsrl8B(vecD dst, vecD src, vecX shift) %{ +// Right shifts with vector shift count on aarch64 SIMD are implemented +// as left shift by negative shift count. +// There are two cases for vector shift count. +// +// Case 1: The vector shift count is from replication. +// | | +// LoadVector RShiftCntV +// | / +// RShiftVI +// Note: In inner loop, multiple neg instructions are used, which can be +// moved to outer loop and merge into one neg instruction. +// +// Case 2: The vector shift count is from loading. +// This case isn't supported by middle-end now. But it's supported by +// panama/vectorIntrinsics(JEP 338: Vector API). +// | | +// LoadVector LoadVector +// | / +// RShiftVI +// + +instruct vsra8B(vecD dst, vecD src, vecD shift, vecD tmp) %{ predicate(n->as_Vector()->length() == 4 || n->as_Vector()->length() == 8); - match(Set dst (URShiftVB src shift)); + match(Set dst (RShiftVB src shift)); ins_cost(INSN_COST); - format %{ "ushl $dst,$src,$shift\t# vector (8B)" %} + effect(TEMP tmp); + format %{ "negr $tmp,$shift\t" + "sshl $dst,$src,$tmp\t# vector (8B)" %} ins_encode %{ - __ ushl(as_FloatRegister($dst$$reg), __ T8B, - as_FloatRegister($src$$reg), + __ negr(as_FloatRegister($tmp$$reg), __ T8B, as_FloatRegister($shift$$reg)); + __ sshl(as_FloatRegister($dst$$reg), __ T8B, + as_FloatRegister($src$$reg), + as_FloatRegister($tmp$$reg)); %} ins_pipe(vshift64); %} -instruct vsrl16B(vecX dst, vecX src, vecX shift) %{ +instruct vsra16B(vecX dst, vecX src, vecX shift, vecX tmp) %{ + predicate(n->as_Vector()->length() == 16); + match(Set dst (RShiftVB src shift)); + ins_cost(INSN_COST); + effect(TEMP tmp); + format %{ "negr $tmp,$shift\t" + "sshl $dst,$src,$tmp\t# vector (16B)" %} + ins_encode %{ + __ negr(as_FloatRegister($tmp$$reg), __ T16B, + as_FloatRegister($shift$$reg)); + __ sshl(as_FloatRegister($dst$$reg), __ T16B, + as_FloatRegister($src$$reg), + as_FloatRegister($tmp$$reg)); + %} + ins_pipe(vshift128); +%} + +instruct vsrl8B(vecD dst, vecD src, vecD shift, vecD tmp) %{ + predicate(n->as_Vector()->length() == 4 || + n->as_Vector()->length() == 8); + match(Set dst (URShiftVB src shift)); + ins_cost(INSN_COST); + effect(TEMP tmp); + format %{ "negr $tmp,$shift\t" + "ushl $dst,$src,$tmp\t# vector (8B)" %} + ins_encode %{ + __ negr(as_FloatRegister($tmp$$reg), __ T8B, + as_FloatRegister($shift$$reg)); + __ ushl(as_FloatRegister($dst$$reg), __ T8B, + as_FloatRegister($src$$reg), + as_FloatRegister($tmp$$reg)); + %} + ins_pipe(vshift64); +%} + +instruct vsrl16B(vecX dst, vecX src, vecX shift, vecX tmp) %{ predicate(n->as_Vector()->length() == 16); match(Set dst (URShiftVB src shift)); ins_cost(INSN_COST); - format %{ "ushl $dst,$src,$shift\t# vector (16B)" %} + effect(TEMP tmp); + format %{ "negr $tmp,$shift\t" + "ushl $dst,$src,$tmp\t# vector (16B)" %} ins_encode %{ + __ negr(as_FloatRegister($tmp$$reg), __ T16B, + as_FloatRegister($shift$$reg)); __ ushl(as_FloatRegister($dst$$reg), __ T16B, as_FloatRegister($src$$reg), - as_FloatRegister($shift$$reg)); + as_FloatRegister($tmp$$reg)); %} ins_pipe(vshift128); %} @@ -16708,11 +16776,10 @@ instruct vsrl16B_imm(vecX dst, vecX src, immI shift) %{ ins_pipe(vshift128_imm); %} -instruct vsll4S(vecD dst, vecD src, vecX shift) %{ +instruct vsll4S(vecD dst, vecD src, vecD shift) %{ predicate(n->as_Vector()->length() == 2 || n->as_Vector()->length() == 4); match(Set dst (LShiftVS src shift)); - match(Set dst (RShiftVS src shift)); ins_cost(INSN_COST); format %{ "sshl $dst,$src,$shift\t# vector (4H)" %} ins_encode %{ @@ -16726,7 +16793,6 @@ instruct vsll4S(vecD dst, vecD src, vecX shift) %{ instruct vsll8S(vecX dst, vecX src, vecX shift) %{ predicate(n->as_Vector()->length() == 8); match(Set dst (LShiftVS src shift)); - match(Set dst (RShiftVS src shift)); ins_cost(INSN_COST); format %{ "sshl $dst,$src,$shift\t# vector (8H)" %} ins_encode %{ @@ -16737,29 +16803,72 @@ instruct vsll8S(vecX dst, vecX src, vecX shift) %{ ins_pipe(vshift128); %} -instruct vsrl4S(vecD dst, vecD src, vecX shift) %{ +instruct vsra4S(vecD dst, vecD src, vecD shift, vecD tmp) %{ predicate(n->as_Vector()->length() == 2 || n->as_Vector()->length() == 4); - match(Set dst (URShiftVS src shift)); + match(Set dst (RShiftVS src shift)); ins_cost(INSN_COST); - format %{ "ushl $dst,$src,$shift\t# vector (4H)" %} + effect(TEMP tmp); + format %{ "negr $tmp,$shift\t" + "sshl $dst,$src,$tmp\t# vector (4H)" %} ins_encode %{ - __ ushl(as_FloatRegister($dst$$reg), __ T4H, - as_FloatRegister($src$$reg), + __ negr(as_FloatRegister($tmp$$reg), __ T8B, as_FloatRegister($shift$$reg)); + __ sshl(as_FloatRegister($dst$$reg), __ T4H, + as_FloatRegister($src$$reg), + as_FloatRegister($tmp$$reg)); %} ins_pipe(vshift64); %} -instruct vsrl8S(vecX dst, vecX src, vecX shift) %{ +instruct vsra8S(vecX dst, vecX src, vecX shift, vecX tmp) %{ + predicate(n->as_Vector()->length() == 8); + match(Set dst (RShiftVS src shift)); + ins_cost(INSN_COST); + effect(TEMP tmp); + format %{ "negr $tmp,$shift\t" + "sshl $dst,$src,$tmp\t# vector (8H)" %} + ins_encode %{ + __ negr(as_FloatRegister($tmp$$reg), __ T16B, + as_FloatRegister($shift$$reg)); + __ sshl(as_FloatRegister($dst$$reg), __ T8H, + as_FloatRegister($src$$reg), + as_FloatRegister($tmp$$reg)); + %} + ins_pipe(vshift128); +%} + +instruct vsrl4S(vecD dst, vecD src, vecD shift, vecD tmp) %{ + predicate(n->as_Vector()->length() == 2 || + n->as_Vector()->length() == 4); + match(Set dst (URShiftVS src shift)); + ins_cost(INSN_COST); + effect(TEMP tmp); + format %{ "negr $tmp,$shift\t" + "ushl $dst,$src,$tmp\t# vector (4H)" %} + ins_encode %{ + __ negr(as_FloatRegister($tmp$$reg), __ T8B, + as_FloatRegister($shift$$reg)); + __ ushl(as_FloatRegister($dst$$reg), __ T4H, + as_FloatRegister($src$$reg), + as_FloatRegister($tmp$$reg)); + %} + ins_pipe(vshift64); +%} + +instruct vsrl8S(vecX dst, vecX src, vecX shift, vecX tmp) %{ predicate(n->as_Vector()->length() == 8); match(Set dst (URShiftVS src shift)); ins_cost(INSN_COST); - format %{ "ushl $dst,$src,$shift\t# vector (8H)" %} + effect(TEMP tmp); + format %{ "negr $tmp,$shift\t" + "ushl $dst,$src,$tmp\t# vector (8H)" %} ins_encode %{ + __ negr(as_FloatRegister($tmp$$reg), __ T16B, + as_FloatRegister($shift$$reg)); __ ushl(as_FloatRegister($dst$$reg), __ T8H, as_FloatRegister($src$$reg), - as_FloatRegister($shift$$reg)); + as_FloatRegister($tmp$$reg)); %} ins_pipe(vshift128); %} @@ -16871,10 +16980,9 @@ instruct vsrl8S_imm(vecX dst, vecX src, immI shift) %{ ins_pipe(vshift128_imm); %} -instruct vsll2I(vecD dst, vecD src, vecX shift) %{ +instruct vsll2I(vecD dst, vecD src, vecD shift) %{ predicate(n->as_Vector()->length() == 2); match(Set dst (LShiftVI src shift)); - match(Set dst (RShiftVI src shift)); ins_cost(INSN_COST); format %{ "sshl $dst,$src,$shift\t# vector (2S)" %} ins_encode %{ @@ -16888,7 +16996,6 @@ instruct vsll2I(vecD dst, vecD src, vecX shift) %{ instruct vsll4I(vecX dst, vecX src, vecX shift) %{ predicate(n->as_Vector()->length() == 4); match(Set dst (LShiftVI src shift)); - match(Set dst (RShiftVI src shift)); ins_cost(INSN_COST); format %{ "sshl $dst,$src,$shift\t# vector (4S)" %} ins_encode %{ @@ -16899,28 +17006,70 @@ instruct vsll4I(vecX dst, vecX src, vecX shift) %{ ins_pipe(vshift128); %} -instruct vsrl2I(vecD dst, vecD src, vecX shift) %{ +instruct vsra2I(vecD dst, vecD src, vecD shift, vecD tmp) %{ predicate(n->as_Vector()->length() == 2); - match(Set dst (URShiftVI src shift)); + match(Set dst (RShiftVI src shift)); ins_cost(INSN_COST); - format %{ "ushl $dst,$src,$shift\t# vector (2S)" %} + effect(TEMP tmp); + format %{ "negr $tmp,$shift\t" + "sshl $dst,$src,$tmp\t# vector (2S)" %} ins_encode %{ - __ ushl(as_FloatRegister($dst$$reg), __ T2S, - as_FloatRegister($src$$reg), + __ negr(as_FloatRegister($tmp$$reg), __ T8B, as_FloatRegister($shift$$reg)); + __ sshl(as_FloatRegister($dst$$reg), __ T2S, + as_FloatRegister($src$$reg), + as_FloatRegister($tmp$$reg)); %} ins_pipe(vshift64); %} -instruct vsrl4I(vecX dst, vecX src, vecX shift) %{ +instruct vsra4I(vecX dst, vecX src, vecX shift, vecX tmp) %{ + predicate(n->as_Vector()->length() == 4); + match(Set dst (RShiftVI src shift)); + ins_cost(INSN_COST); + effect(TEMP tmp); + format %{ "negr $tmp,$shift\t" + "sshl $dst,$src,$tmp\t# vector (4S)" %} + ins_encode %{ + __ negr(as_FloatRegister($tmp$$reg), __ T16B, + as_FloatRegister($shift$$reg)); + __ sshl(as_FloatRegister($dst$$reg), __ T4S, + as_FloatRegister($src$$reg), + as_FloatRegister($tmp$$reg)); + %} + ins_pipe(vshift128); +%} + +instruct vsrl2I(vecD dst, vecD src, vecD shift, vecD tmp) %{ + predicate(n->as_Vector()->length() == 2); + match(Set dst (URShiftVI src shift)); + ins_cost(INSN_COST); + effect(TEMP tmp); + format %{ "negr $tmp,$shift\t" + "ushl $dst,$src,$tmp\t# vector (2S)" %} + ins_encode %{ + __ negr(as_FloatRegister($tmp$$reg), __ T8B, + as_FloatRegister($shift$$reg)); + __ ushl(as_FloatRegister($dst$$reg), __ T2S, + as_FloatRegister($src$$reg), + as_FloatRegister($tmp$$reg)); + %} + ins_pipe(vshift64); +%} + +instruct vsrl4I(vecX dst, vecX src, vecX shift, vecX tmp) %{ predicate(n->as_Vector()->length() == 4); match(Set dst (URShiftVI src shift)); ins_cost(INSN_COST); - format %{ "ushl $dst,$src,$shift\t# vector (4S)" %} + effect(TEMP tmp); + format %{ "negr $tmp,$shift\t" + "ushl $dst,$src,$tmp\t# vector (4S)" %} ins_encode %{ + __ negr(as_FloatRegister($tmp$$reg), __ T16B, + as_FloatRegister($shift$$reg)); __ ushl(as_FloatRegister($dst$$reg), __ T4S, as_FloatRegister($src$$reg), - as_FloatRegister($shift$$reg)); + as_FloatRegister($tmp$$reg)); %} ins_pipe(vshift128); %} @@ -17006,7 +17155,6 @@ instruct vsrl4I_imm(vecX dst, vecX src, immI shift) %{ instruct vsll2L(vecX dst, vecX src, vecX shift) %{ predicate(n->as_Vector()->length() == 2); match(Set dst (LShiftVL src shift)); - match(Set dst (RShiftVL src shift)); ins_cost(INSN_COST); format %{ "sshl $dst,$src,$shift\t# vector (2D)" %} ins_encode %{ @@ -17017,15 +17165,36 @@ instruct vsll2L(vecX dst, vecX src, vecX shift) %{ ins_pipe(vshift128); %} -instruct vsrl2L(vecX dst, vecX src, vecX shift) %{ +instruct vsra2L(vecX dst, vecX src, vecX shift, vecX tmp) %{ + predicate(n->as_Vector()->length() == 2); + match(Set dst (RShiftVL src shift)); + ins_cost(INSN_COST); + effect(TEMP tmp); + format %{ "negr $tmp,$shift\t" + "sshl $dst,$src,$tmp\t# vector (2D)" %} + ins_encode %{ + __ negr(as_FloatRegister($tmp$$reg), __ T16B, + as_FloatRegister($shift$$reg)); + __ sshl(as_FloatRegister($dst$$reg), __ T2D, + as_FloatRegister($src$$reg), + as_FloatRegister($tmp$$reg)); + %} + ins_pipe(vshift128); +%} + +instruct vsrl2L(vecX dst, vecX src, vecX shift, vecX tmp) %{ predicate(n->as_Vector()->length() == 2); match(Set dst (URShiftVL src shift)); ins_cost(INSN_COST); - format %{ "ushl $dst,$src,$shift\t# vector (2D)" %} + effect(TEMP tmp); + format %{ "negr $tmp,$shift\t" + "ushl $dst,$src,$tmp\t# vector (2D)" %} ins_encode %{ + __ negr(as_FloatRegister($tmp$$reg), __ T16B, + as_FloatRegister($shift$$reg)); __ ushl(as_FloatRegister($dst$$reg), __ T2D, as_FloatRegister($src$$reg), - as_FloatRegister($shift$$reg)); + as_FloatRegister($tmp$$reg)); %} ins_pipe(vshift128); %} diff --git a/test/hotspot/jtreg/compiler/c2/cr6340864/TestByteVect.java b/test/hotspot/jtreg/compiler/c2/cr6340864/TestByteVect.java index 42b3ebf9a38..97310e0a31d 100644 --- a/test/hotspot/jtreg/compiler/c2/cr6340864/TestByteVect.java +++ b/test/hotspot/jtreg/compiler/c2/cr6340864/TestByteVect.java @@ -27,6 +27,9 @@ * @summary Implement vectorization optimizations in hotspot-server * * @run main/othervm/timeout=400 -Xbatch -Xmx128m compiler.c2.cr6340864.TestByteVect + * @run main/othervm/timeout=400 -Xbatch -Xmx128m -XX:MaxVectorSize=8 compiler.c2.cr6340864.TestByteVect + * @run main/othervm/timeout=400 -Xbatch -Xmx128m -XX:MaxVectorSize=16 compiler.c2.cr6340864.TestByteVect + * @run main/othervm/timeout=400 -Xbatch -Xmx128m -XX:MaxVectorSize=32 compiler.c2.cr6340864.TestByteVect */ package compiler.c2.cr6340864; diff --git a/test/hotspot/jtreg/compiler/c2/cr6340864/TestDoubleVect.java b/test/hotspot/jtreg/compiler/c2/cr6340864/TestDoubleVect.java index 56fce03cc83..63f4d449f13 100644 --- a/test/hotspot/jtreg/compiler/c2/cr6340864/TestDoubleVect.java +++ b/test/hotspot/jtreg/compiler/c2/cr6340864/TestDoubleVect.java @@ -27,6 +27,9 @@ * @summary Implement vectorization optimizations in hotspot-server * * @run main/othervm/timeout=400 -Xbatch -Xmx128m compiler.c2.cr6340864.TestDoubleVect + * @run main/othervm/timeout=400 -Xbatch -Xmx128m -XX:MaxVectorSize=8 compiler.c2.cr6340864.TestDoubleVect + * @run main/othervm/timeout=400 -Xbatch -Xmx128m -XX:MaxVectorSize=16 compiler.c2.cr6340864.TestDoubleVect + * @run main/othervm/timeout=400 -Xbatch -Xmx128m -XX:MaxVectorSize=32 compiler.c2.cr6340864.TestDoubleVect */ package compiler.c2.cr6340864; diff --git a/test/hotspot/jtreg/compiler/c2/cr6340864/TestFloatVect.java b/test/hotspot/jtreg/compiler/c2/cr6340864/TestFloatVect.java index c64c28ba0ec..5676a77a3cb 100644 --- a/test/hotspot/jtreg/compiler/c2/cr6340864/TestFloatVect.java +++ b/test/hotspot/jtreg/compiler/c2/cr6340864/TestFloatVect.java @@ -27,6 +27,9 @@ * @summary Implement vectorization optimizations in hotspot-server * * @run main/othervm/timeout=400 -Xbatch -Xmx128m compiler.c2.cr6340864.TestFloatVect + * @run main/othervm/timeout=400 -Xbatch -Xmx128m -XX:MaxVectorSize=8 compiler.c2.cr6340864.TestFloatVect + * @run main/othervm/timeout=400 -Xbatch -Xmx128m -XX:MaxVectorSize=16 compiler.c2.cr6340864.TestFloatVect + * @run main/othervm/timeout=400 -Xbatch -Xmx128m -XX:MaxVectorSize=32 compiler.c2.cr6340864.TestFloatVect */ package compiler.c2.cr6340864; diff --git a/test/hotspot/jtreg/compiler/c2/cr6340864/TestIntVect.java b/test/hotspot/jtreg/compiler/c2/cr6340864/TestIntVect.java index 87c8557adb9..40180f74310 100644 --- a/test/hotspot/jtreg/compiler/c2/cr6340864/TestIntVect.java +++ b/test/hotspot/jtreg/compiler/c2/cr6340864/TestIntVect.java @@ -27,6 +27,9 @@ * @summary Implement vectorization optimizations in hotspot-server * * @run main/othervm/timeout=400 -Xbatch -Xmx128m compiler.c2.cr6340864.TestIntVect + * @run main/othervm/timeout=400 -Xbatch -Xmx128m -XX:MaxVectorSize=8 compiler.c2.cr6340864.TestIntVect + * @run main/othervm/timeout=400 -Xbatch -Xmx128m -XX:MaxVectorSize=16 compiler.c2.cr6340864.TestIntVect + * @run main/othervm/timeout=400 -Xbatch -Xmx128m -XX:MaxVectorSize=32 compiler.c2.cr6340864.TestIntVect */ package compiler.c2.cr6340864; diff --git a/test/hotspot/jtreg/compiler/c2/cr6340864/TestLongVect.java b/test/hotspot/jtreg/compiler/c2/cr6340864/TestLongVect.java index 1bd0bf5c91b..da454146d17 100644 --- a/test/hotspot/jtreg/compiler/c2/cr6340864/TestLongVect.java +++ b/test/hotspot/jtreg/compiler/c2/cr6340864/TestLongVect.java @@ -27,6 +27,9 @@ * @summary Implement vectorization optimizations in hotspot-server * * @run main/othervm/timeout=400 -Xbatch -Xmx128m compiler.c2.cr6340864.TestLongVect + * @run main/othervm/timeout=400 -Xbatch -Xmx128m -XX:MaxVectorSize=8 compiler.c2.cr6340864.TestLongVect + * @run main/othervm/timeout=400 -Xbatch -Xmx128m -XX:MaxVectorSize=16 compiler.c2.cr6340864.TestLongVect + * @run main/othervm/timeout=400 -Xbatch -Xmx128m -XX:MaxVectorSize=32 compiler.c2.cr6340864.TestLongVect */ package compiler.c2.cr6340864; diff --git a/test/hotspot/jtreg/compiler/c2/cr6340864/TestShortVect.java b/test/hotspot/jtreg/compiler/c2/cr6340864/TestShortVect.java index e2bf2a90308..1aa3bd86f20 100644 --- a/test/hotspot/jtreg/compiler/c2/cr6340864/TestShortVect.java +++ b/test/hotspot/jtreg/compiler/c2/cr6340864/TestShortVect.java @@ -27,6 +27,9 @@ * @summary Implement vectorization optimizations in hotspot-server * * @run main/othervm/timeout=400 -Xbatch -Xmx128m compiler.c2.cr6340864.TestShortVect + * @run main/othervm/timeout=400 -Xbatch -Xmx128m -XX:MaxVectorSize=8 compiler.c2.cr6340864.TestShortVect + * @run main/othervm/timeout=400 -Xbatch -Xmx128m -XX:MaxVectorSize=16 compiler.c2.cr6340864.TestShortVect + * @run main/othervm/timeout=400 -Xbatch -Xmx128m -XX:MaxVectorSize=32 compiler.c2.cr6340864.TestShortVect */ package compiler.c2.cr6340864; diff --git a/test/hotspot/jtreg/compiler/codegen/TestCharVect2.java b/test/hotspot/jtreg/compiler/codegen/TestCharVect2.java index 6c349239ad7..881bb9052b9 100644 --- a/test/hotspot/jtreg/compiler/codegen/TestCharVect2.java +++ b/test/hotspot/jtreg/compiler/codegen/TestCharVect2.java @@ -27,6 +27,9 @@ * @summary incorrect results of char vectors right shift operaiton * * @run main/othervm/timeout=400 -Xbatch -Xmx128m compiler.codegen.TestCharVect2 + * @run main/othervm/timeout=400 -Xbatch -Xmx128m -XX:MaxVectorSize=8 compiler.codegen.TestCharVect2 + * @run main/othervm/timeout=400 -Xbatch -Xmx128m -XX:MaxVectorSize=16 compiler.codegen.TestCharVect2 + * @run main/othervm/timeout=400 -Xbatch -Xmx128m -XX:MaxVectorSize=32 compiler.codegen.TestCharVect2 */ package compiler.codegen; From 884c808e83b5d9c3199768d714f22a4bb558dccb Mon Sep 17 00:00:00 2001 From: Pavel Rappo Date: Fri, 14 Dec 2018 19:49:03 +0000 Subject: [PATCH 008/101] 8215292: Back out changes for node- and link- local ipv6 multicast address Reviewed-by: chegar, alanb --- .../unix/native/libnet/net_util_md.c | 12 +- .../net/MulticastSocket/PromiscuousIPv6.java | 162 ------------ .../DatagramChannel/PromiscuousIPv6.java | 230 ------------------ 3 files changed, 3 insertions(+), 401 deletions(-) delete mode 100644 test/jdk/java/net/MulticastSocket/PromiscuousIPv6.java delete mode 100644 test/jdk/java/nio/channels/DatagramChannel/PromiscuousIPv6.java diff --git a/src/java.base/unix/native/libnet/net_util_md.c b/src/java.base/unix/native/libnet/net_util_md.c index 86df4978257..5a2473d630c 100644 --- a/src/java.base/unix/native/libnet/net_util_md.c +++ b/src/java.base/unix/native/libnet/net_util_md.c @@ -803,12 +803,8 @@ NET_InetAddressToSockaddr(JNIEnv *env, jobject iaObj, int port, #ifdef __linux__ /* - * On Linux if we are connecting to a - * - * - link-local address - * - multicast interface-local or link-local address - * - * we need to specify the interface in the scope_id. + * On Linux if we are connecting to a link-local address + * we need to specify the interface in the scope_id (2.4 kernel only) * * If the scope was cached then we use the cached value. If not cached but * specified in the Inet6Address we use that, but we first check if the @@ -818,9 +814,7 @@ NET_InetAddressToSockaddr(JNIEnv *env, jobject iaObj, int port, * we try to determine a value from the routing table. In all these * cases the used value is cached for further use. */ - if (IN6_IS_ADDR_LINKLOCAL(&sa->sa6.sin6_addr) - || IN6_IS_ADDR_MC_NODELOCAL(&sa->sa6.sin6_addr) - || IN6_IS_ADDR_MC_LINKLOCAL(&sa->sa6.sin6_addr)) { + if (IN6_IS_ADDR_LINKLOCAL(&sa->sa6.sin6_addr)) { unsigned int cached_scope_id = 0, scope_id = 0; if (ia6_cachedscopeidID) { diff --git a/test/jdk/java/net/MulticastSocket/PromiscuousIPv6.java b/test/jdk/java/net/MulticastSocket/PromiscuousIPv6.java deleted file mode 100644 index 4e22dfef2cb..00000000000 --- a/test/jdk/java/net/MulticastSocket/PromiscuousIPv6.java +++ /dev/null @@ -1,162 +0,0 @@ -/* - * Copyright (c) 2018, 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 8210493 - * @requires os.family == "linux" - * @library /test/lib - * @build jdk.test.lib.NetworkConfiguration - * PromiscuousIPv6 - * @run main PromiscuousIPv6 - */ -import jdk.test.lib.NetworkConfiguration; -import jtreg.SkippedException; - -import java.io.IOException; -import java.net.DatagramPacket; -import java.net.DatagramSocket; -import java.net.InetAddress; -import java.net.InetSocketAddress; -import java.net.MulticastSocket; -import java.net.SocketTimeoutException; - -import static java.lang.System.out; - -/* - * This test was created as a copy of the Promiscuous test and adapted for - * IPv6 node-local and link-local multicast addresses on Linux. - */ -public class PromiscuousIPv6 { - - static final int TIMEOUT = 5 * 1000; // 5 secs - static int id = 1000; - - static void receive(DatagramSocket mc, boolean datagramExpected, int id) - throws IOException - { - byte[] ba = new byte[100]; - DatagramPacket p = new DatagramPacket(ba, ba.length); - try { - mc.receive(p); - int recvId = Integer.parseInt( - new String(p.getData(), 0, p.getLength(), "UTF-8")); - if (datagramExpected) { - if (recvId != id) - throw new RuntimeException("Unexpected id, got " + recvId - + ", expected: " + id); - out.printf("Received message as expected, %s\n", p.getAddress()); - } else { - throw new RuntimeException("Unexpected message received, " - + p.getAddress()); - } - } catch (SocketTimeoutException e) { - if (datagramExpected) - throw new RuntimeException("Expected message not received, " - + e.getMessage()); - else - out.printf("Message not received, as expected\n"); - } - } - - static void test(InetAddress group1, InetAddress group2) - throws IOException - { - try (MulticastSocket mc1 = new MulticastSocket(new InetSocketAddress(group1, 0)); - MulticastSocket mc2 = new MulticastSocket(new InetSocketAddress(group2, mc1.getLocalPort())); - DatagramSocket ds = new DatagramSocket()) { - - final int port = mc1.getLocalPort(); - out.printf("Using port: %d\n", port); - - mc1.setSoTimeout(TIMEOUT); - mc2.setSoTimeout(TIMEOUT); - int nextId = id; - byte[] msg = Integer.toString(nextId).getBytes("UTF-8"); - DatagramPacket p = new DatagramPacket(msg, msg.length); - p.setAddress(group1); - p.setPort(port); - - mc1.joinGroup(group1); - out.printf("mc1 joined the MC group: %s\n", group1); - mc2.joinGroup(group2); - out.printf("mc2 joined the MC group: %s\n", group2); - - out.printf("Sending datagram to: %s/%d\n", group1, port); - ds.send(p); - - // the packet should be received by mc1 only - receive(mc1, true, nextId); - receive(mc2, false, 0); - - nextId = ++id; - msg = Integer.toString(nextId).getBytes("UTF-8"); - p = new DatagramPacket(msg, msg.length); - p.setAddress(group2); - p.setPort(port); - - out.printf("Sending datagram to: %s/%d\n", group2, port); - ds.send(p); - - // the packet should be received by mc2 only - receive(mc2, true, nextId); - receive(mc1, false, 0); - - mc1.leaveGroup(group1); - mc2.leaveGroup(group2); - } - } - - public static void main(String args[]) throws IOException { - String os = System.getProperty("os.name"); - - if (!os.equals("Linux")) { - throw new SkippedException("This test should be run only on Linux"); - } else { - String osVersion = System.getProperty("os.version"); - String prefix = "3.10.0"; - if (osVersion.startsWith(prefix)) { - throw new SkippedException( - String.format("The behavior under test is known NOT to work on '%s' kernels", prefix)); - } - } - - NetworkConfiguration.printSystemConfiguration(System.out); - - if (NetworkConfiguration.probe() - .ip6MulticastInterfaces() - .findAny() - .isEmpty()) { - throw new SkippedException( - "No IPv6 interfaces that support multicast found"); - } - - InetAddress interfaceLocal1 = InetAddress.getByName("ff11::2.3.4.5"); - InetAddress interfaceLocal2 = InetAddress.getByName("ff11::6.7.8.9"); - test(interfaceLocal1, interfaceLocal2); - - InetAddress linkLocal1 = InetAddress.getByName("ff12::2.3.4.5"); - InetAddress linkLocal2 = InetAddress.getByName("ff12::6.7.8.9"); - test(linkLocal1, linkLocal2); - } -} diff --git a/test/jdk/java/nio/channels/DatagramChannel/PromiscuousIPv6.java b/test/jdk/java/nio/channels/DatagramChannel/PromiscuousIPv6.java deleted file mode 100644 index 38da9eba039..00000000000 --- a/test/jdk/java/nio/channels/DatagramChannel/PromiscuousIPv6.java +++ /dev/null @@ -1,230 +0,0 @@ -/* - * Copyright (c) 2018, 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 8210493 - * @requires os.family == "linux" - * @library /test/lib - * @build jdk.test.lib.NetworkConfiguration - * PromiscuousIPv6 - * @run main PromiscuousIPv6 - * @key randomness - */ - -import java.nio.ByteBuffer; -import java.nio.channels.*; -import java.net.*; -import static java.net.StandardProtocolFamily.*; -import java.util.*; -import java.io.IOException; -import java.util.stream.Collectors; - -import jdk.test.lib.NetworkConfiguration; -import jtreg.SkippedException; - -/* - * This test was created as a copy of the Promiscuous test and adapted for - * IPv6 node-local and link-local multicast addresses on Linux. - */ -public class PromiscuousIPv6 { - - static final Random rand = new Random(); - - static final ProtocolFamily UNSPEC = () -> "UNSPEC"; - - /** - * Sends a datagram to the given multicast group - */ - static int sendDatagram(NetworkInterface nif, - InetAddress group, - int port) - throws IOException - { - ProtocolFamily family = (group instanceof Inet6Address) ? - StandardProtocolFamily.INET6 : StandardProtocolFamily.INET; - DatagramChannel dc = DatagramChannel.open(family) - .setOption(StandardSocketOptions.IP_MULTICAST_IF, nif); - int id = rand.nextInt(); - byte[] msg = Integer.toString(id).getBytes("UTF-8"); - ByteBuffer buf = ByteBuffer.wrap(msg); - System.out.format("Send message -> group %s (id=0x%x)\n", - group.getHostAddress(), id); - dc.send(buf, new InetSocketAddress(group, port)); - dc.close(); - return id; - } - - /** - * Wait (with timeout) for datagram. The {@code datagramExpected} - * parameter indicates whether a datagram is expected, and if - * {@code true} then {@code id} is the identifier in the payload. - */ - static void receiveDatagram(DatagramChannel dc, - String name, - boolean datagramExpected, - int id) - throws IOException - { - System.out.println("Checking if received by " + name); - - Selector sel = Selector.open(); - dc.configureBlocking(false); - dc.register(sel, SelectionKey.OP_READ); - ByteBuffer buf = ByteBuffer.allocateDirect(100); - - try { - for (;;) { - System.out.println("Waiting to receive message"); - sel.select(5*1000); - SocketAddress sa = dc.receive(buf); - - // no datagram received - if (sa == null) { - if (datagramExpected) { - throw new RuntimeException("Expected message not received"); - } - System.out.println("No message received (correct)"); - return; - } - - // datagram received - - InetAddress sender = ((InetSocketAddress)sa).getAddress(); - buf.flip(); - byte[] bytes = new byte[buf.remaining()]; - buf.get(bytes); - String s = new String(bytes, "UTF-8"); - int receivedId = -1; - try { - receivedId = Integer.parseInt(s); - System.out.format("Received message from %s (id=0x%x)\n", - sender, receivedId); - } catch (NumberFormatException x) { - System.out.format("Received message from %s (msg=%s)\n", sender, s); - } - - if (!datagramExpected) { - if (receivedId == id) - throw new RuntimeException("Message not expected"); - System.out.println("Message ignored (has wrong id)"); - } else { - if (receivedId == id) { - System.out.println("Message expected"); - return; - } - System.out.println("Message ignored (wrong sender)"); - } - - sel.selectedKeys().clear(); - buf.rewind(); - } - } finally { - sel.close(); - } - } - - static void test(ProtocolFamily family, - NetworkInterface nif, - InetAddress group1, - InetAddress group2) - throws IOException - { - - System.out.format("%nTest family=%s%n", family.name()); - - DatagramChannel dc1 = (family == UNSPEC) ? - DatagramChannel.open() : DatagramChannel.open(family); - DatagramChannel dc2 = (family == UNSPEC) ? - DatagramChannel.open() : DatagramChannel.open(family); - - try { - dc1.setOption(StandardSocketOptions.SO_REUSEADDR, true); - dc2.setOption(StandardSocketOptions.SO_REUSEADDR, true); - - dc1.bind(new InetSocketAddress(group1, 0)); - int port = dc1.socket().getLocalPort(); - dc2.bind(new InetSocketAddress(group2, port)); - - System.out.format("dc1 joining [%s]:%d @ %s\n", - group1.getHostAddress(), port, nif.getName()); - System.out.format("dc2 joining [%s]:%d @ %s\n", - group2.getHostAddress(), port, nif.getName()); - - dc1.join(group1, nif); - dc2.join(group2, nif); - - int id = sendDatagram(nif, group1, port); - - receiveDatagram(dc1, "dc1", true, id); - receiveDatagram(dc2, "dc2", false, id); - - id = sendDatagram(nif, group2, port); - - receiveDatagram(dc1, "dc1", false, id); - receiveDatagram(dc2, "dc2", true, id); - - } finally { - dc1.close(); - dc2.close(); - } - } - - public static void main(String[] args) throws IOException { - - String os = System.getProperty("os.name"); - - if (!os.equals("Linux")) { - throw new SkippedException("This test should be run only on Linux"); - } else { - String osVersion = System.getProperty("os.version"); - String prefix = "3.10.0"; - if (osVersion.startsWith(prefix)) { - throw new SkippedException( - String.format("The behavior under test is known NOT to work on '%s' kernels", prefix)); - } - } - - NetworkConfiguration.printSystemConfiguration(System.out); - - InetAddress interfaceLocal1 = InetAddress.getByName("ff11::2.3.4.5"); - InetAddress interfaceLocal2 = InetAddress.getByName("ff11::6.7.8.9"); - - InetAddress linkLocal1 = InetAddress.getByName("ff12::2.3.4.5"); - InetAddress linkLocal2 = InetAddress.getByName("ff12::6.7.8.9"); - - // get local network configuration to use - NetworkConfiguration config = NetworkConfiguration.probe(); - boolean foundAtLeastOne = false; - for (NetworkInterface nif: config.ip6MulticastInterfaces() - .collect(Collectors.toList())) { - foundAtLeastOne = true; - test(INET6, nif, interfaceLocal1, interfaceLocal2); - test(INET6, nif, linkLocal1, linkLocal2); - } - if (!foundAtLeastOne) { - throw new SkippedException( - "No IPv6 interfaces that support multicast found"); - } - } -} From 516a3b3ec11e7af8a0620b87e8e0d29f9fb38f05 Mon Sep 17 00:00:00 2001 From: Doug Simon Date: Fri, 14 Dec 2018 17:32:16 -0800 Subject: [PATCH 009/101] 8215319: jck lang/INTF/intf049/intf04901 fails in Graal as JIT mode with -Xcomp and AOTed Graal Reviewed-by: iveresov, never, dlong --- .../org/graalvm/compiler/java/BytecodeParser.java | 15 ++++++++------- 1 file changed, 8 insertions(+), 7 deletions(-) diff --git a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.java/src/org/graalvm/compiler/java/BytecodeParser.java b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.java/src/org/graalvm/compiler/java/BytecodeParser.java index 482d12a5183..48adfc457a4 100644 --- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.java/src/org/graalvm/compiler/java/BytecodeParser.java +++ b/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.java/src/org/graalvm/compiler/java/BytecodeParser.java @@ -4195,7 +4195,6 @@ public class BytecodeParser implements GraphBuilderContext { handleIllegalNewInstance(resolvedType); return; } - maybeEagerlyInitialize(resolvedType); ClassInitializationPlugin classInitializationPlugin = graphBuilderConfig.getPlugins().getClassInitializationPlugin(); @@ -4509,7 +4508,6 @@ public class BytecodeParser implements GraphBuilderContext { } ResolvedJavaType holder = resolvedField.getDeclaringClass(); - maybeEagerlyInitialize(holder); ClassInitializationPlugin classInitializationPlugin = this.graphBuilderConfig.getPlugins().getClassInitializationPlugin(); if (classInitializationPlugin != null) { classInitializationPlugin.apply(this, holder, this::createCurrentFrameState); @@ -4545,16 +4543,20 @@ public class BytecodeParser implements GraphBuilderContext { private ResolvedJavaField resolveStaticFieldAccess(JavaField field, ValueNode value) { if (field instanceof ResolvedJavaField) { ResolvedJavaField resolvedField = (ResolvedJavaField) field; - if (resolvedField.getDeclaringClass().isInitialized() || graphBuilderConfig.getPlugins().getClassInitializationPlugin() != null) { + ResolvedJavaType resolvedType = resolvedField.getDeclaringClass(); + maybeEagerlyInitialize(resolvedType); + + if (resolvedType.isInitialized() || graphBuilderConfig.getPlugins().getClassInitializationPlugin() != null) { return resolvedField; } + /* * Static fields have initialization semantics but may be safely accessed under certain * conditions while the class is being initialized. Executing in the clinit or init of - * classes which are subtypes of the field holder are sure to be running in a context - * where the access is safe. + * subclasses (but not implementers) of the field holder are sure to be running in a + * context where the access is safe. */ - if (resolvedField.getDeclaringClass().isAssignableFrom(method.getDeclaringClass())) { + if (!resolvedType.isInterface() && resolvedType.isAssignableFrom(method.getDeclaringClass())) { if (method.isClassInitializer() || method.isConstructor()) { return resolvedField; } @@ -4588,7 +4590,6 @@ public class BytecodeParser implements GraphBuilderContext { ClassInitializationPlugin classInitializationPlugin = this.graphBuilderConfig.getPlugins().getClassInitializationPlugin(); ResolvedJavaType holder = resolvedField.getDeclaringClass(); - maybeEagerlyInitialize(holder); if (classInitializationPlugin != null) { Supplier stateBefore = () -> { JavaKind[] pushedSlotKinds = {field.getJavaKind()}; From e44207a9f6741c54240f0b90b9b7d57cd35eb0c7 Mon Sep 17 00:00:00 2001 From: Xue-Lei Andrew Fan Date: Fri, 14 Dec 2018 17:51:02 -0800 Subject: [PATCH 010/101] 8213782: NullPointerException in sun.security.ssl.OutputRecord.changeWriteCiphers Reviewed-by: ascarpino --- .../sun/security/ssl/ChangeCipherSpec.java | 18 +++++ .../classes/sun/security/ssl/Finished.java | 73 ++++++++++++++----- .../classes/sun/security/ssl/KeyUpdate.java | 18 +++++ .../classes/sun/security/ssl/ServerHello.java | 60 ++++++++++++--- 4 files changed, 140 insertions(+), 29 deletions(-) diff --git a/src/java.base/share/classes/sun/security/ssl/ChangeCipherSpec.java b/src/java.base/share/classes/sun/security/ssl/ChangeCipherSpec.java index 937859f8c98..2983cf615ab 100644 --- a/src/java.base/share/classes/sun/security/ssl/ChangeCipherSpec.java +++ b/src/java.base/share/classes/sun/security/ssl/ChangeCipherSpec.java @@ -105,6 +105,14 @@ final class ChangeCipherSpec { throw new SSLException("Algorithm missing: ", gse); } + if (writeCipher == null) { + hc.conContext.fatal(Alert.ILLEGAL_PARAMETER, + "Illegal cipher suite (" + ncs + + ") and protocol version (" + hc.negotiatedProtocol + ")"); + + return null; + } + if (SSLLogger.isOn && SSLLogger.isOn("ssl,handshake")) { SSLLogger.fine("Produced ChangeCipherSpec message"); } @@ -195,6 +203,16 @@ final class ChangeCipherSpec { // unlikely throw new SSLException("Algorithm missing: ", gse); } + + if (readCipher == null) { + hc.conContext.fatal(Alert.ILLEGAL_PARAMETER, + "Illegal cipher suite (" + hc.negotiatedCipherSuite + + ") and protocol version (" + hc.negotiatedProtocol + + ")"); + + return; + } + tc.inputRecord.changeReadCiphers(readCipher); } else { throw new UnsupportedOperationException("Not supported."); diff --git a/src/java.base/share/classes/sun/security/ssl/Finished.java b/src/java.base/share/classes/sun/security/ssl/Finished.java index 351a6b92d49..47a071027b4 100644 --- a/src/java.base/share/classes/sun/security/ssl/Finished.java +++ b/src/java.base/share/classes/sun/security/ssl/Finished.java @@ -118,7 +118,7 @@ final class Finished { } catch (IOException ioe) { context.conContext.fatal(Alert.ILLEGAL_PARAMETER, "Failed to generate verify_data", ioe); - return; // make the compiler happy + return; } if (!MessageDigest.isEqual(myVerifyData, verifyData)) { context.conContext.fatal(Alert.ILLEGAL_PARAMETER, @@ -681,7 +681,7 @@ final class Finished { // unlikely chc.conContext.fatal(Alert.INTERNAL_ERROR, "no key derivation"); - return null; // make the compiler happy + return null; } SSLTrafficKeyDerivation kdg = @@ -691,7 +691,7 @@ final class Finished { chc.conContext.fatal(Alert.INTERNAL_ERROR, "Not supported key derivation: " + chc.negotiatedProtocol); - return null; // make the compiler happy + return null; } try { @@ -713,6 +713,15 @@ final class Finished { chc.negotiatedProtocol, writeKey, writeIv, chc.sslContext.getSecureRandom()); + if (writeCipher == null) { + chc.conContext.fatal(Alert.ILLEGAL_PARAMETER, + "Illegal cipher suite (" + chc.negotiatedCipherSuite + + ") and protocol version (" + chc.negotiatedProtocol + + ")"); + + return null; + } + chc.baseWriteSecret = writeSecret; chc.conContext.outputRecord.changeWriteCiphers( writeCipher, false); @@ -720,15 +729,16 @@ final class Finished { } catch (GeneralSecurityException gse) { chc.conContext.fatal(Alert.INTERNAL_ERROR, "Failure to derive application secrets", gse); - return null; // make the compiler happy + return null; } // The resumption master secret is stored in the session so // it can be used after the handshake is completed. SSLSecretDerivation sd = ((SSLSecretDerivation) kd).forContext(chc); SecretKey resumptionMasterSecret = sd.deriveKey( - "TlsResumptionMasterSecret", null); - chc.handshakeSession.setResumptionMasterSecret(resumptionMasterSecret); + "TlsResumptionMasterSecret", null); + chc.handshakeSession.setResumptionMasterSecret( + resumptionMasterSecret); chc.conContext.conSession = chc.handshakeSession.finish(); chc.conContext.protocolVersion = chc.negotiatedProtocol; @@ -764,7 +774,7 @@ final class Finished { // unlikely shc.conContext.fatal(Alert.INTERNAL_ERROR, "no key derivation"); - return null; // make the compiler happy + return null; } SSLTrafficKeyDerivation kdg = @@ -774,7 +784,7 @@ final class Finished { shc.conContext.fatal(Alert.INTERNAL_ERROR, "Not supported key derivation: " + shc.negotiatedProtocol); - return null; // make the compiler happy + return null; } // derive salt secret @@ -810,6 +820,15 @@ final class Finished { shc.negotiatedProtocol, writeKey, writeIv, shc.sslContext.getSecureRandom()); + if (writeCipher == null) { + shc.conContext.fatal(Alert.ILLEGAL_PARAMETER, + "Illegal cipher suite (" + shc.negotiatedCipherSuite + + ") and protocol version (" + shc.negotiatedProtocol + + ")"); + + return null; + } + shc.baseWriteSecret = writeSecret; shc.conContext.outputRecord.changeWriteCiphers( writeCipher, false); @@ -819,7 +838,7 @@ final class Finished { } catch (GeneralSecurityException gse) { shc.conContext.fatal(Alert.INTERNAL_ERROR, "Failure to derive application secrets", gse); - return null; // make the compiler happy + return null; } /* @@ -894,7 +913,7 @@ final class Finished { // unlikely chc.conContext.fatal(Alert.INTERNAL_ERROR, "no key derivation"); - return; // make the compiler happy + return; } SSLTrafficKeyDerivation kdg = @@ -904,7 +923,7 @@ final class Finished { chc.conContext.fatal(Alert.INTERNAL_ERROR, "Not supported key derivation: " + chc.negotiatedProtocol); - return; // make the compiler happy + return; } // save the session @@ -947,6 +966,15 @@ final class Finished { chc.negotiatedProtocol, readKey, readIv, chc.sslContext.getSecureRandom()); + if (readCipher == null) { + chc.conContext.fatal(Alert.ILLEGAL_PARAMETER, + "Illegal cipher suite (" + chc.negotiatedCipherSuite + + ") and protocol version (" + chc.negotiatedProtocol + + ")"); + + return; + } + chc.baseReadSecret = readSecret; chc.conContext.inputRecord.changeReadCiphers(readCipher); @@ -955,7 +983,7 @@ final class Finished { } catch (GeneralSecurityException gse) { chc.conContext.fatal(Alert.INTERNAL_ERROR, "Failure to derive application secrets", gse); - return; // make the compiler happy + return; } // @@ -1005,7 +1033,7 @@ final class Finished { // unlikely shc.conContext.fatal(Alert.INTERNAL_ERROR, "no key derivation"); - return; // make the compiler happy + return; } SSLTrafficKeyDerivation kdg = @@ -1015,7 +1043,7 @@ final class Finished { shc.conContext.fatal(Alert.INTERNAL_ERROR, "Not supported key derivation: " + shc.negotiatedProtocol); - return; // make the compiler happy + return; } // save the session @@ -1044,20 +1072,31 @@ final class Finished { shc.negotiatedProtocol, readKey, readIv, shc.sslContext.getSecureRandom()); + if (readCipher == null) { + shc.conContext.fatal(Alert.ILLEGAL_PARAMETER, + "Illegal cipher suite (" + shc.negotiatedCipherSuite + + ") and protocol version (" + shc.negotiatedProtocol + + ")"); + + return; + } + shc.baseReadSecret = readSecret; shc.conContext.inputRecord.changeReadCiphers(readCipher); // The resumption master secret is stored in the session so // it can be used after the handshake is completed. shc.handshakeHash.update(); - SSLSecretDerivation sd = ((SSLSecretDerivation)kd).forContext(shc); + SSLSecretDerivation sd = + ((SSLSecretDerivation)kd).forContext(shc); SecretKey resumptionMasterSecret = sd.deriveKey( "TlsResumptionMasterSecret", null); - shc.handshakeSession.setResumptionMasterSecret(resumptionMasterSecret); + shc.handshakeSession.setResumptionMasterSecret( + resumptionMasterSecret); } catch (GeneralSecurityException gse) { shc.conContext.fatal(Alert.INTERNAL_ERROR, "Failure to derive application secrets", gse); - return; // make the compiler happy + return; } // update connection context diff --git a/src/java.base/share/classes/sun/security/ssl/KeyUpdate.java b/src/java.base/share/classes/sun/security/ssl/KeyUpdate.java index b24d166f361..8477e69db48 100644 --- a/src/java.base/share/classes/sun/security/ssl/KeyUpdate.java +++ b/src/java.base/share/classes/sun/security/ssl/KeyUpdate.java @@ -223,6 +223,16 @@ final class KeyUpdate { Authenticator.valueOf(hc.conContext.protocolVersion), hc.conContext.protocolVersion, key, ivSpec, hc.sslContext.getSecureRandom()); + + if (rc == null) { + hc.conContext.fatal(Alert.ILLEGAL_PARAMETER, + "Illegal cipher suite (" + hc.negotiatedCipherSuite + + ") and protocol version (" + hc.negotiatedProtocol + + ")"); + + return; + } + rc.baseSecret = nplus1; hc.conContext.inputRecord.changeReadCiphers(rc); if (SSLLogger.isOn && SSLLogger.isOn("ssl")) { @@ -303,6 +313,14 @@ final class KeyUpdate { return null; } + if (wc == null) { + hc.conContext.fatal(Alert.ILLEGAL_PARAMETER, + "Illegal cipher suite (" + hc.negotiatedCipherSuite + + ") and protocol version (" + hc.negotiatedProtocol + ")"); + + return null; + } + // Output the handshake message and change the write cipher. // // The KeyUpdate handshake message SHALL be delivered in the diff --git a/src/java.base/share/classes/sun/security/ssl/ServerHello.java b/src/java.base/share/classes/sun/security/ssl/ServerHello.java index ca764f5129a..a8a38a9dfb8 100644 --- a/src/java.base/share/classes/sun/security/ssl/ServerHello.java +++ b/src/java.base/share/classes/sun/security/ssl/ServerHello.java @@ -296,7 +296,7 @@ final class ServerHello { shc.conContext.fatal(Alert.HANDSHAKE_FAILURE, "no cipher suites in common"); - return null; // make the compiler happy + return null; } shc.negotiatedCipherSuite = credentials.cipherSuite; shc.handshakeKeyExchange = credentials.keyExchange; @@ -461,7 +461,7 @@ final class ServerHello { shc.conContext.fatal(Alert.HANDSHAKE_FAILURE, "no cipher suites in common"); - return null; // make the compiler happy. + return null; } private static final class KeyExchangeProperties { @@ -526,7 +526,7 @@ final class ServerHello { if (cipherSuite == null) { shc.conContext.fatal(Alert.HANDSHAKE_FAILURE, "no cipher suites in common"); - return null; // make the compiler happy + return null; } shc.negotiatedCipherSuite = cipherSuite; shc.handshakeSession.setSuite(cipherSuite); @@ -594,7 +594,7 @@ final class ServerHello { // unlikely shc.conContext.fatal(Alert.INTERNAL_ERROR, "Not negotiated key shares"); - return null; // make the compiler happy + return null; } SSLKeyDerivation handshakeKD = ke.createKeyDerivation(shc); @@ -608,7 +608,7 @@ final class ServerHello { shc.conContext.fatal(Alert.INTERNAL_ERROR, "Not supported key derivation: " + shc.negotiatedProtocol); - return null; // make the compiler happy + return null; } SSLKeyDerivation kd = @@ -636,7 +636,16 @@ final class ServerHello { // unlikely shc.conContext.fatal(Alert.HANDSHAKE_FAILURE, "Missing cipher algorithm", gse); - return null; // make the compiler happy + return null; + } + + if (readCipher == null) { + shc.conContext.fatal(Alert.ILLEGAL_PARAMETER, + "Illegal cipher suite (" + shc.negotiatedCipherSuite + + ") and protocol version (" + shc.negotiatedProtocol + + ")"); + + return null; } shc.baseReadSecret = readSecret; @@ -664,7 +673,16 @@ final class ServerHello { // unlikely shc.conContext.fatal(Alert.HANDSHAKE_FAILURE, "Missing cipher algorithm", gse); - return null; // make the compiler happy + return null; + } + + if (writeCipher == null) { + shc.conContext.fatal(Alert.ILLEGAL_PARAMETER, + "Illegal cipher suite (" + shc.negotiatedCipherSuite + + ") and protocol version (" + shc.negotiatedProtocol + + ")"); + + return null; } shc.baseWriteSecret = writeSecret; @@ -748,7 +766,7 @@ final class ServerHello { if (cipherSuite == null) { shc.conContext.fatal(Alert.HANDSHAKE_FAILURE, "no cipher suites in common for hello retry request"); - return null; // make the compiler happy + return null; } ServerHelloMessage hhrm = new ServerHelloMessage(shc, @@ -1244,7 +1262,7 @@ final class ServerHello { // unlikely chc.conContext.fatal(Alert.INTERNAL_ERROR, "Not negotiated key shares"); - return; // make the compiler happy + return; } SSLKeyDerivation handshakeKD = ke.createKeyDerivation(chc); @@ -1257,7 +1275,7 @@ final class ServerHello { chc.conContext.fatal(Alert.INTERNAL_ERROR, "Not supported key derivation: " + chc.negotiatedProtocol); - return; // make the compiler happy + return; } SSLKeyDerivation secretKD = @@ -1286,7 +1304,16 @@ final class ServerHello { // unlikely chc.conContext.fatal(Alert.HANDSHAKE_FAILURE, "Missing cipher algorithm", gse); - return; // make the compiler happy + return; + } + + if (readCipher == null) { + chc.conContext.fatal(Alert.ILLEGAL_PARAMETER, + "Illegal cipher suite (" + chc.negotiatedCipherSuite + + ") and protocol version (" + chc.negotiatedProtocol + + ")"); + + return; } chc.baseReadSecret = readSecret; @@ -1314,7 +1341,16 @@ final class ServerHello { // unlikely chc.conContext.fatal(Alert.HANDSHAKE_FAILURE, "Missing cipher algorithm", gse); - return; // make the compiler happy + return; + } + + if (writeCipher == null) { + chc.conContext.fatal(Alert.ILLEGAL_PARAMETER, + "Illegal cipher suite (" + chc.negotiatedCipherSuite + + ") and protocol version (" + chc.negotiatedProtocol + + ")"); + + return; } chc.baseWriteSecret = writeSecret; From 3933c8477bf3712a5a391e3a5416dc0d50d9cbe8 Mon Sep 17 00:00:00 2001 From: Xue-Lei Andrew Fan Date: Fri, 14 Dec 2018 19:39:39 -0800 Subject: [PATCH 011/101] 8214339: SSLSocketImpl erroneously wraps SocketException Reviewed-by: ascarpino, jnimeh --- .../share/classes/sun/security/ssl/Alert.java | 16 +- .../sun/security/ssl/KeyShareExtension.java | 4 +- .../security/ssl/PreSharedKeyExtension.java | 43 +- .../sun/security/ssl/TransportContext.java | 8 + .../net/ssl/templates/SSLContextTemplate.java | 518 ++++++++++++++++++ .../SSLSocketImpl/SSLExceptionForIOIssue.java | 162 ++++++ 6 files changed, 726 insertions(+), 25 deletions(-) create mode 100644 test/jdk/javax/net/ssl/templates/SSLContextTemplate.java create mode 100644 test/jdk/sun/security/ssl/SSLSocketImpl/SSLExceptionForIOIssue.java diff --git a/src/java.base/share/classes/sun/security/ssl/Alert.java b/src/java.base/share/classes/sun/security/ssl/Alert.java index a8370822e14..09bcfd5afc7 100644 --- a/src/java.base/share/classes/sun/security/ssl/Alert.java +++ b/src/java.base/share/classes/sun/security/ssl/Alert.java @@ -122,11 +122,17 @@ enum Alert { reason = (cause != null) ? cause.getMessage() : ""; } - SSLException ssle = (this == UNEXPECTED_MESSAGE) ? - new SSLProtocolException(reason) : - (handshakeOnly ? - new SSLHandshakeException(reason) : - new SSLException(reason)); + SSLException ssle; + if ((cause != null) && (cause instanceof IOException)) { + ssle = new SSLException(reason); + } else if ((this == UNEXPECTED_MESSAGE)) { + ssle = new SSLProtocolException(reason); + } else if (handshakeOnly) { + ssle = new SSLHandshakeException(reason); + } else { + ssle = new SSLException(reason); + } + if (cause != null) { ssle.initCause(cause); } diff --git a/src/java.base/share/classes/sun/security/ssl/KeyShareExtension.java b/src/java.base/share/classes/sun/security/ssl/KeyShareExtension.java index ac44f2af50e..83dab2cdf3c 100644 --- a/src/java.base/share/classes/sun/security/ssl/KeyShareExtension.java +++ b/src/java.base/share/classes/sun/security/ssl/KeyShareExtension.java @@ -823,8 +823,8 @@ final class KeyShareExtension { } if (selectedGroup == null) { - shc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, - new IOException("No common named group")); + shc.conContext.fatal( + Alert.UNEXPECTED_MESSAGE, "No common named group"); return null; // make the complier happy } diff --git a/src/java.base/share/classes/sun/security/ssl/PreSharedKeyExtension.java b/src/java.base/share/classes/sun/security/ssl/PreSharedKeyExtension.java index 413b934038a..dc313a8cc72 100644 --- a/src/java.base/share/classes/sun/security/ssl/PreSharedKeyExtension.java +++ b/src/java.base/share/classes/sun/security/ssl/PreSharedKeyExtension.java @@ -171,7 +171,7 @@ final class PreSharedKeyExtension { int getIdsEncodedLength() { int idEncodedLength = 0; - for(PskIdentity curId : identities) { + for (PskIdentity curId : identities) { idEncodedLength += curId.getEncodedLength(); } @@ -194,7 +194,7 @@ final class PreSharedKeyExtension { byte[] buffer = new byte[encodedLength]; ByteBuffer m = ByteBuffer.wrap(buffer); Record.putInt16(m, idsEncodedLength); - for(PskIdentity curId : identities) { + for (PskIdentity curId : identities) { curId.writeEncoded(m); } Record.putInt16(m, bindersEncodedLength); @@ -538,12 +538,12 @@ final class PreSharedKeyExtension { } SecretKey psk = pskOpt.get(); - SecretKey binderKey = deriveBinderKey(psk, session); + SecretKey binderKey = deriveBinderKey(shc, psk, session); byte[] computedBinder = - computeBinder(binderKey, session, pskBinderHash); + computeBinder(shc, binderKey, session, pskBinderHash); if (!Arrays.equals(binder, computedBinder)) { shc.conContext.fatal(Alert.ILLEGAL_PARAMETER, - "Incorect PSK binder value"); + "Incorect PSK binder value"); } } @@ -687,13 +687,14 @@ final class PreSharedKeyExtension { ageMillis + chc.resumingSession.getTicketAgeAdd(); identities.add(new PskIdentity(chc.pskIdentity, obfuscatedAge)); - SecretKey binderKey = deriveBinderKey(psk, chc.resumingSession); + SecretKey binderKey = + deriveBinderKey(chc, psk, chc.resumingSession); ClientHelloMessage clientHello = (ClientHelloMessage)message; CHPreSharedKeySpec pskPrototype = createPskPrototype( chc.resumingSession.getSuite().hashAlg.hashLength, identities); HandshakeHash pskBinderHash = chc.handshakeHash.copy(); - byte[] binder = computeBinder(binderKey, pskBinderHash, + byte[] binder = computeBinder(chc, binderKey, pskBinderHash, chc.resumingSession, chc, clientHello, pskPrototype); List binders = new ArrayList<>(); @@ -717,7 +718,8 @@ final class PreSharedKeyExtension { } } - private static byte[] computeBinder(SecretKey binderKey, + private static byte[] computeBinder( + HandshakeContext context, SecretKey binderKey, SSLSessionImpl session, HandshakeHash pskBinderHash) throws IOException { @@ -726,10 +728,11 @@ final class PreSharedKeyExtension { pskBinderHash.update(); byte[] digest = pskBinderHash.digest(); - return computeBinder(binderKey, session, digest); + return computeBinder(context, binderKey, session, digest); } - private static byte[] computeBinder(SecretKey binderKey, + private static byte[] computeBinder( + HandshakeContext context, SecretKey binderKey, HandshakeHash hash, SSLSessionImpl session, HandshakeContext ctx, ClientHello.ClientHelloMessage hello, CHPreSharedKeySpec pskPrototype) throws IOException { @@ -745,10 +748,11 @@ final class PreSharedKeyExtension { hash.update(); byte[] digest = hash.digest(); - return computeBinder(binderKey, session, digest); + return computeBinder(context, binderKey, session, digest); } - private static byte[] computeBinder(SecretKey binderKey, + private static byte[] computeBinder(HandshakeContext context, + SecretKey binderKey, SSLSessionImpl session, byte[] digest) throws IOException { try { CipherSuite.HashAlg hashAlg = session.getSuite().hashAlg; @@ -766,15 +770,17 @@ final class PreSharedKeyExtension { hmac.init(finishedKey); return hmac.doFinal(digest); } catch (NoSuchAlgorithmException | InvalidKeyException ex) { - throw new IOException(ex); + context.conContext.fatal(Alert.INTERNAL_ERROR, ex); + return null; // fatal() always throws, make the compiler happy. } } catch (GeneralSecurityException ex) { - throw new IOException(ex); + context.conContext.fatal(Alert.INTERNAL_ERROR, ex); + return null; // fatal() always throws, make the compiler happy. } } - private static SecretKey deriveBinderKey(SecretKey psk, - SSLSessionImpl session) throws IOException { + private static SecretKey deriveBinderKey(HandshakeContext context, + SecretKey psk, SSLSessionImpl session) throws IOException { try { CipherSuite.HashAlg hashAlg = session.getSuite().hashAlg; HKDF hkdf = new HKDF(hashAlg.name); @@ -788,7 +794,8 @@ final class PreSharedKeyExtension { return hkdf.expand(earlySecret, hkdfInfo, hashAlg.hashLength, "TlsBinderKey"); } catch (GeneralSecurityException ex) { - throw new IOException(ex); + context.conContext.fatal(Alert.INTERNAL_ERROR, ex); + return null; // fatal() always throws, make the compiler happy. } } @@ -844,7 +851,7 @@ final class PreSharedKeyExtension { if (SSLLogger.isOn && SSLLogger.isOn("ssl,handshake")) { SSLLogger.fine( - "Resuming session: ", chc.resumingSession); + "Resuming session: ", chc.resumingSession); } } } diff --git a/src/java.base/share/classes/sun/security/ssl/TransportContext.java b/src/java.base/share/classes/sun/security/ssl/TransportContext.java index 09aed6802e6..811610daa33 100644 --- a/src/java.base/share/classes/sun/security/ssl/TransportContext.java +++ b/src/java.base/share/classes/sun/security/ssl/TransportContext.java @@ -328,6 +328,8 @@ class TransportContext implements ConnectionContext { if (SSLLogger.isOn && SSLLogger.isOn("ssl")) { SSLLogger.warning("Fatal: input record closure failed", ioe); } + + closeReason.addSuppressed(ioe); } // invalidate the session @@ -353,6 +355,8 @@ class TransportContext implements ConnectionContext { SSLLogger.warning( "Fatal: failed to send fatal alert " + alert, ioe); } + + closeReason.addSuppressed(ioe); } } @@ -363,6 +367,8 @@ class TransportContext implements ConnectionContext { if (SSLLogger.isOn && SSLLogger.isOn("ssl")) { SSLLogger.warning("Fatal: output record closure failed", ioe); } + + closeReason.addSuppressed(ioe); } // terminate the handshake context @@ -377,6 +383,8 @@ class TransportContext implements ConnectionContext { if (SSLLogger.isOn && SSLLogger.isOn("ssl")) { SSLLogger.warning("Fatal: transport closure failed", ioe); } + + closeReason.addSuppressed(ioe); } finally { isBroken = true; } diff --git a/test/jdk/javax/net/ssl/templates/SSLContextTemplate.java b/test/jdk/javax/net/ssl/templates/SSLContextTemplate.java new file mode 100644 index 00000000000..65e599bc251 --- /dev/null +++ b/test/jdk/javax/net/ssl/templates/SSLContextTemplate.java @@ -0,0 +1,518 @@ +/* + * Copyright (c) 2018, 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. + */ + +// +// Please run in othervm mode. SunJSSE does not support dynamic system +// properties, no way to re-use system properties in samevm/agentvm mode. +// + +import java.io.ByteArrayInputStream; +import java.security.KeyFactory; +import java.security.KeyStore; +import java.security.PrivateKey; +import java.security.cert.Certificate; +import java.security.cert.CertificateFactory; +import java.security.spec.PKCS8EncodedKeySpec; +import java.util.Base64; +import javax.net.ssl.KeyManagerFactory; +import javax.net.ssl.SSLContext; +import javax.net.ssl.TrustManagerFactory; + +/** + * SSLContext template to speed up JSSE tests. + */ +public interface SSLContextTemplate { + /* + * Create an instance of SSLContext for client use. + */ + default SSLContext createClientSSLContext() throws Exception { + return createSSLContext(trustedCertStrs, + endEntityCertStrs, endEntityPrivateKeys, + endEntityPrivateKeyAlgs, + endEntityPrivateKeyNames, + getClientContextParameters()); + } + + /* + * Create an instance of SSLContext for server use. + */ + default SSLContext createServerSSLContext() throws Exception { + return createSSLContext(trustedCertStrs, + endEntityCertStrs, endEntityPrivateKeys, + endEntityPrivateKeyAlgs, + endEntityPrivateKeyNames, + getServerContextParameters()); + } + + /* + * The parameters used to configure SSLContext. + */ + static final class ContextParameters { + final String contextProtocol; + final String tmAlgorithm; + final String kmAlgorithm; + + ContextParameters(String contextProtocol, + String tmAlgorithm, String kmAlgorithm) { + + this.contextProtocol = contextProtocol; + this.tmAlgorithm = tmAlgorithm; + this.kmAlgorithm = kmAlgorithm; + } + } + + /* + * Get the client side parameters of SSLContext. + */ + default ContextParameters getClientContextParameters() { + return new ContextParameters("TLS", "PKIX", "NewSunX509"); + } + + /* + * Get the server side parameters of SSLContext. + */ + default ContextParameters getServerContextParameters() { + return new ContextParameters("TLS", "PKIX", "NewSunX509"); + } + + /* + * ======================================= + * Certificates and keys used in the test. + */ + // Trusted certificates. + final static String[] trustedCertStrs = { + // SHA256withECDSA, curve prime256v1 + // Validity + // Not Before: May 22 07:18:16 2018 GMT + // Not After : May 17 07:18:16 2038 GMT + // Subject Key Identifier: + // 60:CF:BD:73:FF:FA:1A:30:D2:A4:EC:D3:49:71:46:EF:1A:35:A0:86 + "-----BEGIN CERTIFICATE-----\n" + + "MIIBvjCCAWOgAwIBAgIJAIvFG6GbTroCMAoGCCqGSM49BAMCMDsxCzAJBgNVBAYT\n" + + "AlVTMQ0wCwYDVQQKDARKYXZhMR0wGwYDVQQLDBRTdW5KU1NFIFRlc3QgU2VyaXZj\n" + + "ZTAeFw0xODA1MjIwNzE4MTZaFw0zODA1MTcwNzE4MTZaMDsxCzAJBgNVBAYTAlVT\n" + + "MQ0wCwYDVQQKDARKYXZhMR0wGwYDVQQLDBRTdW5KU1NFIFRlc3QgU2VyaXZjZTBZ\n" + + "MBMGByqGSM49AgEGCCqGSM49AwEHA0IABBz1WeVb6gM2mh85z3QlvaB/l11b5h0v\n" + + "LIzmkC3DKlVukZT+ltH2Eq1oEkpXuf7QmbM0ibrUgtjsWH3mULfmcWmjUDBOMB0G\n" + + "A1UdDgQWBBRgz71z//oaMNKk7NNJcUbvGjWghjAfBgNVHSMEGDAWgBRgz71z//oa\n" + + "MNKk7NNJcUbvGjWghjAMBgNVHRMEBTADAQH/MAoGCCqGSM49BAMCA0kAMEYCIQCG\n" + + "6wluh1r2/T6L31mZXRKf9JxeSf9pIzoLj+8xQeUChQIhAJ09wAi1kV8yePLh2FD9\n" + + "2YEHlSQUAbwwqCDEVB5KxaqP\n" + + "-----END CERTIFICATE-----", + // -----BEGIN PRIVATE KEY----- + // MIGHAgEAMBMGByqGSM49AgEGCCqGSM49AwEHBG0wawIBAQQg/HcHdoLJCdq3haVd + // XZTSKP00YzM3xX97l98vGL/RI1KhRANCAAQc9VnlW+oDNpofOc90Jb2gf5ddW+Yd + // LyyM5pAtwypVbpGU/pbR9hKtaBJKV7n+0JmzNIm61ILY7Fh95lC35nFp + // -----END PRIVATE KEY----- + + // SHA256withRSA, 2048 bits + // Validity + // Not Before: May 22 07:18:16 2018 GMT + // Not After : May 17 07:18:16 2038 GMT + // Subject Key Identifier: + // 0D:DD:93:C9:FE:4B:BD:35:B7:E8:99:78:90:FB:DB:5A:3D:DB:15:4C + "-----BEGIN CERTIFICATE-----\n" + + "MIIDSTCCAjGgAwIBAgIJAI4ZF3iy8zG+MA0GCSqGSIb3DQEBCwUAMDsxCzAJBgNV\n" + + "BAYTAlVTMQ0wCwYDVQQKDARKYXZhMR0wGwYDVQQLDBRTdW5KU1NFIFRlc3QgU2Vy\n" + + "aXZjZTAeFw0xODA1MjIwNzE4MTZaFw0zODA1MTcwNzE4MTZaMDsxCzAJBgNVBAYT\n" + + "AlVTMQ0wCwYDVQQKDARKYXZhMR0wGwYDVQQLDBRTdW5KU1NFIFRlc3QgU2VyaXZj\n" + + "ZTCCASIwDQYJKoZIhvcNAQEBBQADggEPADCCAQoCggEBALpMcY7aWieXDEM1/YJf\n" + + "JW27b4nRIFZyEYhEloyGsKTuQiiQjc8cqRZFNXe2vwziDB4IyTEl0Hjl5QF6ZaQE\n" + + "huPzzwvQm1pv64KrRXrmj3FisQK8B5OWLty9xp6xDqsaMRoyObLK+oIb20T5fSlE\n" + + "evmo1vYjnh8CX0Yzx5Gr5ye6YSEHQvYOWEws8ad17OlyToR2KMeC8w4qo6rs59pW\n" + + "g7Mxn9vo22ImDzrtAbTbXbCias3xlE0Bp0h5luyf+5U4UgksoL9B9r2oP4GrLNEV\n" + + "oJk57t8lwaR0upiv3CnS8LcJELpegZub5ggqLY8ZPYFQPjlK6IzLOm6rXPgZiZ3m\n" + + "RL0CAwEAAaNQME4wHQYDVR0OBBYEFA3dk8n+S701t+iZeJD721o92xVMMB8GA1Ud\n" + + "IwQYMBaAFA3dk8n+S701t+iZeJD721o92xVMMAwGA1UdEwQFMAMBAf8wDQYJKoZI\n" + + "hvcNAQELBQADggEBAJTRC3rKUUhVH07/1+stUungSYgpM08dY4utJq0BDk36BbmO\n" + + "0AnLDMbkwFdHEoqF6hQIfpm7SQTmXk0Fss6Eejm8ynYr6+EXiRAsaXOGOBCzF918\n" + + "/RuKOzqABfgSU4UBKECLM5bMfQTL60qx+HdbdVIpnikHZOFfmjCDVxoHsGyXc1LW\n" + + "Jhkht8IGOgc4PMGvyzTtRFjz01kvrVQZ75aN2E0GQv6dCxaEY0i3ypSzjUWAKqDh\n" + + "3e2OLwUSvumcdaxyCdZAOUsN6pDBQ+8VRG7KxnlRlY1SMEk46QgQYLbPDe/+W/yH\n" + + "ca4PejicPeh+9xRAwoTpiE2gulfT7Lm+fVM7Ruc=\n" + + "-----END CERTIFICATE-----", + + // -----BEGIN PRIVATE KEY----- + // MIIEvAIBADANBgkqhkiG9w0BAQEFAASCBKYwggSiAgEAAoIBAQC6THGO2lonlwxD + // Nf2CXyVtu2+J0SBWchGIRJaMhrCk7kIokI3PHKkWRTV3tr8M4gweCMkxJdB45eUB + // emWkBIbj888L0Jtab+uCq0V65o9xYrECvAeTli7cvcaesQ6rGjEaMjmyyvqCG9tE + // +X0pRHr5qNb2I54fAl9GM8eRq+cnumEhB0L2DlhMLPGndezpck6EdijHgvMOKqOq + // 7OfaVoOzMZ/b6NtiJg867QG0212womrN8ZRNAadIeZbsn/uVOFIJLKC/Qfa9qD+B + // qyzRFaCZOe7fJcGkdLqYr9wp0vC3CRC6XoGbm+YIKi2PGT2BUD45SuiMyzpuq1z4 + // GYmd5kS9AgMBAAECggEAFHSoU2MuWwJ+2jJnb5U66t2V1bAcuOE1g5zkWvG/G5z9 + // rq6Qo5kmB8f5ovdx6tw3MGUOklLwnRXBG3RxDJ1iokz3AvkY1clMNsDPlDsUrQKF + // JSO4QUBQTPSZhnsyfR8XHSU+qJ8Y+ohMfzpVv95BEoCzebtXdVgxVegBlcEmVHo2 + // kMmkRN+bYNsr8eb2r+b0EpyumS39ZgKYh09+cFb78y3T6IFMGcVJTP6nlGBFkmA/ + // 25pYeCF2tSki08qtMJZQAvKfw0Kviibk7ZxRbJqmc7B1yfnOEHP6ftjuvKl2+RP/ + // +5P5f8CfIP6gtA0LwSzAqQX/hfIKrGV5j0pCqrD0kQKBgQDeNR6Xi4sXVq79lihO + // a1bSeV7r8yoQrS8x951uO+ox+UIZ1MsAULadl7zB/P0er92p198I9M/0Jth3KBuS + // zj45mucvpiiGvmQlMKMEfNq4nN7WHOu55kufPswQB2mR4J3xmwI+4fM/nl1zc82h + // De8JSazRldJXNhfx0RGFPmgzbwKBgQDWoVXrXLbCAn41oVnWB8vwY9wjt92ztDqJ + // HMFA/SUohjePep9UDq6ooHyAf/Lz6oE5NgeVpPfTDkgvrCFVKnaWdwALbYoKXT2W + // 9FlyJox6eQzrtHAacj3HJooXWuXlphKSizntfxj3LtMR9BmrmRJOfK+SxNOVJzW2 + // +MowT20EkwKBgHmpB8jdZBgxI7o//m2BI5Y1UZ1KE5vx1kc7VXzHXSBjYqeV9FeF + // 2ZZLP9POWh/1Fh4pzTmwIDODGT2UPhSQy0zq3O0fwkyT7WzXRknsuiwd53u/dejg + // iEL2NPAJvulZ2+AuiHo5Z99LK8tMeidV46xoJDDUIMgTG+UQHNGhK5gNAoGAZn/S + // Cn7SgMC0CWSvBHnguULXZO9wH1wZAFYNLL44OqwuaIUFBh2k578M9kkke7woTmwx + // HxQTjmWpr6qimIuY6q6WBN8hJ2Xz/d1fwhYKzIp20zHuv5KDUlJjbFfqpsuy3u1C + // kts5zwI7pr1ObRbDGVyOdKcu7HI3QtR5qqyjwaUCgYABo7Wq6oHva/9V34+G3Goh + // 63bYGUnRw2l5BD11yhQv8XzGGZFqZVincD8gltNThB0Dc/BI+qu3ky4YdgdZJZ7K + // z51GQGtaHEbrHS5caV79yQ8QGY5mUVH3E+VXSxuIqb6pZq2DH4sTAEFHyncddmOH + // zoXBInYwRG9KE/Bw5elhUw== + // -----END PRIVATE KEY----- + + + // SHA256withDSA, 2048 bits + // Validity + // Not Before: May 22 07:18:18 2018 GMT + // Not After : May 17 07:18:18 2038 GMT + // Subject Key Identifier: + // 76:66:9E:F7:3B:DD:45:E5:3B:D9:72:3C:3F:F0:54:39:86:31:26:53 + "-----BEGIN CERTIFICATE-----\n" + + "MIIErjCCBFSgAwIBAgIJAOktYLNCbr02MAsGCWCGSAFlAwQDAjA7MQswCQYDVQQG\n" + + "EwJVUzENMAsGA1UECgwESmF2YTEdMBsGA1UECwwUU3VuSlNTRSBUZXN0IFNlcml2\n" + + "Y2UwHhcNMTgwNTIyMDcxODE4WhcNMzgwNTE3MDcxODE4WjA7MQswCQYDVQQGEwJV\n" + + "UzENMAsGA1UECgwESmF2YTEdMBsGA1UECwwUU3VuSlNTRSBUZXN0IFNlcml2Y2Uw\n" + + "ggNHMIICOQYHKoZIzjgEATCCAiwCggEBAO5GyPhSm0ze3LSu+gicdULLj05iOfTL\n" + + "UvZQ29sYz41zmqrLBQbdKiHqgJu2Re9sgTb5suLNjF047TOLPnU3jhPtWm2X8Xzi\n" + + "VGIcHym/Q/MeZxStt/88seqroI3WOKzIML2GcrishT+lcGrtH36Tf1+ue2Snn3PS\n" + + "WyxygNqPjllP5uUjYmFLvAf4QLMldkd/D2VxcwsHjB8y5iUZsXezc/LEhRZS/02m\n" + + "ivqlRw3AMkq/OVe/ZtxFWsP0nsfxEGdZuaUFpppGfixxFvymrB3+J51cTt+pZBDq\n" + + "D2y0DYfc+88iCs4jwHTfcDIpLb538HBjBj2rEgtQESQmB0ooD/+wsPsCIQC1bYch\n" + + "gElNtDYL3FgpLgNSUYp7gIWv9ehaC7LO2z7biQKCAQBitvFOnDkUja8NAF7lDpOV\n" + + "b5ipQ8SicBLW3kQamxhyuyxgZyy/PojZ/oPorkqW/T/A0rhnG6MssEpAtdiwVB+c\n" + + "rBYGo3bcwmExJhdOJ6dYuKFppPWhCwKMHs9npK+lqBMl8l5j58xlcFeC7ZfGf8GY\n" + + "GkhFW0c44vEQhMMbac6ZTTP4mw+1t7xJfmDMlLEyIpTXaAAk8uoVLWzQWnR40sHi\n" + + "ybvS0u3JxQkb7/y8tOOZu8qlz/YOS7lQ6UxUGX27Ce1E0+agfPphetoRAlS1cezq\n" + + "Wa7r64Ga0nkj1kwkcRqjgTiJx0NwnUXr78VAXFhVF95+O3lfqhvdtEGtkhDGPg7N\n" + + "A4IBBgACggEBAMmSHQK0w2i+iqUjOPzn0yNEZrzepLlLeQ1tqtn0xnlv5vBAeefD\n" + + "Pm9dd3tZOjufVWP7hhEz8xPobb1CS4e3vuQiv5UBfhdPL3f3l9T7JMAKPH6C9Vve\n" + + "OQXE5eGqbjsySbcmseHoYUt1WCSnSda1opX8zchX04e7DhGfE2/L9flpYEoSt8lI\n" + + "vMNjgOwvKdW3yvPt1/eBBHYNFG5gWPv/Q5KoyCtHS03uqGm4rNc/wZTIEEfd66C+\n" + + "QRaUltjOaHmtwOdDHaNqwhYZSVOip+Mo+TfyzHFREcdHLapo7ZXqbdYkRGxRR3d+\n" + + "3DfHaraJO0OKoYlPkr3JMvM/MSGR9AnZOcejUDBOMB0GA1UdDgQWBBR2Zp73O91F\n" + + "5TvZcjw/8FQ5hjEmUzAfBgNVHSMEGDAWgBR2Zp73O91F5TvZcjw/8FQ5hjEmUzAM\n" + + "BgNVHRMEBTADAQH/MAsGCWCGSAFlAwQDAgNHADBEAiBzriYE41M2y9Hy5ppkL0Qn\n" + + "dIlNc8JhXT/PHW7GDtViagIgMko8Qoj9gDGPK3+O9E8DC3wGiiF9CObM4LN387ok\n" + + "J+g=\n" + + "-----END CERTIFICATE-----" + // -----BEGIN PRIVATE KEY----- + // MIICZQIBADCCAjkGByqGSM44BAEwggIsAoIBAQDuRsj4UptM3ty0rvoInHVCy49O + // Yjn0y1L2UNvbGM+Nc5qqywUG3Soh6oCbtkXvbIE2+bLizYxdOO0ziz51N44T7Vpt + // l/F84lRiHB8pv0PzHmcUrbf/PLHqq6CN1jisyDC9hnK4rIU/pXBq7R9+k39frntk + // p59z0lsscoDaj45ZT+blI2JhS7wH+ECzJXZHfw9lcXMLB4wfMuYlGbF3s3PyxIUW + // Uv9Npor6pUcNwDJKvzlXv2bcRVrD9J7H8RBnWbmlBaaaRn4scRb8pqwd/iedXE7f + // qWQQ6g9stA2H3PvPIgrOI8B033AyKS2+d/BwYwY9qxILUBEkJgdKKA//sLD7AiEA + // tW2HIYBJTbQ2C9xYKS4DUlGKe4CFr/XoWguyzts+24kCggEAYrbxTpw5FI2vDQBe + // 5Q6TlW+YqUPEonAS1t5EGpsYcrssYGcsvz6I2f6D6K5Klv0/wNK4ZxujLLBKQLXY + // sFQfnKwWBqN23MJhMSYXTienWLihaaT1oQsCjB7PZ6SvpagTJfJeY+fMZXBXgu2X + // xn/BmBpIRVtHOOLxEITDG2nOmU0z+JsPtbe8SX5gzJSxMiKU12gAJPLqFS1s0Fp0 + // eNLB4sm70tLtycUJG+/8vLTjmbvKpc/2Dku5UOlMVBl9uwntRNPmoHz6YXraEQJU + // tXHs6lmu6+uBmtJ5I9ZMJHEao4E4icdDcJ1F6+/FQFxYVRfefjt5X6ob3bRBrZIQ + // xj4OzQQjAiEAsceWOM8do4etxp2zgnoNXV8PUUyqWhz1+0srcKV7FR4= + // -----END PRIVATE KEY----- + }; + + // End entity certificate. + final static String[] endEntityCertStrs = { + // SHA256withECDSA, curve prime256v1 + // Validity + // Not Before: May 22 07:18:16 2018 GMT + // Not After : May 17 07:18:16 2038 GMT + // Authority Key Identifier: + // 60:CF:BD:73:FF:FA:1A:30:D2:A4:EC:D3:49:71:46:EF:1A:35:A0:86 + "-----BEGIN CERTIFICATE-----\n" + + "MIIBqjCCAVCgAwIBAgIJAPLY8qZjgNRAMAoGCCqGSM49BAMCMDsxCzAJBgNVBAYT\n" + + "AlVTMQ0wCwYDVQQKDARKYXZhMR0wGwYDVQQLDBRTdW5KU1NFIFRlc3QgU2VyaXZj\n" + + "ZTAeFw0xODA1MjIwNzE4MTZaFw0zODA1MTcwNzE4MTZaMFUxCzAJBgNVBAYTAlVT\n" + + "MQ0wCwYDVQQKDARKYXZhMR0wGwYDVQQLDBRTdW5KU1NFIFRlc3QgU2VyaXZjZTEY\n" + + "MBYGA1UEAwwPUmVncmVzc2lvbiBUZXN0MFkwEwYHKoZIzj0CAQYIKoZIzj0DAQcD\n" + + "QgAEb+9n05qfXnfHUb0xtQJNS4JeSi6IjOfW5NqchvKnfJey9VkJzR7QHLuOESdf\n" + + "xlR7q8YIWgih3iWLGfB+wxHiOqMjMCEwHwYDVR0jBBgwFoAUYM+9c//6GjDSpOzT\n" + + "SXFG7xo1oIYwCgYIKoZIzj0EAwIDSAAwRQIgWpRegWXMheiD3qFdd8kMdrkLxRbq\n" + + "1zj8nQMEwFTUjjQCIQDRIrAjZX+YXHN9b0SoWWLPUq0HmiFIi8RwMnO//wJIGQ==\n" + + "-----END CERTIFICATE-----", + + // SHA256withRSA, 2048 bits + // Validity + // Not Before: May 22 07:18:16 2018 GMT + // Not After : May 17 07:18:16 2038 GMT + // Authority Key Identifier: + // 0D:DD:93:C9:FE:4B:BD:35:B7:E8:99:78:90:FB:DB:5A:3D:DB:15:4C + "-----BEGIN CERTIFICATE-----\n" + + "MIIDNjCCAh6gAwIBAgIJAO2+yPcFryUTMA0GCSqGSIb3DQEBCwUAMDsxCzAJBgNV\n" + + "BAYTAlVTMQ0wCwYDVQQKDARKYXZhMR0wGwYDVQQLDBRTdW5KU1NFIFRlc3QgU2Vy\n" + + "aXZjZTAeFw0xODA1MjIwNzE4MTZaFw0zODA1MTcwNzE4MTZaMFUxCzAJBgNVBAYT\n" + + "AlVTMQ0wCwYDVQQKDARKYXZhMR0wGwYDVQQLDBRTdW5KU1NFIFRlc3QgU2VyaXZj\n" + + "ZTEYMBYGA1UEAwwPUmVncmVzc2lvbiBUZXN0MIIBIjANBgkqhkiG9w0BAQEFAAOC\n" + + "AQ8AMIIBCgKCAQEAszfBobWfZIp8AgC6PiWDDavP65mSvgCXUGxACbxVNAfkLhNR\n" + + "QOsHriRB3X1Q3nvO9PetC6wKlvE9jlnDDj7D+1j1r1CHO7ms1fq8rfcQYdkanDtu\n" + + "4AlHo8v+SSWX16MIXFRYDj2VVHmyPtgbltcg4zGAuwT746FdLI94uXjJjq1IOr/v\n" + + "0VIlwE5ORWH5Xc+5Tj+oFWK0E4a4GHDgtKKhn2m72hN56/GkPKGkguP5NRS1qYYV\n" + + "/EFkdyQMOV8J1M7HaicSft4OL6eKjTrgo93+kHk+tv0Dc6cpVBnalX3TorG8QI6B\n" + + "cHj1XQd78oAlAC+/jF4pc0mwi0un49kdK9gRfQIDAQABoyMwITAfBgNVHSMEGDAW\n" + + "gBQN3ZPJ/ku9NbfomXiQ+9taPdsVTDANBgkqhkiG9w0BAQsFAAOCAQEApXS0nKwm\n" + + "Kp8gpmO2yG1rpd1+2wBABiMU4JZaTqmma24DQ3RzyS+V2TeRb29dl5oTUEm98uc0\n" + + "GPZvhK8z5RFr4YE17dc04nI/VaNDCw4y1NALXGs+AHkjoPjLyGbWpi1S+gfq2sNB\n" + + "Ekkjp6COb/cb9yiFXOGVls7UOIjnVZVd0r7KaPFjZhYh82/f4PA/A1SnIKd1+nfH\n" + + "2yk7mSJNC7Z3qIVDL8MM/jBVwiC3uNe5GPB2uwhd7k5LGAVN3j4HQQGB0Sz+VC1h\n" + + "92oi6xDa+YBva2fvHuCd8P50DDjxmp9CemC7rnZ5j8egj88w14X44Xjb/Fd/ApG9\n" + + "e57NnbT7KM+Grw==\n" + + "-----END CERTIFICATE-----", + + // SHA256withRSA, curv prime256v1 + // Validity + // Not Before: May 22 07:18:16 2018 GMT + // Not After : May 21 07:18:16 2028 GMT + // Authority Key Identifier: + // 0D:DD:93:C9:FE:4B:BD:35:B7:E8:99:78:90:FB:DB:5A:3D:DB:15:4C + "-----BEGIN CERTIFICATE-----\n" + + "MIICazCCAVOgAwIBAgIJAO2+yPcFryUUMA0GCSqGSIb3DQEBCwUAMDsxCzAJBgNV\n" + + "BAYTAlVTMQ0wCwYDVQQKDARKYXZhMR0wGwYDVQQLDBRTdW5KU1NFIFRlc3QgU2Vy\n" + + "aXZjZTAeFw0xODA1MjIwNzE4MTZaFw0yODA1MjEwNzE4MTZaMFUxCzAJBgNVBAYT\n" + + "AlVTMQ0wCwYDVQQKDARKYXZhMR0wGwYDVQQLDBRTdW5KU1NFIFRlc3QgU2VyaXZj\n" + + "ZTEYMBYGA1UEAwwPUmVncmVzc2lvbiBUZXN0MFkwEwYHKoZIzj0CAQYIKoZIzj0D\n" + + "AQcDQgAE59MERNTlVZ1eeps8Z3Oue5ZkgQdPtD+WIE6tj3PbIKpxGPDxvfNP959A\n" + + "yQjEK/ehWQVrCMmNoEkIzY+IIBgB06MjMCEwHwYDVR0jBBgwFoAUDd2Tyf5LvTW3\n" + + "6Jl4kPvbWj3bFUwwDQYJKoZIhvcNAQELBQADggEBAFOTVEqs70ykhZiIdrEsF1Ra\n" + + "I3B2rLvwXZk52uSltk2/bzVvewA577ZCoxQ1pL7ynkisPfBN1uVYtHjM1VA3RC+4\n" + + "+TAK78dnI7otYjWoHp5rvs4l6c/IbOspS290IlNuDUxMErEm5wxIwj+Aukx/1y68\n" + + "hOyCvHBLMY2c1LskH1MMBbDuS1aI+lnGpToi+MoYObxGcV458vxuT8+wwV8Fkpvd\n" + + "ll8IIFmeNPRv+1E+lXbES6CSNCVaZ/lFhPgdgYKleN7sfspiz50DG4dqafuEAaX5\n" + + "xaK1NWXJxTRz0ROH/IUziyuDW6jphrlgit4+3NCzp6vP9hAJQ8Vhcj0n15BKHIQ=\n" + + "-----END CERTIFICATE-----", + + // SHA256withDSA, 2048 bits + // Validity + // Not Before: May 22 07:18:20 2018 GMT + // Not After : May 17 07:18:20 2038 GMT + // Authority Key Identifier: + // 76:66:9E:F7:3B:DD:45:E5:3B:D9:72:3C:3F:F0:54:39:86:31:26:53 + "-----BEGIN CERTIFICATE-----\n" + + "MIIEnDCCBEGgAwIBAgIJAP/jh1qVhNVjMAsGCWCGSAFlAwQDAjA7MQswCQYDVQQG\n" + + "EwJVUzENMAsGA1UECgwESmF2YTEdMBsGA1UECwwUU3VuSlNTRSBUZXN0IFNlcml2\n" + + "Y2UwHhcNMTgwNTIyMDcxODIwWhcNMzgwNTE3MDcxODIwWjBVMQswCQYDVQQGEwJV\n" + + "UzENMAsGA1UECgwESmF2YTEdMBsGA1UECwwUU3VuSlNTRSBUZXN0IFNlcml2Y2Ux\n" + + "GDAWBgNVBAMMD1JlZ3Jlc3Npb24gVGVzdDCCA0cwggI6BgcqhkjOOAQBMIICLQKC\n" + + "AQEAmlavgoJrMcjqWRVcDE2dmWAPREgnzQvneEDef68cprDzjSwvOs5QeFyx75ib\n" + + "ado1e6jO/rW1prCGWHDD1oA/Tn4Pk3vu0nUxzvl1qATc+aJbpUU5Op0bvp6LbCsQ\n" + + "QslV9FeRh7Eb7bP6gpc/kHCBzEgC1VCK7prccXWy+t6SMOHbND3h+UbckfSaUuaV\n" + + "sVJNTD1D6GElfRj4Nmz1BGPfSYvKorwNZEU3gXwFgtDoAcGx7tcyClLpDHfqRfw/\n" + + "7yiqLyeiP7D4hl5lMNouJWDlAdMFp0FMgS3s9VDFinIcr6VtBWMTG7+4+czHAB+3\n" + + "fvrwlqNzhBn3uFHrekN/w8fNxwIhAJo7Sae1za7IMW0Q6hE5B4b+s2B/FaKPoA4E\n" + + "jtZu13B9AoIBAQCOZqLMKfvqZWUgT0PQ3QjR7dAFdd06I9Y3+TOQzZk1+j+vw/6E\n" + + "X4vFItX4gihb/u5Q9CdmpwhVGi7bvo+7+/IKeTgoQ6f5+PSug7SrWWUQ5sPwaZui\n" + + "zXZJ5nTeZDucFc2yFx0wgnjbPwiUxZklOT7xGiOMtzOTa2koCz5KuIBL+/wPKKxm\n" + + "ypo9VoY9xfbdU6LMXZv/lpD5XTM9rYHr/vUTNkukvV6Hpm0YMEWhVZKUJiqCqTqG\n" + + "XHaleOxSw6uQWB/+TznifcC7gB48UOQjCqOKf5VuwQneJLhlhU/jhRV3xtr+hLZa\n" + + "hW1wYhVi8cjLDrZFKlgEQqhB4crnJU0mJY+tA4IBBQACggEAID0ezl00/X8mv7eb\n" + + "bzovum1+DEEP7FM57k6HZEG2N3ve4CW+0m9Cd+cWPz8wkZ+M0j/Eqa6F0IdbkXEc\n" + + "Q7CuzvUyJ57xQ3L/WCgXsiS+Bh8O4Mz7GwW22CGmHqafbVv+hKBfr8MkskO6GJUt\n" + + "SUF/CVLzB4gMIvZMH26tBP2xK+i7FeEK9kT+nGdzQSZBAhFYpEVCBplHZO24/OYq\n" + + "1DNoU327nUuXIhmsfA8N0PjiWbIZIjTPwBGr9H0LpATI7DIDNcvRRvtROP+pBU9y\n" + + "fuykPkptg9C0rCM9t06bukpOSaEz/2VIQdLE8fHYFA6pHZ6CIc2+5cfvMgTPhcjz\n" + + "W2jCt6MjMCEwHwYDVR0jBBgwFoAUdmae9zvdReU72XI8P/BUOYYxJlMwCwYJYIZI\n" + + "AWUDBAMCA0gAMEUCIQCeI5fN08b9BpOaHdc3zQNGjp24FOL/RxlBLeBAorswJgIg\n" + + "JEZ8DhYxQy1O7mmZ2UIT7op6epWMB4dENjs0qWPmcKo=\n" + + "-----END CERTIFICATE-----" + }; + + // Private key in the format of PKCS#8. + final static String[] endEntityPrivateKeys = { + // + // EC private key related to cert endEntityCertStrs[0]. + // + "MIGHAgEAMBMGByqGSM49AgEGCCqGSM49AwEHBG0wawIBAQQgn5K03bpTLjEtFQRa\n" + + "JUtx22gtmGEvvSUSQdimhGthdtihRANCAARv72fTmp9ed8dRvTG1Ak1Lgl5KLoiM\n" + + "59bk2pyG8qd8l7L1WQnNHtAcu44RJ1/GVHurxghaCKHeJYsZ8H7DEeI6", + + // + // RSA private key related to cert endEntityCertStrs[1]. + // + "MIIEvQIBADANBgkqhkiG9w0BAQEFAASCBKcwggSjAgEAAoIBAQCzN8GhtZ9kinwC\n" + + "ALo+JYMNq8/rmZK+AJdQbEAJvFU0B+QuE1FA6weuJEHdfVDee870960LrAqW8T2O\n" + + "WcMOPsP7WPWvUIc7uazV+ryt9xBh2RqcO27gCUejy/5JJZfXowhcVFgOPZVUebI+\n" + + "2BuW1yDjMYC7BPvjoV0sj3i5eMmOrUg6v+/RUiXATk5FYfldz7lOP6gVYrQThrgY\n" + + "cOC0oqGfabvaE3nr8aQ8oaSC4/k1FLWphhX8QWR3JAw5XwnUzsdqJxJ+3g4vp4qN\n" + + "OuCj3f6QeT62/QNzpylUGdqVfdOisbxAjoFwePVdB3vygCUAL7+MXilzSbCLS6fj\n" + + "2R0r2BF9AgMBAAECggEASIkPkMCuw4WdTT44IwERus3IOIYOs2IP3BgEDyyvm4B6\n" + + "JP/iihDWKfA4zEl1Gqcni1RXMHswSglXra682J4kui02Ov+vzEeJIY37Ibn2YnP5\n" + + "ZjRT2s9GtI/S2o4hl8A/mQb2IMViFC+xKehTukhV4j5d6NPKk0XzLR7gcMjnYxwn\n" + + "l21fS6D2oM1xRG/di7sL+uLF8EXLRzfiWDNi12uQv4nwtxPKvuKhH6yzHt7YqMH0\n" + + "46pmDKDaxV4w1JdycjCb6NrCJOYZygoQobuZqOQ30UZoZsPJrtovkncFr1e+lNcO\n" + + "+aWDfOLCtTH046dEQh5oCShyXMybNlry/QHsOtHOwQKBgQDh2iIjs+FPpQy7Z3EX\n" + + "DGEvHYqPjrYO9an2KSRr1m9gzRlWYxKY46WmPKwjMerYtra0GP+TBHrgxsfO8tD2\n" + + "wUAII6sd1qup0a/Sutgf2JxVilLykd0+Ge4/Cs51tCdJ8EqDV2B6WhTewOY2EGvg\n" + + "JiKYkeNwgRX/9M9CFSAMAk0hUQKBgQDLJAartL3DoGUPjYtpJnfgGM23yAGl6G5r\n" + + "NSXDn80BiYIC1p0bG3N0xm3yAjqOtJAUj9jZbvDNbCe3GJfLARMr23legX4tRrgZ\n" + + "nEdKnAFKAKL01oM+A5/lHdkwaZI9yyv+hgSVdYzUjB8rDmzeVQzo1BT7vXypt2yV\n" + + "6O1OnUpCbQKBgA/0rzDChopv6KRcvHqaX0tK1P0rYeVQqb9ATNhpf9jg5Idb3HZ8\n" + + "rrk91BNwdVz2G5ZBpdynFl9G69rNAMJOCM4KZw5mmh4XOEq09Ivba8AHU7DbaTv3\n" + + "7QL7KnbaUWRB26HHzIMYVh0el6T+KADf8NXCiMTr+bfpfbL3dxoiF3zhAoGAbCJD\n" + + "Qse1dBs/cKYCHfkSOsI5T6kx52Tw0jS6Y4X/FOBjyqr/elyEexbdk8PH9Ar931Qr\n" + + "NKMvn8oA4iA/PRrXX7M2yi3YQrWwbkGYWYjtzrzEAdzmg+5eARKAeJrZ8/bg9l3U\n" + + "ttKaItJsDPlizn8rngy3FsJpR9aSAMK6/+wOiYkCgYEA1tZkI1rD1W9NYZtbI9BE\n" + + "qlJVFi2PBOJMKNuWdouPX3HLQ72GJSQff2BFzLTELjweVVJ0SvY4IipzpQOHQOBy\n" + + "5qh/p6izXJZh3IHtvwVBjHoEVplg1b2+I5e3jDCfqnwcQw82dW5SxOJMg1h/BD0I\n" + + "qAL3go42DYeYhu/WnECMeis=", + + // + // EC private key related to cert endEntityCertStrs[2]. + // + "MIGHAgEAMBMGByqGSM49AgEGCCqGSM49AwEHBG0wawIBAQQgGVc7hICpmp91jbYe\n" + + "nrr8nYHD37RZP3VENY+szuA7WjuhRANCAATn0wRE1OVVnV56mzxnc657lmSBB0+0\n" + + "P5YgTq2Pc9sgqnEY8PG980/3n0DJCMQr96FZBWsIyY2gSQjNj4ggGAHT", + + // + // DSA private key related to cert endEntityCertStrs[3]. + // + "MIICZQIBADCCAjoGByqGSM44BAEwggItAoIBAQCaVq+CgmsxyOpZFVwMTZ2ZYA9E\n" + + "SCfNC+d4QN5/rxymsPONLC86zlB4XLHvmJtp2jV7qM7+tbWmsIZYcMPWgD9Ofg+T\n" + + "e+7SdTHO+XWoBNz5olulRTk6nRu+notsKxBCyVX0V5GHsRvts/qClz+QcIHMSALV\n" + + "UIrumtxxdbL63pIw4ds0PeH5RtyR9JpS5pWxUk1MPUPoYSV9GPg2bPUEY99Ji8qi\n" + + "vA1kRTeBfAWC0OgBwbHu1zIKUukMd+pF/D/vKKovJ6I/sPiGXmUw2i4lYOUB0wWn\n" + + "QUyBLez1UMWKchyvpW0FYxMbv7j5zMcAH7d++vCWo3OEGfe4Uet6Q3/Dx83HAiEA\n" + + "mjtJp7XNrsgxbRDqETkHhv6zYH8Voo+gDgSO1m7XcH0CggEBAI5moswp++plZSBP\n" + + "Q9DdCNHt0AV13Toj1jf5M5DNmTX6P6/D/oRfi8Ui1fiCKFv+7lD0J2anCFUaLtu+\n" + + "j7v78gp5OChDp/n49K6DtKtZZRDmw/Bpm6LNdknmdN5kO5wVzbIXHTCCeNs/CJTF\n" + + "mSU5PvEaI4y3M5NraSgLPkq4gEv7/A8orGbKmj1Whj3F9t1Tosxdm/+WkPldMz2t\n" + + "gev+9RM2S6S9XoembRgwRaFVkpQmKoKpOoZcdqV47FLDq5BYH/5POeJ9wLuAHjxQ\n" + + "5CMKo4p/lW7BCd4kuGWFT+OFFXfG2v6EtlqFbXBiFWLxyMsOtkUqWARCqEHhyucl\n" + + "TSYlj60EIgIgLfA75+8KcKxdN8mr6gzGjQe7jPFGG42Ejhd7Q2F4wuw=" + }; + + // Private key algorithm of endEntityPrivateKeys. + final static String[] endEntityPrivateKeyAlgs = { + "EC", + "RSA", + "EC", + "DSA", + }; + + // Private key names of endEntityPrivateKeys. + static final String[] endEntityPrivateKeyNames = { + "ecdsa", + "rsa", + "ec-rsa", + "dsa", + }; + + /* + * Create an instance of SSLContext with the specified trust/key materials. + */ + private SSLContext createSSLContext( + String[] trustedMaterials, + String[] keyMaterialCerts, + String[] keyMaterialKeys, + String[] keyMaterialKeyAlgs, + String[] keyMaterialKeyNames, + ContextParameters params) throws Exception { + + KeyStore ts = null; // trust store + KeyStore ks = null; // key store + char passphrase[] = "passphrase".toCharArray(); + + // Generate certificate from cert string. + CertificateFactory cf = CertificateFactory.getInstance("X.509"); + + // Import the trused certs. + ByteArrayInputStream is; + if (trustedMaterials != null && trustedMaterials.length != 0) { + ts = KeyStore.getInstance("JKS"); + ts.load(null, null); + + Certificate[] trustedCert = + new Certificate[trustedMaterials.length]; + for (int i = 0; i < trustedMaterials.length; i++) { + String trustedCertStr = trustedMaterials[i]; + + is = new ByteArrayInputStream(trustedCertStr.getBytes()); + try { + trustedCert[i] = cf.generateCertificate(is); + } finally { + is.close(); + } + + ts.setCertificateEntry("trusted-cert-" + i, trustedCert[i]); + } + } + + // Import the key materials. + // + // Note that certification pathes bigger than one are not supported yet. + boolean hasKeyMaterials = + (keyMaterialCerts != null) && (keyMaterialCerts.length != 0) && + (keyMaterialKeys != null) && (keyMaterialKeys.length != 0) && + (keyMaterialKeyAlgs != null) && (keyMaterialKeyAlgs.length != 0) && + (keyMaterialCerts.length == keyMaterialKeys.length) && + (keyMaterialCerts.length == keyMaterialKeyAlgs.length); + if (hasKeyMaterials) { + ks = KeyStore.getInstance("JKS"); + ks.load(null, null); + + for (int i = 0; i < keyMaterialCerts.length; i++) { + String keyCertStr = keyMaterialCerts[i]; + + // generate the private key. + PKCS8EncodedKeySpec priKeySpec = new PKCS8EncodedKeySpec( + Base64.getMimeDecoder().decode(keyMaterialKeys[i])); + KeyFactory kf = + KeyFactory.getInstance(keyMaterialKeyAlgs[i]); + PrivateKey priKey = kf.generatePrivate(priKeySpec); + + // generate certificate chain + is = new ByteArrayInputStream(keyCertStr.getBytes()); + Certificate keyCert = null; + try { + keyCert = cf.generateCertificate(is); + } finally { + is.close(); + } + + Certificate[] chain = new Certificate[] { keyCert }; + + // import the key entry. + ks.setKeyEntry("cert-" + keyMaterialKeyNames[i], + priKey, passphrase, chain); + } + } + + // Create an SSLContext object. + TrustManagerFactory tmf = + TrustManagerFactory.getInstance(params.tmAlgorithm); + tmf.init(ts); + + SSLContext context = SSLContext.getInstance(params.contextProtocol); + if (hasKeyMaterials && ks != null) { + KeyManagerFactory kmf = + KeyManagerFactory.getInstance(params.kmAlgorithm); + kmf.init(ks, passphrase); + + context.init(kmf.getKeyManagers(), tmf.getTrustManagers(), null); + } else { + context.init(null, tmf.getTrustManagers(), null); + } + + return context; + } +} diff --git a/test/jdk/sun/security/ssl/SSLSocketImpl/SSLExceptionForIOIssue.java b/test/jdk/sun/security/ssl/SSLSocketImpl/SSLExceptionForIOIssue.java new file mode 100644 index 00000000000..3e626a2577c --- /dev/null +++ b/test/jdk/sun/security/ssl/SSLSocketImpl/SSLExceptionForIOIssue.java @@ -0,0 +1,162 @@ +/* + * Copyright (c) 2018, 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. + */ + +// +// Please run in othervm mode. SunJSSE does not support dynamic system +// properties, no way to re-use system properties in samevm/agentvm mode. +// + +/* + * @test + * @bug 8214339 + * @summary SSLSocketImpl erroneously wraps SocketException + * @library /javax/net/ssl/templates + * @run main/othervm SSLExceptionForIOIssue + */ + +import javax.net.ssl.*; +import java.io.*; +import java.net.InetAddress; + +public class SSLExceptionForIOIssue implements SSLContextTemplate { + + public static void main(String[] args) throws Exception { + System.err.println("==================================="); + new SSLExceptionForIOIssue().test(); + } + + private void test() throws Exception { + SSLServerSocket listenSocket = null; + SSLSocket serverSocket = null; + ClientSocket clientSocket = null; + try { + SSLServerSocketFactory serversocketfactory = + createServerSSLContext().getServerSocketFactory(); + listenSocket = + (SSLServerSocket)serversocketfactory.createServerSocket(0); + listenSocket.setNeedClientAuth(false); + listenSocket.setEnableSessionCreation(true); + listenSocket.setUseClientMode(false); + + System.err.println("Starting client"); + clientSocket = new ClientSocket(listenSocket.getLocalPort()); + clientSocket.start(); + + System.err.println("Accepting client requests"); + serverSocket = (SSLSocket)listenSocket.accept(); + + if (!clientSocket.isDone) { + System.err.println("Waiting 3 seconds for client "); + Thread.sleep(3000); + } + + System.err.println("Sending data to client ..."); + String serverData = "Hi, I am server"; + BufferedWriter os = new BufferedWriter( + new OutputStreamWriter(serverSocket.getOutputStream())); + os.write(serverData, 0, serverData.length()); + os.newLine(); + os.flush(); + } catch (SSLProtocolException | SSLHandshakeException sslhe) { + throw sslhe; + } catch (SSLException ssle) { + // the expected exception, ignore it + System.err.println("server exception: " + ssle); + } finally { + if (listenSocket != null) { + listenSocket.close(); + } + + if (serverSocket != null) { + serverSocket.close(); + } + } + + if (clientSocket != null && clientSocket.clientException != null) { + throw clientSocket.clientException; + } + } + + + + private class ClientSocket extends Thread{ + boolean isDone = false; + int serverPort = 0; + Exception clientException; + + public ClientSocket(int serverPort) { + this.serverPort = serverPort; + } + + @Override + public void run() { + SSLSocket clientSocket = null; + String clientData = "Hi, I am client"; + try { + System.err.println( + "Connecting to server at port " + serverPort); + SSLSocketFactory sslSocketFactory = + createClientSSLContext().getSocketFactory(); + clientSocket = (SSLSocket)sslSocketFactory.createSocket( + InetAddress.getLocalHost(), serverPort); + clientSocket.setSoLinger(true, 3); + clientSocket.setSoTimeout(100); + + + System.err.println("Sending data to server ..."); + + BufferedWriter os = new BufferedWriter( + new OutputStreamWriter(clientSocket.getOutputStream())); + os.write(clientData, 0, clientData.length()); + os.newLine(); + os.flush(); + + System.err.println("Reading data from server"); + BufferedReader is = new BufferedReader( + new InputStreamReader(clientSocket.getInputStream())); + String data = is.readLine(); + System.err.println("Received Data from server: " + data); + } catch (SSLProtocolException | SSLHandshakeException sslhe) { + clientException = sslhe; + System.err.println("unexpected client exception: " + sslhe); + } catch (SSLException ssle) { + // the expected exception, ignore it + System.err.println("expected client exception: " + ssle); + } catch (Exception e) { + clientException = e; + System.err.println("unexpected client exception: " + e); + } finally { + if (clientSocket != null) { + try { + clientSocket.close(); + System.err.println("client socket closed"); + } catch (IOException ioe) { + clientException = ioe; + } + } + + isDone = true; + } + } + } +} From 1af1a69fbcd3169cb8aea56ea0886a6f3c7ec091 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Per=20Lid=C3=A9n?= Date: Sat, 15 Dec 2018 20:03:06 +0100 Subject: [PATCH 012/101] 8215395: Allow null oops in Dictionary and JNIHandle verification Reviewed-by: eosterlund, kbarrett, coleenp --- src/hotspot/share/classfile/dictionary.hpp | 2 +- src/hotspot/share/runtime/jniHandles.cpp | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/src/hotspot/share/classfile/dictionary.hpp b/src/hotspot/share/classfile/dictionary.hpp index 4c0e91046c2..087b5e6c62f 100644 --- a/src/hotspot/share/classfile/dictionary.hpp +++ b/src/hotspot/share/classfile/dictionary.hpp @@ -173,7 +173,7 @@ class DictionaryEntry : public HashtableEntry { for (ProtectionDomainEntry* current = pd_set(); // accessed at a safepoint current != NULL; current = current->_next) { - guarantee(oopDesc::is_oop(current->_pd_cache->object_no_keepalive()), "Invalid oop"); + guarantee(oopDesc::is_oop_or_null(current->_pd_cache->object_no_keepalive()), "Invalid oop"); } } diff --git a/src/hotspot/share/runtime/jniHandles.cpp b/src/hotspot/share/runtime/jniHandles.cpp index 6b1692fe189..ca09c575150 100644 --- a/src/hotspot/share/runtime/jniHandles.cpp +++ b/src/hotspot/share/runtime/jniHandles.cpp @@ -318,7 +318,7 @@ void JNIHandles::print_on(outputStream* st) { class VerifyJNIHandles: public OopClosure { public: virtual void do_oop(oop* root) { - guarantee(oopDesc::is_oop(RawAccess<>::oop_load(root)), "Invalid oop"); + guarantee(oopDesc::is_oop_or_null(RawAccess<>::oop_load(root)), "Invalid oop"); } virtual void do_oop(narrowOop* root) { ShouldNotReachHere(); } }; From 2ebc9f99e5f344b19a6172ee5cab26e13a899d0d Mon Sep 17 00:00:00 2001 From: Robert Field Date: Sat, 15 Dec 2018 17:13:39 -0800 Subject: [PATCH 013/101] 8215099: jshell tool: /help representation of ctrl/meta characters inconsistent Reviewed-by: jlahoda --- .../jshell/tool/resources/l10n.properties | 206 +++++++++--------- .../jdk/jshell/ToolShiftTabTest.java | 2 +- test/langtools/jdk/jshell/ToolSimpleTest.java | 5 +- 3 files changed, 111 insertions(+), 102 deletions(-) diff --git a/src/jdk.jshell/share/classes/jdk/internal/jshell/tool/resources/l10n.properties b/src/jdk.jshell/share/classes/jdk/internal/jshell/tool/resources/l10n.properties index e4cddba1142..bfb85ed6a7d 100644 --- a/src/jdk.jshell/share/classes/jdk/internal/jshell/tool/resources/l10n.properties +++ b/src/jdk.jshell/share/classes/jdk/internal/jshell/tool/resources/l10n.properties @@ -187,12 +187,12 @@ jshell.console.create.method = Create method jshell.console.resolvable = \nThe identifier is resolvable in this context. jshell.console.no.candidate = \nNo candidate fully qualified names found to import. jshell.console.incomplete = \nResults may be incomplete; try again later for complete results. -jshell.console.erroneous = \nIncomplete or erroneous. A single valid expression or statement must proceed Shift- m. -jshell.console.exprstmt = \nA single valid expression or statement must proceed Shift- m. -jshell.console.empty = \nEmpty entry. A single valid expression or statement must proceed Shift- m.. +jshell.console.erroneous = \nIncomplete or erroneous. A single valid expression or statement must precede Shift+Tab m. +jshell.console.exprstmt = \nA single valid expression or statement must precede Shift+Tab m. +jshell.console.empty = \nEmpty entry. A single valid expression or statement must precede Shift+Tab m. jshell.fix.wrong.shortcut =\ -Unexpected character after Shift-Tab.\n\ +Unexpected character after Shift+Tab.\n\ Use "i" for auto-import, "v" for variable creation, or "m" for method creation.\n\ For more information see:\n\ /help shortcuts @@ -579,92 +579,92 @@ or prior commands/snippets can be retrieved from history, edited, and executed.\ This support is similar to readline/editline with simple emacs-like bindings.\n\ There are also jshell tool specific key sequences.\n\ \n\ ---- Line and history navigation ---\n\ -\n\ -Return\n\t\ - Enters the current snippet\n\ -Left-arrow or Ctrl+B\n\t\ - Moves backward one character\n\ -Right-arrow or Ctrl+F\n\t\ - Moves forward one character\n\ -Up-arrow or Ctrl+P\n\t\ - Moves up one line, backward through history\n\ -Down arrow or Ctrl+N\n\t\ - Moves down one line, forward through history\n\ -Ctrl+A\n\t\ - Moves to the beginning of the line\n\ -Ctrl+E\n\t\ - Moves to the end of the line\n\ -Meta+B\n\t\ - Moves backward one word\n\ -Meta+F\n\t\ - Moves forward one word\n\ -Ctrl+R\n\t\ +Line and history navigation:\n\ +\n\t\ +Return\n\t\t\ + Enters the current snippet\n\t\ +Left-arrow or Ctrl+B\n\t\t\ + Moves backward one character\n\t\ +Right-arrow or Ctrl+F\n\t\t\ + Moves forward one character\n\t\ +Up-arrow or Ctrl+P\n\t\t\ + Moves up one line, backward through history\n\t\ +Down arrow or Ctrl+N\n\t\t\ + Moves down one line, forward through history\n\t\ +Ctrl+A\n\t\t\ + Moves to the beginning of the line\n\t\ +Ctrl+E\n\t\t\ + Moves to the end of the line\n\t\ +Meta+B\n\t\t\ + Moves backward one word\n\t\ +Meta+F\n\t\t\ + Moves forward one word\n\t\ +Ctrl+R\n\t\t\ Search backward through history\n\ \n\ \n\ ---- Line and history basic editing ---\n\ -\n\ -Meta+Return or Ctrl+Return (depending on platform)\n\t\ - Insert a new line in snippet\n\ -Ctrl+_ (underscore may require shift key) or Ctrl+X then Ctrl+U\n\t\ - Undo edit - repeat to undo more edits\n\ -Delete\n\t\ - Deletes the character at or after the cursor, depending on the operating system\n\ -Backspace\n\t\ - Deletes the character before the cursor\n\ -Ctrl+K\n\t\ - Deletes the text from the cursor to the end of the line\n\ -Meta+D\n\t\ - Deletes the text from the cursor to the end of the word\n\ -Ctrl+W\n\t\ - Deletes the text from the cursor to the previous white space\n\ -Ctrl+Y\n\t\ - Pastes the most recently deleted text into the line\n\ -Meta+Y\n\t\ - After Ctrl+Y, Meta+Y cycles through previously deleted text\n\ -Ctrl+X then Ctrl+K\n\t\ +Line and history basic editing:\n\ +\n\t\ +Meta+Return or Ctrl+Return (depending on platform)\n\t\t\ + Insert a new line in snippet\n\t\ +Ctrl+_ (underscore may require the Shift key) or Ctrl+X then Ctrl+U\n\t\t\ + Undo edit - repeat to undo more edits\n\t\ +Delete\n\t\t\ + Deletes the character at or after the cursor, depending on the operating system\n\t\ +Backspace\n\t\t\ + Deletes the character before the cursor\n\t\ +Ctrl+K\n\t\t\ + Deletes the text from the cursor to the end of the line\n\t\ +Meta+D\n\t\t\ + Deletes the text from the cursor to the end of the word\n\t\ +Ctrl+W\n\t\t\ + Deletes the text from the cursor to the previous white space\n\t\ +Ctrl+Y\n\t\t\ + Pastes the most recently deleted text into the line\n\t\ +Meta+Y\n\t\t\ + After Ctrl+Y, Meta+Y cycles through previously deleted text\n\t\ +Ctrl+X then Ctrl+K\n\t\t\ Delete whole snippet\n\ \n\ \n\ ---- Shortcuts for jshell tool ---\n\ -\n\ +Shortcuts for jshell tool:\n\ +\n\t\ For details, see: /help shortcuts\n\ -\n\ -Tab\n\t\ - Complete Java identifier or jshell command\n\ -Shift+Tab then v\n\t\ - Convert expression to variable declaration\n\ -Shift+Tab then m\n\t\ - Convert statement to method declaration\n\ -Shift+Tab then i\n\t\ - Add imports for this identifier\n\ +\n\t\ +Tab\n\t\t\ + Complete Java identifier or jshell command\n\t\ +Shift+Tab then v\n\t\t\ + Convert expression to variable declaration\n\t\ +Shift+Tab then m\n\t\t\ + Convert statement to method declaration\n\t\ +Shift+Tab then i\n\t\t\ + Add imports for this identifier\n\t\ \n\ \n\ ---- More line and history editing ---\n\ -\n\ -Ctrl+L\n\t\ - Clear screen and reprint snippet\n\ -Ctrl+U\n\t\ - Kill whole line\n\ -Ctrl+T\n\t\ - Transpose characters\n\ -Ctrl+X then Ctrl+B\n\t\ - Navigate to matching bracket, parenthesis, ...\n\ -Ctrl+X then =\n\t\ - Enter show current character position mode\n\ -Ctrl+X then Ctrl+O\n\t\ - Toggle overwrite characters vs insert characters\n\ -Meta+C\n\t\ - Capitalize word\n\ -Meta+U\n\t\ - Convert word to uppercase\n\ -Meta+L\n\t\ - Convert word to lowercase\n\ -Meta+0 through Meta+9 then key\n\t\ +More line and history editing:\n\ +\n\t\ +Ctrl+L\n\t\t\ + Clear screen and reprint snippet\n\t\ +Ctrl+U\n\t\t\ + Kill whole line\n\t\ +Ctrl+T\n\t\t\ + Transpose characters\n\t\ +Ctrl+X then Ctrl+B\n\t\t\ + Navigate to matching bracket, parenthesis, ...\n\t\ +Ctrl+X then =\n\t\t\ + Enter show current character position mode\n\t\ +Ctrl+X then Ctrl+O\n\t\t\ + Toggle overwrite characters vs insert characters\n\t\ +Meta+C\n\t\t\ + Capitalize word\n\t\ +Meta+U\n\t\t\ + Convert word to uppercase\n\t\ +Meta+L\n\t\t\ + Convert word to lowercase\n\t\ +Meta+0 through Meta+9 then key\n\t\t\ Repeat the specified number of times\n\ \n\ -Where, for example, "Ctrl+A" means hold down the control key and press A.\n\ +Where, for example, "Ctrl+A" means hold down the Control key and press A.\n\ Where "Meta" is "Alt" on many keyboards.\n\ Line editing support is derived from JLine 3. @@ -672,26 +672,34 @@ help.shortcuts.summary = a description of keystrokes for snippet and command com information access, and automatic code generation help.shortcuts =\ Supported shortcuts include:\n\ -\n\ -\n\t\t\ +\n\t\ +Tab\n\t\t\ After entering the first few letters of a Java identifier,\n\t\t\ - a jshell tool command, or, in some cases, a jshell tool command argument,\n\t\t\ - press the key to complete the input.\n\t\t\ - If there is more than one completion, then possible completions will be shown.\n\t\t\ - Will show documentation if available and appropriate.\n\n\ -Shift- v\n\t\t\ - After a complete expression, hold down while pressing ,\n\t\t\ - then release and press "v", the expression will be converted to\n\t\t\ - a variable declaration whose type is based on the type of the expression.\n\n\ -Shift- m\n\t\t\ - After a complete expression or statement, hold down while pressing ,\n\t\t\ - then release and press "m", the expression or statement will be converted to\n\t\t\ - a method declaration. If an expression, the return type is based on the type\n\t\t\ - of the expression.\n\n\ -Shift- i\n\t\t\ - After an unresolvable identifier, hold down while pressing ,\n\t\t\ - then release and press "i", and the jshell tool will propose possible imports\n\t\t\ - which will resolve the identifier based on the content of the specified classpath. + a jshell tool command, or, in some cases, a jshell tool\n\t\t\ + command argument, press the Tab key to complete the input.\n\t\t\ + If there is more than one completion, then possible completions\n\t\t\ + will be shown.\n\t\t\ + Another Tab will show documentation if available and appropriate.\n\n\t\ +Shift+Tab then v\n\t\t\ + After a complete expression, hold down the Shift key while\n\t\t\ + pressing the Tab key, then release and press the "v" key,\n\t\t\ + the expression will be converted to a variable declaration \n\t\t\ + whose type is based on the type of the expression.\n\n\t\ +Shift+Tab then m\n\t\t\ + After a complete expression or statement, hold down the\n\t\t\ + Shift key while pressing the Tab key, then release and press\n\t\t\ + the "m" key, the expression or statement will be converted\n\t\t\ + to a method declaration. If it is an expression, the method\n\t\t\ + return type will be based on the type of the expression.\n\n\t\ +Shift+Tab then i\n\t\t\ + After an unresolvable identifier, hold down the Shift key\n\t\t\ + while pressing the Tab key, then release and press the "i" key,\n\t\t\ + and the jshell tool will propose possible imports which will\n\t\t\ + resolve the identifier based on the content of the specified\n\t\t\ + classpath. Enter the digit corresponding to the desired import,\n\t\t\ + or press the "0" key to add no imports.\n\ +\n\ +For information onother special keys see: /help keys help.context.summary = a description of the evaluation context options for /env /reload and /reset help.context =\ @@ -789,7 +797,7 @@ Startup and error snippets maybe used, e.g.: s3-s9 or e1-e4\n\ Any number of IDs or ID ranges may be used, e.g.: /3-7 s4 14-16 e2\n\ See also '/help id'.\n\ \n\ -Finally, you can search backwards through history by entering ctrl-R followed by the string to search for. +Finally, you can search backwards through history by entering Ctrl+R followed by the string to search for. help.set._retain = \ The '-retain' option saves a setting so that it is used in future sessions.\n\ diff --git a/test/langtools/jdk/jshell/ToolShiftTabTest.java b/test/langtools/jdk/jshell/ToolShiftTabTest.java index 8ed785537c0..b668ff500c1 100644 --- a/test/langtools/jdk/jshell/ToolShiftTabTest.java +++ b/test/langtools/jdk/jshell/ToolShiftTabTest.java @@ -126,7 +126,7 @@ public class ToolShiftTabTest extends UITesting { doRunTest((inputSink, out) -> { inputSink.write("123"); inputSink.write(FIX + "z"); - waitOutput(out, "Unexpected character after Shift-Tab"); + waitOutput(out, "Unexpected character after Shift\\+Tab"); }); } } diff --git a/test/langtools/jdk/jshell/ToolSimpleTest.java b/test/langtools/jdk/jshell/ToolSimpleTest.java index 86aae2b490c..cb1c964a3a8 100644 --- a/test/langtools/jdk/jshell/ToolSimpleTest.java +++ b/test/langtools/jdk/jshell/ToolSimpleTest.java @@ -23,7 +23,7 @@ /* * @test - * @bug 8153716 8143955 8151754 8150382 8153920 8156910 8131024 8160089 8153897 8167128 8154513 8170015 8170368 8172102 8172103 8165405 8173073 8173848 8174041 8173916 8174028 8174262 8174797 8177079 8180508 8177466 8172154 8192979 8191842 8198573 8198801 8210596 8210959 + * @bug 8153716 8143955 8151754 8150382 8153920 8156910 8131024 8160089 8153897 8167128 8154513 8170015 8170368 8172102 8172103 8165405 8173073 8173848 8174041 8173916 8174028 8174262 8174797 8177079 8180508 8177466 8172154 8192979 8191842 8198573 8198801 8210596 8210959 8215099 * @summary Simple jshell tool tests * @modules jdk.compiler/com.sun.tools.javac.api * jdk.compiler/com.sun.tools.javac.main @@ -448,7 +448,8 @@ public class ToolSimpleTest extends ReplToolTesting { test( (a) -> assertHelp(a, "/?", "/list", "/help", "/exit", "intro"), (a) -> assertHelp(a, "/help", "/list", "/help", "/exit", "intro"), - (a) -> assertHelp(a, "/help short", "shortcuts", ""), + (a) -> assertHelp(a, "/help short", "shortcuts", "Tab"), + (a) -> assertHelp(a, "/help keys", "line", "Shift", "imports", "history"), (a) -> assertHelp(a, "/? /li", "/list -all", "snippets"), (a) -> assertHelp(a, "/help /set prompt", "optionally contain '%s'", "quoted"), (a) -> assertHelp(a, "/help /help", "/help "), From 2e989366ea703f6eaa330f8a611ba01f27ef9997 Mon Sep 17 00:00:00 2001 From: Tobias Hartmann Date: Mon, 17 Dec 2018 08:25:57 +0100 Subject: [PATCH 014/101] 8215410: Regression test for JDK-8214994 Added a regression test for 8214994 which was fixed by accident with 8211451. Reviewed-by: kvn --- .../compiler/loopopts/TestSplitIfOpaque1.java | 97 +++++++++++++++++++ 1 file changed, 97 insertions(+) create mode 100644 test/hotspot/jtreg/compiler/loopopts/TestSplitIfOpaque1.java diff --git a/test/hotspot/jtreg/compiler/loopopts/TestSplitIfOpaque1.java b/test/hotspot/jtreg/compiler/loopopts/TestSplitIfOpaque1.java new file mode 100644 index 00000000000..e1896824d05 --- /dev/null +++ b/test/hotspot/jtreg/compiler/loopopts/TestSplitIfOpaque1.java @@ -0,0 +1,97 @@ +/* + * Copyright (c) 2018, 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 8214994 + * @summary The split-if optimization fails because an unexpected Opaque1 is encountered. + * @run main/othervm -XX:+IgnoreUnrecognizedVMOptions -XX:-UseOnStackReplacement -XX:-BackgroundCompilation + * compiler.loopopts.TestSplitIfOpaque1 + */ + +package compiler.loopopts; + +public class TestSplitIfOpaque1 { + + static class MyClass { + int f; + MyClass(int f) { + this.f = f; + } + } + + // The inner loop is found to be a counted loop and a loop limit check + // is added by updating the Opaque1 input of the predicate template. + // This Opaque1 node is then pushed upwards to before the predicates of + // the outer loop and ends up right after the second 'if (b)'. + // The split-if optimization kicks in and splits the first predicate of + // the outer loop (an "obj != NULL" check) through the 'if (b)' region. + // We fail because the region contains an unexpected Opaque1 node. + static int test1(boolean b, int limit, MyClass obj) { + int res = 0; + MyClass notNull = new MyClass(42); + if (b) { + limit = 100; + } + if (b) { + obj = notNull; + } + for (int i = 0; i < 1000; ++i) { + res += obj.f; + for (int j = 0; j <= limit; ++j) { + // Empty + } + } + return res; + } + + // Same as test1 but triggers slightly different failure mode + static int test2(boolean b, int limit, MyClass obj, int[] array) { + int res = 0; + MyClass notNull = new MyClass(12); + if (b) { + limit = 100; + } + if (b) { + obj = notNull; + } + for (int i = 0; i < 1000; ++i) { + res += obj.f; + for (int j = 0; j <= limit; ++j) { + array[j] = j; + } + } + return res; + } + + public static void main(String[] args) { + MyClass obj = new MyClass(42); + int[] array = new int[101]; + for (int i = 0; i < 20_000; i++) { + test1(true, 50, obj); + test1(false, 100, obj); + test2(true, 50, obj, array); + test2(false, 100, obj, array); + } + } +} From 6a9c6a0392a0121db8c8e5c6d3e5637fb8db0f2a Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Per=20Lid=C3=A9n?= Date: Mon, 17 Dec 2018 16:55:17 +0100 Subject: [PATCH 015/101] 8215451: JNI IsSameObject should not keep objects alive Reviewed-by: eosterlund, kbarrett --- src/hotspot/share/prims/jni.cpp | 4 +--- src/hotspot/share/runtime/jniHandles.cpp | 8 +------ src/hotspot/share/runtime/jniHandles.hpp | 9 +++++-- .../share/runtime/jniHandles.inline.hpp | 24 +++++++++++++++---- .../MyPackage/HeapMonitorThreadTest.java | 1 - 5 files changed, 28 insertions(+), 18 deletions(-) diff --git a/src/hotspot/share/prims/jni.cpp b/src/hotspot/share/prims/jni.cpp index 1278f1d9f3a..00f66da0219 100644 --- a/src/hotspot/share/prims/jni.cpp +++ b/src/hotspot/share/prims/jni.cpp @@ -823,9 +823,7 @@ JNI_QUICK_ENTRY(jboolean, jni_IsSameObject(JNIEnv *env, jobject r1, jobject r2)) HOTSPOT_JNI_ISSAMEOBJECT_ENTRY(env, r1, r2); - oop a = JNIHandles::resolve(r1); - oop b = JNIHandles::resolve(r2); - jboolean ret = oopDesc::equals(a, b) ? JNI_TRUE : JNI_FALSE; + jboolean ret = JNIHandles::is_same_object(r1, r2) ? JNI_TRUE : JNI_FALSE; HOTSPOT_JNI_ISSAMEOBJECT_RETURN(ret); return ret; diff --git a/src/hotspot/share/runtime/jniHandles.cpp b/src/hotspot/share/runtime/jniHandles.cpp index ca09c575150..a165e1e1948 100644 --- a/src/hotspot/share/runtime/jniHandles.cpp +++ b/src/hotspot/share/runtime/jniHandles.cpp @@ -152,17 +152,11 @@ jobject JNIHandles::make_weak_global(Handle obj, AllocFailType alloc_failmode) { oop JNIHandles::resolve_external_guard(jobject handle) { oop result = NULL; if (handle != NULL) { - result = resolve_impl(handle); + result = resolve_impl<0 /* decorators */, true /* external_guard */>(handle); } return result; } -oop JNIHandles::resolve_jweak(jweak handle) { - assert(handle != NULL, "precondition"); - assert(is_jweak(handle), "precondition"); - return NativeAccess::oop_load(jweak_ptr(handle)); -} - bool JNIHandles::is_global_weak_cleared(jweak handle) { assert(handle != NULL, "precondition"); assert(is_jweak(handle), "not a weak handle"); diff --git a/src/hotspot/share/runtime/jniHandles.hpp b/src/hotspot/share/runtime/jniHandles.hpp index 3c6be3b6b1a..4c8c9fc1148 100644 --- a/src/hotspot/share/runtime/jniHandles.hpp +++ b/src/hotspot/share/runtime/jniHandles.hpp @@ -42,8 +42,10 @@ class JNIHandles : AllStatic { inline static oop* jobject_ptr(jobject handle); // NOT jweak! inline static oop* jweak_ptr(jobject handle); - template inline static oop resolve_impl(jobject handle); - static oop resolve_jweak(jweak handle); + template inline static oop resolve_impl(jobject handle); + + // Resolve handle into oop, without keeping the object alive + inline static oop resolve_no_keepalive(jobject handle); // This method is not inlined in order to avoid circular includes between // this header file and thread.hpp. @@ -70,6 +72,9 @@ class JNIHandles : AllStatic { // Resolve externally provided handle into oop with some guards static oop resolve_external_guard(jobject handle); + // Check for equality without keeping objects alive + static bool is_same_object(jobject handle1, jobject handle2); + // Local handles static jobject make_local(oop obj); static jobject make_local(JNIEnv* env, oop obj); // Fast version when env is known diff --git a/src/hotspot/share/runtime/jniHandles.inline.hpp b/src/hotspot/share/runtime/jniHandles.inline.hpp index d8b9c7770b1..43e284196c0 100644 --- a/src/hotspot/share/runtime/jniHandles.inline.hpp +++ b/src/hotspot/share/runtime/jniHandles.inline.hpp @@ -49,15 +49,15 @@ inline oop* JNIHandles::jweak_ptr(jobject handle) { } // external_guard is true if called from resolve_external_guard. -template +template inline oop JNIHandles::resolve_impl(jobject handle) { assert(handle != NULL, "precondition"); assert(!current_thread_in_native(), "must not be in native"); oop result; if (is_jweak(handle)) { // Unlikely - result = resolve_jweak(handle); + result = NativeAccess::oop_load(jweak_ptr(handle)); } else { - result = NativeAccess<>::oop_load(jobject_ptr(handle)); + result = NativeAccess::oop_load(jobject_ptr(handle)); // Construction of jobjects canonicalize a null value into a null // jobject, so for non-jweak the pointee should never be null. assert(external_guard || result != NULL, "Invalid JNI handle"); @@ -68,14 +68,28 @@ inline oop JNIHandles::resolve_impl(jobject handle) { inline oop JNIHandles::resolve(jobject handle) { oop result = NULL; if (handle != NULL) { - result = resolve_impl(handle); + result = resolve_impl<0 /* decorators */, false /* external_guard */>(handle); } return result; } +inline oop JNIHandles::resolve_no_keepalive(jobject handle) { + oop result = NULL; + if (handle != NULL) { + result = resolve_impl(handle); + } + return result; +} + +inline bool JNIHandles::is_same_object(jobject handle1, jobject handle2) { + oop obj1 = resolve_no_keepalive(handle1); + oop obj2 = resolve_no_keepalive(handle2); + return oopDesc::equals(obj1, obj2); +} + inline oop JNIHandles::resolve_non_null(jobject handle) { assert(handle != NULL, "JNI handle should not be null"); - oop result = resolve_impl(handle); + oop result = resolve_impl<0 /* decorators */, false /* external_guard */>(handle); assert(result != NULL, "NULL read from jni handle"); return result; } diff --git a/test/hotspot/jtreg/serviceability/jvmti/HeapMonitor/MyPackage/HeapMonitorThreadTest.java b/test/hotspot/jtreg/serviceability/jvmti/HeapMonitor/MyPackage/HeapMonitorThreadTest.java index 31e38e1e646..12892dca542 100644 --- a/test/hotspot/jtreg/serviceability/jvmti/HeapMonitor/MyPackage/HeapMonitorThreadTest.java +++ b/test/hotspot/jtreg/serviceability/jvmti/HeapMonitor/MyPackage/HeapMonitorThreadTest.java @@ -30,7 +30,6 @@ package MyPackage; * @summary Verifies the JVMTI Heap Monitor Thread information sanity. * @compile HeapMonitorThreadTest.java * @run main/othervm/native -Xmx512m -agentlib:HeapMonitorTest MyPackage.HeapMonitorThreadTest - * @requires !vm.gc.Z */ import java.util.List; From a3bb5ead7eb3fd3465e97196adfbaaa622672203 Mon Sep 17 00:00:00 2001 From: Nick Gasson Date: Mon, 17 Dec 2018 10:36:07 -0800 Subject: [PATCH 016/101] 8214512: ARM32: Jtreg test compiler/c2/Test8062950.java fails on ARM Reviewed-by: dlong, enevill, bulasevich --- src/hotspot/cpu/arm/arm.ad | 18 +++++++++++++++++- src/hotspot/cpu/arm/macroAssembler_arm.cpp | 13 +++++++++---- src/hotspot/cpu/arm/macroAssembler_arm.hpp | 10 +++++----- 3 files changed, 31 insertions(+), 10 deletions(-) diff --git a/src/hotspot/cpu/arm/arm.ad b/src/hotspot/cpu/arm/arm.ad index 43d771ea050..49394291888 100644 --- a/src/hotspot/cpu/arm/arm.ad +++ b/src/hotspot/cpu/arm/arm.ad @@ -8945,9 +8945,10 @@ instruct partialSubtypeCheck( R0RegP index, R1RegP sub, R2RegP super, flagsRegP instruct cmpFastLock(flagsRegP pcc, iRegP object, iRegP box, iRegP scratch2, iRegP scratch ) %{ match(Set pcc (FastLock object box)); + predicate(!(UseBiasedLocking && !UseOptoBiasInlining)); effect(TEMP scratch, TEMP scratch2); - ins_cost(100); + ins_cost(DEFAULT_COST*3); format %{ "FASTLOCK $object, $box; KILL $scratch, $scratch2" %} ins_encode %{ @@ -8956,6 +8957,21 @@ instruct cmpFastLock(flagsRegP pcc, iRegP object, iRegP box, iRegP scratch2, iRe ins_pipe(long_memory_op); %} +instruct cmpFastLock_noBiasInline(flagsRegP pcc, iRegP object, iRegP box, iRegP scratch2, + iRegP scratch, iRegP scratch3) %{ + match(Set pcc (FastLock object box)); + predicate(UseBiasedLocking && !UseOptoBiasInlining); + + effect(TEMP scratch, TEMP scratch2, TEMP scratch3); + ins_cost(DEFAULT_COST*5); + + format %{ "FASTLOCK $object, $box; KILL $scratch, $scratch2, $scratch3" %} + ins_encode %{ + __ fast_lock($object$$Register, $box$$Register, $scratch$$Register, $scratch2$$Register, $scratch3$$Register); + %} + ins_pipe(long_memory_op); +%} + instruct cmpFastUnlock(flagsRegP pcc, iRegP object, iRegP box, iRegP scratch2, iRegP scratch ) %{ match(Set pcc (FastUnlock object box)); diff --git a/src/hotspot/cpu/arm/macroAssembler_arm.cpp b/src/hotspot/cpu/arm/macroAssembler_arm.cpp index 0840f584ac9..a09fa56e5a9 100644 --- a/src/hotspot/cpu/arm/macroAssembler_arm.cpp +++ b/src/hotspot/cpu/arm/macroAssembler_arm.cpp @@ -1971,7 +1971,7 @@ void MacroAssembler::resolve(DecoratorSet decorators, Register obj) { #ifdef COMPILER2 -void MacroAssembler::fast_lock(Register Roop, Register Rbox, Register Rscratch, Register Rscratch2) +void MacroAssembler::fast_lock(Register Roop, Register Rbox, Register Rscratch, Register Rscratch2, Register scratch3) { assert(VM_Version::supports_ldrex(), "unsupported, yet?"); @@ -1985,11 +1985,13 @@ void MacroAssembler::fast_lock(Register Roop, Register Rbox, Register Rscratch, Label fast_lock, done; if (UseBiasedLocking && !UseOptoBiasInlining) { - Label failed; - biased_locking_enter(Roop, Rmark, Rscratch, false, noreg, done, failed); - bind(failed); + assert(scratch3 != noreg, "need extra temporary for -XX:-UseOptoBiasInlining"); + biased_locking_enter(Roop, Rmark, Rscratch, false, scratch3, done, done); + // Fall through if lock not biased otherwise branch to done } + // Invariant: Rmark loaded below does not contain biased lock pattern + ldr(Rmark, Address(Roop, oopDesc::mark_offset_in_bytes())); tst(Rmark, markOopDesc::unlocked_value); b(fast_lock, ne); @@ -2016,6 +2018,9 @@ void MacroAssembler::fast_lock(Register Roop, Register Rbox, Register Rscratch, bind(done); + // At this point flags are set as follows: + // EQ -> Success + // NE -> Failure, branch to slow path } void MacroAssembler::fast_unlock(Register Roop, Register Rbox, Register Rscratch, Register Rscratch2) diff --git a/src/hotspot/cpu/arm/macroAssembler_arm.hpp b/src/hotspot/cpu/arm/macroAssembler_arm.hpp index 86c4f81f2bb..114246a85aa 100644 --- a/src/hotspot/cpu/arm/macroAssembler_arm.hpp +++ b/src/hotspot/cpu/arm/macroAssembler_arm.hpp @@ -371,10 +371,10 @@ public: // lock_reg and obj_reg must be loaded up with the appropriate values. // swap_reg must be supplied. // tmp_reg must be supplied. - // Optional slow case is for implementations (interpreter and C1) which branch to - // slow case directly. If slow_case is NULL, then leaves condition - // codes set (for C2's Fast_Lock node) and jumps to done label. - // Falls through for the fast locking attempt. + // Done label is branched to with condition code EQ set if the lock is + // biased and we acquired it. Slow case label is branched to with + // condition code NE set if the lock is biased but we failed to acquire + // it. Otherwise fall through. // Returns offset of first potentially-faulting instruction for null // check info (currently consumed only by C1). If // swap_reg_contains_mark is true then returns -1 as it is assumed @@ -1073,7 +1073,7 @@ public: void restore_default_fp_mode(); #ifdef COMPILER2 - void fast_lock(Register obj, Register box, Register scratch, Register scratch2); + void fast_lock(Register obj, Register box, Register scratch, Register scratch2, Register scratch3 = noreg); void fast_unlock(Register obj, Register box, Register scratch, Register scratch2); #endif From 2fa63331130ea38d984805fe57b4b2c57e60dd10 Mon Sep 17 00:00:00 2001 From: Muthusamy Chinnathambi Date: Tue, 18 Dec 2018 12:59:39 +0530 Subject: [PATCH 017/101] 8215397: jsig.c missing classpath exception Reviewed-by: dholmes --- src/java.base/unix/native/libjsig/jsig.c | 4 +++- 1 file changed, 3 insertions(+), 1 deletion(-) diff --git a/src/java.base/unix/native/libjsig/jsig.c b/src/java.base/unix/native/libjsig/jsig.c index e0972174191..a8ab454dbe7 100644 --- a/src/java.base/unix/native/libjsig/jsig.c +++ b/src/java.base/unix/native/libjsig/jsig.c @@ -5,7 +5,9 @@ * * 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. + * published by the Free Software Foundation. Oracle designates this + * particular file as subject to the "Classpath" exception as provided + * by Oracle in the LICENSE file that accompanied this code. * * This code is distributed in the hope that it will be useful, but WITHOUT * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or From 5a6385b363da3b5faab35a5a53f2ffe55f93c1ba Mon Sep 17 00:00:00 2001 From: Rahul Raghavan Date: Tue, 18 Dec 2018 19:13:54 +0530 Subject: [PATCH 018/101] 8211698: Crash in C2 compiled code during execution of double array heavy processing code Correctly registered new Opaque4Node in add_range_check_predicate Reviewed-by: roland, thartmann --- src/hotspot/share/opto/compile.cpp | 3 +- src/hotspot/share/opto/loopTransform.cpp | 4 +- src/hotspot/share/opto/loopnode.hpp | 2 +- src/hotspot/share/opto/node.cpp | 2 +- .../jtreg/compiler/loopopts/Test8211698.java | 57 +++++++++++++++++++ 5 files changed, 62 insertions(+), 6 deletions(-) create mode 100644 test/hotspot/jtreg/compiler/loopopts/Test8211698.java diff --git a/src/hotspot/share/opto/compile.cpp b/src/hotspot/share/opto/compile.cpp index b113715aed9..42dc07b3b7c 100644 --- a/src/hotspot/share/opto/compile.cpp +++ b/src/hotspot/share/opto/compile.cpp @@ -3469,8 +3469,7 @@ void Compile::final_graph_reshaping_main_switch(Node* n, Final_Reshape_Counts& f } case Op_CmpUL: { if (!Matcher::has_match_rule(Op_CmpUL)) { - // We don't support unsigned long comparisons. Set 'max_idx_expr' - // to max_julong if < 0 to make the signed comparison fail. + // No support for unsigned long comparisons ConINode* sign_pos = new ConINode(TypeInt::make(BitsPerLong - 1)); Node* sign_bit_mask = new RShiftLNode(n->in(1), sign_pos); Node* orl = new OrLNode(n->in(1), sign_bit_mask); diff --git a/src/hotspot/share/opto/loopTransform.cpp b/src/hotspot/share/opto/loopTransform.cpp index 31e362ef6c1..209acbdffe0 100644 --- a/src/hotspot/share/opto/loopTransform.cpp +++ b/src/hotspot/share/opto/loopTransform.cpp @@ -2289,9 +2289,9 @@ Node* PhaseIdealLoop::add_range_check_predicate(IdealLoopTree* loop, CountedLoop register_new_node(opaque_bol, predicate_proj); IfNode* new_iff = NULL; if (overflow) { - new_iff = new IfNode(predicate_proj, bol, PROB_MAX, COUNT_UNKNOWN); + new_iff = new IfNode(predicate_proj, opaque_bol, PROB_MAX, COUNT_UNKNOWN); } else { - new_iff = new RangeCheckNode(predicate_proj, bol, PROB_MAX, COUNT_UNKNOWN); + new_iff = new RangeCheckNode(predicate_proj, opaque_bol, PROB_MAX, COUNT_UNKNOWN); } register_control(new_iff, loop->_parent, predicate_proj); Node* iffalse = new IfFalseNode(new_iff); diff --git a/src/hotspot/share/opto/loopnode.hpp b/src/hotspot/share/opto/loopnode.hpp index bf030aed428..064ced8f9e2 100644 --- a/src/hotspot/share/opto/loopnode.hpp +++ b/src/hotspot/share/opto/loopnode.hpp @@ -302,7 +302,7 @@ public: void set_slp_max_unroll(int unroll_factor) { _slp_maximum_unroll_factor = unroll_factor; } int slp_max_unroll() const { return _slp_maximum_unroll_factor; } - virtual LoopNode* skip_strip_mined(int expect_opaq = 1); + virtual LoopNode* skip_strip_mined(int expect_skeleton = 1); OuterStripMinedLoopNode* outer_loop() const; virtual IfTrueNode* outer_loop_tail() const; virtual OuterStripMinedLoopEndNode* outer_loop_end() const; diff --git a/src/hotspot/share/opto/node.cpp b/src/hotspot/share/opto/node.cpp index 6976fc91efb..d68aa4c0dee 100644 --- a/src/hotspot/share/opto/node.cpp +++ b/src/hotspot/share/opto/node.cpp @@ -1367,7 +1367,7 @@ static void kill_dead_code( Node *dead, PhaseIterGVN *igvn ) { igvn->C->remove_range_check_cast(cast); } if (dead->Opcode() == Op_Opaque4) { - igvn->C->remove_range_check_cast(dead); + igvn->C->remove_opaque4_node(dead); } BarrierSetC2* bs = BarrierSet::barrier_set()->barrier_set_c2(); bs->unregister_potential_barrier_node(dead); diff --git a/test/hotspot/jtreg/compiler/loopopts/Test8211698.java b/test/hotspot/jtreg/compiler/loopopts/Test8211698.java new file mode 100644 index 00000000000..37cf52e4648 --- /dev/null +++ b/test/hotspot/jtreg/compiler/loopopts/Test8211698.java @@ -0,0 +1,57 @@ +/* + * Copyright (c) 2018, 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 8211698 + * @summary Crash in C2 compiled code during execution of double array heavy processing code + * + * @run main/othervm -XX:CompileOnly=Test8211698.test Test8211698 + * + */ + +public class Test8211698 { + + public static void main(String[] args) { + Test8211698 issue = new Test8211698(); + for (int i = 0; i < 10000; i++) { + issue.test(); + } + } + + public void test() { + int[] iarr1 = new int[888]; + for (int i = 5; i > 0; i--) { + for (int j = 0; j <= i - 1; j++) { + int istep = 2 * j - i; + int iadj = 0; + if (istep < 0) { + iadj = iarr1[-istep]; + } else { + iadj = iarr1[istep]; + } + } + } + } +} + From 84105b36fd8156c572523beb695685cf096eedda Mon Sep 17 00:00:00 2001 From: Xue-Lei Andrew Fan Date: Tue, 18 Dec 2018 12:08:51 -0800 Subject: [PATCH 019/101] 8215443: The use of TransportContext.fatal() leads to bad coding style Reviewed-by: ascarpino --- .../share/classes/sun/security/ssl/Alert.java | 10 +- .../sun/security/ssl/AlpnExtension.java | 29 +++--- .../security/ssl/CertSignAlgsExtension.java | 6 +- .../sun/security/ssl/CertStatusExtension.java | 21 ++-- .../sun/security/ssl/CertificateMessage.java | 51 +++++----- .../sun/security/ssl/CertificateRequest.java | 22 ++--- .../sun/security/ssl/CertificateStatus.java | 9 +- .../sun/security/ssl/CertificateVerify.java | 68 ++++++------- .../sun/security/ssl/ChangeCipherSpec.java | 16 ++- .../classes/sun/security/ssl/ClientHello.java | 35 +++---- .../sun/security/ssl/ClientKeyExchange.java | 5 +- .../sun/security/ssl/CookieExtension.java | 9 +- .../sun/security/ssl/DHClientKeyExchange.java | 18 ++-- .../sun/security/ssl/DHKeyExchange.java | 2 +- .../sun/security/ssl/DHServerKeyExchange.java | 36 +++---- .../security/ssl/ECDHClientKeyExchange.java | 38 ++++--- .../sun/security/ssl/ECDHKeyExchange.java | 10 +- .../security/ssl/ECDHServerKeyExchange.java | 46 ++++----- .../security/ssl/ECPointFormatsExtension.java | 12 +-- .../sun/security/ssl/EncryptedExtensions.java | 2 +- .../ssl/ExtendedMasterSecretExtension.java | 22 ++--- .../classes/sun/security/ssl/Finished.java | 63 ++++-------- .../sun/security/ssl/HandshakeContext.java | 17 +--- .../sun/security/ssl/HelloRequest.java | 4 +- .../sun/security/ssl/HelloVerifyRequest.java | 4 +- .../sun/security/ssl/KeyShareExtension.java | 57 ++++------- .../classes/sun/security/ssl/KeyUpdate.java | 32 +++--- .../sun/security/ssl/MaxFragExtension.java | 23 ++--- .../sun/security/ssl/NewSessionTicket.java | 8 +- .../security/ssl/PostHandshakeContext.java | 7 +- .../security/ssl/PreSharedKeyExtension.java | 40 ++++---- .../ssl/PskKeyExchangeModesExtension.java | 5 +- .../security/ssl/RSAClientKeyExchange.java | 24 +++-- .../sun/security/ssl/RSAKeyExchange.java | 2 +- .../security/ssl/RSAServerKeyExchange.java | 21 ++-- .../sun/security/ssl/RenegoInfoExtension.java | 40 ++++---- .../sun/security/ssl/SSLEngineImpl.java | 26 +++-- .../sun/security/ssl/SSLExtensions.java | 7 +- .../sun/security/ssl/SSLSocketImpl.java | 11 ++- .../sun/security/ssl/SSLTransport.java | 10 +- .../classes/sun/security/ssl/ServerHello.java | 99 ++++++++----------- .../sun/security/ssl/ServerHelloDone.java | 2 +- .../sun/security/ssl/ServerKeyExchange.java | 5 +- .../sun/security/ssl/ServerNameExtension.java | 13 ++- .../ssl/SignatureAlgorithmsExtension.java | 16 ++- .../ssl/SupportedGroupsExtension.java | 6 +- .../ssl/SupportedVersionsExtension.java | 9 +- .../sun/security/ssl/TransportContext.java | 21 ++-- 48 files changed, 454 insertions(+), 585 deletions(-) diff --git a/src/java.base/share/classes/sun/security/ssl/Alert.java b/src/java.base/share/classes/sun/security/ssl/Alert.java index 09bcfd5afc7..4fc4d20302d 100644 --- a/src/java.base/share/classes/sun/security/ssl/Alert.java +++ b/src/java.base/share/classes/sun/security/ssl/Alert.java @@ -193,7 +193,7 @@ enum Alert { // AlertDescription description; // } Alert; if (m.remaining() != 2) { - context.fatal(Alert.ILLEGAL_PARAMETER, + throw context.fatal(Alert.ILLEGAL_PARAMETER, "Invalid Alert message: no sufficient data"); } @@ -247,14 +247,14 @@ enum Alert { if (tc.peerUserCanceled) { tc.closeOutbound(); } else if (tc.handshakeContext != null) { - tc.fatal(Alert.UNEXPECTED_MESSAGE, + throw tc.fatal(Alert.UNEXPECTED_MESSAGE, "Received close_notify during handshake"); } } else if (alert == Alert.USER_CANCELED) { if (level == Level.WARNING) { tc.peerUserCanceled = true; } else { - tc.fatal(alert, + throw tc.fatal(alert, "Received fatal close_notify alert", true, null); } } else if ((level == Level.WARNING) && (alert != null)) { @@ -269,7 +269,7 @@ enum Alert { alert != Alert.NO_CERTIFICATE || (tc.sslConfig.clientAuthType != ClientAuthType.CLIENT_AUTH_REQUESTED)) { - tc.fatal(Alert.HANDSHAKE_FAILURE, + throw tc.fatal(Alert.HANDSHAKE_FAILURE, "received handshake warning: " + alert.description); } // Otherwise, ignore the warning } // Otherwise, ignore the warning. @@ -282,7 +282,7 @@ enum Alert { diagnostic = "Received fatal alert: " + alert.description; } - tc.fatal(alert, diagnostic, true, null); + throw tc.fatal(alert, diagnostic, true, null); } } } diff --git a/src/java.base/share/classes/sun/security/ssl/AlpnExtension.java b/src/java.base/share/classes/sun/security/ssl/AlpnExtension.java index 582ea3d1b02..e62a330adaa 100644 --- a/src/java.base/share/classes/sun/security/ssl/AlpnExtension.java +++ b/src/java.base/share/classes/sun/security/ssl/AlpnExtension.java @@ -174,7 +174,8 @@ final class AlpnExtension { SSLLogger.severe( "Application protocol name cannot be empty"); } - chc.conContext.fatal(Alert.ILLEGAL_PARAMETER, + + throw chc.conContext.fatal(Alert.ILLEGAL_PARAMETER, "Application protocol name cannot be empty"); } @@ -189,7 +190,8 @@ final class AlpnExtension { ") exceeds the size limit (" + MAX_AP_LENGTH + " bytes)"); } - chc.conContext.fatal(Alert.ILLEGAL_PARAMETER, + + throw chc.conContext.fatal(Alert.ILLEGAL_PARAMETER, "Application protocol name (" + ap + ") exceeds the size limit (" + MAX_AP_LENGTH + " bytes)"); @@ -204,7 +206,8 @@ final class AlpnExtension { ") exceed the size limit (" + MAX_AP_LIST_LENGTH + " bytes)"); } - chc.conContext.fatal(Alert.ILLEGAL_PARAMETER, + + throw chc.conContext.fatal(Alert.ILLEGAL_PARAMETER, "The configured application protocols (" + Arrays.toString(laps) + ") exceed the size limit (" + @@ -283,8 +286,7 @@ final class AlpnExtension { try { spec = new AlpnSpec(buffer); } catch (IOException ioe) { - shc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, ioe); - return; // fatal() always throws, make the compiler happy. + throw shc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, ioe); } // Update the context. @@ -302,7 +304,7 @@ final class AlpnExtension { } if (!matched) { - shc.conContext.fatal(Alert.NO_APPLICATION_PROTOCOL, + throw shc.conContext.fatal(Alert.NO_APPLICATION_PROTOCOL, "No matching application layer protocol values"); } } // Otherwise, applicationProtocol will be set by the @@ -379,7 +381,8 @@ final class AlpnExtension { if ((shc.applicationProtocol == null) || (!shc.applicationProtocol.isEmpty() && !alps.contains(shc.applicationProtocol))) { - shc.conContext.fatal(Alert.NO_APPLICATION_PROTOCOL, + throw shc.conContext.fatal( + Alert.NO_APPLICATION_PROTOCOL, "No matching application layer protocol values"); } } @@ -391,7 +394,8 @@ final class AlpnExtension { if ((shc.applicationProtocol == null) || (!shc.applicationProtocol.isEmpty() && !alps.contains(shc.applicationProtocol))) { - shc.conContext.fatal(Alert.NO_APPLICATION_PROTOCOL, + throw shc.conContext.fatal( + Alert.NO_APPLICATION_PROTOCOL, "No matching application layer protocol values"); } } @@ -454,7 +458,7 @@ final class AlpnExtension { if (requestedAlps == null || requestedAlps.applicationProtocols == null || requestedAlps.applicationProtocols.isEmpty()) { - chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, + throw chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, "Unexpected " + SSLExtension.CH_ALPN.name + " extension"); } @@ -463,13 +467,12 @@ final class AlpnExtension { try { spec = new AlpnSpec(buffer); } catch (IOException ioe) { - chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, ioe); - return; // fatal() always throws, make the compiler happy. + throw chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, ioe); } // Only one application protocol is allowed. if (spec.applicationProtocols.size() != 1) { - chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, + throw chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, "Invalid " + SSLExtension.CH_ALPN.name + " extension: " + "Only one application protocol name " + "is allowed in ServerHello message"); @@ -478,7 +481,7 @@ final class AlpnExtension { // The respond application protocol must be one of the requested. if (!requestedAlps.applicationProtocols.containsAll( spec.applicationProtocols)) { - chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, + throw chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, "Invalid " + SSLExtension.CH_ALPN.name + " extension: " + "Only client specified application protocol " + "is allowed in ServerHello message"); diff --git a/src/java.base/share/classes/sun/security/ssl/CertSignAlgsExtension.java b/src/java.base/share/classes/sun/security/ssl/CertSignAlgsExtension.java index ba94f0695cd..ba39722e45d 100644 --- a/src/java.base/share/classes/sun/security/ssl/CertSignAlgsExtension.java +++ b/src/java.base/share/classes/sun/security/ssl/CertSignAlgsExtension.java @@ -153,8 +153,7 @@ final class CertSignAlgsExtension { try { spec = new SignatureSchemesSpec(buffer); } catch (IOException ioe) { - shc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, ioe); - return; // fatal() always throws, make the compiler happy. + throw shc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, ioe); } // Update the context. @@ -297,8 +296,7 @@ final class CertSignAlgsExtension { try { spec = new SignatureSchemesSpec(buffer); } catch (IOException ioe) { - chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, ioe); - return; // fatal() always throws, make the compiler happy. + throw chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, ioe); } // Update the context. diff --git a/src/java.base/share/classes/sun/security/ssl/CertStatusExtension.java b/src/java.base/share/classes/sun/security/ssl/CertStatusExtension.java index 2ca3eb8cb05..714cf4f6cc0 100644 --- a/src/java.base/share/classes/sun/security/ssl/CertStatusExtension.java +++ b/src/java.base/share/classes/sun/security/ssl/CertStatusExtension.java @@ -606,8 +606,7 @@ final class CertStatusExtension { try { spec = new CertStatusRequestSpec(buffer); } catch (IOException ioe) { - shc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, ioe); - return; // fatal() always throws, make the compiler happy. + throw shc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, ioe); } // Update the context. @@ -711,13 +710,13 @@ final class CertStatusExtension { CertStatusRequestSpec requestedCsr = (CertStatusRequestSpec) chc.handshakeExtensions.get(CH_STATUS_REQUEST); if (requestedCsr == null) { - chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, + throw chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, "Unexpected status_request extension in ServerHello"); } // Parse the extension. if (buffer.hasRemaining()) { - chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, + throw chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, "Invalid status_request extension in ServerHello message: " + "the extension data must be empty"); } @@ -964,8 +963,7 @@ final class CertStatusExtension { try { spec = new CertStatusRequestV2Spec(buffer); } catch (IOException ioe) { - shc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, ioe); - return; // fatal() always throws, make the compiler happy. + throw shc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, ioe); } // Update the context. @@ -1067,13 +1065,13 @@ final class CertStatusExtension { CertStatusRequestV2Spec requestedCsr = (CertStatusRequestV2Spec) chc.handshakeExtensions.get(CH_STATUS_REQUEST_V2); if (requestedCsr == null) { - chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, + throw chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, "Unexpected status_request_v2 extension in ServerHello"); } // Parse the extension. if (buffer.hasRemaining()) { - chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, + throw chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, "Invalid status_request_v2 extension in ServerHello: " + "the extension data must be empty"); } @@ -1157,10 +1155,10 @@ final class CertStatusExtension { respBytes); producedData = certResp.toByteArray(); } catch (CertificateException ce) { - shc.conContext.fatal(Alert.BAD_CERTIFICATE, + throw shc.conContext.fatal(Alert.BAD_CERTIFICATE, "Failed to parse server certificates", ce); } catch (IOException ioe) { - shc.conContext.fatal(Alert.BAD_CERT_STATUS_RESPONSE, + throw shc.conContext.fatal(Alert.BAD_CERT_STATUS_RESPONSE, "Failed to parse certificate status response", ioe); } @@ -1188,8 +1186,7 @@ final class CertStatusExtension { try { spec = new CertStatusResponseSpec(buffer); } catch (IOException ioe) { - chc.conContext.fatal(Alert.DECODE_ERROR, ioe); - return; // fatal() always throws, make the compiler happy. + throw chc.conContext.fatal(Alert.DECODE_ERROR, ioe); } if (chc.sslContext.isStaplingEnabled(true)) { diff --git a/src/java.base/share/classes/sun/security/ssl/CertificateMessage.java b/src/java.base/share/classes/sun/security/ssl/CertificateMessage.java index b0d82dadbf4..a38cfbfe486 100644 --- a/src/java.base/share/classes/sun/security/ssl/CertificateMessage.java +++ b/src/java.base/share/classes/sun/security/ssl/CertificateMessage.java @@ -111,10 +111,10 @@ final class CertificateMessage { encodedCerts.add(cert.getEncoded()); } catch (CertificateEncodingException cee) { // unlikely - handshakeContext.conContext.fatal(Alert.INTERNAL_ERROR, + throw handshakeContext.conContext.fatal( + Alert.INTERNAL_ERROR, "Could not encode certificate (" + cert.getSubjectX500Principal() + ")", cee); - break; } } @@ -127,7 +127,8 @@ final class CertificateMessage { int listLen = Record.getInt24(m); if (listLen > m.remaining()) { - handshakeContext.conContext.fatal(Alert.ILLEGAL_PARAMETER, + throw handshakeContext.conContext.fatal( + Alert.ILLEGAL_PARAMETER, "Error parsing certificate message:no sufficient data"); } if (listLen > 0) { @@ -248,10 +249,8 @@ final class CertificateMessage { } if (x509Possession == null) { // unlikely - shc.conContext.fatal(Alert.INTERNAL_ERROR, + throw shc.conContext.fatal(Alert.INTERNAL_ERROR, "No expected X.509 certificate for server authentication"); - - return null; // make the compiler happy } shc.handshakeSession.setLocalPrivateKey( @@ -375,7 +374,7 @@ final class CertificateMessage { if (shc.sslConfig.clientAuthType != ClientAuthType.CLIENT_AUTH_REQUESTED) { // unexpected or require client authentication - shc.conContext.fatal(Alert.BAD_CERTIFICATE, + throw shc.conContext.fatal(Alert.BAD_CERTIFICATE, "Empty server certificate chain"); } else { return; @@ -392,7 +391,7 @@ final class CertificateMessage { new ByteArrayInputStream(encodedCert)); } } catch (CertificateException ce) { - shc.conContext.fatal(Alert.BAD_CERTIFICATE, + throw shc.conContext.fatal(Alert.BAD_CERTIFICATE, "Failed to parse server certificates", ce); } @@ -410,7 +409,7 @@ final class CertificateMessage { T12CertificateMessage certificateMessage) throws IOException { List encodedCerts = certificateMessage.encodedCertChain; if (encodedCerts == null || encodedCerts.isEmpty()) { - chc.conContext.fatal(Alert.BAD_CERTIFICATE, + throw chc.conContext.fatal(Alert.BAD_CERTIFICATE, "Empty server certificate chain"); } @@ -424,7 +423,7 @@ final class CertificateMessage { new ByteArrayInputStream(encodedCert)); } } catch (CertificateException ce) { - chc.conContext.fatal(Alert.BAD_CERTIFICATE, + throw chc.conContext.fatal(Alert.BAD_CERTIFICATE, "Failed to parse server certificates", ce); } @@ -443,7 +442,7 @@ final class CertificateMessage { if ((identityAlg == null || identityAlg.isEmpty()) && !isIdentityEquivalent(x509Certs[0], chc.reservedServerCerts[0])) { - chc.conContext.fatal(Alert.BAD_CERTIFICATE, + throw chc.conContext.fatal(Alert.BAD_CERTIFICATE, "server certificate change is restricted " + "during renegotiation"); } @@ -639,7 +638,7 @@ final class CertificateMessage { // the certificate chain in the TLS session. chc.handshakeSession.setPeerCertificates(certs); } catch (CertificateException ce) { - chc.conContext.fatal(getCertificateAlert(chc, ce), ce); + throw chc.conContext.fatal(getCertificateAlert(chc, ce), ce); } } @@ -685,7 +684,7 @@ final class CertificateMessage { "Improper X509TrustManager implementation"); } } catch (CertificateException ce) { - shc.conContext.fatal(Alert.CERTIFICATE_UNKNOWN, ce); + throw shc.conContext.fatal(Alert.CERTIFICATE_UNKNOWN, ce); } } @@ -942,22 +941,20 @@ final class CertificateMessage { SSLPossession pos = choosePossession(shc, clientHello); if (pos == null) { - shc.conContext.fatal(Alert.HANDSHAKE_FAILURE, + throw shc.conContext.fatal(Alert.HANDSHAKE_FAILURE, "No available authentication scheme"); - return null; // make the complier happy } if (!(pos instanceof X509Possession)) { - shc.conContext.fatal(Alert.HANDSHAKE_FAILURE, + throw shc.conContext.fatal(Alert.HANDSHAKE_FAILURE, "No X.509 certificate for server authentication"); } X509Possession x509Possession = (X509Possession)pos; X509Certificate[] localCerts = x509Possession.popCerts; if (localCerts == null || localCerts.length == 0) { - shc.conContext.fatal(Alert.HANDSHAKE_FAILURE, + throw shc.conContext.fatal(Alert.HANDSHAKE_FAILURE, "No X.509 certificate for server authentication"); - return null; // make the complier happy } // update the context @@ -969,9 +966,8 @@ final class CertificateMessage { try { cm = new T13CertificateMessage(shc, (new byte[0]), localCerts); } catch (SSLException | CertificateException ce) { - shc.conContext.fatal(Alert.HANDSHAKE_FAILURE, + throw shc.conContext.fatal(Alert.HANDSHAKE_FAILURE, "Failed to produce server Certificate message", ce); - return null; // make the complier happy } // Check the OCSP stapling extensions and attempt @@ -1108,9 +1104,8 @@ final class CertificateMessage { cm = new T13CertificateMessage( chc, chc.certRequestContext, localCerts); } catch (SSLException | CertificateException ce) { - chc.conContext.fatal(Alert.HANDSHAKE_FAILURE, + throw chc.conContext.fatal(Alert.HANDSHAKE_FAILURE, "Failed to produce client Certificate message", ce); - return null; } if (SSLLogger.isOn && SSLLogger.isOn("ssl,handshake")) { SSLLogger.fine("Produced client Certificate message", cm); @@ -1163,7 +1158,7 @@ final class CertificateMessage { if (certificateMessage.certEntries == null || certificateMessage.certEntries.isEmpty()) { if (shc.sslConfig.clientAuthType == CLIENT_AUTH_REQUIRED) { - shc.conContext.fatal(Alert.BAD_CERTIFICATE, + throw shc.conContext.fatal(Alert.BAD_CERTIFICATE, "Empty client certificate chain"); } else { // optional client authentication @@ -1187,7 +1182,7 @@ final class CertificateMessage { T13CertificateMessage certificateMessage )throws IOException { if (certificateMessage.certEntries == null || certificateMessage.certEntries.isEmpty()) { - chc.conContext.fatal(Alert.BAD_CERTIFICATE, + throw chc.conContext.fatal(Alert.BAD_CERTIFICATE, "Empty server certificate chain"); } @@ -1224,7 +1219,7 @@ final class CertificateMessage { new ByteArrayInputStream(entry.encoded)); } } catch (CertificateException ce) { - shc.conContext.fatal(Alert.BAD_CERTIFICATE, + throw shc.conContext.fatal(Alert.BAD_CERTIFICATE, "Failed to parse server certificates", ce); } @@ -1270,7 +1265,7 @@ final class CertificateMessage { // the certificate chain in the TLS session. shc.handshakeSession.setPeerCertificates(certs); } catch (CertificateException ce) { - shc.conContext.fatal(Alert.CERTIFICATE_UNKNOWN, ce); + throw shc.conContext.fatal(Alert.CERTIFICATE_UNKNOWN, ce); } return certs; @@ -1289,7 +1284,7 @@ final class CertificateMessage { new ByteArrayInputStream(entry.encoded)); } } catch (CertificateException ce) { - chc.conContext.fatal(Alert.BAD_CERTIFICATE, + throw chc.conContext.fatal(Alert.BAD_CERTIFICATE, "Failed to parse server certificates", ce); } @@ -1326,7 +1321,7 @@ final class CertificateMessage { // the certificate chain in the TLS session. chc.handshakeSession.setPeerCertificates(certs); } catch (CertificateException ce) { - chc.conContext.fatal(getCertificateAlert(chc, ce), ce); + throw chc.conContext.fatal(getCertificateAlert(chc, ce), ce); } return certs; diff --git a/src/java.base/share/classes/sun/security/ssl/CertificateRequest.java b/src/java.base/share/classes/sun/security/ssl/CertificateRequest.java index 14f76107ca4..ba35385d7e0 100644 --- a/src/java.base/share/classes/sun/security/ssl/CertificateRequest.java +++ b/src/java.base/share/classes/sun/security/ssl/CertificateRequest.java @@ -171,14 +171,14 @@ final class CertificateRequest { // DistinguishedName certificate_authorities<0..2^16-1>; // } CertificateRequest; if (m.remaining() < 4) { - handshakeContext.conContext.fatal(Alert.ILLEGAL_PARAMETER, + throw handshakeContext.conContext.fatal(Alert.ILLEGAL_PARAMETER, "Incorrect CertificateRequest message: no sufficient data"); } this.types = Record.getBytes8(m); int listLen = Record.getInt16(m); if (listLen > m.remaining()) { - handshakeContext.conContext.fatal(Alert.ILLEGAL_PARAMETER, + throw handshakeContext.conContext.fatal(Alert.ILLEGAL_PARAMETER, "Incorrect CertificateRequest message:no sufficient data"); } @@ -407,7 +407,7 @@ final class CertificateRequest { this.types = ClientCertificateType.CERT_TYPES; if (signatureSchemes == null || signatureSchemes.isEmpty()) { - handshakeContext.conContext.fatal(Alert.ILLEGAL_PARAMETER, + throw handshakeContext.conContext.fatal(Alert.ILLEGAL_PARAMETER, "No signature algorithms specified for " + "CertificateRequest hanshake message"); } @@ -437,7 +437,7 @@ final class CertificateRequest { // certificate_authorities if (m.remaining() < 8) { - handshakeContext.conContext.fatal(Alert.ILLEGAL_PARAMETER, + throw handshakeContext.conContext.fatal(Alert.ILLEGAL_PARAMETER, "Invalid CertificateRequest handshake message: " + "no sufficient data"); } @@ -445,14 +445,14 @@ final class CertificateRequest { // supported_signature_algorithms if (m.remaining() < 6) { - handshakeContext.conContext.fatal(Alert.ILLEGAL_PARAMETER, + throw handshakeContext.conContext.fatal(Alert.ILLEGAL_PARAMETER, "Invalid CertificateRequest handshake message: " + "no sufficient data"); } byte[] algs = Record.getBytes16(m); if (algs == null || algs.length == 0 || (algs.length & 0x01) != 0) { - handshakeContext.conContext.fatal(Alert.ILLEGAL_PARAMETER, + throw handshakeContext.conContext.fatal(Alert.ILLEGAL_PARAMETER, "Invalid CertificateRequest handshake message: " + "incomplete signature algorithms"); } @@ -466,14 +466,14 @@ final class CertificateRequest { // certificate_authorities if (m.remaining() < 2) { - handshakeContext.conContext.fatal(Alert.ILLEGAL_PARAMETER, + throw handshakeContext.conContext.fatal(Alert.ILLEGAL_PARAMETER, "Invalid CertificateRequest handshake message: " + "no sufficient data"); } int listLen = Record.getInt16(m); if (listLen > m.remaining()) { - handshakeContext.conContext.fatal(Alert.ILLEGAL_PARAMETER, + throw handshakeContext.conContext.fatal(Alert.ILLEGAL_PARAMETER, "Invalid CertificateRequest message: no sufficient data"); } @@ -597,7 +597,7 @@ final class CertificateRequest { if (shc.localSupportedSignAlgs == null || shc.localSupportedSignAlgs.isEmpty()) { - shc.conContext.fatal(Alert.HANDSHAKE_FAILURE, + throw shc.conContext.fatal(Alert.HANDSHAKE_FAILURE, "No supported signature algorithm"); } @@ -783,14 +783,14 @@ final class CertificateRequest { // Extension extensions<2..2^16-1>; // } CertificateRequest; if (m.remaining() < 5) { - handshakeContext.conContext.fatal(Alert.ILLEGAL_PARAMETER, + throw handshakeContext.conContext.fatal(Alert.ILLEGAL_PARAMETER, "Invalid CertificateRequest handshake message: " + "no sufficient data"); } this.requestContext = Record.getBytes8(m); if (m.remaining() < 4) { - handshakeContext.conContext.fatal(Alert.ILLEGAL_PARAMETER, + throw handshakeContext.conContext.fatal(Alert.ILLEGAL_PARAMETER, "Invalid CertificateRequest handshake message: " + "no sufficient extensions data"); } diff --git a/src/java.base/share/classes/sun/security/ssl/CertificateStatus.java b/src/java.base/share/classes/sun/security/ssl/CertificateStatus.java index c4087aab6ed..3f457ec051b 100644 --- a/src/java.base/share/classes/sun/security/ssl/CertificateStatus.java +++ b/src/java.base/share/classes/sun/security/ssl/CertificateStatus.java @@ -154,7 +154,8 @@ final class CertificateStatus { encodedResponses.add(respDER); encodedResponsesLen = 3 + respDER.length; } else { - handshakeContext.conContext.fatal(Alert.HANDSHAKE_FAILURE, + throw handshakeContext.conContext.fatal( + Alert.HANDSHAKE_FAILURE, "Zero-length OCSP Response"); } } else if (statusType == CertStatusRequestType.OCSP_MULTI) { @@ -172,11 +173,13 @@ final class CertificateStatus { } if (respListLen != 0) { - handshakeContext.conContext.fatal(Alert.INTERNAL_ERROR, + throw handshakeContext.conContext.fatal( + Alert.INTERNAL_ERROR, "Bad OCSP response list length"); } } else { - handshakeContext.conContext.fatal(Alert.HANDSHAKE_FAILURE, + throw handshakeContext.conContext.fatal( + Alert.HANDSHAKE_FAILURE, "Unsupported StatusResponseType: " + statusType); } messageLength = messageLength(); diff --git a/src/java.base/share/classes/sun/security/ssl/CertificateVerify.java b/src/java.base/share/classes/sun/security/ssl/CertificateVerify.java index bcc90f1796e..b5c6c9f2301 100644 --- a/src/java.base/share/classes/sun/security/ssl/CertificateVerify.java +++ b/src/java.base/share/classes/sun/security/ssl/CertificateVerify.java @@ -83,11 +83,11 @@ final class CertificateVerify { signer.update(hashes); temproary = signer.sign(); } catch (NoSuchAlgorithmException nsae) { - chc.conContext.fatal(Alert.INTERNAL_ERROR, + throw chc.conContext.fatal(Alert.INTERNAL_ERROR, "Unsupported signature algorithm (" + algorithm + ") used in CertificateVerify handshake message", nsae); } catch (GeneralSecurityException gse) { - chc.conContext.fatal(Alert.HANDSHAKE_FAILURE, + throw chc.conContext.fatal(Alert.HANDSHAKE_FAILURE, "Cannot produce CertificateVerify signature", gse); } @@ -112,7 +112,7 @@ final class CertificateVerify { // }; // } Signature; if (m.remaining() < 2) { - shc.conContext.fatal(Alert.ILLEGAL_PARAMETER, + throw shc.conContext.fatal(Alert.ILLEGAL_PARAMETER, "Invalid CertificateVerify message: no sufficient data"); } @@ -128,7 +128,7 @@ final class CertificateVerify { if (x509Credentials == null || x509Credentials.popPublicKey == null) { - shc.conContext.fatal(Alert.HANDSHAKE_FAILURE, + throw shc.conContext.fatal(Alert.HANDSHAKE_FAILURE, "No X509 credentials negotiated for CertificateVerify"); } @@ -140,15 +140,15 @@ final class CertificateVerify { shc.handshakeSession.getMasterSecret()); signer.update(hashes); if (!signer.verify(signature)) { - shc.conContext.fatal(Alert.HANDSHAKE_FAILURE, + throw shc.conContext.fatal(Alert.HANDSHAKE_FAILURE, "Invalid CertificateVerify message: invalid signature"); } } catch (NoSuchAlgorithmException nsae) { - shc.conContext.fatal(Alert.INTERNAL_ERROR, + throw shc.conContext.fatal(Alert.INTERNAL_ERROR, "Unsupported signature algorithm (" + algorithm + ") used in CertificateVerify handshake message", nsae); } catch (GeneralSecurityException gse) { - shc.conContext.fatal(Alert.HANDSHAKE_FAILURE, + throw shc.conContext.fatal(Alert.HANDSHAKE_FAILURE, "Cannot verify CertificateVerify signature", gse); } } @@ -327,11 +327,11 @@ final class CertificateVerify { signer.update(hashes); temproary = signer.sign(); } catch (NoSuchAlgorithmException nsae) { - chc.conContext.fatal(Alert.INTERNAL_ERROR, + throw chc.conContext.fatal(Alert.INTERNAL_ERROR, "Unsupported signature algorithm (" + algorithm + ") used in CertificateVerify handshake message", nsae); } catch (GeneralSecurityException gse) { - chc.conContext.fatal(Alert.HANDSHAKE_FAILURE, + throw chc.conContext.fatal(Alert.HANDSHAKE_FAILURE, "Cannot produce CertificateVerify signature", gse); } @@ -356,7 +356,7 @@ final class CertificateVerify { // }; // } Signature; if (m.remaining() < 2) { - shc.conContext.fatal(Alert.ILLEGAL_PARAMETER, + throw shc.conContext.fatal(Alert.ILLEGAL_PARAMETER, "Invalid CertificateVerify message: no sufficient data"); } @@ -372,7 +372,7 @@ final class CertificateVerify { if (x509Credentials == null || x509Credentials.popPublicKey == null) { - shc.conContext.fatal(Alert.HANDSHAKE_FAILURE, + throw shc.conContext.fatal(Alert.HANDSHAKE_FAILURE, "No X509 credentials negotiated for CertificateVerify"); } @@ -383,15 +383,15 @@ final class CertificateVerify { byte[] hashes = shc.handshakeHash.digest(algorithm); signer.update(hashes); if (!signer.verify(signature)) { - shc.conContext.fatal(Alert.HANDSHAKE_FAILURE, + throw shc.conContext.fatal(Alert.HANDSHAKE_FAILURE, "Invalid CertificateVerify message: invalid signature"); } } catch (NoSuchAlgorithmException nsae) { - shc.conContext.fatal(Alert.INTERNAL_ERROR, + throw shc.conContext.fatal(Alert.INTERNAL_ERROR, "Unsupported signature algorithm (" + algorithm + ") used in CertificateVerify handshake message", nsae); } catch (GeneralSecurityException gse) { - shc.conContext.fatal(Alert.HANDSHAKE_FAILURE, + throw shc.conContext.fatal(Alert.HANDSHAKE_FAILURE, "Cannot verify CertificateVerify signature", gse); } } @@ -570,7 +570,7 @@ final class CertificateVerify { if (signatureScheme == null) { // Unlikely, the credentials generator should have // selected the preferable signature algorithm properly. - chc.conContext.fatal(Alert.INTERNAL_ERROR, + throw chc.conContext.fatal(Alert.INTERNAL_ERROR, "No preferred signature algorithm for CertificateVerify"); } @@ -582,12 +582,12 @@ final class CertificateVerify { temproary = signer.sign(); } catch (NoSuchAlgorithmException | InvalidAlgorithmParameterException nsae) { - chc.conContext.fatal(Alert.INTERNAL_ERROR, + throw chc.conContext.fatal(Alert.INTERNAL_ERROR, "Unsupported signature algorithm (" + signatureScheme.name + ") used in CertificateVerify handshake message", nsae); } catch (InvalidKeyException | SignatureException ikse) { - chc.conContext.fatal(Alert.HANDSHAKE_FAILURE, + throw chc.conContext.fatal(Alert.HANDSHAKE_FAILURE, "Cannot produce CertificateVerify signature", ikse); } @@ -607,7 +607,7 @@ final class CertificateVerify { // opaque signature<0..2^16-1>; // } DigitallySigned; if (m.remaining() < 4) { - shc.conContext.fatal(Alert.ILLEGAL_PARAMETER, + throw shc.conContext.fatal(Alert.ILLEGAL_PARAMETER, "Invalid CertificateVerify message: no sufficient data"); } @@ -615,13 +615,13 @@ final class CertificateVerify { int ssid = Record.getInt16(m); this.signatureScheme = SignatureScheme.valueOf(ssid); if (signatureScheme == null) { - shc.conContext.fatal(Alert.HANDSHAKE_FAILURE, + throw shc.conContext.fatal(Alert.HANDSHAKE_FAILURE, "Invalid signature algorithm (" + ssid + ") used in CertificateVerify handshake message"); } if (!shc.localSupportedSignAlgs.contains(signatureScheme)) { - shc.conContext.fatal(Alert.HANDSHAKE_FAILURE, + throw shc.conContext.fatal(Alert.HANDSHAKE_FAILURE, "Unsupported signature algorithm (" + signatureScheme.name + ") used in CertificateVerify handshake message"); @@ -638,7 +638,7 @@ final class CertificateVerify { if (x509Credentials == null || x509Credentials.popPublicKey == null) { - shc.conContext.fatal(Alert.HANDSHAKE_FAILURE, + throw shc.conContext.fatal(Alert.HANDSHAKE_FAILURE, "No X509 credentials negotiated for CertificateVerify"); } @@ -649,17 +649,17 @@ final class CertificateVerify { signatureScheme.getSignature(x509Credentials.popPublicKey); signer.update(shc.handshakeHash.archived()); if (!signer.verify(signature)) { - shc.conContext.fatal(Alert.HANDSHAKE_FAILURE, + throw shc.conContext.fatal(Alert.HANDSHAKE_FAILURE, "Invalid CertificateVerify signature"); } } catch (NoSuchAlgorithmException | InvalidAlgorithmParameterException nsae) { - shc.conContext.fatal(Alert.INTERNAL_ERROR, + throw shc.conContext.fatal(Alert.INTERNAL_ERROR, "Unsupported signature algorithm (" + signatureScheme.name + ") used in CertificateVerify handshake message", nsae); } catch (InvalidKeyException | SignatureException ikse) { - shc.conContext.fatal(Alert.HANDSHAKE_FAILURE, + throw shc.conContext.fatal(Alert.HANDSHAKE_FAILURE, "Cannot verify CertificateVerify signature", ikse); } } @@ -871,7 +871,7 @@ final class CertificateVerify { if (signatureScheme == null) { // Unlikely, the credentials generator should have // selected the preferable signature algorithm properly. - context.conContext.fatal(Alert.INTERNAL_ERROR, + throw context.conContext.fatal(Alert.INTERNAL_ERROR, "No preferred signature algorithm for CertificateVerify"); } @@ -897,12 +897,12 @@ final class CertificateVerify { temproary = signer.sign(); } catch (NoSuchAlgorithmException | InvalidAlgorithmParameterException nsae) { - context.conContext.fatal(Alert.INTERNAL_ERROR, + throw context.conContext.fatal(Alert.INTERNAL_ERROR, "Unsupported signature algorithm (" + signatureScheme.name + ") used in CertificateVerify handshake message", nsae); } catch (InvalidKeyException | SignatureException ikse) { - context.conContext.fatal(Alert.HANDSHAKE_FAILURE, + throw context.conContext.fatal(Alert.HANDSHAKE_FAILURE, "Cannot produce CertificateVerify signature", ikse); } @@ -918,7 +918,7 @@ final class CertificateVerify { // opaque signature<0..2^16-1>; // } DigitallySigned; if (m.remaining() < 4) { - context.conContext.fatal(Alert.ILLEGAL_PARAMETER, + throw context.conContext.fatal(Alert.ILLEGAL_PARAMETER, "Invalid CertificateVerify message: no sufficient data"); } @@ -926,13 +926,13 @@ final class CertificateVerify { int ssid = Record.getInt16(m); this.signatureScheme = SignatureScheme.valueOf(ssid); if (signatureScheme == null) { - context.conContext.fatal(Alert.HANDSHAKE_FAILURE, + throw context.conContext.fatal(Alert.HANDSHAKE_FAILURE, "Invalid signature algorithm (" + ssid + ") used in CertificateVerify handshake message"); } if (!context.localSupportedSignAlgs.contains(signatureScheme)) { - context.conContext.fatal(Alert.HANDSHAKE_FAILURE, + throw context.conContext.fatal(Alert.HANDSHAKE_FAILURE, "Unsupported signature algorithm (" + signatureScheme.name + ") used in CertificateVerify handshake message"); @@ -949,7 +949,7 @@ final class CertificateVerify { if (x509Credentials == null || x509Credentials.popPublicKey == null) { - context.conContext.fatal(Alert.HANDSHAKE_FAILURE, + throw context.conContext.fatal(Alert.HANDSHAKE_FAILURE, "No X509 credentials negotiated for CertificateVerify"); } @@ -975,17 +975,17 @@ final class CertificateVerify { signatureScheme.getSignature(x509Credentials.popPublicKey); signer.update(contentCovered); if (!signer.verify(signature)) { - context.conContext.fatal(Alert.HANDSHAKE_FAILURE, + throw context.conContext.fatal(Alert.HANDSHAKE_FAILURE, "Invalid CertificateVerify signature"); } } catch (NoSuchAlgorithmException | InvalidAlgorithmParameterException nsae) { - context.conContext.fatal(Alert.INTERNAL_ERROR, + throw context.conContext.fatal(Alert.INTERNAL_ERROR, "Unsupported signature algorithm (" + signatureScheme.name + ") used in CertificateVerify handshake message", nsae); } catch (InvalidKeyException | SignatureException ikse) { - context.conContext.fatal(Alert.HANDSHAKE_FAILURE, + throw context.conContext.fatal(Alert.HANDSHAKE_FAILURE, "Cannot verify CertificateVerify signature", ikse); } } diff --git a/src/java.base/share/classes/sun/security/ssl/ChangeCipherSpec.java b/src/java.base/share/classes/sun/security/ssl/ChangeCipherSpec.java index 2983cf615ab..fad46fb4178 100644 --- a/src/java.base/share/classes/sun/security/ssl/ChangeCipherSpec.java +++ b/src/java.base/share/classes/sun/security/ssl/ChangeCipherSpec.java @@ -106,11 +106,9 @@ final class ChangeCipherSpec { } if (writeCipher == null) { - hc.conContext.fatal(Alert.ILLEGAL_PARAMETER, + throw hc.conContext.fatal(Alert.ILLEGAL_PARAMETER, "Illegal cipher suite (" + ncs + ") and protocol version (" + hc.negotiatedProtocol + ")"); - - return null; } if (SSLLogger.isOn && SSLLogger.isOn("ssl,handshake")) { @@ -144,7 +142,7 @@ final class ChangeCipherSpec { // parse if (message.remaining() != 1 || message.get() != 1) { - tc.fatal(Alert.UNEXPECTED_MESSAGE, + throw tc.fatal(Alert.UNEXPECTED_MESSAGE, "Malformed or unexpected ChangeCipherSpec message"); } if (SSLLogger.isOn && SSLLogger.isOn("ssl,handshake")) { @@ -153,7 +151,7 @@ final class ChangeCipherSpec { // validate if (tc.handshakeContext == null) { - tc.fatal(Alert.HANDSHAKE_FAILURE, + throw tc.fatal(Alert.HANDSHAKE_FAILURE, "Unexpected ChangeCipherSpec message"); } @@ -161,7 +159,7 @@ final class ChangeCipherSpec { HandshakeContext hc = tc.handshakeContext; if (hc.handshakeKeyDerivation == null) { - tc.fatal(Alert.UNEXPECTED_MESSAGE, + throw tc.fatal(Alert.UNEXPECTED_MESSAGE, "Unexpected ChangeCipherSpec message"); } @@ -205,12 +203,10 @@ final class ChangeCipherSpec { } if (readCipher == null) { - hc.conContext.fatal(Alert.ILLEGAL_PARAMETER, + throw hc.conContext.fatal(Alert.ILLEGAL_PARAMETER, "Illegal cipher suite (" + hc.negotiatedCipherSuite + ") and protocol version (" + hc.negotiatedProtocol + ")"); - - return; } tc.inputRecord.changeReadCiphers(readCipher); @@ -243,7 +239,7 @@ final class ChangeCipherSpec { // parse if (message.remaining() != 1 || message.get() != 1) { - tc.fatal(Alert.UNEXPECTED_MESSAGE, + throw tc.fatal(Alert.UNEXPECTED_MESSAGE, "Malformed or unexpected ChangeCipherSpec message"); } if (SSLLogger.isOn && SSLLogger.isOn("ssl,handshake")) { diff --git a/src/java.base/share/classes/sun/security/ssl/ClientHello.java b/src/java.base/share/classes/sun/security/ssl/ClientHello.java index 96bbef5db59..6fbac3103e5 100644 --- a/src/java.base/share/classes/sun/security/ssl/ClientHello.java +++ b/src/java.base/share/classes/sun/security/ssl/ClientHello.java @@ -144,8 +144,8 @@ final class ClientHello { if (id == SSLExtension.CH_PRE_SHARED_KEY.id) { // ensure pre_shared_key is the last extension if (remaining > 0) { - tc.fatal(Alert.ILLEGAL_PARAMETER, - "pre_shared_key extension is not last"); + throw tc.fatal(Alert.ILLEGAL_PARAMETER, + "pre_shared_key extension is not last"); } // read only up to the IDs Record.getBytes16(m); @@ -169,7 +169,8 @@ final class ClientHello { try { sessionId.checkLength(clientVersion); } catch (SSLProtocolException ex) { - handshakeContext.conContext.fatal(Alert.ILLEGAL_PARAMETER, ex); + throw handshakeContext.conContext.fatal( + Alert.ILLEGAL_PARAMETER, ex); } if (isDTLS) { this.cookie = Record.getBytes8(m); @@ -179,8 +180,9 @@ final class ClientHello { byte[] encodedIds = Record.getBytes16(m); if (encodedIds.length == 0 || (encodedIds.length & 0x01) != 0) { - handshakeContext.conContext.fatal(Alert.ILLEGAL_PARAMETER, - "Invalid ClientHello message"); + throw handshakeContext.conContext.fatal( + Alert.ILLEGAL_PARAMETER, + "Invalid ClientHello message"); } this.cipherSuiteIds = new int[encodedIds.length >> 1]; @@ -702,7 +704,8 @@ final class ClientHello { try { chc.kickstart(); } catch (IOException ioe) { - chc.conContext.fatal(Alert.HANDSHAKE_FAILURE, ioe); + throw chc.conContext.fatal( + Alert.HANDSHAKE_FAILURE, ioe); } // The handshake message has been delivered. @@ -790,7 +793,7 @@ final class ClientHello { // clean up this consumer shc.handshakeConsumers.remove(SSLHandshake.CLIENT_HELLO.id); if (!shc.handshakeConsumers.isEmpty()) { - shc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, + throw shc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, "No more handshake message allowed " + "in a ClientHello flight"); } @@ -877,7 +880,7 @@ final class ClientHello { context.activeProtocols, chv); if (pv == null || pv == ProtocolVersion.NONE || pv == ProtocolVersion.SSL20Hello) { - context.conContext.fatal(Alert.PROTOCOL_VERSION, + throw context.conContext.fatal(Alert.PROTOCOL_VERSION, "Client requested protocol " + ProtocolVersion.nameOf(clientHelloVersion) + " is not enabled or supported in server context"); @@ -910,13 +913,11 @@ final class ClientHello { } // No protocol version can be negotiated. - context.conContext.fatal(Alert.PROTOCOL_VERSION, + throw context.conContext.fatal(Alert.PROTOCOL_VERSION, "The client supported protocol versions " + Arrays.toString( ProtocolVersion.toStringArray(clientSupportedVersions)) + " are not accepted by server preferences " + context.activeProtocols); - - return null; // make the compiler happy } } @@ -957,13 +958,13 @@ final class ClientHello { if (shc.conContext.isNegotiated) { if (!shc.conContext.secureRenegotiation && !HandshakeContext.allowUnsafeRenegotiation) { - shc.conContext.fatal(Alert.HANDSHAKE_FAILURE, + throw shc.conContext.fatal(Alert.HANDSHAKE_FAILURE, "Unsafe renegotiation is not allowed"); } if (ServerHandshakeContext.rejectClientInitiatedRenego && !shc.kickstartMessageDelivered) { - shc.conContext.fatal(Alert.HANDSHAKE_FAILURE, + throw shc.conContext.fatal(Alert.HANDSHAKE_FAILURE, "Client initiated renegotiation is not allowed"); } } @@ -1170,13 +1171,13 @@ final class ClientHello { handshakeProducer.produce(shc, clientHello); } else { // unlikely - shc.conContext.fatal(Alert.HANDSHAKE_FAILURE, + throw shc.conContext.fatal(Alert.HANDSHAKE_FAILURE, "No HelloRetryRequest producer: " + shc.handshakeProducers); } if (!shc.handshakeProducers.isEmpty()) { // unlikely, but please double check. - shc.conContext.fatal(Alert.HANDSHAKE_FAILURE, + throw shc.conContext.fatal(Alert.HANDSHAKE_FAILURE, "unknown handshake producers: " + shc.handshakeProducers); } } @@ -1264,13 +1265,13 @@ final class ClientHello { if (shc.conContext.isNegotiated) { if (!shc.conContext.secureRenegotiation && !HandshakeContext.allowUnsafeRenegotiation) { - shc.conContext.fatal(Alert.HANDSHAKE_FAILURE, + throw shc.conContext.fatal(Alert.HANDSHAKE_FAILURE, "Unsafe renegotiation is not allowed"); } if (ServerHandshakeContext.rejectClientInitiatedRenego && !shc.kickstartMessageDelivered) { - shc.conContext.fatal(Alert.HANDSHAKE_FAILURE, + throw shc.conContext.fatal(Alert.HANDSHAKE_FAILURE, "Client initiated renegotiation is not allowed"); } } diff --git a/src/java.base/share/classes/sun/security/ssl/ClientKeyExchange.java b/src/java.base/share/classes/sun/security/ssl/ClientKeyExchange.java index 8f442004450..040a604ae21 100644 --- a/src/java.base/share/classes/sun/security/ssl/ClientKeyExchange.java +++ b/src/java.base/share/classes/sun/security/ssl/ClientKeyExchange.java @@ -68,9 +68,8 @@ final class ClientKeyExchange { } // not consumer defined. - chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, + throw chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, "Unexpected ClientKeyExchange handshake message."); - return null; // make the compiler happe } } @@ -105,7 +104,7 @@ final class ClientKeyExchange { } // not consumer defined. - shc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, + throw shc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, "Unexpected ClientKeyExchange handshake message."); } } diff --git a/src/java.base/share/classes/sun/security/ssl/CookieExtension.java b/src/java.base/share/classes/sun/security/ssl/CookieExtension.java index a81695fcf4a..71df50c1310 100644 --- a/src/java.base/share/classes/sun/security/ssl/CookieExtension.java +++ b/src/java.base/share/classes/sun/security/ssl/CookieExtension.java @@ -163,8 +163,7 @@ public class CookieExtension { try { spec = new CookieSpec(buffer); } catch (IOException ioe) { - shc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, ioe); - return; // fatal() always throws, make the compiler happy. + throw shc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, ioe); } shc.handshakeExtensions.put(SSLExtension.CH_COOKIE, spec); @@ -201,9 +200,8 @@ public class CookieExtension { HelloCookieManager hcm = shc.sslContext.getHelloCookieManager(shc.negotiatedProtocol); if (!hcm.isCookieValid(shc, clientHello, spec.cookie)) { - shc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, + throw shc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, "unrecognized cookie"); - return; // fatal() always throws, make the compiler happy. } } } @@ -270,8 +268,7 @@ public class CookieExtension { try { spec = new CookieSpec(buffer); } catch (IOException ioe) { - chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, ioe); - return; // fatal() always throws, make the compiler happy. + throw chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, ioe); } chc.handshakeExtensions.put(SSLExtension.HRR_COOKIE, spec); diff --git a/src/java.base/share/classes/sun/security/ssl/DHClientKeyExchange.java b/src/java.base/share/classes/sun/security/ssl/DHClientKeyExchange.java index 17f3acba17b..86e2c618a95 100644 --- a/src/java.base/share/classes/sun/security/ssl/DHClientKeyExchange.java +++ b/src/java.base/share/classes/sun/security/ssl/DHClientKeyExchange.java @@ -87,7 +87,7 @@ final class DHClientKeyExchange { if (dhePossession == null) { // unlikely - chc.conContext.fatal(Alert.HANDSHAKE_FAILURE, + throw chc.conContext.fatal(Alert.HANDSHAKE_FAILURE, "No DHE credentials negotiated for client key exchange"); } @@ -104,14 +104,14 @@ final class DHClientKeyExchange { (ServerHandshakeContext)handshakeContext; if (m.remaining() < 3) { - shc.conContext.fatal(Alert.HANDSHAKE_FAILURE, + throw shc.conContext.fatal(Alert.HANDSHAKE_FAILURE, "Invalid DH ClientKeyExchange message: insufficient data"); } this.y = Record.getBytes16(m); if (m.hasRemaining()) { - shc.conContext.fatal(Alert.HANDSHAKE_FAILURE, + throw shc.conContext.fatal(Alert.HANDSHAKE_FAILURE, "Invalid DH ClientKeyExchange message: unknown extra data"); } } @@ -177,7 +177,7 @@ final class DHClientKeyExchange { } if (dheCredentials == null) { - chc.conContext.fatal(Alert.HANDSHAKE_FAILURE, + throw chc.conContext.fatal(Alert.HANDSHAKE_FAILURE, "No DHE credentials negotiated for client key exchange"); } @@ -202,7 +202,7 @@ final class DHClientKeyExchange { chc.negotiatedProtocol); if (ke == null) { // unlikely - chc.conContext.fatal(Alert.INTERNAL_ERROR, + throw chc.conContext.fatal(Alert.INTERNAL_ERROR, "Not supported key exchange type"); } else { SSLKeyDerivation masterKD = ke.createKeyDerivation(chc); @@ -214,7 +214,7 @@ final class DHClientKeyExchange { SSLTrafficKeyDerivation.valueOf(chc.negotiatedProtocol); if (kd == null) { // unlikely - chc.conContext.fatal(Alert.INTERNAL_ERROR, + throw chc.conContext.fatal(Alert.INTERNAL_ERROR, "Not supported key derivation: " + chc.negotiatedProtocol); } else { @@ -254,7 +254,7 @@ final class DHClientKeyExchange { if (dhePossession == null) { // unlikely - shc.conContext.fatal(Alert.HANDSHAKE_FAILURE, + throw shc.conContext.fatal(Alert.HANDSHAKE_FAILURE, "No expected DHE possessions for client key exchange"); } @@ -263,7 +263,7 @@ final class DHClientKeyExchange { shc.negotiatedProtocol); if (ke == null) { // unlikely - shc.conContext.fatal(Alert.INTERNAL_ERROR, + throw shc.conContext.fatal(Alert.INTERNAL_ERROR, "Not supported key exchange type"); } @@ -310,7 +310,7 @@ final class DHClientKeyExchange { SSLTrafficKeyDerivation.valueOf(shc.negotiatedProtocol); if (kd == null) { // unlikely - shc.conContext.fatal(Alert.INTERNAL_ERROR, + throw shc.conContext.fatal(Alert.INTERNAL_ERROR, "Not supported key derivation: " + shc.negotiatedProtocol); } else { shc.handshakeKeyDerivation = diff --git a/src/java.base/share/classes/sun/security/ssl/DHKeyExchange.java b/src/java.base/share/classes/sun/security/ssl/DHKeyExchange.java index 8bc3e95bcac..a6668f816cc 100644 --- a/src/java.base/share/classes/sun/security/ssl/DHKeyExchange.java +++ b/src/java.base/share/classes/sun/security/ssl/DHKeyExchange.java @@ -438,7 +438,7 @@ final class DHKeyExchange { } if (dhePossession == null || dheCredentials == null) { - context.conContext.fatal(Alert.HANDSHAKE_FAILURE, + throw context.conContext.fatal(Alert.HANDSHAKE_FAILURE, "No sufficient DHE key agreement parameters negotiated"); } diff --git a/src/java.base/share/classes/sun/security/ssl/DHServerKeyExchange.java b/src/java.base/share/classes/sun/security/ssl/DHServerKeyExchange.java index f12d6be3f4e..d09bf5d4a80 100644 --- a/src/java.base/share/classes/sun/security/ssl/DHServerKeyExchange.java +++ b/src/java.base/share/classes/sun/security/ssl/DHServerKeyExchange.java @@ -106,7 +106,7 @@ final class DHServerKeyExchange { if (dhePossession == null) { // unlikely - shc.conContext.fatal(Alert.ILLEGAL_PARAMETER, + throw shc.conContext.fatal(Alert.ILLEGAL_PARAMETER, "No DHE credentials negotiated for server key exchange"); } DHPublicKey publicKey = dhePossession.publicKey; @@ -132,7 +132,7 @@ final class DHServerKeyExchange { if (signatureScheme == null) { // Unlikely, the credentials generator should have // selected the preferable signature algorithm properly. - shc.conContext.fatal(Alert.INTERNAL_ERROR, + throw shc.conContext.fatal(Alert.INTERNAL_ERROR, "No preferred signature algorithm"); } try { @@ -140,7 +140,7 @@ final class DHServerKeyExchange { x509Possession.popPrivateKey); } catch (NoSuchAlgorithmException | InvalidKeyException | InvalidAlgorithmParameterException nsae) { - shc.conContext.fatal(Alert.INTERNAL_ERROR, + throw shc.conContext.fatal(Alert.INTERNAL_ERROR, "Unsupported signature algorithm: " + signatureScheme.name, nsae); } @@ -151,7 +151,7 @@ final class DHServerKeyExchange { x509Possession.popPrivateKey.getAlgorithm(), x509Possession.popPrivateKey); } catch (NoSuchAlgorithmException | InvalidKeyException e) { - shc.conContext.fatal(Alert.INTERNAL_ERROR, + throw shc.conContext.fatal(Alert.INTERNAL_ERROR, "Unsupported signature algorithm: " + x509Possession.popPrivateKey.getAlgorithm(), e); } @@ -163,7 +163,7 @@ final class DHServerKeyExchange { shc.serverHelloRandom.randomBytes); signature = signer.sign(); } catch (SignatureException ex) { - shc.conContext.fatal(Alert.INTERNAL_ERROR, + throw shc.conContext.fatal(Alert.INTERNAL_ERROR, "Failed to sign dhe parameters: " + x509Possession.popPrivateKey.getAlgorithm(), ex); } @@ -189,7 +189,7 @@ final class DHServerKeyExchange { new BigInteger(1, p), new BigInteger(1, p))); } catch (InvalidKeyException ike) { - chc.conContext.fatal(Alert.HANDSHAKE_FAILURE, + throw chc.conContext.fatal(Alert.HANDSHAKE_FAILURE, "Invalid DH ServerKeyExchange: invalid parameters", ike); } @@ -204,7 +204,7 @@ final class DHServerKeyExchange { if (x509Credentials == null) { // anonymous, no authentication, no signature if (m.hasRemaining()) { - chc.conContext.fatal(Alert.HANDSHAKE_FAILURE, + throw chc.conContext.fatal(Alert.HANDSHAKE_FAILURE, "Invalid DH ServerKeyExchange: unknown extra data"); } @@ -221,13 +221,13 @@ final class DHServerKeyExchange { int ssid = Record.getInt16(m); signatureScheme = SignatureScheme.valueOf(ssid); if (signatureScheme == null) { - chc.conContext.fatal(Alert.HANDSHAKE_FAILURE, + throw chc.conContext.fatal(Alert.HANDSHAKE_FAILURE, "Invalid signature algorithm (" + ssid + ") used in DH ServerKeyExchange handshake message"); } if (!chc.localSupportedSignAlgs.contains(signatureScheme)) { - chc.conContext.fatal(Alert.HANDSHAKE_FAILURE, + throw chc.conContext.fatal(Alert.HANDSHAKE_FAILURE, "Unsupported signature algorithm (" + signatureScheme.name + ") used in DH ServerKeyExchange handshake message"); @@ -245,11 +245,9 @@ final class DHServerKeyExchange { x509Credentials.popPublicKey); } catch (NoSuchAlgorithmException | InvalidKeyException | InvalidAlgorithmParameterException nsae) { - chc.conContext.fatal(Alert.INTERNAL_ERROR, + throw chc.conContext.fatal(Alert.INTERNAL_ERROR, "Unsupported signature algorithm: " + signatureScheme.name, nsae); - - return; // make the compiler happe } } else { try { @@ -257,11 +255,9 @@ final class DHServerKeyExchange { x509Credentials.popPublicKey.getAlgorithm(), x509Credentials.popPublicKey); } catch (NoSuchAlgorithmException | InvalidKeyException e) { - chc.conContext.fatal(Alert.INTERNAL_ERROR, + throw chc.conContext.fatal(Alert.INTERNAL_ERROR, "Unsupported signature algorithm: " + x509Credentials.popPublicKey.getAlgorithm(), e); - - return; // make the compiler happe } } @@ -271,11 +267,11 @@ final class DHServerKeyExchange { chc.serverHelloRandom.randomBytes); if (!signer.verify(paramsSignature)) { - chc.conContext.fatal(Alert.HANDSHAKE_FAILURE, + throw chc.conContext.fatal(Alert.HANDSHAKE_FAILURE, "Invalid signature on DH ServerKeyExchange message"); } } catch (SignatureException ex) { - chc.conContext.fatal(Alert.HANDSHAKE_FAILURE, + throw chc.conContext.fatal(Alert.HANDSHAKE_FAILURE, "Cannot verify DH ServerKeyExchange signature", ex); } } @@ -535,15 +531,13 @@ final class DHServerKeyExchange { new BigInteger(1, skem.g)); publicKey = (DHPublicKey)kf.generatePublic(spec); } catch (GeneralSecurityException gse) { - chc.conContext.fatal(Alert.INSUFFICIENT_SECURITY, + throw chc.conContext.fatal(Alert.INSUFFICIENT_SECURITY, "Could not generate DHPublicKey", gse); - - return; // make the compiler happy } if (!chc.algorithmConstraints.permits( EnumSet.of(CryptoPrimitive.KEY_AGREEMENT), publicKey)) { - chc.conContext.fatal(Alert.INSUFFICIENT_SECURITY, + throw chc.conContext.fatal(Alert.INSUFFICIENT_SECURITY, "DH ServerKeyExchange does not comply to " + "algorithm constraints"); } diff --git a/src/java.base/share/classes/sun/security/ssl/ECDHClientKeyExchange.java b/src/java.base/share/classes/sun/security/ssl/ECDHClientKeyExchange.java index 672aeaeca9a..4ee812f2201 100644 --- a/src/java.base/share/classes/sun/security/ssl/ECDHClientKeyExchange.java +++ b/src/java.base/share/classes/sun/security/ssl/ECDHClientKeyExchange.java @@ -190,20 +190,20 @@ final class ECDHClientKeyExchange { } if (x509Credentials == null) { - chc.conContext.fatal(Alert.INTERNAL_ERROR, + throw chc.conContext.fatal(Alert.INTERNAL_ERROR, "No server certificate for ECDH client key exchange"); } PublicKey publicKey = x509Credentials.popPublicKey; if (!publicKey.getAlgorithm().equals("EC")) { - chc.conContext.fatal(Alert.ILLEGAL_PARAMETER, + throw chc.conContext.fatal(Alert.ILLEGAL_PARAMETER, "Not EC server certificate for ECDH client key exchange"); } ECParameterSpec params = ((ECPublicKey)publicKey).getParams(); NamedGroup namedGroup = NamedGroup.valueOf(params); if (namedGroup == null) { - chc.conContext.fatal(Alert.ILLEGAL_PARAMETER, + throw chc.conContext.fatal(Alert.ILLEGAL_PARAMETER, "Unsupported EC server cert for ECDH client key exchange"); } @@ -228,7 +228,7 @@ final class ECDHClientKeyExchange { chc.negotiatedProtocol); if (ke == null) { // unlikely - chc.conContext.fatal(Alert.INTERNAL_ERROR, + throw chc.conContext.fatal(Alert.INTERNAL_ERROR, "Not supported key exchange type"); } else { SSLKeyDerivation masterKD = ke.createKeyDerivation(chc); @@ -240,7 +240,7 @@ final class ECDHClientKeyExchange { SSLTrafficKeyDerivation.valueOf(chc.negotiatedProtocol); if (kd == null) { // unlikely - chc.conContext.fatal(Alert.INTERNAL_ERROR, + throw chc.conContext.fatal(Alert.INTERNAL_ERROR, "Not supported key derivation: " + chc.negotiatedProtocol); } else { @@ -280,15 +280,14 @@ final class ECDHClientKeyExchange { if (x509Possession == null) { // unlikely, have been checked during cipher suite negotiation. - shc.conContext.fatal(Alert.INTERNAL_ERROR, + throw shc.conContext.fatal(Alert.INTERNAL_ERROR, "No expected EC server cert for ECDH client key exchange"); - return; // make the compiler happy } PrivateKey privateKey = x509Possession.popPrivateKey; if (!privateKey.getAlgorithm().equals("EC")) { // unlikely, have been checked during cipher suite negotiation. - shc.conContext.fatal(Alert.ILLEGAL_PARAMETER, + throw shc.conContext.fatal(Alert.ILLEGAL_PARAMETER, "Not EC server cert for ECDH client key exchange"); } @@ -296,7 +295,7 @@ final class ECDHClientKeyExchange { NamedGroup namedGroup = NamedGroup.valueOf(params); if (namedGroup == null) { // unlikely, have been checked during cipher suite negotiation. - shc.conContext.fatal(Alert.ILLEGAL_PARAMETER, + throw shc.conContext.fatal(Alert.ILLEGAL_PARAMETER, "Unsupported EC server cert for ECDH client key exchange"); } @@ -305,9 +304,8 @@ final class ECDHClientKeyExchange { shc.negotiatedProtocol); if (ke == null) { // unlikely - shc.conContext.fatal(Alert.INTERNAL_ERROR, + throw shc.conContext.fatal(Alert.INTERNAL_ERROR, "Not supported key exchange type"); - return; // make the compiler happy } // parse the handshake message @@ -353,7 +351,7 @@ final class ECDHClientKeyExchange { SSLTrafficKeyDerivation.valueOf(shc.negotiatedProtocol); if (kd == null) { // unlikely - shc.conContext.fatal(Alert.INTERNAL_ERROR, + throw shc.conContext.fatal(Alert.INTERNAL_ERROR, "Not supported key derivation: " + shc.negotiatedProtocol); } else { shc.handshakeKeyDerivation = @@ -387,7 +385,7 @@ final class ECDHClientKeyExchange { } if (ecdheCredentials == null) { - chc.conContext.fatal(Alert.INTERNAL_ERROR, + throw chc.conContext.fatal(Alert.INTERNAL_ERROR, "No ECDHE credentials negotiated for client key exchange"); } @@ -412,7 +410,7 @@ final class ECDHClientKeyExchange { chc.negotiatedProtocol); if (ke == null) { // unlikely - chc.conContext.fatal(Alert.INTERNAL_ERROR, + throw chc.conContext.fatal(Alert.INTERNAL_ERROR, "Not supported key exchange type"); } else { SSLKeyDerivation masterKD = ke.createKeyDerivation(chc); @@ -424,7 +422,7 @@ final class ECDHClientKeyExchange { SSLTrafficKeyDerivation.valueOf(chc.negotiatedProtocol); if (kd == null) { // unlikely - chc.conContext.fatal(Alert.INTERNAL_ERROR, + throw chc.conContext.fatal(Alert.INTERNAL_ERROR, "Not supported key derivation: " + chc.negotiatedProtocol); } else { @@ -463,16 +461,15 @@ final class ECDHClientKeyExchange { } if (ecdhePossession == null) { // unlikely - shc.conContext.fatal(Alert.INTERNAL_ERROR, + throw shc.conContext.fatal(Alert.INTERNAL_ERROR, "No expected ECDHE possessions for client key exchange"); - return; // make the compiler happy } ECParameterSpec params = ecdhePossession.publicKey.getParams(); NamedGroup namedGroup = NamedGroup.valueOf(params); if (namedGroup == null) { // unlikely, have been checked during cipher suite negotiation. - shc.conContext.fatal(Alert.ILLEGAL_PARAMETER, + throw shc.conContext.fatal(Alert.ILLEGAL_PARAMETER, "Unsupported EC server cert for ECDHE client key exchange"); } @@ -481,9 +478,8 @@ final class ECDHClientKeyExchange { shc.negotiatedProtocol); if (ke == null) { // unlikely - shc.conContext.fatal(Alert.INTERNAL_ERROR, + throw shc.conContext.fatal(Alert.INTERNAL_ERROR, "Not supported key exchange type"); - return; // make the compiler happy } // parse the handshake message @@ -529,7 +525,7 @@ final class ECDHClientKeyExchange { SSLTrafficKeyDerivation.valueOf(shc.negotiatedProtocol); if (kd == null) { // unlikely - shc.conContext.fatal(Alert.INTERNAL_ERROR, + throw shc.conContext.fatal(Alert.INTERNAL_ERROR, "Not supported key derivation: " + shc.negotiatedProtocol); } else { shc.handshakeKeyDerivation = diff --git a/src/java.base/share/classes/sun/security/ssl/ECDHKeyExchange.java b/src/java.base/share/classes/sun/security/ssl/ECDHKeyExchange.java index 64423669d19..668247cd99e 100644 --- a/src/java.base/share/classes/sun/security/ssl/ECDHKeyExchange.java +++ b/src/java.base/share/classes/sun/security/ssl/ECDHKeyExchange.java @@ -274,7 +274,7 @@ final class ECDHKeyExchange { NamedGroup ng = NamedGroup.valueOf(params); if (ng == null) { // unlikely, have been checked during cipher suite negotiation. - shc.conContext.fatal(Alert.ILLEGAL_PARAMETER, + throw shc.conContext.fatal(Alert.ILLEGAL_PARAMETER, "Unsupported EC server cert for ECDH key exchange"); } @@ -295,7 +295,7 @@ final class ECDHKeyExchange { } if (x509Possession == null || ecdheCredentials == null) { - shc.conContext.fatal(Alert.HANDSHAKE_FAILURE, + throw shc.conContext.fatal(Alert.HANDSHAKE_FAILURE, "No sufficient ECDHE key agreement parameters negotiated"); } @@ -327,7 +327,7 @@ final class ECDHKeyExchange { NamedGroup namedGroup = NamedGroup.valueOf(params); if (namedGroup == null) { // unlikely, should have been checked previously - chc.conContext.fatal(Alert.ILLEGAL_PARAMETER, + throw chc.conContext.fatal(Alert.ILLEGAL_PARAMETER, "Unsupported EC server cert for ECDH key exchange"); } @@ -344,7 +344,7 @@ final class ECDHKeyExchange { } if (ecdhePossession == null || x509Credentials == null) { - chc.conContext.fatal(Alert.HANDSHAKE_FAILURE, + throw chc.conContext.fatal(Alert.HANDSHAKE_FAILURE, "No sufficient ECDH key agreement parameters negotiated"); } @@ -388,7 +388,7 @@ final class ECDHKeyExchange { } if (ecdhePossession == null || ecdheCredentials == null) { - context.conContext.fatal(Alert.HANDSHAKE_FAILURE, + throw context.conContext.fatal(Alert.HANDSHAKE_FAILURE, "No sufficient ECDHE key agreement parameters negotiated"); } diff --git a/src/java.base/share/classes/sun/security/ssl/ECDHServerKeyExchange.java b/src/java.base/share/classes/sun/security/ssl/ECDHServerKeyExchange.java index 62f660e586e..bbd9a6b3ebb 100644 --- a/src/java.base/share/classes/sun/security/ssl/ECDHServerKeyExchange.java +++ b/src/java.base/share/classes/sun/security/ssl/ECDHServerKeyExchange.java @@ -113,7 +113,7 @@ final class ECDHServerKeyExchange { if (ecdhePossession == null) { // unlikely - shc.conContext.fatal(Alert.ILLEGAL_PARAMETER, + throw shc.conContext.fatal(Alert.ILLEGAL_PARAMETER, "No ECDHE credentials negotiated for server key exchange"); } @@ -125,7 +125,7 @@ final class ECDHServerKeyExchange { this.namedGroup = NamedGroup.valueOf(params); if ((namedGroup == null) || (namedGroup.oid == null) ) { // unlikely - shc.conContext.fatal(Alert.ILLEGAL_PARAMETER, + throw shc.conContext.fatal(Alert.ILLEGAL_PARAMETER, "Unnamed EC parameter spec: " + params); } @@ -146,7 +146,7 @@ final class ECDHServerKeyExchange { if (signatureScheme == null) { // Unlikely, the credentials generator should have // selected the preferable signature algorithm properly. - shc.conContext.fatal(Alert.INTERNAL_ERROR, + throw shc.conContext.fatal(Alert.INTERNAL_ERROR, "No preferred signature algorithm for " + x509Possession.popPrivateKey.getAlgorithm() + " key"); @@ -156,7 +156,7 @@ final class ECDHServerKeyExchange { x509Possession.popPrivateKey); } catch (NoSuchAlgorithmException | InvalidKeyException | InvalidAlgorithmParameterException nsae) { - shc.conContext.fatal(Alert.INTERNAL_ERROR, + throw shc.conContext.fatal(Alert.INTERNAL_ERROR, "Unsupported signature algorithm: " + signatureScheme.name, nsae); } @@ -167,7 +167,7 @@ final class ECDHServerKeyExchange { x509Possession.popPrivateKey.getAlgorithm(), x509Possession.popPrivateKey); } catch (NoSuchAlgorithmException | InvalidKeyException e) { - shc.conContext.fatal(Alert.INTERNAL_ERROR, + throw shc.conContext.fatal(Alert.INTERNAL_ERROR, "Unsupported signature algorithm: " + x509Possession.popPrivateKey.getAlgorithm(), e); } @@ -180,7 +180,7 @@ final class ECDHServerKeyExchange { namedGroup.id, publicPoint); signature = signer.sign(); } catch (SignatureException ex) { - shc.conContext.fatal(Alert.INTERNAL_ERROR, + throw shc.conContext.fatal(Alert.INTERNAL_ERROR, "Failed to sign ecdhe parameters: " + x509Possession.popPrivateKey.getAlgorithm(), ex); } @@ -199,37 +199,37 @@ final class ECDHServerKeyExchange { byte curveType = (byte)Record.getInt8(m); if (curveType != CURVE_NAMED_CURVE) { // Unlikely as only the named curves should be negotiated. - chc.conContext.fatal(Alert.ILLEGAL_PARAMETER, + throw chc.conContext.fatal(Alert.ILLEGAL_PARAMETER, "Unsupported ECCurveType: " + curveType); } int namedGroupId = Record.getInt16(m); this.namedGroup = NamedGroup.valueOf(namedGroupId); if (namedGroup == null) { - chc.conContext.fatal(Alert.ILLEGAL_PARAMETER, + throw chc.conContext.fatal(Alert.ILLEGAL_PARAMETER, "Unknown named group ID: " + namedGroupId); } if (!SupportedGroups.isSupported(namedGroup)) { - chc.conContext.fatal(Alert.ILLEGAL_PARAMETER, + throw chc.conContext.fatal(Alert.ILLEGAL_PARAMETER, "Unsupported named group: " + namedGroup); } if (namedGroup.oid == null) { - chc.conContext.fatal(Alert.ILLEGAL_PARAMETER, + throw chc.conContext.fatal(Alert.ILLEGAL_PARAMETER, "Unknown named EC curve: " + namedGroup); } ECParameterSpec parameters = JsseJce.getECParameterSpec(namedGroup.oid); if (parameters == null) { - chc.conContext.fatal(Alert.ILLEGAL_PARAMETER, + throw chc.conContext.fatal(Alert.ILLEGAL_PARAMETER, "No supported EC parameter: " + namedGroup); } publicPoint = Record.getBytes8(m); if (publicPoint.length == 0) { - chc.conContext.fatal(Alert.ILLEGAL_PARAMETER, + throw chc.conContext.fatal(Alert.ILLEGAL_PARAMETER, "Insufficient ECPoint data: " + namedGroup); } @@ -242,7 +242,7 @@ final class ECDHServerKeyExchange { new ECPublicKeySpec(point, parameters)); } catch (NoSuchAlgorithmException | InvalidKeySpecException | IOException ex) { - chc.conContext.fatal(Alert.ILLEGAL_PARAMETER, + throw chc.conContext.fatal(Alert.ILLEGAL_PARAMETER, "Invalid ECPoint: " + namedGroup, ex); } @@ -259,7 +259,7 @@ final class ECDHServerKeyExchange { if (x509Credentials == null) { // anonymous, no authentication, no signature if (m.hasRemaining()) { - chc.conContext.fatal(Alert.HANDSHAKE_FAILURE, + throw chc.conContext.fatal(Alert.HANDSHAKE_FAILURE, "Invalid DH ServerKeyExchange: unknown extra data"); } this.signatureScheme = null; @@ -275,13 +275,13 @@ final class ECDHServerKeyExchange { int ssid = Record.getInt16(m); signatureScheme = SignatureScheme.valueOf(ssid); if (signatureScheme == null) { - chc.conContext.fatal(Alert.HANDSHAKE_FAILURE, + throw chc.conContext.fatal(Alert.HANDSHAKE_FAILURE, "Invalid signature algorithm (" + ssid + ") used in ECDH ServerKeyExchange handshake message"); } if (!chc.localSupportedSignAlgs.contains(signatureScheme)) { - chc.conContext.fatal(Alert.HANDSHAKE_FAILURE, + throw chc.conContext.fatal(Alert.HANDSHAKE_FAILURE, "Unsupported signature algorithm (" + signatureScheme.name + ") used in ECDH ServerKeyExchange handshake message"); @@ -299,11 +299,9 @@ final class ECDHServerKeyExchange { x509Credentials.popPublicKey); } catch (NoSuchAlgorithmException | InvalidKeyException | InvalidAlgorithmParameterException nsae) { - chc.conContext.fatal(Alert.INTERNAL_ERROR, + throw chc.conContext.fatal(Alert.INTERNAL_ERROR, "Unsupported signature algorithm: " + signatureScheme.name, nsae); - - return; // make the compiler happe } } else { try { @@ -311,11 +309,9 @@ final class ECDHServerKeyExchange { x509Credentials.popPublicKey.getAlgorithm(), x509Credentials.popPublicKey); } catch (NoSuchAlgorithmException | InvalidKeyException e) { - chc.conContext.fatal(Alert.INTERNAL_ERROR, + throw chc.conContext.fatal(Alert.INTERNAL_ERROR, "Unsupported signature algorithm: " + x509Credentials.popPublicKey.getAlgorithm(), e); - - return; // make the compiler happe } } @@ -326,11 +322,11 @@ final class ECDHServerKeyExchange { namedGroup.id, publicPoint); if (!signer.verify(paramsSignature)) { - chc.conContext.fatal(Alert.HANDSHAKE_FAILURE, + throw chc.conContext.fatal(Alert.HANDSHAKE_FAILURE, "Invalid ECDH ServerKeyExchange signature"); } } catch (SignatureException ex) { - chc.conContext.fatal(Alert.HANDSHAKE_FAILURE, + throw chc.conContext.fatal(Alert.HANDSHAKE_FAILURE, "Cannot verify ECDH ServerKeyExchange signature", ex); } } @@ -546,7 +542,7 @@ final class ECDHServerKeyExchange { if (!chc.algorithmConstraints.permits( EnumSet.of(CryptoPrimitive.KEY_AGREEMENT), skem.publicKey)) { - chc.conContext.fatal(Alert.INSUFFICIENT_SECURITY, + throw chc.conContext.fatal(Alert.INSUFFICIENT_SECURITY, "ECDH ServerKeyExchange does not comply " + "to algorithm constraints"); } diff --git a/src/java.base/share/classes/sun/security/ssl/ECPointFormatsExtension.java b/src/java.base/share/classes/sun/security/ssl/ECPointFormatsExtension.java index 0986d2e87eb..f6dc7db7674 100644 --- a/src/java.base/share/classes/sun/security/ssl/ECPointFormatsExtension.java +++ b/src/java.base/share/classes/sun/security/ssl/ECPointFormatsExtension.java @@ -231,13 +231,12 @@ final class ECPointFormatsExtension { try { spec = new ECPointFormatsSpec(buffer); } catch (IOException ioe) { - shc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, ioe); - return; // fatal() always throws, make the compiler happy. + throw shc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, ioe); } // per RFC 4492, uncompressed points must always be supported. if (!spec.hasUncompressedFormat()) { - shc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, + throw shc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, "Invalid ec_point_formats extension data: " + "peer does not support uncompressed points"); } @@ -272,7 +271,7 @@ final class ECPointFormatsExtension { ECPointFormatsSpec requestedSpec = (ECPointFormatsSpec) chc.handshakeExtensions.get(CH_EC_POINT_FORMATS); if (requestedSpec == null) { - chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, + throw chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, "Unexpected ec_point_formats extension in ServerHello"); } @@ -281,13 +280,12 @@ final class ECPointFormatsExtension { try { spec = new ECPointFormatsSpec(buffer); } catch (IOException ioe) { - chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, ioe); - return; // fatal() always throws, make the compiler happy. + throw chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, ioe); } // per RFC 4492, uncompressed points must always be supported. if (!spec.hasUncompressedFormat()) { - chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, + throw chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, "Invalid ec_point_formats extension data: " + "peer does not support uncompressed points"); } diff --git a/src/java.base/share/classes/sun/security/ssl/EncryptedExtensions.java b/src/java.base/share/classes/sun/security/ssl/EncryptedExtensions.java index ec631d3f1f2..0792d97248d 100644 --- a/src/java.base/share/classes/sun/security/ssl/EncryptedExtensions.java +++ b/src/java.base/share/classes/sun/security/ssl/EncryptedExtensions.java @@ -60,7 +60,7 @@ final class EncryptedExtensions { // Extension extensions<0..2^16-1>; // } EncryptedExtensions; if (m.remaining() < 2) { - handshakeContext.conContext.fatal(Alert.ILLEGAL_PARAMETER, + throw handshakeContext.conContext.fatal(Alert.ILLEGAL_PARAMETER, "Invalid EncryptedExtensions handshake message: " + "no sufficient data"); } diff --git a/src/java.base/share/classes/sun/security/ssl/ExtendedMasterSecretExtension.java b/src/java.base/share/classes/sun/security/ssl/ExtendedMasterSecretExtension.java index 85d31f344d0..7db4840996e 100644 --- a/src/java.base/share/classes/sun/security/ssl/ExtendedMasterSecretExtension.java +++ b/src/java.base/share/classes/sun/security/ssl/ExtendedMasterSecretExtension.java @@ -172,8 +172,7 @@ final class ExtendedMasterSecretExtension { try { spec = new ExtendedMasterSecretSpec(buffer); } catch (IOException ioe) { - shc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, ioe); - return; // fatal() always throws, make the compiler happy. + throw shc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, ioe); } if (shc.isResumption && shc.resumingSession != null && @@ -232,7 +231,7 @@ final class ExtendedMasterSecretExtension { // // As if extended master extension is required for full // handshake, it MUST be used in abbreviated handshake too. - shc.conContext.fatal(Alert.HANDSHAKE_FAILURE, + throw shc.conContext.fatal(Alert.HANDSHAKE_FAILURE, "Extended Master Secret extension is required"); } @@ -242,7 +241,7 @@ final class ExtendedMasterSecretExtension { // session used the "extended_master_secret" extension // but the new ClientHello does not contain it, the // server MUST abort the abbreviated handshake. - shc.conContext.fatal(Alert.HANDSHAKE_FAILURE, + throw shc.conContext.fatal(Alert.HANDSHAKE_FAILURE, "Missing Extended Master Secret extension " + "on session resumption"); } else { @@ -250,7 +249,7 @@ final class ExtendedMasterSecretExtension { // original session nor the new ClientHello uses the // extension, the server SHOULD abort the handshake. if (!SSLConfiguration.allowLegacyResumption) { - shc.conContext.fatal(Alert.HANDSHAKE_FAILURE, + throw shc.conContext.fatal(Alert.HANDSHAKE_FAILURE, "Missing Extended Master Secret extension " + "on session resumption"); } else { // Otherwise, continue with a full handshake. @@ -318,7 +317,7 @@ final class ExtendedMasterSecretExtension { ExtendedMasterSecretSpec requstedSpec = (ExtendedMasterSecretSpec) chc.handshakeExtensions.get(CH_EXTENDED_MASTER_SECRET); if (requstedSpec == null) { - chc.conContext.fatal(Alert.UNSUPPORTED_EXTENSION, + throw chc.conContext.fatal(Alert.UNSUPPORTED_EXTENSION, "Server sent the extended_master_secret " + "extension improperly"); } @@ -328,13 +327,12 @@ final class ExtendedMasterSecretExtension { try { spec = new ExtendedMasterSecretSpec(buffer); } catch (IOException ioe) { - chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, ioe); - return; // fatal() always throws, make the compiler happy. + throw chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, ioe); } if (chc.isResumption && chc.resumingSession != null && !chc.resumingSession.useExtendedMasterSecret) { - chc.conContext.fatal(Alert.UNSUPPORTED_EXTENSION, + throw chc.conContext.fatal(Alert.UNSUPPORTED_EXTENSION, "Server sent an unexpected extended_master_secret " + "extension on session resumption"); } @@ -364,7 +362,7 @@ final class ExtendedMasterSecretExtension { // For full handshake, if a client receives a ServerHello // without the extension, it SHOULD abort the handshake if // it does not wish to interoperate with legacy servers. - chc.conContext.fatal(Alert.HANDSHAKE_FAILURE, + throw chc.conContext.fatal(Alert.HANDSHAKE_FAILURE, "Extended Master Secret extension is required"); } @@ -374,14 +372,14 @@ final class ExtendedMasterSecretExtension { // the "extended_master_secret" extension but the new // ServerHello does not contain the extension, the client // MUST abort the handshake. - chc.conContext.fatal(Alert.HANDSHAKE_FAILURE, + throw chc.conContext.fatal(Alert.HANDSHAKE_FAILURE, "Missing Extended Master Secret extension " + "on session resumption"); } else if (SSLConfiguration.useExtendedMasterSecret && !SSLConfiguration.allowLegacyResumption && chc.negotiatedProtocol.useTLS10PlusSpec()) { // Unlikely, abbreviated handshake should be discarded. - chc.conContext.fatal(Alert.HANDSHAKE_FAILURE, + throw chc.conContext.fatal(Alert.HANDSHAKE_FAILURE, "Extended Master Secret extension is required"); } } diff --git a/src/java.base/share/classes/sun/security/ssl/Finished.java b/src/java.base/share/classes/sun/security/ssl/Finished.java index 47a071027b4..aabf09a290f 100644 --- a/src/java.base/share/classes/sun/security/ssl/Finished.java +++ b/src/java.base/share/classes/sun/security/ssl/Finished.java @@ -83,7 +83,7 @@ final class Finished { try { vd = vds.createVerifyData(context, false); } catch (IOException ioe) { - context.conContext.fatal(Alert.ILLEGAL_PARAMETER, + throw context.conContext.fatal(Alert.ILLEGAL_PARAMETER, "Failed to generate verify_data", ioe); } @@ -102,7 +102,7 @@ final class Finished { } if (m.remaining() != verifyDataLen) { - context.conContext.fatal(Alert.ILLEGAL_PARAMETER, + throw context.conContext.fatal(Alert.ILLEGAL_PARAMETER, "Inappropriate finished message: need " + verifyDataLen + " but remaining " + m.remaining() + " bytes verify_data"); } @@ -116,12 +116,11 @@ final class Finished { try { myVerifyData = vd.createVerifyData(context, true); } catch (IOException ioe) { - context.conContext.fatal(Alert.ILLEGAL_PARAMETER, + throw context.conContext.fatal(Alert.ILLEGAL_PARAMETER, "Failed to generate verify_data", ioe); - return; } if (!MessageDigest.isEqual(myVerifyData, verifyData)) { - context.conContext.fatal(Alert.ILLEGAL_PARAMETER, + throw context.conContext.fatal(Alert.ILLEGAL_PARAMETER, "The Finished message cannot be verified."); } } @@ -518,7 +517,7 @@ final class Finished { // we have received ChangeCipherSpec if (hc.conContext.consumers.containsKey( ContentType.CHANGE_CIPHER_SPEC.id)) { - hc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, + throw hc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, "Missing ChangeCipherSpec message"); } @@ -679,19 +678,17 @@ final class Finished { SSLKeyDerivation kd = chc.handshakeKeyDerivation; if (kd == null) { // unlikely - chc.conContext.fatal(Alert.INTERNAL_ERROR, + throw chc.conContext.fatal(Alert.INTERNAL_ERROR, "no key derivation"); - return null; } SSLTrafficKeyDerivation kdg = SSLTrafficKeyDerivation.valueOf(chc.negotiatedProtocol); if (kdg == null) { // unlikely - chc.conContext.fatal(Alert.INTERNAL_ERROR, + throw chc.conContext.fatal(Alert.INTERNAL_ERROR, "Not supported key derivation: " + chc.negotiatedProtocol); - return null; } try { @@ -714,12 +711,10 @@ final class Finished { chc.sslContext.getSecureRandom()); if (writeCipher == null) { - chc.conContext.fatal(Alert.ILLEGAL_PARAMETER, + throw chc.conContext.fatal(Alert.ILLEGAL_PARAMETER, "Illegal cipher suite (" + chc.negotiatedCipherSuite + ") and protocol version (" + chc.negotiatedProtocol + ")"); - - return null; } chc.baseWriteSecret = writeSecret; @@ -727,9 +722,8 @@ final class Finished { writeCipher, false); } catch (GeneralSecurityException gse) { - chc.conContext.fatal(Alert.INTERNAL_ERROR, + throw chc.conContext.fatal(Alert.INTERNAL_ERROR, "Failure to derive application secrets", gse); - return null; } // The resumption master secret is stored in the session so @@ -772,19 +766,17 @@ final class Finished { SSLKeyDerivation kd = shc.handshakeKeyDerivation; if (kd == null) { // unlikely - shc.conContext.fatal(Alert.INTERNAL_ERROR, + throw shc.conContext.fatal(Alert.INTERNAL_ERROR, "no key derivation"); - return null; } SSLTrafficKeyDerivation kdg = SSLTrafficKeyDerivation.valueOf(shc.negotiatedProtocol); if (kdg == null) { // unlikely - shc.conContext.fatal(Alert.INTERNAL_ERROR, + throw shc.conContext.fatal(Alert.INTERNAL_ERROR, "Not supported key derivation: " + shc.negotiatedProtocol); - return null; } // derive salt secret @@ -821,12 +813,10 @@ final class Finished { shc.sslContext.getSecureRandom()); if (writeCipher == null) { - shc.conContext.fatal(Alert.ILLEGAL_PARAMETER, + throw shc.conContext.fatal(Alert.ILLEGAL_PARAMETER, "Illegal cipher suite (" + shc.negotiatedCipherSuite + ") and protocol version (" + shc.negotiatedProtocol + ")"); - - return null; } shc.baseWriteSecret = writeSecret; @@ -836,9 +826,8 @@ final class Finished { // update the context for the following key derivation shc.handshakeKeyDerivation = secretKD; } catch (GeneralSecurityException gse) { - shc.conContext.fatal(Alert.INTERNAL_ERROR, + throw shc.conContext.fatal(Alert.INTERNAL_ERROR, "Failure to derive application secrets", gse); - return null; } /* @@ -911,19 +900,17 @@ final class Finished { SSLKeyDerivation kd = chc.handshakeKeyDerivation; if (kd == null) { // unlikely - chc.conContext.fatal(Alert.INTERNAL_ERROR, + throw chc.conContext.fatal(Alert.INTERNAL_ERROR, "no key derivation"); - return; } SSLTrafficKeyDerivation kdg = SSLTrafficKeyDerivation.valueOf(chc.negotiatedProtocol); if (kdg == null) { // unlikely - chc.conContext.fatal(Alert.INTERNAL_ERROR, + throw chc.conContext.fatal(Alert.INTERNAL_ERROR, "Not supported key derivation: " + chc.negotiatedProtocol); - return; } // save the session @@ -967,12 +954,10 @@ final class Finished { chc.sslContext.getSecureRandom()); if (readCipher == null) { - chc.conContext.fatal(Alert.ILLEGAL_PARAMETER, + throw chc.conContext.fatal(Alert.ILLEGAL_PARAMETER, "Illegal cipher suite (" + chc.negotiatedCipherSuite + ") and protocol version (" + chc.negotiatedProtocol + ")"); - - return; } chc.baseReadSecret = readSecret; @@ -981,9 +966,8 @@ final class Finished { // update the context for the following key derivation chc.handshakeKeyDerivation = secretKD; } catch (GeneralSecurityException gse) { - chc.conContext.fatal(Alert.INTERNAL_ERROR, + throw chc.conContext.fatal(Alert.INTERNAL_ERROR, "Failure to derive application secrets", gse); - return; } // @@ -1031,19 +1015,17 @@ final class Finished { SSLKeyDerivation kd = shc.handshakeKeyDerivation; if (kd == null) { // unlikely - shc.conContext.fatal(Alert.INTERNAL_ERROR, + throw shc.conContext.fatal(Alert.INTERNAL_ERROR, "no key derivation"); - return; } SSLTrafficKeyDerivation kdg = SSLTrafficKeyDerivation.valueOf(shc.negotiatedProtocol); if (kdg == null) { // unlikely - shc.conContext.fatal(Alert.INTERNAL_ERROR, + throw shc.conContext.fatal(Alert.INTERNAL_ERROR, "Not supported key derivation: " + shc.negotiatedProtocol); - return; } // save the session @@ -1073,12 +1055,10 @@ final class Finished { shc.sslContext.getSecureRandom()); if (readCipher == null) { - shc.conContext.fatal(Alert.ILLEGAL_PARAMETER, + throw shc.conContext.fatal(Alert.ILLEGAL_PARAMETER, "Illegal cipher suite (" + shc.negotiatedCipherSuite + ") and protocol version (" + shc.negotiatedProtocol + ")"); - - return; } shc.baseReadSecret = readSecret; @@ -1094,9 +1074,8 @@ final class Finished { shc.handshakeSession.setResumptionMasterSecret( resumptionMasterSecret); } catch (GeneralSecurityException gse) { - shc.conContext.fatal(Alert.INTERNAL_ERROR, + throw shc.conContext.fatal(Alert.INTERNAL_ERROR, "Failure to derive application secrets", gse); - return; } // update connection context diff --git a/src/java.base/share/classes/sun/security/ssl/HandshakeContext.java b/src/java.base/share/classes/sun/security/ssl/HandshakeContext.java index fd7b7784a5a..a95c691b943 100644 --- a/src/java.base/share/classes/sun/security/ssl/HandshakeContext.java +++ b/src/java.base/share/classes/sun/security/ssl/HandshakeContext.java @@ -365,26 +365,20 @@ abstract class HandshakeContext implements ConnectionContext { // } Handshake; if (plaintext.contentType != ContentType.HANDSHAKE.id) { - conContext.fatal(Alert.INTERNAL_ERROR, + throw conContext.fatal(Alert.INTERNAL_ERROR, "Unexpected operation for record: " + plaintext.contentType); - - return 0; } if (plaintext.fragment == null || plaintext.fragment.remaining() < 4) { - conContext.fatal(Alert.UNEXPECTED_MESSAGE, + throw conContext.fatal(Alert.UNEXPECTED_MESSAGE, "Invalid handshake message: insufficient data"); - - return 0; } byte handshakeType = (byte)Record.getInt8(plaintext.fragment); int handshakeLen = Record.getInt24(plaintext.fragment); if (handshakeLen != plaintext.fragment.remaining()) { - conContext.fatal(Alert.UNEXPECTED_MESSAGE, + throw conContext.fatal(Alert.UNEXPECTED_MESSAGE, "Invalid handshake message: insufficient handshake body"); - - return 0; } return handshakeType; @@ -438,16 +432,15 @@ abstract class HandshakeContext implements ConnectionContext { } if (consumer == null) { - conContext.fatal(Alert.UNEXPECTED_MESSAGE, + throw conContext.fatal(Alert.UNEXPECTED_MESSAGE, "Unexpected handshake message: " + SSLHandshake.nameOf(handshakeType)); - return; } try { consumer.consume(this, fragment); } catch (UnsupportedOperationException unsoe) { - conContext.fatal(Alert.UNEXPECTED_MESSAGE, + throw conContext.fatal(Alert.UNEXPECTED_MESSAGE, "Unsupported handshake message: " + SSLHandshake.nameOf(handshakeType), unsoe); } diff --git a/src/java.base/share/classes/sun/security/ssl/HelloRequest.java b/src/java.base/share/classes/sun/security/ssl/HelloRequest.java index 26560fd908b..395451a01ae 100644 --- a/src/java.base/share/classes/sun/security/ssl/HelloRequest.java +++ b/src/java.base/share/classes/sun/security/ssl/HelloRequest.java @@ -59,7 +59,7 @@ final class HelloRequest { ByteBuffer m) throws IOException { super(handshakeContext); if (m.hasRemaining()) { - handshakeContext.conContext.fatal(Alert.ILLEGAL_PARAMETER, + throw handshakeContext.conContext.fatal(Alert.ILLEGAL_PARAMETER, "Error parsing HelloRequest message: not empty"); } } @@ -185,7 +185,7 @@ final class HelloRequest { if (!chc.kickstartMessageDelivered) { if (!chc.conContext.secureRenegotiation && !HandshakeContext.allowUnsafeRenegotiation) { - chc.conContext.fatal(Alert.HANDSHAKE_FAILURE, + throw chc.conContext.fatal(Alert.HANDSHAKE_FAILURE, "Unsafe renegotiation is not allowed"); } diff --git a/src/java.base/share/classes/sun/security/ssl/HelloVerifyRequest.java b/src/java.base/share/classes/sun/security/ssl/HelloVerifyRequest.java index e0f673ed1a7..1417087e4bc 100644 --- a/src/java.base/share/classes/sun/security/ssl/HelloVerifyRequest.java +++ b/src/java.base/share/classes/sun/security/ssl/HelloVerifyRequest.java @@ -73,7 +73,7 @@ final class HelloVerifyRequest { // opaque cookie<0..2^8-1>; // } HelloVerifyRequest; if (m.remaining() < 3) { - chc.conContext.fatal(Alert.ILLEGAL_PARAMETER, + throw chc.conContext.fatal(Alert.ILLEGAL_PARAMETER, "Invalid HelloVerifyRequest: no sufficient data"); } @@ -186,7 +186,7 @@ final class HelloVerifyRequest { chc.handshakeConsumers.remove(SSLHandshake.SERVER_HELLO.id); } if (!chc.handshakeConsumers.isEmpty()) { - chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, + throw chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, "No more message expected before " + "HelloVerifyRequest is processed"); } diff --git a/src/java.base/share/classes/sun/security/ssl/KeyShareExtension.java b/src/java.base/share/classes/sun/security/ssl/KeyShareExtension.java index 83dab2cdf3c..c54bf66f380 100644 --- a/src/java.base/share/classes/sun/security/ssl/KeyShareExtension.java +++ b/src/java.base/share/classes/sun/security/ssl/KeyShareExtension.java @@ -337,8 +337,7 @@ final class KeyShareExtension { try { spec = new CHKeyShareSpec(buffer); } catch (IOException ioe) { - shc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, ioe); - return; // fatal() always throws, make the compiler happy. + throw shc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, ioe); } List credentials = new LinkedList<>(); @@ -610,16 +609,14 @@ final class KeyShareExtension { if (chc.clientRequestedNamedGroups == null || chc.clientRequestedNamedGroups.isEmpty()) { // No supported groups. - chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, + throw chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, "Unexpected key_share extension in ServerHello"); - return; // fatal() always throws, make the compiler happy. } // Is it a supported and enabled extension? if (!chc.sslConfig.isAvailable(SSLExtension.SH_KEY_SHARE)) { - chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, + throw chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, "Unsupported key_share extension in ServerHello"); - return; // fatal() always throws, make the compiler happy. } // Parse the extension @@ -627,25 +624,22 @@ final class KeyShareExtension { try { spec = new SHKeyShareSpec(buffer); } catch (IOException ioe) { - chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, ioe); - return; // fatal() always throws, make the compiler happy. + throw chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, ioe); } KeyShareEntry keyShare = spec.serverShare; NamedGroup ng = NamedGroup.valueOf(keyShare.namedGroupId); if (ng == null || !SupportedGroups.isActivatable( chc.sslConfig.algorithmConstraints, ng)) { - chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, + throw chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, "Unsupported named group: " + NamedGroup.nameOf(keyShare.namedGroupId)); - return; // fatal() always throws, make the compiler happy. } SSLKeyExchange ke = SSLKeyExchange.valueOf(ng); if (ke == null) { - chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, + throw chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, "No key exchange for named group " + ng.name); - return; // fatal() always throws, make the compiler happy. } SSLCredentials credentials = null; @@ -657,7 +651,7 @@ final class KeyShareExtension { if (!chc.algorithmConstraints.permits( EnumSet.of(CryptoPrimitive.KEY_AGREEMENT), ecdhec.popPublicKey)) { - chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, + throw chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, "ECDHE key share entry does not " + "comply to algorithm constraints"); } else { @@ -665,7 +659,7 @@ final class KeyShareExtension { } } } catch (IOException | GeneralSecurityException ex) { - chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, + throw chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, "Cannot decode named group: " + NamedGroup.nameOf(keyShare.namedGroupId)); } @@ -677,7 +671,7 @@ final class KeyShareExtension { if (!chc.algorithmConstraints.permits( EnumSet.of(CryptoPrimitive.KEY_AGREEMENT), dhec.popPublicKey)) { - chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, + throw chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, "DHE key share entry does not " + "comply to algorithm constraints"); } else { @@ -685,18 +679,18 @@ final class KeyShareExtension { } } } catch (IOException | GeneralSecurityException ex) { - chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, + throw chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, "Cannot decode named group: " + NamedGroup.nameOf(keyShare.namedGroupId)); } } else { - chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, + throw chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, "Unsupported named group: " + NamedGroup.nameOf(keyShare.namedGroupId)); } if (credentials == null) { - chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, + throw chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, "Unsupported named group: " + ng.name); } @@ -794,17 +788,15 @@ final class KeyShareExtension { // Is it a supported and enabled extension? if (!shc.sslConfig.isAvailable(SSLExtension.HRR_KEY_SHARE)) { - shc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, + throw shc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, "Unsupported key_share extension in HelloRetryRequest"); - return null; // make the compiler happy. } if (shc.clientRequestedNamedGroups == null || shc.clientRequestedNamedGroups.isEmpty()) { // No supported groups. - shc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, + throw shc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, "Unexpected key_share extension in HelloRetryRequest"); - return null; // make the compiler happy. } NamedGroup selectedGroup = null; @@ -823,9 +815,8 @@ final class KeyShareExtension { } if (selectedGroup == null) { - shc.conContext.fatal( + throw shc.conContext.fatal( Alert.UNEXPECTED_MESSAGE, "No common named group"); - return null; // make the complier happy } byte[] extdata = new byte[] { @@ -861,9 +852,8 @@ final class KeyShareExtension { // Is it a supported and enabled extension? if (!shc.sslConfig.isAvailable(SSLExtension.HRR_KEY_SHARE)) { - shc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, + throw shc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, "Unsupported key_share extension in HelloRetryRequest"); - return null; // make the compiler happy. } CHKeyShareSpec spec = (CHKeyShareSpec)shc.handshakeExtensions.get( @@ -903,17 +893,15 @@ final class KeyShareExtension { // Is it a supported and enabled extension? if (!chc.sslConfig.isAvailable(SSLExtension.HRR_KEY_SHARE)) { - chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, + throw chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, "Unsupported key_share extension in HelloRetryRequest"); - return; // make the compiler happy. } if (chc.clientRequestedNamedGroups == null || chc.clientRequestedNamedGroups.isEmpty()) { // No supported groups. - chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, + throw chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, "Unexpected key_share extension in HelloRetryRequest"); - return; // make the compiler happy. } // Parse the extension @@ -921,23 +909,20 @@ final class KeyShareExtension { try { spec = new HRRKeyShareSpec(buffer); } catch (IOException ioe) { - chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, ioe); - return; // fatal() always throws, make the compiler happy. + throw chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, ioe); } NamedGroup serverGroup = NamedGroup.valueOf(spec.selectedGroup); if (serverGroup == null) { - chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, + throw chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, "Unsupported HelloRetryRequest selected group: " + NamedGroup.nameOf(spec.selectedGroup)); - return; // fatal() always throws, make the compiler happy. } if (!chc.clientRequestedNamedGroups.contains(serverGroup)) { - chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, + throw chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, "Unexpected HelloRetryRequest selected group: " + serverGroup.name); - return; // fatal() always throws, make the compiler happy. } // update the context diff --git a/src/java.base/share/classes/sun/security/ssl/KeyUpdate.java b/src/java.base/share/classes/sun/security/ssl/KeyUpdate.java index 8477e69db48..13063444ee6 100644 --- a/src/java.base/share/classes/sun/security/ssl/KeyUpdate.java +++ b/src/java.base/share/classes/sun/security/ssl/KeyUpdate.java @@ -78,7 +78,7 @@ final class KeyUpdate { super(context); if (m.remaining() != 1) { - context.conContext.fatal(Alert.ILLEGAL_PARAMETER, + throw context.conContext.fatal(Alert.ILLEGAL_PARAMETER, "KeyUpdate has an unexpected length of "+ m.remaining()); } @@ -86,7 +86,7 @@ final class KeyUpdate { byte request = m.get(); this.status = KeyUpdateRequest.valueOf(request); if (status == null) { - context.conContext.fatal(Alert.ILLEGAL_PARAMETER, + throw context.conContext.fatal(Alert.ILLEGAL_PARAMETER, "Invalid KeyUpdate message value: " + KeyUpdateRequest.nameOf(request)); } @@ -198,18 +198,17 @@ final class KeyUpdate { SSLTrafficKeyDerivation.valueOf(hc.conContext.protocolVersion); if (kdg == null) { // unlikely - hc.conContext.fatal(Alert.INTERNAL_ERROR, + throw hc.conContext.fatal(Alert.INTERNAL_ERROR, "Not supported key derivation: " + hc.conContext.protocolVersion); - return; } SSLKeyDerivation skd = kdg.createKeyDerivation(hc, hc.conContext.inputRecord.readCipher.baseSecret); if (skd == null) { // unlikely - hc.conContext.fatal(Alert.INTERNAL_ERROR, "no key derivation"); - return; + throw hc.conContext.fatal( + Alert.INTERNAL_ERROR, "no key derivation"); } SecretKey nplus1 = skd.deriveKey("TlsUpdateNplus1", null); @@ -225,12 +224,10 @@ final class KeyUpdate { hc.sslContext.getSecureRandom()); if (rc == null) { - hc.conContext.fatal(Alert.ILLEGAL_PARAMETER, + throw hc.conContext.fatal(Alert.ILLEGAL_PARAMETER, "Illegal cipher suite (" + hc.negotiatedCipherSuite + ") and protocol version (" + hc.negotiatedProtocol + ")"); - - return; } rc.baseSecret = nplus1; @@ -239,9 +236,8 @@ final class KeyUpdate { SSLLogger.fine("KeyUpdate: read key updated"); } } catch (GeneralSecurityException gse) { - hc.conContext.fatal(Alert.INTERNAL_ERROR, + throw hc.conContext.fatal(Alert.INTERNAL_ERROR, "Failure to derive read secrets", gse); - return; } if (km.status == KeyUpdateRequest.REQUESTED) { @@ -281,18 +277,17 @@ final class KeyUpdate { SSLTrafficKeyDerivation.valueOf(hc.conContext.protocolVersion); if (kdg == null) { // unlikely - hc.conContext.fatal(Alert.INTERNAL_ERROR, + throw hc.conContext.fatal(Alert.INTERNAL_ERROR, "Not supported key derivation: " + hc.conContext.protocolVersion); - return null; } SSLKeyDerivation skd = kdg.createKeyDerivation(hc, hc.conContext.outputRecord.writeCipher.baseSecret); if (skd == null) { // unlikely - hc.conContext.fatal(Alert.INTERNAL_ERROR, "no key derivation"); - return null; + throw hc.conContext.fatal( + Alert.INTERNAL_ERROR, "no key derivation"); } SecretKey nplus1 = skd.deriveKey("TlsUpdateNplus1", null); @@ -308,17 +303,14 @@ final class KeyUpdate { hc.conContext.protocolVersion, key, ivSpec, hc.sslContext.getSecureRandom()); } catch (GeneralSecurityException gse) { - hc.conContext.fatal(Alert.INTERNAL_ERROR, + throw hc.conContext.fatal(Alert.INTERNAL_ERROR, "Failure to derive write secrets", gse); - return null; } if (wc == null) { - hc.conContext.fatal(Alert.ILLEGAL_PARAMETER, + throw hc.conContext.fatal(Alert.ILLEGAL_PARAMETER, "Illegal cipher suite (" + hc.negotiatedCipherSuite + ") and protocol version (" + hc.negotiatedProtocol + ")"); - - return null; } // Output the handshake message and change the write cipher. diff --git a/src/java.base/share/classes/sun/security/ssl/MaxFragExtension.java b/src/java.base/share/classes/sun/security/ssl/MaxFragExtension.java index 73428e39b10..4a3bb8181bd 100644 --- a/src/java.base/share/classes/sun/security/ssl/MaxFragExtension.java +++ b/src/java.base/share/classes/sun/security/ssl/MaxFragExtension.java @@ -253,13 +253,12 @@ final class MaxFragExtension { try { spec = new MaxFragLenSpec(buffer); } catch (IOException ioe) { - shc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, ioe); - return; // fatal() always throws, make the compiler happy. + throw shc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, ioe); } MaxFragLenEnum mfle = MaxFragLenEnum.valueOf(spec.id); if (mfle == null) { - shc.conContext.fatal(Alert.ILLEGAL_PARAMETER, + throw shc.conContext.fatal(Alert.ILLEGAL_PARAMETER, "the requested maximum fragment length is other " + "than the allowed values"); } @@ -359,7 +358,7 @@ final class MaxFragExtension { MaxFragLenSpec requestedSpec = (MaxFragLenSpec) chc.handshakeExtensions.get(CH_MAX_FRAGMENT_LENGTH); if (requestedSpec == null) { - chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, + throw chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, "Unexpected max_fragment_length extension in ServerHello"); } @@ -368,18 +367,17 @@ final class MaxFragExtension { try { spec = new MaxFragLenSpec(buffer); } catch (IOException ioe) { - chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, ioe); - return; // fatal() always throws, make the compiler happy. + throw chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, ioe); } if (spec.id != requestedSpec.id) { - chc.conContext.fatal(Alert.ILLEGAL_PARAMETER, + throw chc.conContext.fatal(Alert.ILLEGAL_PARAMETER, "The maximum fragment length response is not requested"); } MaxFragLenEnum mfle = MaxFragLenEnum.valueOf(spec.id); if (mfle == null) { - chc.conContext.fatal(Alert.ILLEGAL_PARAMETER, + throw chc.conContext.fatal(Alert.ILLEGAL_PARAMETER, "the requested maximum fragment length is other " + "than the allowed values"); } @@ -532,7 +530,7 @@ final class MaxFragExtension { MaxFragLenSpec requestedSpec = (MaxFragLenSpec) chc.handshakeExtensions.get(CH_MAX_FRAGMENT_LENGTH); if (requestedSpec == null) { - chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, + throw chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, "Unexpected max_fragment_length extension in ServerHello"); } @@ -541,18 +539,17 @@ final class MaxFragExtension { try { spec = new MaxFragLenSpec(buffer); } catch (IOException ioe) { - chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, ioe); - return; // fatal() always throws, make the compiler happy. + throw chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, ioe); } if (spec.id != requestedSpec.id) { - chc.conContext.fatal(Alert.ILLEGAL_PARAMETER, + throw chc.conContext.fatal(Alert.ILLEGAL_PARAMETER, "The maximum fragment length response is not requested"); } MaxFragLenEnum mfle = MaxFragLenEnum.valueOf(spec.id); if (mfle == null) { - chc.conContext.fatal(Alert.ILLEGAL_PARAMETER, + throw chc.conContext.fatal(Alert.ILLEGAL_PARAMETER, "the requested maximum fragment length is other " + "than the allowed values"); } diff --git a/src/java.base/share/classes/sun/security/ssl/NewSessionTicket.java b/src/java.base/share/classes/sun/security/ssl/NewSessionTicket.java index f40bef36ce5..bf89ac48e4f 100644 --- a/src/java.base/share/classes/sun/security/ssl/NewSessionTicket.java +++ b/src/java.base/share/classes/sun/security/ssl/NewSessionTicket.java @@ -86,7 +86,7 @@ final class NewSessionTicket { // Extension extensions<0..2^16-2>; // } NewSessionTicket; if (m.remaining() < 14) { - context.conContext.fatal(Alert.ILLEGAL_PARAMETER, + throw context.conContext.fatal(Alert.ILLEGAL_PARAMETER, "Invalid NewSessionTicket message: no sufficient data"); } @@ -95,18 +95,18 @@ final class NewSessionTicket { this.ticketNonce = Record.getBytes8(m); if (m.remaining() < 5) { - context.conContext.fatal(Alert.ILLEGAL_PARAMETER, + throw context.conContext.fatal(Alert.ILLEGAL_PARAMETER, "Invalid NewSessionTicket message: no sufficient data"); } this.ticket = Record.getBytes16(m); if (ticket.length == 0) { - context.conContext.fatal(Alert.ILLEGAL_PARAMETER, + throw context.conContext.fatal(Alert.ILLEGAL_PARAMETER, "No ticket in the NewSessionTicket handshake message"); } if (m.remaining() < 2) { - context.conContext.fatal(Alert.ILLEGAL_PARAMETER, + throw context.conContext.fatal(Alert.ILLEGAL_PARAMETER, "Invalid NewSessionTicket message: no sufficient data"); } diff --git a/src/java.base/share/classes/sun/security/ssl/PostHandshakeContext.java b/src/java.base/share/classes/sun/security/ssl/PostHandshakeContext.java index 1c899751a09..3b4044d8a4c 100644 --- a/src/java.base/share/classes/sun/security/ssl/PostHandshakeContext.java +++ b/src/java.base/share/classes/sun/security/ssl/PostHandshakeContext.java @@ -43,7 +43,7 @@ final class PostHandshakeContext extends HandshakeContext { super(context); if (!negotiatedProtocol.useTLS13PlusSpec()) { - conContext.fatal(Alert.UNEXPECTED_MESSAGE, + throw conContext.fatal(Alert.UNEXPECTED_MESSAGE, "Post-handshake not supported in " + negotiatedProtocol.name); } @@ -63,16 +63,15 @@ final class PostHandshakeContext extends HandshakeContext { void dispatch(byte handshakeType, ByteBuffer fragment) throws IOException { SSLConsumer consumer = handshakeConsumers.get(handshakeType); if (consumer == null) { - conContext.fatal(Alert.UNEXPECTED_MESSAGE, + throw conContext.fatal(Alert.UNEXPECTED_MESSAGE, "Unexpected post-handshake message: " + SSLHandshake.nameOf(handshakeType)); - return; } try { consumer.consume(this, fragment); } catch (UnsupportedOperationException unsoe) { - conContext.fatal(Alert.UNEXPECTED_MESSAGE, + throw conContext.fatal(Alert.UNEXPECTED_MESSAGE, "Unsupported post-handshake message: " + SSLHandshake.nameOf(handshakeType), unsoe); } diff --git a/src/java.base/share/classes/sun/security/ssl/PreSharedKeyExtension.java b/src/java.base/share/classes/sun/security/ssl/PreSharedKeyExtension.java index dc313a8cc72..de485e23d89 100644 --- a/src/java.base/share/classes/sun/security/ssl/PreSharedKeyExtension.java +++ b/src/java.base/share/classes/sun/security/ssl/PreSharedKeyExtension.java @@ -111,14 +111,14 @@ final class PreSharedKeyExtension { // PskBinderEntry binders<33..2^16-1>; // } OfferedPsks; if (m.remaining() < 44) { - context.conContext.fatal(Alert.ILLEGAL_PARAMETER, + throw context.conContext.fatal(Alert.ILLEGAL_PARAMETER, "Invalid pre_shared_key extension: " + "insufficient data (length=" + m.remaining() + ")"); } int idEncodedLength = Record.getInt16(m); if (idEncodedLength < 7) { - context.conContext.fatal(Alert.ILLEGAL_PARAMETER, + throw context.conContext.fatal(Alert.ILLEGAL_PARAMETER, "Invalid pre_shared_key extension: " + "insufficient identities (length=" + idEncodedLength + ")"); } @@ -128,7 +128,7 @@ final class PreSharedKeyExtension { while (idReadLength < idEncodedLength) { byte[] id = Record.getBytes16(m); if (id.length < 1) { - context.conContext.fatal(Alert.ILLEGAL_PARAMETER, + throw context.conContext.fatal(Alert.ILLEGAL_PARAMETER, "Invalid pre_shared_key extension: " + "insufficient identity (length=" + id.length + ")"); } @@ -140,7 +140,7 @@ final class PreSharedKeyExtension { } if (m.remaining() < 35) { - context.conContext.fatal(Alert.ILLEGAL_PARAMETER, + throw context.conContext.fatal(Alert.ILLEGAL_PARAMETER, "Invalid pre_shared_key extension: " + "insufficient binders data (length=" + m.remaining() + ")"); @@ -148,7 +148,7 @@ final class PreSharedKeyExtension { int bindersEncodedLen = Record.getInt16(m); if (bindersEncodedLen < 33) { - context.conContext.fatal(Alert.ILLEGAL_PARAMETER, + throw context.conContext.fatal(Alert.ILLEGAL_PARAMETER, "Invalid pre_shared_key extension: " + "insufficient binders (length=" + bindersEncodedLen + ")"); @@ -159,7 +159,7 @@ final class PreSharedKeyExtension { while (bindersReadLength < bindersEncodedLen) { byte[] binder = Record.getBytes8(m); if (binder.length < 32) { - context.conContext.fatal(Alert.ILLEGAL_PARAMETER, + throw context.conContext.fatal(Alert.ILLEGAL_PARAMETER, "Invalid pre_shared_key extension: " + "insufficient binder entry (length=" + binder.length + ")"); @@ -271,7 +271,7 @@ final class PreSharedKeyExtension { SHPreSharedKeySpec(HandshakeContext context, ByteBuffer m) throws IOException { if (m.remaining() < 2) { - context.conContext.fatal(Alert.ILLEGAL_PARAMETER, + throw context.conContext.fatal(Alert.ILLEGAL_PARAMETER, "Invalid pre_shared_key extension: " + "insufficient selected_identity (length=" + m.remaining() + ")"); @@ -348,21 +348,20 @@ final class PreSharedKeyExtension { try { pskSpec = new CHPreSharedKeySpec(shc, buffer); } catch (IOException ioe) { - shc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, ioe); - return; // fatal() always throws, make the compiler happy. + throw shc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, ioe); } // The "psk_key_exchange_modes" extension should have been loaded. if (!shc.handshakeExtensions.containsKey( SSLExtension.PSK_KEY_EXCHANGE_MODES)) { - shc.conContext.fatal(Alert.ILLEGAL_PARAMETER, + throw shc.conContext.fatal(Alert.ILLEGAL_PARAMETER, "Client sent PSK but not PSK modes, or the PSK " + "extension is not the last extension"); } // error if id and binder lists are not the same length if (pskSpec.identities.size() != pskSpec.binders.size()) { - shc.conContext.fatal(Alert.ILLEGAL_PARAMETER, + throw shc.conContext.fatal(Alert.ILLEGAL_PARAMETER, "PSK extension has incorrect number of binders"); } @@ -506,7 +505,7 @@ final class PreSharedKeyExtension { SHPreSharedKeySpec shPsk = (SHPreSharedKeySpec) shc.handshakeExtensions.get(SSLExtension.SH_PRE_SHARED_KEY); if (chPsk == null || shPsk == null) { - shc.conContext.fatal(Alert.INTERNAL_ERROR, + throw shc.conContext.fatal(Alert.INTERNAL_ERROR, "Required extensions are unavailable"); } @@ -533,7 +532,7 @@ final class PreSharedKeyExtension { HandshakeHash pskBinderHash, byte[] binder) throws IOException { Optional pskOpt = session.getPreSharedKey(); if (!pskOpt.isPresent()) { - shc.conContext.fatal(Alert.INTERNAL_ERROR, + throw shc.conContext.fatal(Alert.INTERNAL_ERROR, "Session has no PSK"); } SecretKey psk = pskOpt.get(); @@ -542,7 +541,7 @@ final class PreSharedKeyExtension { byte[] computedBinder = computeBinder(shc, binderKey, session, pskBinderHash); if (!Arrays.equals(binder, computedBinder)) { - shc.conContext.fatal(Alert.ILLEGAL_PARAMETER, + throw shc.conContext.fatal(Alert.ILLEGAL_PARAMETER, "Incorect PSK binder value"); } } @@ -770,12 +769,10 @@ final class PreSharedKeyExtension { hmac.init(finishedKey); return hmac.doFinal(digest); } catch (NoSuchAlgorithmException | InvalidKeyException ex) { - context.conContext.fatal(Alert.INTERNAL_ERROR, ex); - return null; // fatal() always throws, make the compiler happy. + throw context.conContext.fatal(Alert.INTERNAL_ERROR, ex); } } catch (GeneralSecurityException ex) { - context.conContext.fatal(Alert.INTERNAL_ERROR, ex); - return null; // fatal() always throws, make the compiler happy. + throw context.conContext.fatal(Alert.INTERNAL_ERROR, ex); } } @@ -794,8 +791,7 @@ final class PreSharedKeyExtension { return hkdf.expand(earlySecret, hkdfInfo, hashAlg.hashLength, "TlsBinderKey"); } catch (GeneralSecurityException ex) { - context.conContext.fatal(Alert.INTERNAL_ERROR, ex); - return null; // fatal() always throws, make the compiler happy. + throw context.conContext.fatal(Alert.INTERNAL_ERROR, ex); } } @@ -834,7 +830,7 @@ final class PreSharedKeyExtension { // Is it a response of the specific request? if (!chc.handshakeExtensions.containsKey( SSLExtension.CH_PRE_SHARED_KEY)) { - chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, + throw chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, "Server sent unexpected pre_shared_key extension"); } @@ -845,7 +841,7 @@ final class PreSharedKeyExtension { } if (shPsk.selectedIdentity != 0) { - chc.conContext.fatal(Alert.ILLEGAL_PARAMETER, + throw chc.conContext.fatal(Alert.ILLEGAL_PARAMETER, "Selected identity index is not in correct range."); } diff --git a/src/java.base/share/classes/sun/security/ssl/PskKeyExchangeModesExtension.java b/src/java.base/share/classes/sun/security/ssl/PskKeyExchangeModesExtension.java index 456db23a4c1..8680c6ceeed 100644 --- a/src/java.base/share/classes/sun/security/ssl/PskKeyExchangeModesExtension.java +++ b/src/java.base/share/classes/sun/security/ssl/PskKeyExchangeModesExtension.java @@ -201,8 +201,7 @@ final class PskKeyExchangeModesExtension { try { spec = new PskKeyExchangeModesSpec(buffer); } catch (IOException ioe) { - shc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, ioe); - return; // fatal() always throws, make the compiler happy. + throw shc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, ioe); } // Update the context. @@ -324,7 +323,7 @@ final class PskKeyExchangeModesExtension { SSLExtensionSpec spec = shc.handshakeExtensions.get(SSLExtension.CH_PRE_SHARED_KEY); if (spec != null) { - shc.conContext.fatal(Alert.HANDSHAKE_FAILURE, + throw shc.conContext.fatal(Alert.HANDSHAKE_FAILURE, "pre_shared_key key extension is offered " + "without a psk_key_exchange_modes extension"); } diff --git a/src/java.base/share/classes/sun/security/ssl/RSAClientKeyExchange.java b/src/java.base/share/classes/sun/security/ssl/RSAClientKeyExchange.java index 912e7de6b40..de26cf91f2e 100644 --- a/src/java.base/share/classes/sun/security/ssl/RSAClientKeyExchange.java +++ b/src/java.base/share/classes/sun/security/ssl/RSAClientKeyExchange.java @@ -75,7 +75,7 @@ final class RSAClientKeyExchange { super(context); if (m.remaining() < 2) { - context.conContext.fatal(Alert.HANDSHAKE_FAILURE, + throw context.conContext.fatal(Alert.HANDSHAKE_FAILURE, "Invalid RSA ClientKeyExchange message: insufficient data"); } @@ -167,14 +167,14 @@ final class RSAClientKeyExchange { } if (rsaCredentials == null && x509Credentials == null) { - chc.conContext.fatal(Alert.ILLEGAL_PARAMETER, + throw chc.conContext.fatal(Alert.ILLEGAL_PARAMETER, "No RSA credentials negotiated for client key exchange"); } PublicKey publicKey = (rsaCredentials != null) ? rsaCredentials.popPublicKey : x509Credentials.popPublicKey; if (!publicKey.getAlgorithm().equals("RSA")) { // unlikely - chc.conContext.fatal(Alert.ILLEGAL_PARAMETER, + throw chc.conContext.fatal(Alert.ILLEGAL_PARAMETER, "Not RSA public key for client key exchange"); } @@ -186,10 +186,8 @@ final class RSAClientKeyExchange { ckem = new RSAClientKeyExchangeMessage( chc, premaster, publicKey); } catch (GeneralSecurityException gse) { - chc.conContext.fatal(Alert.ILLEGAL_PARAMETER, + throw chc.conContext.fatal(Alert.ILLEGAL_PARAMETER, "Cannot generate RSA premaster secret", gse); - - return null; // make the compiler happy } if (SSLLogger.isOn && SSLLogger.isOn("ssl,handshake")) { SSLLogger.fine( @@ -205,7 +203,7 @@ final class RSAClientKeyExchange { chc.negotiatedCipherSuite.keyExchange, chc.negotiatedProtocol); if (ke == null) { // unlikely - chc.conContext.fatal(Alert.INTERNAL_ERROR, + throw chc.conContext.fatal(Alert.INTERNAL_ERROR, "Not supported key exchange type"); } else { SSLKeyDerivation masterKD = ke.createKeyDerivation(chc); @@ -217,7 +215,7 @@ final class RSAClientKeyExchange { SSLTrafficKeyDerivation kd = SSLTrafficKeyDerivation.valueOf(chc.negotiatedProtocol); if (kd == null) { // unlikely - chc.conContext.fatal(Alert.INTERNAL_ERROR, + throw chc.conContext.fatal(Alert.INTERNAL_ERROR, "Not supported key derivation: " + chc.negotiatedProtocol); } else { @@ -262,14 +260,14 @@ final class RSAClientKeyExchange { } if (rsaPossession == null && x509Possession == null) { // unlikely - shc.conContext.fatal(Alert.ILLEGAL_PARAMETER, + throw shc.conContext.fatal(Alert.ILLEGAL_PARAMETER, "No RSA possessions negotiated for client key exchange"); } PrivateKey privateKey = (rsaPossession != null) ? rsaPossession.popPrivateKey : x509Possession.popPrivateKey; if (!privateKey.getAlgorithm().equals("RSA")) { // unlikely - shc.conContext.fatal(Alert.ILLEGAL_PARAMETER, + throw shc.conContext.fatal(Alert.ILLEGAL_PARAMETER, "Not RSA private key for client key exchange"); } @@ -287,7 +285,7 @@ final class RSAClientKeyExchange { RSAPremasterSecret.decode(shc, privateKey, ckem.encrypted); shc.handshakeCredentials.add(premaster); } catch (GeneralSecurityException gse) { - shc.conContext.fatal(Alert.ILLEGAL_PARAMETER, + throw shc.conContext.fatal(Alert.ILLEGAL_PARAMETER, "Cannot decode RSA premaster secret", gse); } @@ -296,7 +294,7 @@ final class RSAClientKeyExchange { shc.negotiatedCipherSuite.keyExchange, shc.negotiatedProtocol); if (ke == null) { // unlikely - shc.conContext.fatal(Alert.INTERNAL_ERROR, + throw shc.conContext.fatal(Alert.INTERNAL_ERROR, "Not supported key exchange type"); } else { SSLKeyDerivation masterKD = ke.createKeyDerivation(shc); @@ -308,7 +306,7 @@ final class RSAClientKeyExchange { SSLTrafficKeyDerivation kd = SSLTrafficKeyDerivation.valueOf(shc.negotiatedProtocol); if (kd == null) { // unlikely - shc.conContext.fatal(Alert.INTERNAL_ERROR, + throw shc.conContext.fatal(Alert.INTERNAL_ERROR, "Not supported key derivation: " + shc.negotiatedProtocol); } else { diff --git a/src/java.base/share/classes/sun/security/ssl/RSAKeyExchange.java b/src/java.base/share/classes/sun/security/ssl/RSAKeyExchange.java index 3d73111cc41..555d9aeaf84 100644 --- a/src/java.base/share/classes/sun/security/ssl/RSAKeyExchange.java +++ b/src/java.base/share/classes/sun/security/ssl/RSAKeyExchange.java @@ -274,7 +274,7 @@ final class RSAKeyExchange { } if (premaster == null) { - context.conContext.fatal(Alert.HANDSHAKE_FAILURE, + throw context.conContext.fatal(Alert.HANDSHAKE_FAILURE, "No sufficient RSA key agreement parameters negotiated"); } diff --git a/src/java.base/share/classes/sun/security/ssl/RSAServerKeyExchange.java b/src/java.base/share/classes/sun/security/ssl/RSAServerKeyExchange.java index 5af690a7b78..ebe82a5a6e4 100644 --- a/src/java.base/share/classes/sun/security/ssl/RSAServerKeyExchange.java +++ b/src/java.base/share/classes/sun/security/ssl/RSAServerKeyExchange.java @@ -94,7 +94,7 @@ final class RSAServerKeyExchange { signature = signer.sign(); } catch (NoSuchAlgorithmException | InvalidKeyException | SignatureException ex) { - shc.conContext.fatal(Alert.INTERNAL_ERROR, + throw shc.conContext.fatal(Alert.INTERNAL_ERROR, "Failed to sign ephemeral RSA parameters", ex); } @@ -122,7 +122,7 @@ final class RSAServerKeyExchange { } if (x509Credentials == null) { - chc.conContext.fatal(Alert.ILLEGAL_PARAMETER, + throw chc.conContext.fatal(Alert.ILLEGAL_PARAMETER, "No RSA credentials negotiated for server key exchange"); } @@ -133,12 +133,12 @@ final class RSAServerKeyExchange { chc.clientHelloRandom.randomBytes, chc.serverHelloRandom.randomBytes); if (!signer.verify(paramsSignature)) { - chc.conContext.fatal(Alert.HANDSHAKE_FAILURE, + throw chc.conContext.fatal(Alert.HANDSHAKE_FAILURE, "Invalid signature of RSA ServerKeyExchange message"); } } catch (NoSuchAlgorithmException | InvalidKeyException | SignatureException ex) { - chc.conContext.fatal(Alert.INTERNAL_ERROR, + throw chc.conContext.fatal(Alert.INTERNAL_ERROR, "Failed to sign ephemeral RSA parameters", ex); } } @@ -250,12 +250,12 @@ final class RSAServerKeyExchange { return null; } else if (x509Possession == null) { // unlikely - shc.conContext.fatal(Alert.ILLEGAL_PARAMETER, + throw shc.conContext.fatal(Alert.ILLEGAL_PARAMETER, "No RSA certificate negotiated for server key exchange"); } else if (!"RSA".equals( x509Possession.popPrivateKey.getAlgorithm())) { // unlikely - shc.conContext.fatal(Alert.ILLEGAL_PARAMETER, + throw shc.conContext.fatal(Alert.ILLEGAL_PARAMETER, "No X.509 possession can be used for " + "ephemeral RSA ServerKeyExchange"); } @@ -312,15 +312,13 @@ final class RSAServerKeyExchange { new BigInteger(1, skem.exponent)); publicKey = (RSAPublicKey)kf.generatePublic(spec); } catch (GeneralSecurityException gse) { - chc.conContext.fatal(Alert.INSUFFICIENT_SECURITY, + throw chc.conContext.fatal(Alert.INSUFFICIENT_SECURITY, "Could not generate RSAPublicKey", gse); - - return; // make the compiler happy } if (!chc.algorithmConstraints.permits( EnumSet.of(CryptoPrimitive.KEY_AGREEMENT), publicKey)) { - chc.conContext.fatal(Alert.INSUFFICIENT_SECURITY, + throw chc.conContext.fatal(Alert.INSUFFICIENT_SECURITY, "RSA ServerKeyExchange does not comply to " + "algorithm constraints"); } @@ -328,7 +326,8 @@ final class RSAServerKeyExchange { // // update // - chc.handshakeCredentials.add(new EphemeralRSACredentials(publicKey)); + chc.handshakeCredentials.add( + new EphemeralRSACredentials(publicKey)); // // produce diff --git a/src/java.base/share/classes/sun/security/ssl/RenegoInfoExtension.java b/src/java.base/share/classes/sun/security/ssl/RenegoInfoExtension.java index c1e684ff733..27b978e6402 100644 --- a/src/java.base/share/classes/sun/security/ssl/RenegoInfoExtension.java +++ b/src/java.base/share/classes/sun/security/ssl/RenegoInfoExtension.java @@ -185,12 +185,10 @@ final class RenegoInfoExtension { return null; } else { // terminate the session. - chc.conContext.fatal(Alert.HANDSHAKE_FAILURE, + throw chc.conContext.fatal(Alert.HANDSHAKE_FAILURE, "insecure renegotiation is not allowed"); } } - - return null; } } @@ -226,14 +224,13 @@ final class RenegoInfoExtension { try { spec = new RenegotiationInfoSpec(buffer); } catch (IOException ioe) { - shc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, ioe); - return; // fatal() always throws, make the compiler happy. + throw shc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, ioe); } if (!shc.conContext.isNegotiated) { // initial handshaking. if (spec.renegotiatedConnection.length != 0) { - shc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, + throw shc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, "Invalid renegotiation_info extension data: not empty"); } shc.conContext.secureRenegotiation = true; @@ -241,14 +238,14 @@ final class RenegoInfoExtension { if (!shc.conContext.secureRenegotiation) { // Unexpected RI extension for insecure renegotiation, // abort the handshake with a fatal handshake_failure alert. - shc.conContext.fatal(Alert.HANDSHAKE_FAILURE, + throw shc.conContext.fatal(Alert.HANDSHAKE_FAILURE, "The renegotiation_info is present in a insecure " + "renegotiation"); } else { // verify the client_verify_data value if (!Arrays.equals(shc.conContext.clientVerifyData, spec.renegotiatedConnection)) { - shc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, + throw shc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, "Invalid renegotiation_info extension data: " + "incorrect verify data in ClientHello"); } @@ -295,7 +292,7 @@ final class RenegoInfoExtension { } if (!HandshakeContext.allowLegacyHelloMessages) { - shc.conContext.fatal(Alert.HANDSHAKE_FAILURE, + throw shc.conContext.fatal(Alert.HANDSHAKE_FAILURE, "Failed to negotiate the use of secure renegotiation"); } // otherwise, allow legacy hello message @@ -307,7 +304,7 @@ final class RenegoInfoExtension { shc.conContext.secureRenegotiation = false; } else if (shc.conContext.secureRenegotiation) { // Require secure renegotiation, terminate the connection. - shc.conContext.fatal(Alert.HANDSHAKE_FAILURE, + throw shc.conContext.fatal(Alert.HANDSHAKE_FAILURE, "Inconsistent secure renegotiation indication"); } else { // renegotiation, not secure if (HandshakeContext.allowUnsafeRenegotiation) { @@ -320,7 +317,7 @@ final class RenegoInfoExtension { if (SSLLogger.isOn && SSLLogger.isOn("ssl,handshake")) { SSLLogger.fine("Terminate insecure renegotiation"); } - shc.conContext.fatal(Alert.HANDSHAKE_FAILURE, + throw shc.conContext.fatal(Alert.HANDSHAKE_FAILURE, "Unsafe renegotiation is not allowed"); } } @@ -430,7 +427,7 @@ final class RenegoInfoExtension { if (requestedSpec == null && !chc.activeCipherSuites.contains( CipherSuite.TLS_EMPTY_RENEGOTIATION_INFO_SCSV)) { - chc.conContext.fatal(Alert.INTERNAL_ERROR, + throw chc.conContext.fatal(Alert.INTERNAL_ERROR, "Missing renegotiation_info and SCSV detected in " + "ClientHello"); } @@ -440,8 +437,7 @@ final class RenegoInfoExtension { try { spec = new RenegotiationInfoSpec(buffer); } catch (IOException ioe) { - chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, ioe); - return; // fatal() always throws, make the compiler happy. + throw chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, ioe); } @@ -452,7 +448,7 @@ final class RenegoInfoExtension { // and if it is not, MUST abort the handshake (by sending // a fatal handshake_failure alert). [RFC 5746] if (spec.renegotiatedConnection.length != 0) { - chc.conContext.fatal(Alert.HANDSHAKE_FAILURE, + throw chc.conContext.fatal(Alert.HANDSHAKE_FAILURE, "Invalid renegotiation_info in ServerHello: " + "not empty renegotiated_connection"); } @@ -467,7 +463,7 @@ final class RenegoInfoExtension { int infoLen = chc.conContext.clientVerifyData.length + chc.conContext.serverVerifyData.length; if (spec.renegotiatedConnection.length != infoLen) { - chc.conContext.fatal(Alert.HANDSHAKE_FAILURE, + throw chc.conContext.fatal(Alert.HANDSHAKE_FAILURE, "Invalid renegotiation_info in ServerHello: " + "invalid renegotiated_connection length (" + spec.renegotiatedConnection.length + ")"); @@ -476,14 +472,14 @@ final class RenegoInfoExtension { byte[] cvd = chc.conContext.clientVerifyData; if (!Arrays.equals(spec.renegotiatedConnection, 0, cvd.length, cvd, 0, cvd.length)) { - chc.conContext.fatal(Alert.HANDSHAKE_FAILURE, + throw chc.conContext.fatal(Alert.HANDSHAKE_FAILURE, "Invalid renegotiation_info in ServerHello: " + "unmatched client_verify_data value"); } byte[] svd = chc.conContext.serverVerifyData; if (!Arrays.equals(spec.renegotiatedConnection, cvd.length, infoLen, svd, 0, svd.length)) { - chc.conContext.fatal(Alert.HANDSHAKE_FAILURE, + throw chc.conContext.fatal(Alert.HANDSHAKE_FAILURE, "Invalid renegotiation_info in ServerHello: " + "unmatched server_verify_data value"); } @@ -516,7 +512,7 @@ final class RenegoInfoExtension { if (requestedSpec == null && !chc.activeCipherSuites.contains( CipherSuite.TLS_EMPTY_RENEGOTIATION_INFO_SCSV)) { - chc.conContext.fatal(Alert.INTERNAL_ERROR, + throw chc.conContext.fatal(Alert.INTERNAL_ERROR, "Missing renegotiation_info and SCSV detected in " + "ClientHello"); } @@ -524,7 +520,7 @@ final class RenegoInfoExtension { if (!chc.conContext.isNegotiated) { // initial handshaking. if (!HandshakeContext.allowLegacyHelloMessages) { - chc.conContext.fatal(Alert.HANDSHAKE_FAILURE, + throw chc.conContext.fatal(Alert.HANDSHAKE_FAILURE, "Failed to negotiate the use of secure renegotiation"); } // otherwise, allow legacy hello message @@ -536,7 +532,7 @@ final class RenegoInfoExtension { chc.conContext.secureRenegotiation = false; } else if (chc.conContext.secureRenegotiation) { // Require secure renegotiation, terminate the connection. - chc.conContext.fatal(Alert.HANDSHAKE_FAILURE, + throw chc.conContext.fatal(Alert.HANDSHAKE_FAILURE, "Inconsistent secure renegotiation indication"); } else { // renegotiation, not secure if (HandshakeContext.allowUnsafeRenegotiation) { @@ -549,7 +545,7 @@ final class RenegoInfoExtension { if (SSLLogger.isOn && SSLLogger.isOn("ssl,handshake")) { SSLLogger.fine("Terminate insecure renegotiation"); } - chc.conContext.fatal(Alert.HANDSHAKE_FAILURE, + throw chc.conContext.fatal(Alert.HANDSHAKE_FAILURE, "Unsafe renegotiation is not allowed"); } } diff --git a/src/java.base/share/classes/sun/security/ssl/SSLEngineImpl.java b/src/java.base/share/classes/sun/security/ssl/SSLEngineImpl.java index a4f2b5dd247..c263caba011 100644 --- a/src/java.base/share/classes/sun/security/ssl/SSLEngineImpl.java +++ b/src/java.base/share/classes/sun/security/ssl/SSLEngineImpl.java @@ -102,10 +102,10 @@ final class SSLEngineImpl extends SSLEngine implements SSLTransport { try { conContext.kickstart(); } catch (IOException ioe) { - conContext.fatal(Alert.HANDSHAKE_FAILURE, + throw conContext.fatal(Alert.HANDSHAKE_FAILURE, "Couldn't kickstart handshaking", ioe); } catch (Exception ex) { // including RuntimeException - conContext.fatal(Alert.INTERNAL_ERROR, + throw conContext.fatal(Alert.INTERNAL_ERROR, "Fail to begin handshake", ex); } } @@ -137,16 +137,14 @@ final class SSLEngineImpl extends SSLEngine implements SSLTransport { srcs, srcsOffset, srcsLength, dsts, dstsOffset, dstsLength); } catch (SSLProtocolException spe) { // may be an unexpected handshake message - conContext.fatal(Alert.UNEXPECTED_MESSAGE, spe); + throw conContext.fatal(Alert.UNEXPECTED_MESSAGE, spe); } catch (IOException ioe) { - conContext.fatal(Alert.INTERNAL_ERROR, + throw conContext.fatal(Alert.INTERNAL_ERROR, "problem wrapping app data", ioe); } catch (Exception ex) { // including RuntimeException - conContext.fatal(Alert.INTERNAL_ERROR, + throw conContext.fatal(Alert.INTERNAL_ERROR, "Fail to wrap application data", ex); } - - return null; // make compiler happy } private SSLEngineResult writeRecord( @@ -275,9 +273,9 @@ final class SSLEngineImpl extends SSLEngine implements SSLTransport { srcs, srcsOffset, srcsLength, dsts, dstsOffset, dstsLength); } catch (SSLHandshakeException she) { // may be record sequence number overflow - conContext.fatal(Alert.HANDSHAKE_FAILURE, she); + throw conContext.fatal(Alert.HANDSHAKE_FAILURE, she); } catch (IOException e) { - conContext.fatal(Alert.UNEXPECTED_MESSAGE, e); + throw conContext.fatal(Alert.UNEXPECTED_MESSAGE, e); } if (ciphertext == null) { @@ -444,7 +442,7 @@ final class SSLEngineImpl extends SSLEngine implements SSLTransport { srcs, srcsOffset, srcsLength, dsts, dstsOffset, dstsLength); } catch (SSLProtocolException spe) { // may be an unexpected handshake message - conContext.fatal(Alert.UNEXPECTED_MESSAGE, + throw conContext.fatal(Alert.UNEXPECTED_MESSAGE, spe.getMessage(), spe); } catch (IOException ioe) { /* @@ -453,14 +451,12 @@ final class SSLEngineImpl extends SSLEngine implements SSLTransport { * got us into this situation, so report that much back. * Our days of consuming are now over anyway. */ - conContext.fatal(Alert.INTERNAL_ERROR, + throw conContext.fatal(Alert.INTERNAL_ERROR, "problem unwrapping net record", ioe); } catch (Exception ex) { // including RuntimeException - conContext.fatal(Alert.INTERNAL_ERROR, + throw conContext.fatal(Alert.INTERNAL_ERROR, "Fail to unwrap network record", ex); } - - return null; // make compiler happy } private SSLEngineResult readRecord( @@ -721,7 +717,7 @@ final class SSLEngineImpl extends SSLEngine implements SSLTransport { if (!conContext.isInputCloseNotified && (conContext.isNegotiated || conContext.handshakeContext != null)) { - conContext.fatal(Alert.INTERNAL_ERROR, + throw conContext.fatal(Alert.INTERNAL_ERROR, "closing inbound before receiving peer's close_notify"); } diff --git a/src/java.base/share/classes/sun/security/ssl/SSLExtensions.java b/src/java.base/share/classes/sun/security/ssl/SSLExtensions.java index aa5ed01c7de..76c72f610af 100644 --- a/src/java.base/share/classes/sun/security/ssl/SSLExtensions.java +++ b/src/java.base/share/classes/sun/security/ssl/SSLExtensions.java @@ -60,7 +60,8 @@ final class SSLExtensions { int extId = Record.getInt16(m); int extLen = Record.getInt16(m); if (extLen > m.remaining()) { - hm.handshakeContext.conContext.fatal(Alert.ILLEGAL_PARAMETER, + throw hm.handshakeContext.conContext.fatal( + Alert.ILLEGAL_PARAMETER, "Error parsing extension (" + extId + "): no sufficient data"); } @@ -86,7 +87,7 @@ final class SSLExtensions { "in the ServerHello handshake message"); } } else { - hm.handshakeContext.conContext.fatal( + throw hm.handshakeContext.conContext.fatal( Alert.UNSUPPORTED_EXTENSION, "extension (" + extId + ") should not be presented in " + handshakeType.name); @@ -102,7 +103,7 @@ final class SSLExtensions { } if (extension.handshakeType != handshakeType) { - hm.handshakeContext.conContext.fatal( + throw hm.handshakeContext.conContext.fatal( Alert.UNSUPPORTED_EXTENSION, "extension (" + extId + ") should not be " + "presented in " + handshakeType.name); diff --git a/src/java.base/share/classes/sun/security/ssl/SSLSocketImpl.java b/src/java.base/share/classes/sun/security/ssl/SSLSocketImpl.java index 0a19de26169..1bed7ee184e 100644 --- a/src/java.base/share/classes/sun/security/ssl/SSLSocketImpl.java +++ b/src/java.base/share/classes/sun/security/ssl/SSLSocketImpl.java @@ -402,7 +402,7 @@ public final class SSLSocketImpl readHandshakeRecord(); } } catch (IOException ioe) { - conContext.fatal(Alert.HANDSHAKE_FAILURE, + throw conContext.fatal(Alert.HANDSHAKE_FAILURE, "Couldn't kickstart handshaking", ioe); } catch (Exception oe) { // including RuntimeException handleException(oe); @@ -642,7 +642,7 @@ public final class SSLSocketImpl if (checkCloseNotify && !conContext.isInputCloseNotified && (conContext.isNegotiated || conContext.handshakeContext != null)) { - conContext.fatal(Alert.INTERNAL_ERROR, + throw conContext.fatal(Alert.INTERNAL_ERROR, "closing inbound before receiving peer's close_notify"); } @@ -982,9 +982,9 @@ public final class SSLSocketImpl conContext.outputRecord.deliver(b, off, len); } catch (SSLHandshakeException she) { // may be record sequence number overflow - conContext.fatal(Alert.HANDSHAKE_FAILURE, she); + throw conContext.fatal(Alert.HANDSHAKE_FAILURE, she); } catch (IOException e) { - conContext.fatal(Alert.UNEXPECTED_MESSAGE, e); + throw conContext.fatal(Alert.UNEXPECTED_MESSAGE, e); } // Is the sequence number is nearly overflow, or has the key usage @@ -1309,7 +1309,8 @@ public final class SSLSocketImpl alert = Alert.INTERNAL_ERROR; } } - conContext.fatal(alert, cause); + + throw conContext.fatal(alert, cause); } private Plaintext handleEOF(EOFException eofe) throws IOException { diff --git a/src/java.base/share/classes/sun/security/ssl/SSLTransport.java b/src/java.base/share/classes/sun/security/ssl/SSLTransport.java index dd3a899ae2e..92615f4185f 100644 --- a/src/java.base/share/classes/sun/security/ssl/SSLTransport.java +++ b/src/java.base/share/classes/sun/security/ssl/SSLTransport.java @@ -115,7 +115,7 @@ interface SSLTransport { } } - context.fatal(Alert.UNEXPECTED_MESSAGE, unsoe); + throw context.fatal(Alert.UNEXPECTED_MESSAGE, unsoe); } catch (BadPaddingException bpe) { /* * The basic SSLv3 record protection involves (optional) @@ -126,15 +126,15 @@ interface SSLTransport { Alert alert = (context.handshakeContext != null) ? Alert.HANDSHAKE_FAILURE : Alert.BAD_RECORD_MAC; - context.fatal(alert, bpe); + throw context.fatal(alert, bpe); } catch (SSLHandshakeException she) { // may be record sequence number overflow - context.fatal(Alert.HANDSHAKE_FAILURE, she); + throw context.fatal(Alert.HANDSHAKE_FAILURE, she); } catch (EOFException eofe) { // rethrow EOFException, the call will handle it if neede. throw eofe; } catch (IOException ioe) { - context.fatal(Alert.UNEXPECTED_MESSAGE, ioe); + throw context.fatal(Alert.UNEXPECTED_MESSAGE, ioe); } if (plaintexts == null || plaintexts.length == 0) { @@ -191,7 +191,7 @@ interface SSLTransport { } if (remains > 0) { - context.fatal(Alert.INTERNAL_ERROR, + throw context.fatal(Alert.INTERNAL_ERROR, "no sufficient room in the destination buffers"); } } diff --git a/src/java.base/share/classes/sun/security/ssl/ServerHello.java b/src/java.base/share/classes/sun/security/ssl/ServerHello.java index a8a38a9dfb8..da5dd5b17e7 100644 --- a/src/java.base/share/classes/sun/security/ssl/ServerHello.java +++ b/src/java.base/share/classes/sun/security/ssl/ServerHello.java @@ -133,7 +133,7 @@ final class ServerHello { this.serverVersion = ProtocolVersion.valueOf(major, minor); if (this.serverVersion == null) { // The client should only request for known protocol versions. - context.conContext.fatal(Alert.PROTOCOL_VERSION, + throw context.conContext.fatal(Alert.PROTOCOL_VERSION, "Unsupported protocol version: " + ProtocolVersion.nameOf(major, minor)); } @@ -143,20 +143,21 @@ final class ServerHello { try { sessionId.checkLength(serverVersion.id); } catch (SSLProtocolException ex) { - handshakeContext.conContext.fatal(Alert.ILLEGAL_PARAMETER, ex); + throw handshakeContext.conContext.fatal( + Alert.ILLEGAL_PARAMETER, ex); } int cipherSuiteId = Record.getInt16(m); this.cipherSuite = CipherSuite.valueOf(cipherSuiteId); if (cipherSuite == null || !context.isNegotiable(cipherSuite)) { - context.conContext.fatal(Alert.ILLEGAL_PARAMETER, + throw context.conContext.fatal(Alert.ILLEGAL_PARAMETER, "Server selected improper ciphersuite " + CipherSuite.nameOf(cipherSuiteId)); } this.compressionMethod = m.get(); if (compressionMethod != 0) { - context.conContext.fatal(Alert.ILLEGAL_PARAMETER, + throw context.conContext.fatal(Alert.ILLEGAL_PARAMETER, "compression type not supported, " + compressionMethod); } @@ -293,10 +294,8 @@ final class ServerHello { KeyExchangeProperties credentials = chooseCipherSuite(shc, clientHello); if (credentials == null) { - shc.conContext.fatal(Alert.HANDSHAKE_FAILURE, + throw shc.conContext.fatal(Alert.HANDSHAKE_FAILURE, "no cipher suites in common"); - - return null; } shc.negotiatedCipherSuite = credentials.cipherSuite; shc.handshakeKeyExchange = credentials.keyExchange; @@ -374,7 +373,7 @@ final class ServerHello { SSLTrafficKeyDerivation.valueOf(shc.negotiatedProtocol); if (kdg == null) { // unlikely - shc.conContext.fatal(Alert.INTERNAL_ERROR, + throw shc.conContext.fatal(Alert.INTERNAL_ERROR, "Not supported key derivation: " + shc.negotiatedProtocol); } else { @@ -458,10 +457,8 @@ final class ServerHello { } } - shc.conContext.fatal(Alert.HANDSHAKE_FAILURE, + throw shc.conContext.fatal(Alert.HANDSHAKE_FAILURE, "no cipher suites in common"); - - return null; } private static final class KeyExchangeProperties { @@ -524,9 +521,8 @@ final class ServerHello { // negotiate the cipher suite. CipherSuite cipherSuite = chooseCipherSuite(shc, clientHello); if (cipherSuite == null) { - shc.conContext.fatal(Alert.HANDSHAKE_FAILURE, + throw shc.conContext.fatal(Alert.HANDSHAKE_FAILURE, "no cipher suites in common"); - return null; } shc.negotiatedCipherSuite = cipherSuite; shc.handshakeSession.setSuite(cipherSuite); @@ -592,9 +588,8 @@ final class ServerHello { SSLKeyExchange ke = shc.handshakeKeyExchange; if (ke == null) { // unlikely - shc.conContext.fatal(Alert.INTERNAL_ERROR, + throw shc.conContext.fatal(Alert.INTERNAL_ERROR, "Not negotiated key shares"); - return null; } SSLKeyDerivation handshakeKD = ke.createKeyDerivation(shc); @@ -605,10 +600,9 @@ final class ServerHello { SSLTrafficKeyDerivation.valueOf(shc.negotiatedProtocol); if (kdg == null) { // unlikely - shc.conContext.fatal(Alert.INTERNAL_ERROR, + throw shc.conContext.fatal(Alert.INTERNAL_ERROR, "Not supported key derivation: " + shc.negotiatedProtocol); - return null; } SSLKeyDerivation kd = @@ -634,18 +628,15 @@ final class ServerHello { shc.sslContext.getSecureRandom()); } catch (GeneralSecurityException gse) { // unlikely - shc.conContext.fatal(Alert.HANDSHAKE_FAILURE, + throw shc.conContext.fatal(Alert.HANDSHAKE_FAILURE, "Missing cipher algorithm", gse); - return null; } if (readCipher == null) { - shc.conContext.fatal(Alert.ILLEGAL_PARAMETER, + throw shc.conContext.fatal(Alert.ILLEGAL_PARAMETER, "Illegal cipher suite (" + shc.negotiatedCipherSuite + ") and protocol version (" + shc.negotiatedProtocol + ")"); - - return null; } shc.baseReadSecret = readSecret; @@ -671,18 +662,15 @@ final class ServerHello { shc.sslContext.getSecureRandom()); } catch (GeneralSecurityException gse) { // unlikely - shc.conContext.fatal(Alert.HANDSHAKE_FAILURE, + throw shc.conContext.fatal(Alert.HANDSHAKE_FAILURE, "Missing cipher algorithm", gse); - return null; } if (writeCipher == null) { - shc.conContext.fatal(Alert.ILLEGAL_PARAMETER, + throw shc.conContext.fatal(Alert.ILLEGAL_PARAMETER, "Illegal cipher suite (" + shc.negotiatedCipherSuite + ") and protocol version (" + shc.negotiatedProtocol + ")"); - - return null; } shc.baseWriteSecret = writeSecret; @@ -764,9 +752,8 @@ final class ServerHello { CipherSuite cipherSuite = T13ServerHelloProducer.chooseCipherSuite(shc, clientHello); if (cipherSuite == null) { - shc.conContext.fatal(Alert.HANDSHAKE_FAILURE, + throw shc.conContext.fatal(Alert.HANDSHAKE_FAILURE, "no cipher suites in common for hello retry request"); - return null; } ServerHelloMessage hhrm = new ServerHelloMessage(shc, @@ -875,7 +862,7 @@ final class ServerHello { SSLHandshake.HELLO_VERIFY_REQUEST.id); } if (!chc.handshakeConsumers.isEmpty()) { - chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, + throw chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, "No more message expected before ServerHello is processed"); } @@ -913,14 +900,14 @@ final class ServerHello { } if (!chc.activeProtocols.contains(serverVersion)) { - chc.conContext.fatal(Alert.PROTOCOL_VERSION, + throw chc.conContext.fatal(Alert.PROTOCOL_VERSION, "The server selected protocol version " + serverVersion + " is not accepted by client preferences " + chc.activeProtocols); } if (!serverVersion.useTLS13PlusSpec()) { - chc.conContext.fatal(Alert.PROTOCOL_VERSION, + throw chc.conContext.fatal(Alert.PROTOCOL_VERSION, "Unexpected HelloRetryRequest for " + serverVersion.name); } @@ -965,7 +952,7 @@ final class ServerHello { } if (!chc.activeProtocols.contains(serverVersion)) { - chc.conContext.fatal(Alert.PROTOCOL_VERSION, + throw chc.conContext.fatal(Alert.PROTOCOL_VERSION, "The server selected protocol version " + serverVersion + " is not accepted by client preferences " + chc.activeProtocols); @@ -982,7 +969,7 @@ final class ServerHello { } if (serverHello.serverRandom.isVersionDowngrade(chc)) { - chc.conContext.fatal(Alert.ILLEGAL_PARAMETER, + throw chc.conContext.fatal(Alert.ILLEGAL_PARAMETER, "A potential protocol version downgrade attack"); } @@ -1025,7 +1012,7 @@ final class ServerHello { ClientHandshakeContext chc = (ClientHandshakeContext)context; ServerHelloMessage serverHello = (ServerHelloMessage)message; if (!chc.isNegotiable(serverHello.serverVersion)) { - chc.conContext.fatal(Alert.PROTOCOL_VERSION, + throw chc.conContext.fatal(Alert.PROTOCOL_VERSION, "Server chose " + serverHello.serverVersion + ", but that protocol version is not enabled or " + "not supported by the client."); @@ -1037,7 +1024,7 @@ final class ServerHello { chc.negotiatedProtocol, chc.negotiatedCipherSuite); chc.serverHelloRandom = serverHello.serverRandom; if (chc.negotiatedCipherSuite.keyExchange == null) { - chc.conContext.fatal(Alert.PROTOCOL_VERSION, + throw chc.conContext.fatal(Alert.PROTOCOL_VERSION, "TLS 1.2 or prior version does not support the " + "server cipher suite: " + chc.negotiatedCipherSuite.name); } @@ -1063,7 +1050,7 @@ final class ServerHello { // Verify that the session ciphers are unchanged. CipherSuite sessionSuite = chc.resumingSession.getSuite(); if (chc.negotiatedCipherSuite != sessionSuite) { - chc.conContext.fatal(Alert.PROTOCOL_VERSION, + throw chc.conContext.fatal(Alert.PROTOCOL_VERSION, "Server returned wrong cipher suite for session"); } @@ -1071,7 +1058,7 @@ final class ServerHello { ProtocolVersion sessionVersion = chc.resumingSession.getProtocolVersion(); if (chc.negotiatedProtocol != sessionVersion) { - chc.conContext.fatal(Alert.PROTOCOL_VERSION, + throw chc.conContext.fatal(Alert.PROTOCOL_VERSION, "Server resumed with wrong protocol version"); } @@ -1090,7 +1077,7 @@ final class ServerHello { } chc.isResumption = false; if (!chc.sslConfig.enableSessionCreation) { - chc.conContext.fatal(Alert.PROTOCOL_VERSION, + throw chc.conContext.fatal(Alert.PROTOCOL_VERSION, "New session creation is disabled"); } } @@ -1109,7 +1096,7 @@ final class ServerHello { } if (!chc.sslConfig.enableSessionCreation) { - chc.conContext.fatal(Alert.PROTOCOL_VERSION, + throw chc.conContext.fatal(Alert.PROTOCOL_VERSION, "New session creation is disabled"); } chc.handshakeSession = new SSLSessionImpl(chc, @@ -1130,7 +1117,7 @@ final class ServerHello { SSLTrafficKeyDerivation.valueOf(chc.negotiatedProtocol); if (kdg == null) { // unlikely - chc.conContext.fatal(Alert.INTERNAL_ERROR, + throw chc.conContext.fatal(Alert.INTERNAL_ERROR, "Not supported key derivation: " + chc.negotiatedProtocol); } else { @@ -1201,7 +1188,7 @@ final class ServerHello { ClientHandshakeContext chc = (ClientHandshakeContext)context; ServerHelloMessage serverHello = (ServerHelloMessage)message; if (serverHello.serverVersion != ProtocolVersion.TLS12) { - chc.conContext.fatal(Alert.PROTOCOL_VERSION, + throw chc.conContext.fatal(Alert.PROTOCOL_VERSION, "The ServerHello.legacy_version field is not TLS 1.2"); } @@ -1226,7 +1213,7 @@ final class ServerHello { } if (!chc.sslConfig.enableSessionCreation) { - chc.conContext.fatal(Alert.PROTOCOL_VERSION, + throw chc.conContext.fatal(Alert.PROTOCOL_VERSION, "New session creation is disabled"); } chc.handshakeSession = new SSLSessionImpl(chc, @@ -1239,7 +1226,7 @@ final class ServerHello { Optional psk = chc.resumingSession.consumePreSharedKey(); if(!psk.isPresent()) { - chc.conContext.fatal(Alert.INTERNAL_ERROR, + throw chc.conContext.fatal(Alert.INTERNAL_ERROR, "No PSK available. Unable to resume."); } @@ -1260,9 +1247,8 @@ final class ServerHello { SSLKeyExchange ke = chc.handshakeKeyExchange; if (ke == null) { // unlikely - chc.conContext.fatal(Alert.INTERNAL_ERROR, + throw chc.conContext.fatal(Alert.INTERNAL_ERROR, "Not negotiated key shares"); - return; } SSLKeyDerivation handshakeKD = ke.createKeyDerivation(chc); @@ -1272,10 +1258,9 @@ final class ServerHello { SSLTrafficKeyDerivation.valueOf(chc.negotiatedProtocol); if (kdg == null) { // unlikely - chc.conContext.fatal(Alert.INTERNAL_ERROR, + throw chc.conContext.fatal(Alert.INTERNAL_ERROR, "Not supported key derivation: " + chc.negotiatedProtocol); - return; } SSLKeyDerivation secretKD = @@ -1302,18 +1287,15 @@ final class ServerHello { chc.sslContext.getSecureRandom()); } catch (GeneralSecurityException gse) { // unlikely - chc.conContext.fatal(Alert.HANDSHAKE_FAILURE, + throw chc.conContext.fatal(Alert.HANDSHAKE_FAILURE, "Missing cipher algorithm", gse); - return; } if (readCipher == null) { - chc.conContext.fatal(Alert.ILLEGAL_PARAMETER, + throw chc.conContext.fatal(Alert.ILLEGAL_PARAMETER, "Illegal cipher suite (" + chc.negotiatedCipherSuite + ") and protocol version (" + chc.negotiatedProtocol + ")"); - - return; } chc.baseReadSecret = readSecret; @@ -1339,18 +1321,15 @@ final class ServerHello { chc.sslContext.getSecureRandom()); } catch (GeneralSecurityException gse) { // unlikely - chc.conContext.fatal(Alert.HANDSHAKE_FAILURE, + throw chc.conContext.fatal(Alert.HANDSHAKE_FAILURE, "Missing cipher algorithm", gse); - return; } if (writeCipher == null) { - chc.conContext.fatal(Alert.ILLEGAL_PARAMETER, + throw chc.conContext.fatal(Alert.ILLEGAL_PARAMETER, "Illegal cipher suite (" + chc.negotiatedCipherSuite + ") and protocol version (" + chc.negotiatedProtocol + ")"); - - return; } chc.baseWriteSecret = writeSecret; @@ -1412,7 +1391,7 @@ final class ServerHello { ClientHandshakeContext chc = (ClientHandshakeContext)context; ServerHelloMessage helloRetryRequest = (ServerHelloMessage)message; if (helloRetryRequest.serverVersion != ProtocolVersion.TLS12) { - chc.conContext.fatal(Alert.PROTOCOL_VERSION, + throw chc.conContext.fatal(Alert.PROTOCOL_VERSION, "The HelloRetryRequest.legacy_version is not TLS 1.2"); } @@ -1442,7 +1421,7 @@ final class ServerHello { chc.initialClientHelloMsg.write(hos); } catch (IOException ioe) { // unlikely - chc.conContext.fatal(Alert.HANDSHAKE_FAILURE, + throw chc.conContext.fatal(Alert.HANDSHAKE_FAILURE, "Failed to construct message hash", ioe); } chc.handshakeHash.deliver(hos.toByteArray()); diff --git a/src/java.base/share/classes/sun/security/ssl/ServerHelloDone.java b/src/java.base/share/classes/sun/security/ssl/ServerHelloDone.java index ed0f23bb76f..0d368f90ecc 100644 --- a/src/java.base/share/classes/sun/security/ssl/ServerHelloDone.java +++ b/src/java.base/share/classes/sun/security/ssl/ServerHelloDone.java @@ -50,7 +50,7 @@ final class ServerHelloDone { ByteBuffer m) throws IOException { super(handshakeContext); if (m.hasRemaining()) { - handshakeContext.conContext.fatal(Alert.ILLEGAL_PARAMETER, + throw handshakeContext.conContext.fatal(Alert.ILLEGAL_PARAMETER, "Error parsing ServerHelloDone message: not empty"); } } diff --git a/src/java.base/share/classes/sun/security/ssl/ServerKeyExchange.java b/src/java.base/share/classes/sun/security/ssl/ServerKeyExchange.java index cb5403ebd70..273bf669723 100644 --- a/src/java.base/share/classes/sun/security/ssl/ServerKeyExchange.java +++ b/src/java.base/share/classes/sun/security/ssl/ServerKeyExchange.java @@ -68,9 +68,8 @@ final class ServerKeyExchange { } // not producer defined. - shc.conContext.fatal(Alert.HANDSHAKE_FAILURE, + throw shc.conContext.fatal(Alert.HANDSHAKE_FAILURE, "No ServerKeyExchange handshake message can be produced."); - return null; // make the compiler happe } } @@ -107,7 +106,7 @@ final class ServerKeyExchange { } // no consumer defined. - chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, + throw chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, "Unexpected ServerKeyExchange handshake message."); } } diff --git a/src/java.base/share/classes/sun/security/ssl/ServerNameExtension.java b/src/java.base/share/classes/sun/security/ssl/ServerNameExtension.java index aae5f933b63..9859bf64c33 100644 --- a/src/java.base/share/classes/sun/security/ssl/ServerNameExtension.java +++ b/src/java.base/share/classes/sun/security/ssl/ServerNameExtension.java @@ -295,8 +295,7 @@ final class ServerNameExtension { try { spec = new CHServerNamesSpec(buffer); } catch (IOException ioe) { - shc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, ioe); - return; // fatal() always throws, make the compiler happy. + throw shc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, ioe); } // Update the context. @@ -314,7 +313,7 @@ final class ServerNameExtension { } } else { // We do not reject client without SNI extension currently. - shc.conContext.fatal(Alert.UNRECOGNIZED_NAME, + throw shc.conContext.fatal(Alert.UNRECOGNIZED_NAME, "Unrecognized server name indication"); } } else { @@ -483,13 +482,13 @@ final class ServerNameExtension { CHServerNamesSpec spec = (CHServerNamesSpec) chc.handshakeExtensions.get(CH_SERVER_NAME); if (spec == null) { - chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, + throw chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, "Unexpected ServerHello server_name extension"); } // Parse the extension. if (buffer.remaining() != 0) { - chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, + throw chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, "Invalid ServerHello server_name extension"); } @@ -570,13 +569,13 @@ final class ServerNameExtension { CHServerNamesSpec spec = (CHServerNamesSpec) chc.handshakeExtensions.get(CH_SERVER_NAME); if (spec == null) { - chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, + throw chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, "Unexpected EncryptedExtensions server_name extension"); } // Parse the extension. if (buffer.remaining() != 0) { - chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, + throw chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, "Invalid EncryptedExtensions server_name extension"); } diff --git a/src/java.base/share/classes/sun/security/ssl/SignatureAlgorithmsExtension.java b/src/java.base/share/classes/sun/security/ssl/SignatureAlgorithmsExtension.java index 6e48134b1d1..27638b48b33 100644 --- a/src/java.base/share/classes/sun/security/ssl/SignatureAlgorithmsExtension.java +++ b/src/java.base/share/classes/sun/security/ssl/SignatureAlgorithmsExtension.java @@ -238,8 +238,7 @@ final class SignatureAlgorithmsExtension { try { spec = new SignatureSchemesSpec(buffer); } catch (IOException ioe) { - shc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, ioe); - return; // fatal() always throws, make the compiler happy. + throw shc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, ioe); } // Update the context. @@ -329,7 +328,7 @@ final class SignatureAlgorithmsExtension { // We may support the server authentication other than X.509 // certificate later. if (shc.negotiatedProtocol.useTLS13PlusSpec()) { - shc.conContext.fatal(Alert.MISSING_EXTENSION, + throw shc.conContext.fatal(Alert.MISSING_EXTENSION, "No mandatory signature_algorithms extension in the " + "received CertificateRequest handshake message"); } @@ -403,10 +402,9 @@ final class SignatureAlgorithmsExtension { // handshake message in TLS 1.3. if (!shc.sslConfig.isAvailable( SSLExtension.CR_SIGNATURE_ALGORITHMS)) { - shc.conContext.fatal(Alert.MISSING_EXTENSION, + throw shc.conContext.fatal(Alert.MISSING_EXTENSION, "No available signature_algorithms extension " + "for client certificate authentication"); - return null; // make the compiler happy } // Produce the extension. @@ -454,10 +452,9 @@ final class SignatureAlgorithmsExtension { // handshake message in TLS 1.3. if (!chc.sslConfig.isAvailable( SSLExtension.CR_SIGNATURE_ALGORITHMS)) { - chc.conContext.fatal(Alert.HANDSHAKE_FAILURE, + throw chc.conContext.fatal(Alert.HANDSHAKE_FAILURE, "No available signature_algorithms extension " + "for client certificate authentication"); - return; // make the compiler happy } // Parse the extension. @@ -465,8 +462,7 @@ final class SignatureAlgorithmsExtension { try { spec = new SignatureSchemesSpec(buffer); } catch (IOException ioe) { - chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, ioe); - return; // fatal() always throws, make the compiler happy. + throw chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, ioe); } List knownSignatureSchemes = new LinkedList<>(); @@ -545,7 +541,7 @@ final class SignatureAlgorithmsExtension { // This is a mandatory extension for CertificateRequest handshake // message in TLS 1.3. - chc.conContext.fatal(Alert.MISSING_EXTENSION, + throw chc.conContext.fatal(Alert.MISSING_EXTENSION, "No mandatory signature_algorithms extension in the " + "received CertificateRequest handshake message"); } diff --git a/src/java.base/share/classes/sun/security/ssl/SupportedGroupsExtension.java b/src/java.base/share/classes/sun/security/ssl/SupportedGroupsExtension.java index b1c11a66ea7..1044b6d13de 100644 --- a/src/java.base/share/classes/sun/security/ssl/SupportedGroupsExtension.java +++ b/src/java.base/share/classes/sun/security/ssl/SupportedGroupsExtension.java @@ -900,8 +900,7 @@ final class SupportedGroupsExtension { try { spec = new SupportedGroupsSpec(buffer); } catch (IOException ioe) { - shc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, ioe); - return; // fatal() always throws, make the compiler happy. + throw shc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, ioe); } // Update the context. @@ -1024,8 +1023,7 @@ final class SupportedGroupsExtension { try { spec = new SupportedGroupsSpec(buffer); } catch (IOException ioe) { - chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, ioe); - return; // fatal() always throws, make the compiler happy. + throw chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, ioe); } // Update the context. diff --git a/src/java.base/share/classes/sun/security/ssl/SupportedVersionsExtension.java b/src/java.base/share/classes/sun/security/ssl/SupportedVersionsExtension.java index cb3f654b82c..3318910abb8 100644 --- a/src/java.base/share/classes/sun/security/ssl/SupportedVersionsExtension.java +++ b/src/java.base/share/classes/sun/security/ssl/SupportedVersionsExtension.java @@ -225,8 +225,7 @@ final class SupportedVersionsExtension { try { spec = new CHSupportedVersionsSpec(buffer); } catch (IOException ioe) { - shc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, ioe); - return; // fatal() always throws, make the compiler happy. + throw shc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, ioe); } // Update the context. @@ -368,8 +367,7 @@ final class SupportedVersionsExtension { try { spec = new SHSupportedVersionsSpec(buffer); } catch (IOException ioe) { - chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, ioe); - return; // fatal() always throws, make the compiler happy. + throw chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, ioe); } // Update the context. @@ -458,8 +456,7 @@ final class SupportedVersionsExtension { try { spec = new SHSupportedVersionsSpec(buffer); } catch (IOException ioe) { - chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, ioe); - return; // fatal() always throws, make the compiler happy. + throw chc.conContext.fatal(Alert.UNEXPECTED_MESSAGE, ioe); } // Update the context. diff --git a/src/java.base/share/classes/sun/security/ssl/TransportContext.java b/src/java.base/share/classes/sun/security/ssl/TransportContext.java index 811610daa33..e0c9fcd4543 100644 --- a/src/java.base/share/classes/sun/security/ssl/TransportContext.java +++ b/src/java.base/share/classes/sun/security/ssl/TransportContext.java @@ -148,9 +148,8 @@ class TransportContext implements ConnectionContext { ContentType ct = ContentType.valueOf(plaintext.contentType); if (ct == null) { - fatal(Alert.UNEXPECTED_MESSAGE, + throw fatal(Alert.UNEXPECTED_MESSAGE, "Unknown content type: " + plaintext.contentType); - return; } switch (ct) { @@ -164,7 +163,7 @@ class TransportContext implements ConnectionContext { protocolVersion.useTLS13PlusSpec()) { handshakeContext = new PostHandshakeContext(this); } else { - fatal(Alert.UNEXPECTED_MESSAGE, + throw fatal(Alert.UNEXPECTED_MESSAGE, "Unexpected post-handshake message: " + SSLHandshake.nameOf(type)); } @@ -185,7 +184,7 @@ class TransportContext implements ConnectionContext { if (consumer != null) { consumer.consume(this, plaintext.fragment); } else { - fatal(Alert.UNEXPECTED_MESSAGE, + throw fatal(Alert.UNEXPECTED_MESSAGE, "Unexpected content: " + plaintext.contentType); } } @@ -250,22 +249,22 @@ class TransportContext implements ConnectionContext { } } - void fatal(Alert alert, + SSLException fatal(Alert alert, String diagnostic) throws SSLException { - fatal(alert, diagnostic, null); + return fatal(alert, diagnostic, null); } - void fatal(Alert alert, Throwable cause) throws SSLException { - fatal(alert, null, cause); + SSLException fatal(Alert alert, Throwable cause) throws SSLException { + return fatal(alert, null, cause); } - void fatal(Alert alert, + SSLException fatal(Alert alert, String diagnostic, Throwable cause) throws SSLException { - fatal(alert, diagnostic, false, cause); + return fatal(alert, diagnostic, false, cause); } // Note: close_notify is not delivered via fatal() methods. - void fatal(Alert alert, String diagnostic, + SSLException fatal(Alert alert, String diagnostic, boolean recvFatalAlert, Throwable cause) throws SSLException { // If we've already shutdown because of an error, there is nothing we // can do except rethrow the exception. From 32ea2e53435e3d0b8740f18e45514f328d2bf966 Mon Sep 17 00:00:00 2001 From: Dean Long Date: Tue, 18 Dec 2018 12:36:27 -0800 Subject: [PATCH 020/101] 8215205: javaVFrame much slower than vframeStream Reviewed-by: mchung, thartmann --- src/hotspot/share/code/scopeDesc.cpp | 12 ++++++-- src/hotspot/share/code/scopeDesc.hpp | 9 +++++- src/hotspot/share/runtime/vframe.cpp | 33 +++++++++++++++++++++ src/hotspot/share/runtime/vframe.hpp | 6 ++++ src/hotspot/share/runtime/vframe.inline.hpp | 7 +++++ src/hotspot/share/runtime/vframe_hp.cpp | 8 +++++ src/hotspot/share/runtime/vframe_hp.hpp | 3 ++ 7 files changed, 75 insertions(+), 3 deletions(-) diff --git a/src/hotspot/share/code/scopeDesc.cpp b/src/hotspot/share/code/scopeDesc.cpp index 4807f61d74c..c7d0b7d37e7 100644 --- a/src/hotspot/share/code/scopeDesc.cpp +++ b/src/hotspot/share/code/scopeDesc.cpp @@ -51,9 +51,9 @@ ScopeDesc::ScopeDesc(const CompiledMethod* code, int decode_offset, bool reexecu } -ScopeDesc::ScopeDesc(const ScopeDesc* parent) { +void ScopeDesc::initialize(const ScopeDesc* parent, int decode_offset) { _code = parent->_code; - _decode_offset = parent->_sender_decode_offset; + _decode_offset = decode_offset; _objects = parent->_objects; _reexecute = false; //reexecute only applies to the first scope _rethrow_exception = false; @@ -61,6 +61,14 @@ ScopeDesc::ScopeDesc(const ScopeDesc* parent) { decode_body(); } +ScopeDesc::ScopeDesc(const ScopeDesc* parent) { + initialize(parent, parent->_sender_decode_offset); +} + +ScopeDesc::ScopeDesc(const ScopeDesc* parent, int decode_offset) { + initialize(parent, decode_offset); +} + void ScopeDesc::decode_body() { if (decode_offset() == DebugInformationRecorder::serialized_null) { diff --git a/src/hotspot/share/code/scopeDesc.hpp b/src/hotspot/share/code/scopeDesc.hpp index d1eaaac7426..f7ed903f771 100644 --- a/src/hotspot/share/code/scopeDesc.hpp +++ b/src/hotspot/share/code/scopeDesc.hpp @@ -67,6 +67,9 @@ class ScopeDesc : public ResourceObj { // avoid a .hpp-.hpp dependency.) ScopeDesc(const CompiledMethod* code, int decode_offset, bool reexecute, bool rethrow_exception, bool return_oop); + // Direct access to scope + ScopeDesc* at_offset(int decode_offset) { return new ScopeDesc(this, decode_offset); } + // JVM state Method* method() const { return _method; } int bci() const { return _bci; } @@ -85,12 +88,16 @@ class ScopeDesc : public ResourceObj { // Returns where the scope was decoded int decode_offset() const { return _decode_offset; } + int sender_decode_offset() const { return _sender_decode_offset; } + // Tells whether sender() returns NULL bool is_top() const; private: - // Alternative constructor + void initialize(const ScopeDesc* parent, int decode_offset); + // Alternative constructors ScopeDesc(const ScopeDesc* parent); + ScopeDesc(const ScopeDesc* parent, int decode_offset); // JVM state Method* _method; diff --git a/src/hotspot/share/runtime/vframe.cpp b/src/hotspot/share/runtime/vframe.cpp index 452f48bc5cf..76de1313629 100644 --- a/src/hotspot/share/runtime/vframe.cpp +++ b/src/hotspot/share/runtime/vframe.cpp @@ -452,8 +452,10 @@ vframeStream::vframeStream(JavaThread* thread, frame top_frame, _stop_at_java_call_stub = stop_at_java_call_stub; // skip top frame, as it may not be at safepoint + _prev_frame = top_frame; _frame = top_frame.sender(&_reg_map); while (!fill_from_frame()) { + _prev_frame = _frame; _frame = _frame.sender(&_reg_map); } } @@ -534,6 +536,37 @@ void vframeStreamCommon::skip_reflection_related_frames() { } } +javaVFrame* vframeStreamCommon::asJavaVFrame() { + javaVFrame* result = NULL; + if (_mode == compiled_mode) { + guarantee(_frame.is_compiled_frame(), "expected compiled Java frame"); + + // lazy update to register map + bool update_map = true; + RegisterMap map(_thread, update_map); + frame f = _prev_frame.sender(&map); + + guarantee(f.is_compiled_frame(), "expected compiled Java frame"); + + compiledVFrame* cvf = compiledVFrame::cast(vframe::new_vframe(&f, &map, _thread)); + + guarantee(cvf->cb() == cb(), "wrong code blob"); + + // get the same scope as this stream + cvf = cvf->at_scope(_decode_offset, _vframe_id); + + guarantee(cvf->scope()->decode_offset() == _decode_offset, "wrong scope"); + guarantee(cvf->scope()->sender_decode_offset() == _sender_decode_offset, "wrong scope"); + guarantee(cvf->vframe_id() == _vframe_id, "wrong vframe"); + + result = cvf; + } else { + result = javaVFrame::cast(vframe::new_vframe(&_frame, &_reg_map, _thread)); + } + guarantee(result->method() == method(), "wrong method"); + return result; +} + #ifndef PRODUCT void vframe::print() { diff --git a/src/hotspot/share/runtime/vframe.hpp b/src/hotspot/share/runtime/vframe.hpp index 2ee93d7b948..3c723f61557 100644 --- a/src/hotspot/share/runtime/vframe.hpp +++ b/src/hotspot/share/runtime/vframe.hpp @@ -278,12 +278,16 @@ class MonitorInfo : public ResourceObj { class vframeStreamCommon : StackObj { protected: // common + frame _prev_frame; frame _frame; JavaThread* _thread; RegisterMap _reg_map; enum { interpreted_mode, compiled_mode, at_end_mode } _mode; + // For compiled_mode + int _decode_offset; int _sender_decode_offset; + int _vframe_id; // Cached information Method* _method; @@ -320,6 +324,8 @@ class vframeStreamCommon : StackObj { return (CompiledMethod*) cb(); } + javaVFrame* asJavaVFrame(); + // Frame type inline bool is_interpreted_frame() const; inline bool is_entry_frame() const; diff --git a/src/hotspot/share/runtime/vframe.inline.hpp b/src/hotspot/share/runtime/vframe.inline.hpp index e6402e3db6d..83ab8f06d2a 100644 --- a/src/hotspot/share/runtime/vframe.inline.hpp +++ b/src/hotspot/share/runtime/vframe.inline.hpp @@ -44,6 +44,7 @@ inline void vframeStreamCommon::next() { // handle general case do { + _prev_frame = _frame; _frame = _frame.sender(&_reg_map); } while (!fill_from_frame()); } @@ -59,6 +60,7 @@ inline vframeStream::vframeStream(JavaThread* thread, bool stop_at_java_call_stu _frame = _thread->last_frame(); while (!fill_from_frame()) { + _prev_frame = _frame; _frame = _frame.sender(&_reg_map); } } @@ -68,12 +70,14 @@ inline bool vframeStreamCommon::fill_in_compiled_inlined_sender() { return false; } fill_from_compiled_frame(_sender_decode_offset); + ++_vframe_id; return true; } inline void vframeStreamCommon::fill_from_compiled_frame(int decode_offset) { _mode = compiled_mode; + _decode_offset = decode_offset; // Range check to detect ridiculous offsets. if (decode_offset == DebugInformationRecorder::serialized_null || @@ -118,6 +122,8 @@ inline void vframeStreamCommon::fill_from_compiled_frame(int decode_offset) { inline void vframeStreamCommon::fill_from_compiled_native_frame() { _mode = compiled_mode; _sender_decode_offset = DebugInformationRecorder::serialized_null; + _decode_offset = DebugInformationRecorder::serialized_null; + _vframe_id = 0; _method = nm()->method(); _bci = 0; } @@ -187,6 +193,7 @@ inline bool vframeStreamCommon::fill_from_frame() { decode_offset = pc_desc->scope_decode_offset(); } fill_from_compiled_frame(decode_offset); + _vframe_id = 0; } return true; } diff --git a/src/hotspot/share/runtime/vframe_hp.cpp b/src/hotspot/share/runtime/vframe_hp.cpp index 3f4bfd44217..fcc3e7bae31 100644 --- a/src/hotspot/share/runtime/vframe_hp.cpp +++ b/src/hotspot/share/runtime/vframe_hp.cpp @@ -252,6 +252,14 @@ compiledVFrame::compiledVFrame(const frame* fr, const RegisterMap* reg_map, Java guarantee(_scope != NULL, "scope must be present"); } +compiledVFrame* compiledVFrame::at_scope(int decode_offset, int vframe_id) { + if (scope()->decode_offset() != decode_offset) { + ScopeDesc* scope = this->scope()->at_offset(decode_offset); + return new compiledVFrame(frame_pointer(), register_map(), thread(), scope, vframe_id); + } + assert(_vframe_id == vframe_id, "wrong frame id"); + return this; +} bool compiledVFrame::is_top() const { // FIX IT: Remove this when new native stubs are in place diff --git a/src/hotspot/share/runtime/vframe_hp.hpp b/src/hotspot/share/runtime/vframe_hp.hpp index ab7db10549e..522c097514b 100644 --- a/src/hotspot/share/runtime/vframe_hp.hpp +++ b/src/hotspot/share/runtime/vframe_hp.hpp @@ -72,6 +72,9 @@ class compiledVFrame: public javaVFrame { // Returns the scopeDesc ScopeDesc* scope() const { return _scope; } + // Return the compiledVFrame for the desired scope + compiledVFrame* at_scope(int decode_offset, int vframe_id); + // Returns SynchronizationEntryBCI or bci() (used for synchronization) int raw_bci() const; From d1597bb1cbd8a213bb73bf1efb3db0584b40e7dc Mon Sep 17 00:00:00 2001 From: Dean Long Date: Tue, 18 Dec 2018 12:45:07 -0800 Subject: [PATCH 021/101] 8214329: SwingMark SubMenus 9% regression in 12-b19 on Linux client Reviewed-by: thartmann, mullan --- src/hotspot/share/prims/jvm.cpp | 11 ++++------- 1 file changed, 4 insertions(+), 7 deletions(-) diff --git a/src/hotspot/share/prims/jvm.cpp b/src/hotspot/share/prims/jvm.cpp index 429f2d7d469..26772e6294a 100644 --- a/src/hotspot/share/prims/jvm.cpp +++ b/src/hotspot/share/prims/jvm.cpp @@ -1230,11 +1230,10 @@ JVM_ENTRY(jobject, JVM_GetStackAccessControlContext(JNIEnv *env, jclass cls)) oop protection_domain = NULL; // Iterate through Java frames - RegisterMap reg_map(thread); - javaVFrame *vf = thread->last_java_vframe(®_map); - for (; vf != NULL; vf = vf->java_sender()) { + vframeStream vfst(thread); + for(; !vfst.at_end(); vfst.next()) { // get method of frame - Method* method = vf->method(); + Method* method = vfst.method(); // stop at the first privileged frame if (method->method_holder() == SystemDictionary::AccessController_klass() && @@ -1243,9 +1242,7 @@ JVM_ENTRY(jobject, JVM_GetStackAccessControlContext(JNIEnv *env, jclass cls)) // this frame is privileged is_privileged = true; - javaVFrame *priv = vf; // executePrivileged - javaVFrame *caller_fr = priv->java_sender(); // doPrivileged - caller_fr = caller_fr->java_sender(); // caller + javaVFrame *priv = vfst.asJavaVFrame(); // executePrivileged StackValueCollection* locals = priv->locals(); privileged_context = locals->obj_at(1); From ad47b4c4cc15a12dc467c7910602218399b2b387 Mon Sep 17 00:00:00 2001 From: Xue-Lei Andrew Fan Date: Tue, 18 Dec 2018 15:18:44 -0800 Subject: [PATCH 022/101] 8209333: Socket reset issue for TLS 1.3 socket close Reviewed-by: jnimeh --- .../sun/security/ssl/SSLConfiguration.java | 2 - .../sun/security/ssl/SSLSocketImpl.java | 31 +++- .../security/ssl/SSLSocketInputRecord.java | 13 ++ .../SSLSocketBruceForceClose.java | 147 ++++++++++++++++ .../ssl/SSLSocketImpl/SSLSocketClose.java | 159 ++++++++++++++++++ 5 files changed, 349 insertions(+), 3 deletions(-) create mode 100644 test/jdk/sun/security/ssl/SSLSocketImpl/SSLSocketBruceForceClose.java create mode 100644 test/jdk/sun/security/ssl/SSLSocketImpl/SSLSocketClose.java diff --git a/src/java.base/share/classes/sun/security/ssl/SSLConfiguration.java b/src/java.base/share/classes/sun/security/ssl/SSLConfiguration.java index be717717331..bd3dd3cb1b5 100644 --- a/src/java.base/share/classes/sun/security/ssl/SSLConfiguration.java +++ b/src/java.base/share/classes/sun/security/ssl/SSLConfiguration.java @@ -29,8 +29,6 @@ import java.security.AccessControlContext; import java.security.AccessController; import java.security.AlgorithmConstraints; import java.security.NoSuchAlgorithmException; -import java.security.PrivilegedActionException; -import java.security.PrivilegedExceptionAction; import java.util.ArrayList; import java.util.Arrays; import java.util.Collection; diff --git a/src/java.base/share/classes/sun/security/ssl/SSLSocketImpl.java b/src/java.base/share/classes/sun/security/ssl/SSLSocketImpl.java index 1bed7ee184e..b60a84d4636 100644 --- a/src/java.base/share/classes/sun/security/ssl/SSLSocketImpl.java +++ b/src/java.base/share/classes/sun/security/ssl/SSLSocketImpl.java @@ -608,7 +608,12 @@ public final class SSLSocketImpl } } else { if (!conContext.isInboundClosed()) { - conContext.inputRecord.close(); + try (conContext.inputRecord) { + // Try the best to use up the input records and close the + // socket gracefully, without impact the performance too + // much. + appInput.deplete(); + } } if ((autoClose || !isLayered()) && !super.isInputShutdown()) { @@ -907,6 +912,30 @@ public final class SSLSocketImpl return false; } + + /** + * Try the best to use up the input records so as to close the + * socket gracefully, without impact the performance too much. + */ + private synchronized void deplete() { + if (!conContext.isInboundClosed()) { + if (!(conContext.inputRecord instanceof SSLSocketInputRecord)) { + return; + } + + SSLSocketInputRecord socketInputRecord = + (SSLSocketInputRecord)conContext.inputRecord; + try { + socketInputRecord.deplete( + conContext.isNegotiated && (getSoTimeout() > 0)); + } catch (IOException ioe) { + if (SSLLogger.isOn && SSLLogger.isOn("ssl")) { + SSLLogger.warning( + "input stream close depletion failed", ioe); + } + } + } + } } @Override diff --git a/src/java.base/share/classes/sun/security/ssl/SSLSocketInputRecord.java b/src/java.base/share/classes/sun/security/ssl/SSLSocketInputRecord.java index 7a4bbac91dc..57524dc4d64 100644 --- a/src/java.base/share/classes/sun/security/ssl/SSLSocketInputRecord.java +++ b/src/java.base/share/classes/sun/security/ssl/SSLSocketInputRecord.java @@ -463,4 +463,17 @@ final class SSLSocketInputRecord extends InputRecord implements SSLRecord { return n; } + + // Try to use up the input stream without impact the performance too much. + void deplete(boolean tryToRead) throws IOException { + int remaining = is.available(); + if (tryToRead && (remaining == 0)) { + // try to wait and read one byte if no buffered input + is.read(); + } + + while ((remaining = is.available()) != 0) { + is.skip(remaining); + } + } } diff --git a/test/jdk/sun/security/ssl/SSLSocketImpl/SSLSocketBruceForceClose.java b/test/jdk/sun/security/ssl/SSLSocketImpl/SSLSocketBruceForceClose.java new file mode 100644 index 00000000000..b220a3d434b --- /dev/null +++ b/test/jdk/sun/security/ssl/SSLSocketImpl/SSLSocketBruceForceClose.java @@ -0,0 +1,147 @@ +/* + * Copyright (c) 2018, 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. + */ + +// +// Please run in othervm mode. SunJSSE does not support dynamic system +// properties, no way to re-use system properties in samevm/agentvm mode. +// + +/* + * @test + * @bug 8209333 + * @summary Socket reset issue for TLS 1.3 socket close + * @library /javax/net/ssl/templates + * @run main/othervm SSLSocketBruceForceClose + */ + +import javax.net.ssl.*; +import java.io.*; +import java.net.InetAddress; + +public class SSLSocketBruceForceClose implements SSLContextTemplate { + + public static void main(String[] args) throws Exception { + for (int i = 0; i<= 10; i++) { + System.err.println("==================================="); + System.err.println("loop " + i); + System.err.println("==================================="); + new SSLSocketBruceForceClose().test(); + } + } + + private void test() throws Exception { + SSLServerSocket listenSocket = null; + SSLSocket serverSocket = null; + ClientSocket clientSocket = null; + try { + SSLServerSocketFactory serversocketfactory = + createServerSSLContext().getServerSocketFactory(); + listenSocket = + (SSLServerSocket)serversocketfactory.createServerSocket(0); + listenSocket.setNeedClientAuth(false); + listenSocket.setEnableSessionCreation(true); + listenSocket.setUseClientMode(false); + + + System.err.println("Starting client"); + clientSocket = new ClientSocket(listenSocket.getLocalPort()); + clientSocket.start(); + + System.err.println("Accepting client requests"); + serverSocket = (SSLSocket) listenSocket.accept(); + + System.err.println("Reading data from client"); + BufferedReader serverReader = new BufferedReader( + new InputStreamReader(serverSocket.getInputStream())); + String data = serverReader.readLine(); + System.err.println("Received data from client: " + data); + + System.err.println("Reading more data from client"); + data = serverReader.readLine(); + System.err.println("Received data from client: " + data); + } finally { + if (listenSocket != null) { + listenSocket.close(); + } + + if (serverSocket != null) { + serverSocket.close(); + } + } + + if (clientSocket != null && clientSocket.clientException != null) { + throw clientSocket.clientException; + } + } + + private class ClientSocket extends Thread{ + int serverPort = 0; + Exception clientException; + + public ClientSocket(int serverPort) { + this.serverPort = serverPort; + } + + @Override + public void run() { + SSLSocket clientSocket = null; + String clientData = "Hi, I am client"; + try { + System.err.println( + "Connecting to server at port " + serverPort); + SSLSocketFactory sslSocketFactory = + createClientSSLContext().getSocketFactory(); + clientSocket = (SSLSocket)sslSocketFactory.createSocket( + InetAddress.getLocalHost(), serverPort); + clientSocket.setSoLinger(true, 3); + clientSocket.setSoTimeout(1000); + + + System.err.println("Sending data to server ..."); + + BufferedWriter os = new BufferedWriter( + new OutputStreamWriter(clientSocket.getOutputStream())); + os.write(clientData, 0, clientData.length()); + os.newLine(); + os.flush(); + + System.err.println("Sending more data to server ..."); + os.write(clientData, 0, clientData.length()); + os.newLine(); + os.flush(); + } catch (Exception e) { + clientException = e; + } finally { + if (clientSocket != null) { + try{ + clientSocket.close(); + System.err.println("client socket closed"); + } catch (IOException ioe) { + clientException = ioe; + } + } + } + } + } +} + diff --git a/test/jdk/sun/security/ssl/SSLSocketImpl/SSLSocketClose.java b/test/jdk/sun/security/ssl/SSLSocketImpl/SSLSocketClose.java new file mode 100644 index 00000000000..47a06093e3f --- /dev/null +++ b/test/jdk/sun/security/ssl/SSLSocketImpl/SSLSocketClose.java @@ -0,0 +1,159 @@ +/* + * Copyright (c) 2018, 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. + */ + +// +// Please run in othervm mode. SunJSSE does not support dynamic system +// properties, no way to re-use system properties in samevm/agentvm mode. +// + +/* + * @test + * @bug 8209333 + * @summary Socket reset issue for TLS 1.3 socket close + * @library /javax/net/ssl/templates + * @run main/othervm SSLSocketClose + */ + +import javax.net.ssl.*; +import java.io.*; +import java.net.InetAddress; + +public class SSLSocketClose implements SSLContextTemplate { + + public static void main(String[] args) throws Exception { + for (int i = 0; i<= 10; i++) { + System.err.println("==================================="); + System.err.println("loop " + i); + System.err.println("==================================="); + new SSLSocketClose().test(); + } + } + + private void test() throws Exception { + SSLServerSocket listenSocket = null; + SSLSocket serverSocket = null; + ClientSocket clientSocket = null; + try { + SSLServerSocketFactory serversocketfactory = + createServerSSLContext().getServerSocketFactory(); + listenSocket = + (SSLServerSocket)serversocketfactory.createServerSocket(0); + listenSocket.setNeedClientAuth(false); + listenSocket.setEnableSessionCreation(true); + listenSocket.setUseClientMode(false); + + + System.err.println("Starting client"); + clientSocket = new ClientSocket(listenSocket.getLocalPort()); + clientSocket.start(); + + System.err.println("Accepting client requests"); + serverSocket = (SSLSocket) listenSocket.accept(); + + System.err.println("Reading data from client"); + BufferedReader serverReader = new BufferedReader( + new InputStreamReader(serverSocket.getInputStream())); + String data = serverReader.readLine(); + System.err.println("Received data from client: " + data); + + System.err.println("Sending data to client ..."); + String serverData = "Hi, I am server"; + BufferedWriter os = new BufferedWriter( + new OutputStreamWriter(serverSocket.getOutputStream())); + os.write(serverData, 0, serverData.length()); + os.newLine(); + os.flush(); + + System.err.println("Reading more data from client"); + data = serverReader.readLine(); + System.err.println("Received data from client: " + data); + } finally { + if (listenSocket != null) { + listenSocket.close(); + } + + if (serverSocket != null) { + serverSocket.close(); + } + } + + if (clientSocket != null && clientSocket.clientException != null) { + throw clientSocket.clientException; + } + } + + private class ClientSocket extends Thread{ + int serverPort = 0; + Exception clientException; + + public ClientSocket(int serverPort) { + this.serverPort = serverPort; + } + + @Override + public void run() { + SSLSocket clientSocket = null; + String clientData = "Hi, I am client"; + try { + System.err.println( + "Connecting to server at port " + serverPort); + SSLSocketFactory sslSocketFactory = + createClientSSLContext().getSocketFactory(); + clientSocket = (SSLSocket)sslSocketFactory.createSocket( + InetAddress.getLocalHost(), serverPort); + clientSocket.setSoLinger(true, 3); + + System.err.println("Sending data to server ..."); + + BufferedWriter os = new BufferedWriter( + new OutputStreamWriter(clientSocket.getOutputStream())); + os.write(clientData, 0, clientData.length()); + os.newLine(); + os.flush(); + + System.err.println("Reading data from server"); + BufferedReader is = new BufferedReader( + new InputStreamReader(clientSocket.getInputStream())); + String data = is.readLine(); + System.err.println("Received Data from server: " + data); + + System.err.println("Sending more data to server ..."); + os.write(clientData, 0, clientData.length()); + os.newLine(); + os.flush(); + } catch (Exception e) { + clientException = e; + } finally { + if (clientSocket != null) { + try{ + clientSocket.close(); + System.err.println("client socket closed"); + } catch (IOException ioe) { + clientException = ioe; + } + } + } + } + } +} + From 57dc039131937dded0e25fcdb61efce6127d5743 Mon Sep 17 00:00:00 2001 From: Dean Long Date: Tue, 18 Dec 2018 16:36:26 -0800 Subject: [PATCH 023/101] 8214583: AccessController.getContext may return wrong value after JDK-8212605 Reviewed-by: mchung, redestad --- src/hotspot/share/include/jvm.h | 9 +++ src/hotspot/share/prims/jvm.cpp | 9 ++- .../java/security/AccessController.java | 19 +++++- .../share/native/libjava/AccessController.c | 9 +++ .../security/AccessController/DoPriv.java | 68 +++++++++++++++++++ .../jdk/javax/security/auth/Subject/DoAs.java | 57 ++++++++++++++++ 6 files changed, 167 insertions(+), 4 deletions(-) create mode 100644 test/jdk/java/security/AccessController/DoPriv.java create mode 100644 test/jdk/javax/security/auth/Subject/DoAs.java diff --git a/src/hotspot/share/include/jvm.h b/src/hotspot/share/include/jvm.h index 5fa112eb3d9..d0ba86e0a33 100644 --- a/src/hotspot/share/include/jvm.h +++ b/src/hotspot/share/include/jvm.h @@ -625,6 +625,15 @@ JVM_GetMethodParameters(JNIEnv *env, jobject method); JNIEXPORT jobject JNICALL JVM_GetInheritedAccessControlContext(JNIEnv *env, jclass cls); +/* + * Ensure that code doing a stackwalk and using javaVFrame::locals() to + * get the value will see a materialized value and not a scalar-replaced + * null value. + */ +#define JVM_EnsureMaterializedForStackWalk(env, value) \ + do {} while(0) // Nothing to do. The fact that the value escaped + // through a native method is enough. + JNIEXPORT jobject JNICALL JVM_GetStackAccessControlContext(JNIEnv *env, jclass cls); diff --git a/src/hotspot/share/prims/jvm.cpp b/src/hotspot/share/prims/jvm.cpp index 26772e6294a..e1e0fd394e1 100644 --- a/src/hotspot/share/prims/jvm.cpp +++ b/src/hotspot/share/prims/jvm.cpp @@ -61,6 +61,7 @@ #include "runtime/handles.inline.hpp" #include "runtime/init.hpp" #include "runtime/interfaceSupport.inline.hpp" +#include "runtime/deoptimization.hpp" #include "runtime/java.hpp" #include "runtime/javaCalls.hpp" #include "runtime/jfieldIDWorkaround.hpp" @@ -1245,8 +1246,12 @@ JVM_ENTRY(jobject, JVM_GetStackAccessControlContext(JNIEnv *env, jclass cls)) javaVFrame *priv = vfst.asJavaVFrame(); // executePrivileged StackValueCollection* locals = priv->locals(); - privileged_context = locals->obj_at(1); - Handle caller = locals->obj_at(2); + StackValue* ctx_sv = locals->at(1); // AccessControlContext context + StackValue* clr_sv = locals->at(2); // Class caller + assert(!ctx_sv->obj_is_scalar_replaced(), "found scalar-replaced object"); + assert(!clr_sv->obj_is_scalar_replaced(), "found scalar-replaced object"); + privileged_context = ctx_sv->get_obj(); + Handle caller = clr_sv->get_obj(); Klass *caller_klass = java_lang_Class::as_Klass(caller()); protection_domain = caller_klass->protection_domain(); diff --git a/src/java.base/share/classes/java/security/AccessController.java b/src/java.base/share/classes/java/security/AccessController.java index 5cb1fef74e2..f8548639765 100644 --- a/src/java.base/share/classes/java/security/AccessController.java +++ b/src/java.base/share/classes/java/security/AccessController.java @@ -709,6 +709,13 @@ public final class AccessController { return context; } + /** + * The value needs to be physically located in the frame, so that it + * can be found by a stack walk. + */ + @Hidden + private static native void ensureMaterializedForStackWalk(Object o); + /** * Sanity check that the caller context is indeed privileged. * @@ -734,6 +741,11 @@ public final class AccessController { AccessControlContext context, Class caller) { + // Ensure context has a physical value in the frame + if (context != null) { + ensureMaterializedForStackWalk(context); + } + assert isPrivileged(); // sanity check invariant T result = action.run(); assert isPrivileged(); // sanity check invariant @@ -742,7 +754,6 @@ public final class AccessController { // retrieved by getStackAccessControlContext(). Reference.reachabilityFence(context); Reference.reachabilityFence(caller); - Reference.reachabilityFence(action); return result; } @@ -761,6 +772,11 @@ public final class AccessController { Class caller) throws Exception { + // Ensure context has a physical value in the frame + if (context != null) { + ensureMaterializedForStackWalk(context); + } + assert isPrivileged(); // sanity check invariant T result = action.run(); assert isPrivileged(); // sanity check invariant @@ -769,7 +785,6 @@ public final class AccessController { // retrieved by getStackAccessControlContext(). Reference.reachabilityFence(context); Reference.reachabilityFence(caller); - Reference.reachabilityFence(action); return result; } diff --git a/src/java.base/share/native/libjava/AccessController.c b/src/java.base/share/native/libjava/AccessController.c index 9d6a812505d..3fb85be9ccb 100644 --- a/src/java.base/share/native/libjava/AccessController.c +++ b/src/java.base/share/native/libjava/AccessController.c @@ -59,3 +59,12 @@ Java_java_security_AccessController_getInheritedAccessControlContext( { return JVM_GetInheritedAccessControlContext(env, this); } + +JNIEXPORT void JNICALL +Java_java_security_AccessController_ensureMaterializedForStackWalk( + JNIEnv *env, + jclass cls, + jobject value) +{ + JVM_EnsureMaterializedForStackWalk(env, value); +} diff --git a/test/jdk/java/security/AccessController/DoPriv.java b/test/jdk/java/security/AccessController/DoPriv.java new file mode 100644 index 00000000000..e99b8f62468 --- /dev/null +++ b/test/jdk/java/security/AccessController/DoPriv.java @@ -0,0 +1,68 @@ +/* + * Copyright (c) 2018, 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 8214583 + * @summary Check that getContext works after JIT compiler escape analysis. + */ +import java.security.AccessControlContext; +import java.security.AccessController; +import java.security.DomainCombiner; +import java.security.ProtectionDomain; +import java.security.PrivilegedActionException; +import java.security.PrivilegedExceptionAction; +import java.net.URL; +import java.util.Collections; +import java.util.HashSet; +import java.util.Set; + +public class DoPriv { + + static void go(final DomainCombiner dc0, final AccessControlContext co, final int index) throws Exception { + final AccessControlContext ci = new AccessControlContext(co, dc0); + AccessController.doPrivileged((PrivilegedExceptionAction)() -> { + AccessControlContext c1 = AccessController.getContext(); + DomainCombiner dc = c1.getDomainCombiner(); + if (dc != dc0 || dc == null) { + throw new AssertionError("iteration " + index + " " + dc + " != " + dc0); + } + return 0; + }, ci); + } + + public static void main(String[] args) throws Exception { + final DomainCombiner dc0 = new DomainCombiner() { + public ProtectionDomain[] combine(ProtectionDomain[] currentDomains, + ProtectionDomain[] assignedDomains) { + return null; + } + }; + + final AccessControlContext co = AccessController.getContext(); + + for (int i = 0; i < 500_000; ++i) { + go(dc0, co, i); + } + } +} diff --git a/test/jdk/javax/security/auth/Subject/DoAs.java b/test/jdk/javax/security/auth/Subject/DoAs.java new file mode 100644 index 00000000000..7f378be80a0 --- /dev/null +++ b/test/jdk/javax/security/auth/Subject/DoAs.java @@ -0,0 +1,57 @@ +/* + * Copyright (c) 2018, 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 8214583 + * @summary Check that getSubject works after JIT compiler escape analysis. + */ +import java.security.AccessControlContext; +import java.security.AccessController; +import java.security.PrivilegedAction; +import java.security.PrivilegedExceptionAction; +import java.util.Arrays; +import java.util.Collections; +import java.util.HashSet; +import java.util.Set; +import javax.security.auth.Subject; + +public class DoAs { + + public static void main(String[] args) throws Exception { + final Set outer = new HashSet<>(Arrays.asList("Outer")); + final Subject subject = new Subject(true, Collections.EMPTY_SET, outer, Collections.EMPTY_SET); + + for (int i = 0; i < 100_000; ++i) { + final int index = i; + Subject.doAs(subject, (PrivilegedExceptionAction)() -> { + AccessControlContext c1 = AccessController.getContext(); + Subject s = Subject.getSubject(c1); + if (s != subject) { + throw new AssertionError("outer Oops! " + "iteration " + index + " " + s + " != " + subject); + } + return 0; + }); + } + } +} From cfcd43da04c68f0d74f298a6841bc0a83bf14f08 Mon Sep 17 00:00:00 2001 From: Valerie Peng Date: Wed, 19 Dec 2018 02:27:44 +0000 Subject: [PATCH 024/101] 8214096: sun.security.util.SignatureUtil passes null parameter, so JCE validation fails Changed SignatureUtil.specialSetParameter to ignore null signature parameters Reviewed-by: mullan, weijun --- .../sun/security/util/SignatureUtil.java | 18 +---- .../security/util/misc/SetNullSigParams.java | 78 +++++++++++++++++++ 2 files changed, 82 insertions(+), 14 deletions(-) create mode 100644 test/jdk/sun/security/util/misc/SetNullSigParams.java diff --git a/src/java.base/share/classes/sun/security/util/SignatureUtil.java b/src/java.base/share/classes/sun/security/util/SignatureUtil.java index 2d85566e48d..9fd8d18598b 100644 --- a/src/java.base/share/classes/sun/security/util/SignatureUtil.java +++ b/src/java.base/share/classes/sun/security/util/SignatureUtil.java @@ -28,8 +28,6 @@ package sun.security.util; import java.io.IOException; import java.security.*; import java.security.spec.*; -import sun.security.util.ObjectIdentifier; -import sun.security.x509.AlgorithmId; import sun.security.rsa.RSAUtil; /** @@ -86,13 +84,12 @@ public class SignatureUtil { // specified Signature object as signature parameters. public static void specialSetParameter(Signature sig, byte[] paramBytes) throws InvalidAlgorithmParameterException, ProviderException { - - AlgorithmParameters params = null; if (paramBytes != null) { String sigName = sig.getAlgorithm(); - params = createAlgorithmParameters(sigName, paramBytes); + AlgorithmParameters params = + createAlgorithmParameters(sigName, paramBytes); + specialSetParameter(sig, params); } - specialSetParameter(sig, params); } // Special method for setting the specified AlgorithmParameter object @@ -100,16 +97,9 @@ public class SignatureUtil { public static void specialSetParameter(Signature sig, AlgorithmParameters params) throws InvalidAlgorithmParameterException, ProviderException { - - String sigName = sig.getAlgorithm(); if (params != null) { + String sigName = sig.getAlgorithm(); sig.setParameter(getParamSpec(sigName, params)); - } else { - try { - sig.setParameter(null); - } catch (UnsupportedOperationException e) { - // ignore for maintaining backward compatibility - } } } } diff --git a/test/jdk/sun/security/util/misc/SetNullSigParams.java b/test/jdk/sun/security/util/misc/SetNullSigParams.java new file mode 100644 index 00000000000..f8e9ce0b0ee --- /dev/null +++ b/test/jdk/sun/security/util/misc/SetNullSigParams.java @@ -0,0 +1,78 @@ +/* + * Copyright (c) 2018, 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 8214096 + * @summary Make sure SignatureUtil can accept null algorithm parameters + * @modules java.base/sun.security.util + */ +import java.security.*; +import java.security.spec.AlgorithmParameterSpec; +import sun.security.util.SignatureUtil; + +public class SetNullSigParams { + + public static void main(String[] args) throws Exception { + Signature sig = new SpecialSigImpl(); + SignatureUtil.specialSetParameter(sig, (byte[]) null); + SignatureUtil.specialSetParameter(sig, (AlgorithmParameters) null); + } + + // Sample Signature impl class which simulates 3rd party provider behavior + // and throws NPE when given null algorithm parameters + // For max backward-compatibility, sun.security.util.SignatureUtil class + // now calls setParameter() only when algorithm parameters is non-null + private static class SpecialSigImpl extends Signature { + SpecialSigImpl() { + super("ANY"); + } + @Override + protected void engineInitVerify(PublicKey publicKey) + throws InvalidKeyException {} + @Override + protected void engineInitSign(PrivateKey privateKey) + throws InvalidKeyException {} + @Override + protected void engineUpdate(byte b) throws SignatureException {} + @Override + protected void engineUpdate(byte[] b, int off, int len) + throws SignatureException {} + @Override + protected byte[] engineSign() throws SignatureException { return null; } + @Override + protected boolean engineVerify(byte[] sigBytes) + throws SignatureException { return false; } + @Override + protected void engineSetParameter(String param, Object value) + throws InvalidParameterException {} + @Override + protected void engineSetParameter(AlgorithmParameterSpec params) + throws InvalidAlgorithmParameterException { + if (params == null) throw new NullPointerException("Test Failed"); + } + @Override + protected Object engineGetParameter(String param) + throws InvalidParameterException { return null; } + } +} From fd6cbffc4bffe9b025f75054af80d8c9a0f6f02f Mon Sep 17 00:00:00 2001 From: Priya Lakshmi Muthuswamy Date: Wed, 19 Dec 2018 11:08:08 +0530 Subject: [PATCH 025/101] 8214570: Use {@systemProperty} for definitions of system properties Reviewed-by: alanb, mullan --- .../share/classes/javax/smartcardio/TerminalFactory.java | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/src/java.smartcardio/share/classes/javax/smartcardio/TerminalFactory.java b/src/java.smartcardio/share/classes/javax/smartcardio/TerminalFactory.java index e37fe33a74f..21b8459c947 100644 --- a/src/java.smartcardio/share/classes/javax/smartcardio/TerminalFactory.java +++ b/src/java.smartcardio/share/classes/javax/smartcardio/TerminalFactory.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2005, 2016, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2005, 2018, 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 @@ -182,7 +182,7 @@ public final class TerminalFactory { *

It is determined as follows: * * when this class is initialized, the system property - * javax.smartcardio.TerminalFactory.DefaultType + * {@systemProperty javax.smartcardio.TerminalFactory.DefaultType} * is examined. If it is set, a TerminalFactory of this type is * instantiated by calling the {@linkplain #getInstance * getInstance(String,Object)} method passing From b756562a67171906039b5a8b00bd0d3283131251 Mon Sep 17 00:00:00 2001 From: Igor Ignatyev Date: Tue, 18 Dec 2018 13:37:06 -0800 Subject: [PATCH 026/101] 8215322: add @file support to jaotc Reviewed-by: kvn --- .../src/jdk/tools/jaotc/Main.java | 32 +++++++++- .../compiler/aot/cli/jaotc/AtFileTest.java | 60 +++++++++++++++++++ 2 files changed, 91 insertions(+), 1 deletion(-) create mode 100644 test/hotspot/jtreg/compiler/aot/cli/jaotc/AtFileTest.java diff --git a/src/jdk.aot/share/classes/jdk.tools.jaotc/src/jdk/tools/jaotc/Main.java b/src/jdk.aot/share/classes/jdk.tools.jaotc/src/jdk/tools/jaotc/Main.java index ed384d5689f..31dc6bf441a 100644 --- a/src/jdk.aot/share/classes/jdk.tools.jaotc/src/jdk/tools/jaotc/Main.java +++ b/src/jdk.aot/share/classes/jdk.tools.jaotc/src/jdk/tools/jaotc/Main.java @@ -29,11 +29,18 @@ import static org.graalvm.compiler.core.common.GraalOptions.GeneratePIC; import static org.graalvm.compiler.core.common.GraalOptions.ImmutableCode; import static org.graalvm.compiler.hotspot.meta.HotSpotAOTProfilingPlugin.Options.TieredAOT; +import java.io.IOException; import java.io.PrintWriter; import java.text.MessageFormat; +import java.util.ArrayList; +import java.util.Collections; import java.util.List; import java.util.ListIterator; import java.util.Set; +import java.util.StringTokenizer; +import java.util.stream.Stream; +import java.nio.file.Files; +import java.nio.file.Paths; import org.graalvm.compiler.api.replacements.SnippetReflectionProvider; import org.graalvm.compiler.api.runtime.GraalJVMCICompiler; @@ -78,10 +85,33 @@ public final class Main { public static void main(String[] args) throws Exception { Main t = new Main(); - final int exitCode = t.run(args); + final int exitCode = t.run(parse(args)); System.exit(exitCode); } + /** + * Expands '@file' in command line arguments by replacing '@file' with the content of 'file' + * parsed by StringTokenizer. '@' character can be quoted as '@@'. + */ + private static String[] parse(String[] args) throws IOException { + List result = new ArrayList<>(); + for (String arg : args) { + if (arg.length() > 1 && arg.charAt(0) == '@') { + String v = arg.substring(1); + if (v.charAt(0) == '@') { + result.add(v); + } else { + try (Stream file = Files.lines(Paths.get(v))) { + file.map(StringTokenizer::new).map(Collections::list).flatMap(l -> l.stream().map(o -> (String) o)).forEachOrdered(result::add); + } + } + } else { + result.add(arg); + } + } + return result.toArray(String[]::new); + } + private int run(String[] args) { log = new PrintWriter(System.out); printer = new LogPrinter(this, log); diff --git a/test/hotspot/jtreg/compiler/aot/cli/jaotc/AtFileTest.java b/test/hotspot/jtreg/compiler/aot/cli/jaotc/AtFileTest.java new file mode 100644 index 00000000000..af97bc67c85 --- /dev/null +++ b/test/hotspot/jtreg/compiler/aot/cli/jaotc/AtFileTest.java @@ -0,0 +1,60 @@ +/* + * Copyright (c) 2018, 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 + * @summary check at-file jaotc support + * @comment based on CompileClassTest with arguments wrote in 'jaotc.cmd' file + * @requires vm.aot + * @bug 8215322 + * @library / /test/lib /testlibrary + * @modules java.base/jdk.internal.misc + * @build compiler.aot.cli.jaotc.AtFileTest + * @run driver ClassFileInstaller compiler.aot.cli.jaotc.data.HelloWorldOne + * @run driver compiler.aot.cli.jaotc.AtFileTest + */ + +package compiler.aot.cli.jaotc; + +import compiler.aot.cli.jaotc.data.HelloWorldOne; +import java.io.File; +import java.nio.file.Path; +import java.nio.file.Paths; +import java.nio.file.Files; +import java.util.List; +import jdk.test.lib.Asserts; +import jdk.test.lib.process.OutputAnalyzer; + +public class AtFileTest { + public static void main(String[] args) throws Exception { + Path file = Paths.get("jatoc.cmd"); + Files.write(file, List.of("--class-name", + JaotcTestHelper.getClassAotCompilationName(HelloWorldOne.class))); + OutputAnalyzer oa = JaotcTestHelper.compileLibrary("@" + file.toString()); + oa.shouldHaveExitValue(0); + File compiledLibrary = new File(JaotcTestHelper.DEFAULT_LIB_PATH); + Asserts.assertTrue(compiledLibrary.exists(), "Compiled library file missing"); + Asserts.assertGT(compiledLibrary.length(), 0L, "Unexpected compiled library size"); + JaotcTestHelper.checkLibraryUsage(HelloWorldOne.class.getName()); + } +} From c9d7dd082744a120a41daea7a03bd7df3b1e74d3 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Per=20Lid=C3=A9n?= Date: Wed, 19 Dec 2018 08:32:48 +0100 Subject: [PATCH 027/101] 8215487: ZGC: ZRuntimeWorkers incorrectly identify themselves as ZWorkers Reviewed-by: eosterlund --- src/hotspot/share/gc/z/zObjectAllocator.cpp | 6 ++- src/hotspot/share/gc/z/zRuntimeWorkers.cpp | 45 +++++++++++++++++++++ src/hotspot/share/gc/z/zThread.cpp | 14 ++++++- src/hotspot/share/gc/z/zThread.hpp | 11 +++++ src/hotspot/share/gc/z/zWorkers.cpp | 23 +++++++---- 5 files changed, 88 insertions(+), 11 deletions(-) diff --git a/src/hotspot/share/gc/z/zObjectAllocator.cpp b/src/hotspot/share/gc/z/zObjectAllocator.cpp index 15eca0f9f5c..ad0d3098d53 100644 --- a/src/hotspot/share/gc/z/zObjectAllocator.cpp +++ b/src/hotspot/share/gc/z/zObjectAllocator.cpp @@ -128,7 +128,8 @@ uintptr_t ZObjectAllocator::alloc_medium_object(size_t size, ZAllocationFlags fl } uintptr_t ZObjectAllocator::alloc_small_object_from_nonworker(size_t size, ZAllocationFlags flags) { - assert(ZThread::is_java() || ZThread::is_vm(), "Should be a Java or VM thread"); + assert(ZThread::is_java() || ZThread::is_vm() || ZThread::is_runtime_worker(), + "Should be a Java, VM or Runtime worker thread"); // Non-worker small page allocation can never use the reserve flags.set_no_reserve(); @@ -193,7 +194,8 @@ uintptr_t ZObjectAllocator::alloc_object(size_t size) { } uintptr_t ZObjectAllocator::alloc_object_for_relocation(size_t size) { - assert(ZThread::is_java() || ZThread::is_worker() || ZThread::is_vm(), "Unknown thread"); + assert(ZThread::is_java() || ZThread::is_vm() || ZThread::is_worker() || ZThread::is_runtime_worker(), + "Unknown thread"); ZAllocationFlags flags; flags.set_relocation(); diff --git a/src/hotspot/share/gc/z/zRuntimeWorkers.cpp b/src/hotspot/share/gc/z/zRuntimeWorkers.cpp index 00a3e98d9d4..558224daf86 100644 --- a/src/hotspot/share/gc/z/zRuntimeWorkers.cpp +++ b/src/hotspot/share/gc/z/zRuntimeWorkers.cpp @@ -22,7 +22,43 @@ */ #include "precompiled.hpp" +#include "gc/shared/workgroup.hpp" #include "gc/z/zRuntimeWorkers.hpp" +#include "gc/z/zThread.hpp" +#include "runtime/mutexLocker.hpp" + +class ZRuntimeWorkersInitializeTask : public AbstractGangTask { +private: + const uint _nworkers; + uint _started; + Monitor _monitor; + +public: + ZRuntimeWorkersInitializeTask(uint nworkers) : + AbstractGangTask("ZRuntimeWorkersInitializeTask"), + _nworkers(nworkers), + _started(0), + _monitor(Monitor::leaf, + "ZRuntimeWorkersInitialize", + false /* allow_vm_block */, + Monitor::_safepoint_check_never) {} + + virtual void work(uint worker_id) { + // Register as runtime worker + ZThread::set_runtime_worker(); + + // Wait for all threads to start + MonitorLockerEx ml(&_monitor, Monitor::_no_safepoint_check_flag); + if (++_started == _nworkers) { + // All threads started + ml.notify_all(); + } else { + while (_started != _nworkers) { + ml.wait(Monitor::_no_safepoint_check_flag); + } + } + } +}; ZRuntimeWorkers::ZRuntimeWorkers() : _workers("RuntimeWorker", @@ -35,6 +71,15 @@ ZRuntimeWorkers::ZRuntimeWorkers() : // Initialize worker threads _workers.initialize_workers(); _workers.update_active_workers(nworkers()); + if (_workers.active_workers() != nworkers()) { + vm_exit_during_initialization("Failed to create ZRuntimeWorkers"); + } + + // Execute task to register threads as runtime workers. This also + // helps reduce latency in early safepoints, which otherwise would + // have to take on any warmup costs. + ZRuntimeWorkersInitializeTask task(nworkers()); + _workers.run_task(&task); } uint ZRuntimeWorkers::nworkers() const { diff --git a/src/hotspot/share/gc/z/zThread.cpp b/src/hotspot/share/gc/z/zThread.cpp index 5bde1d113e3..a1598671056 100644 --- a/src/hotspot/share/gc/z/zThread.cpp +++ b/src/hotspot/share/gc/z/zThread.cpp @@ -31,6 +31,7 @@ __thread uintptr_t ZThread::_id; __thread bool ZThread::_is_vm; __thread bool ZThread::_is_java; __thread bool ZThread::_is_worker; +__thread bool ZThread::_is_runtime_worker; __thread uint ZThread::_worker_id; void ZThread::initialize() { @@ -40,7 +41,8 @@ void ZThread::initialize() { _id = (uintptr_t)thread; _is_vm = thread->is_VM_thread(); _is_java = thread->is_Java_thread(); - _is_worker = thread->is_Worker_thread(); + _is_worker = false; + _is_runtime_worker = false; _worker_id = (uint)-1; } @@ -56,6 +58,16 @@ const char* ZThread::name() { return "Unknown"; } +void ZThread::set_worker() { + ensure_initialized(); + _is_worker = true; +} + +void ZThread::set_runtime_worker() { + ensure_initialized(); + _is_runtime_worker = true; +} + bool ZThread::has_worker_id() { return _initialized && _is_worker && diff --git a/src/hotspot/share/gc/z/zThread.hpp b/src/hotspot/share/gc/z/zThread.hpp index 365a216caf5..c87ee666d92 100644 --- a/src/hotspot/share/gc/z/zThread.hpp +++ b/src/hotspot/share/gc/z/zThread.hpp @@ -29,6 +29,8 @@ class ZThread : public AllStatic { friend class ZTask; + friend class ZWorkersInitializeTask; + friend class ZRuntimeWorkersInitializeTask; private: static __thread bool _initialized; @@ -36,6 +38,7 @@ private: static __thread bool _is_vm; static __thread bool _is_java; static __thread bool _is_worker; + static __thread bool _is_runtime_worker; static __thread uint _worker_id; static void initialize(); @@ -46,6 +49,9 @@ private: } } + static void set_worker(); + static void set_runtime_worker(); + static bool has_worker_id(); static void set_worker_id(uint worker_id); static void clear_worker_id(); @@ -73,6 +79,11 @@ public: return _is_worker; } + static bool is_runtime_worker() { + ensure_initialized(); + return _is_runtime_worker; + } + static uint worker_id() { assert(has_worker_id(), "Worker id not initialized"); return _worker_id; diff --git a/src/hotspot/share/gc/z/zWorkers.cpp b/src/hotspot/share/gc/z/zWorkers.cpp index 6a0c2561dec..c60c043ad45 100644 --- a/src/hotspot/share/gc/z/zWorkers.cpp +++ b/src/hotspot/share/gc/z/zWorkers.cpp @@ -24,6 +24,7 @@ #include "precompiled.hpp" #include "gc/z/zGlobals.hpp" #include "gc/z/zTask.hpp" +#include "gc/z/zThread.hpp" #include "gc/z/zWorkers.inline.hpp" #include "runtime/os.hpp" #include "runtime/mutexLocker.hpp" @@ -64,20 +65,26 @@ uint ZWorkers::calculate_nconcurrent() { return calculate_nworkers(12.5); } -class ZWorkersWarmupTask : public ZTask { +class ZWorkersInitializeTask : public ZTask { private: const uint _nworkers; uint _started; Monitor _monitor; public: - ZWorkersWarmupTask(uint nworkers) : - ZTask("ZWorkersWarmupTask"), + ZWorkersInitializeTask(uint nworkers) : + ZTask("ZWorkersInitializeTask"), _nworkers(nworkers), _started(0), - _monitor(Monitor::leaf, "ZWorkersWarmup", false, Monitor::_safepoint_check_never) {} + _monitor(Monitor::leaf, + "ZWorkersInitialize", + false /* allow_vm_block */, + Monitor::_safepoint_check_never) {} virtual void work() { + // Register as worker + ZThread::set_worker(); + // Wait for all threads to start MonitorLockerEx ml(&_monitor, Monitor::_no_safepoint_check_flag); if (++_started == _nworkers) { @@ -107,10 +114,10 @@ ZWorkers::ZWorkers() : vm_exit_during_initialization("Failed to create ZWorkers"); } - // Warm up worker threads by having them execute a dummy task. - // This helps reduce latency in early GC pauses, which otherwise - // would have to take on any warmup costs. - ZWorkersWarmupTask task(nworkers()); + // Execute task to register threads as workers. This also helps + // reduce latency in early GC pauses, which otherwise would have + // to take on any warmup costs. + ZWorkersInitializeTask task(nworkers()); run(&task, nworkers()); } From 36d0987f4d52a37b77831768cc0ee31b138bdeeb Mon Sep 17 00:00:00 2001 From: Sean Coffey Date: Wed, 19 Dec 2018 18:21:38 +0000 Subject: [PATCH 028/101] 8214532: Update RFC 2459 references in javadoc to RFC 5280 Reviewed-by: mullan --- .../certpath/DistributionPointFetcher.java | 4 +- .../provider/certpath/ForwardBuilder.java | 4 +- .../share/classes/sun/security/x509/AVA.java | 2 +- .../sun/security/x509/AlgorithmId.java | 8 +-- .../x509/CertificatePoliciesExtension.java | 4 +- .../classes/sun/security/x509/DNSName.java | 2 +- .../sun/security/x509/IPAddressName.java | 55 ++++++++++--------- .../sun/security/x509/KeyIdentifier.java | 4 +- .../x509/NameConstraintsExtension.java | 4 +- .../classes/sun/security/x509/OIDName.java | 4 +- .../classes/sun/security/x509/RFC822Name.java | 25 +++++---- .../classes/sun/security/x509/URIName.java | 4 +- .../sun/security/x509/X400Address.java | 4 +- .../sun/security/x509/X509CertImpl.java | 12 +--- 14 files changed, 66 insertions(+), 70 deletions(-) diff --git a/src/java.base/share/classes/sun/security/provider/certpath/DistributionPointFetcher.java b/src/java.base/share/classes/sun/security/provider/certpath/DistributionPointFetcher.java index 0f16c646a9c..5715842beef 100644 --- a/src/java.base/share/classes/sun/security/provider/certpath/DistributionPointFetcher.java +++ b/src/java.base/share/classes/sun/security/provider/certpath/DistributionPointFetcher.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2002, 2017, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2002, 2018, 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 @@ -770,7 +770,7 @@ public class DistributionPointFetcher { * * In practice, conforming CAs MUST use the key identifier method, * and MUST include authority key identifier extension in all CRLs - * issued. [section 5.2.1, RFC 2459] + * issued. [section 5.2.1, RFC 5280] */ AuthorityKeyIdentifierExtension crlAKID = crl.getAuthKeyIdExtension(); issuerSelector.setSkiAndSerialNumber(crlAKID); diff --git a/src/java.base/share/classes/sun/security/provider/certpath/ForwardBuilder.java b/src/java.base/share/classes/sun/security/provider/certpath/ForwardBuilder.java index 56ecabfc4f3..6122fdaa970 100644 --- a/src/java.base/share/classes/sun/security/provider/certpath/ForwardBuilder.java +++ b/src/java.base/share/classes/sun/security/provider/certpath/ForwardBuilder.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2000, 2016, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2000, 2018, 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 @@ -668,7 +668,7 @@ class ForwardBuilder extends Builder { * Verifies a matching certificate. * * This method executes the validation steps in the PKIX path - * validation algorithm which were + * validation algorithm, RFC 5280, which were * not satisfied by the selection criteria used by getCertificates() * to find the certs and only the steps that can be executed in a * forward direction (target to trust anchor). Those steps that can diff --git a/src/java.base/share/classes/sun/security/x509/AVA.java b/src/java.base/share/classes/sun/security/x509/AVA.java index 107189bc47c..654729528f3 100644 --- a/src/java.base/share/classes/sun/security/x509/AVA.java +++ b/src/java.base/share/classes/sun/security/x509/AVA.java @@ -1045,7 +1045,7 @@ public class AVA implements DerEncoder { if (valStr == null) { - // rfc1779 specifies that attribute values associated + // RFC 1779 specifies that attribute values associated // with non-standard keyword attributes may be represented // using the hex format below. This will be used only // when the value is not a string type diff --git a/src/java.base/share/classes/sun/security/x509/AlgorithmId.java b/src/java.base/share/classes/sun/security/x509/AlgorithmId.java index 825e4ffb81e..5ecb716adf1 100644 --- a/src/java.base/share/classes/sun/security/x509/AlgorithmId.java +++ b/src/java.base/share/classes/sun/security/x509/AlgorithmId.java @@ -166,15 +166,15 @@ public class AlgorithmId implements Serializable, DerEncoder { // Several AlgorithmId should omit the whole parameter part when // it's NULL. They are --- - // rfc3370 2.1: Implementations SHOULD generate SHA-1 + // RFC 3370 2.1: Implementations SHOULD generate SHA-1 // AlgorithmIdentifiers with absent parameters. - // rfc3447 C1: When id-sha1, id-sha224, id-sha256, id-sha384 and + // RFC 3447 C1: When id-sha1, id-sha224, id-sha256, id-sha384 and // id-sha512 are used in an AlgorithmIdentifier the parameters // (which are optional) SHOULD be omitted. - // rfc3279 2.3.2: The id-dsa algorithm syntax includes optional + // RFC 3279 2.3.2: The id-dsa algorithm syntax includes optional // domain parameters... When omitted, the parameters component // MUST be omitted entirely - // rfc3370 3.1: When the id-dsa-with-sha1 algorithm identifier + // RFC 3370 3.1: When the id-dsa-with-sha1 algorithm identifier // is used, the AlgorithmIdentifier parameters field MUST be absent. /*if ( algid.equals((Object)SHA_oid) || diff --git a/src/java.base/share/classes/sun/security/x509/CertificatePoliciesExtension.java b/src/java.base/share/classes/sun/security/x509/CertificatePoliciesExtension.java index 925936658f7..c2f71b3abc8 100644 --- a/src/java.base/share/classes/sun/security/x509/CertificatePoliciesExtension.java +++ b/src/java.base/share/classes/sun/security/x509/CertificatePoliciesExtension.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2000, 2011, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2000, 2018, 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 @@ -45,7 +45,7 @@ import sun.security.util.DerOutputStream; * certificate. *

* Optional qualifiers are not supported in this implementation, as they are - * not recommended by RFC2459. + * not recommended by RFC 5280. * * The ASN.1 syntax for this is (IMPLICIT tagging is defined in the * module definition): diff --git a/src/java.base/share/classes/sun/security/x509/DNSName.java b/src/java.base/share/classes/sun/security/x509/DNSName.java index 5ddce75833b..b2c2da637ec 100644 --- a/src/java.base/share/classes/sun/security/x509/DNSName.java +++ b/src/java.base/share/classes/sun/security/x509/DNSName.java @@ -181,7 +181,7 @@ public class DNSName implements GeneralNameInterface { * For example, www.host.example.com would satisfy the constraint but * host1.example.com would not. *

- * draft-ietf-pkix-new-part1-00.txt: DNSName restrictions are expressed as foo.bar.com. + * RFC 5280: DNSName restrictions are expressed as foo.bar.com. * Any DNSName that * can be constructed by simply adding to the left hand side of the name * satisfies the name constraint. For example, www.foo.bar.com would diff --git a/src/java.base/share/classes/sun/security/x509/IPAddressName.java b/src/java.base/share/classes/sun/security/x509/IPAddressName.java index f9451744059..7c12cae9128 100644 --- a/src/java.base/share/classes/sun/security/x509/IPAddressName.java +++ b/src/java.base/share/classes/sun/security/x509/IPAddressName.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 1997, 2002, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 1997, 2018, 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 @@ -37,27 +37,27 @@ import sun.security.util.DerValue; /** * This class implements the IPAddressName as required by the GeneralNames * ASN.1 object. Both IPv4 and IPv6 addresses are supported using the - * formats specified in IETF PKIX RFC2459. + * formats specified in IETF PKIX RFC 5280. *

- * [RFC2459 4.2.1.7 Subject Alternative Name] - * When the subjectAltName extension contains a iPAddress, the address - * MUST be stored in the octet string in "network byte order," as - * specified in RFC 791. The least significant bit (LSB) of - * each octet is the LSB of the corresponding byte in the network - * address. For IP Version 4, as specified in RFC 791, the octet string - * MUST contain exactly four octets. For IP Version 6, as specified in - * RFC 1883, the octet string MUST contain exactly sixteen octets. + * [RFC 5280 4.2.1.6 Subject Alternative Name] + * When the subjectAltName extension contains an iPAddress, the address + * MUST be stored in the octet string in "network byte order", as + * specified in [RFC791]. The least significant bit (LSB) of each octet + * is the LSB of the corresponding byte in the network address. For IP + * version 4, as specified in [RFC791], the octet string MUST contain + * exactly four octets. For IP version 6, as specified in + * [RFC 2460], the octet string MUST contain exactly sixteen octets. *

- * [RFC2459 4.2.1.11 Name Constraints] - * The syntax of iPAddress MUST be as described in section 4.2.1.7 with - * the following additions specifically for Name Constraints. For IPv4 - * addresses, the ipAddress field of generalName MUST contain eight (8) - * octets, encoded in the style of RFC 1519 (CIDR) to represent an - * address range.[RFC 1519] For IPv6 addresses, the ipAddress field + * [RFC 5280 4.2.1.10 Name Constraints] + * The syntax of iPAddress MUST be as described in Section 4.2.1.6 with + * the following additions specifically for name constraints. For IPv4 + * addresses, the iPAddress field of GeneralName MUST contain eight (8) + * octets, encoded in the style of RFC 4632 (CIDR) to represent an + * address range [RFC 4632]. For IPv6 addresses, the iPAddress field * MUST contain 32 octets similarly encoded. For example, a name - * constraint for "class C" subnet 10.9.8.0 shall be represented as the - * octets 0A 09 08 00 FF FF FF 00, representing the CIDR notation - * 10.9.8.0/255.255.255.0. + * constraint for "class C" subnet 192.0.2.0 is represented as the + * octets C0 00 02 00 FF FF FF 00, representing the CIDR notation + * 192.0.2.0/24 (mask 255.255.255.0). *

* @see GeneralName * @see GeneralNameInterface @@ -376,15 +376,16 @@ public class IPAddressName implements GeneralNameInterface { * . These results are used in checking NameConstraints during * certification path verification. *

- * [RFC2459] The syntax of iPAddress MUST be as described in section - * 4.2.1.7 with the following additions specifically for Name Constraints. - * For IPv4 addresses, the ipAddress field of generalName MUST contain - * eight (8) octets, encoded in the style of RFC 1519 (CIDR) to represent an - * address range.[RFC 1519] For IPv6 addresses, the ipAddress field + * [RFC 5280 4.2.1.10 Name Constraints] + * The syntax of iPAddress MUST be as described in Section 4.2.1.6 with + * the following additions specifically for name constraints. For IPv4 + * addresses, the iPAddress field of GeneralName MUST contain eight (8) + * octets, encoded in the style of RFC 4632 (CIDR) to represent an + * address range [RFC 4632]. For IPv6 addresses, the iPAddress field * MUST contain 32 octets similarly encoded. For example, a name - * constraint for "class C" subnet 10.9.8.0 shall be represented as the - * octets 0A 09 08 00 FF FF FF 00, representing the CIDR notation - * 10.9.8.0/255.255.255.0. + * constraint for "class C" subnet 192.0.2.0 is represented as the + * octets C0 00 02 00 FF FF FF 00, representing the CIDR notation + * 192.0.2.0/24 (mask 255.255.255.0). * * @param inputName to be checked for being constrained * @return constraint type above diff --git a/src/java.base/share/classes/sun/security/x509/KeyIdentifier.java b/src/java.base/share/classes/sun/security/x509/KeyIdentifier.java index 97036e2ea88..d74bc051b96 100644 --- a/src/java.base/share/classes/sun/security/x509/KeyIdentifier.java +++ b/src/java.base/share/classes/sun/security/x509/KeyIdentifier.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 1997, 1999, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 1997, 2018, 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 @@ -63,7 +63,7 @@ public class KeyIdentifier { /** * Creates a KeyIdentifier from a public-key value. * - *

From RFC2459: Two common methods for generating key identifiers from + *

From RFC 5280: Two common methods for generating key identifiers from * the public key are: *

    *
  1. The keyIdentifier is composed of the 160-bit SHA-1 hash of the diff --git a/src/java.base/share/classes/sun/security/x509/NameConstraintsExtension.java b/src/java.base/share/classes/sun/security/x509/NameConstraintsExtension.java index 88f787e0704..e595fbd6811 100644 --- a/src/java.base/share/classes/sun/security/x509/NameConstraintsExtension.java +++ b/src/java.base/share/classes/sun/security/x509/NameConstraintsExtension.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 1997, 2017, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 1997, 2018, 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 @@ -326,7 +326,7 @@ implements CertAttrSet, Cloneable { * expanded by a merge, just remain constant or become more * limiting. *

    - * IETF RFC2459 specifies the processing of Name Constraints as + * IETF RFC 5280 specifies the processing of Name Constraints as * follows: *

    * (j) If permittedSubtrees is present in the certificate, set the diff --git a/src/java.base/share/classes/sun/security/x509/OIDName.java b/src/java.base/share/classes/sun/security/x509/OIDName.java index 6db9d11e2d3..eda3426a985 100644 --- a/src/java.base/share/classes/sun/security/x509/OIDName.java +++ b/src/java.base/share/classes/sun/security/x509/OIDName.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 1997, 2015, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 1997, 2018, 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 @@ -156,7 +156,7 @@ public class OIDName implements GeneralNameInterface { else if (this.equals((OIDName)inputName)) constraintType = NAME_MATCH; else - //widens and narrows not defined in RFC2459 for OIDName (aka registeredID) + //widens and narrows not defined in RFC 5280 for OIDName (aka registeredID) throw new UnsupportedOperationException("Narrowing and widening are not supported for OIDNames"); return constraintType; } diff --git a/src/java.base/share/classes/sun/security/x509/RFC822Name.java b/src/java.base/share/classes/sun/security/x509/RFC822Name.java index 916b1beeecb..312e7a7cf21 100644 --- a/src/java.base/share/classes/sun/security/x509/RFC822Name.java +++ b/src/java.base/share/classes/sun/security/x509/RFC822Name.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 1997, 2011, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 1997, 2018, 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 @@ -68,7 +68,7 @@ public class RFC822Name implements GeneralNameInterface /** * Parse an RFC822Name string to see if it is a valid - * addr-spec according to IETF RFC822 and RFC2459: + * addr-spec according to IETF RFC 822 and RFC 5280: * [local-part@]domain *

    * local-part@ could be empty for an RFC822Name NameConstraint, @@ -131,7 +131,7 @@ public class RFC822Name implements GeneralNameInterface * Compares this name with another, for equality. * * @return true iff the names are equivalent - * according to RFC2459. + * according to RFC 5280. */ public boolean equals(Object obj) { if (this == obj) @@ -142,7 +142,7 @@ public class RFC822Name implements GeneralNameInterface RFC822Name other = (RFC822Name)obj; - // RFC2459 mandates that these names are + // RFC 5280 mandates that these names are // not case-sensitive return name.equalsIgnoreCase(other.name); } @@ -166,14 +166,15 @@ public class RFC822Name implements GeneralNameInterface * . These results are used in checking NameConstraints during * certification path verification. *

    - * [RFC2459] When the subjectAltName extension contains an Internet mail address, - * the address MUST be included as an rfc822Name. The format of an - * rfc822Name is an "addr-spec" as defined in RFC 822 [RFC 822]. An - * addr-spec has the form "local-part@domain". Note that an addr-spec - * has no phrase (such as a common name) before it, has no comment (text + * + * [RFC 5280]: + * When the subjectAltName extension contains an Internet mail address, + * the address MUST be stored in the rfc822Name. The format of an + * rfc822Name is a "Mailbox" as defined in Section 4.1.2 of [RFC2821]. + * A Mailbox has the form "Local-part@Domain". Note that a Mailbox has + * no phrase (such as a common name) before it, has no comment (text * surrounded in parentheses) after it, and is not surrounded by "<" and - * ">". Note that while upper and lower case letters are allowed in an - * RFC 822 addr-spec, no significance is attached to the case. + * ">". * * @param inputName to be checked for being constrained * @return constraint type above @@ -187,7 +188,7 @@ public class RFC822Name implements GeneralNameInterface else if (inputName.getType() != (GeneralNameInterface.NAME_RFC822)) { constraintType = NAME_DIFF_TYPE; } else { - //RFC2459 specifies that case is not significant in RFC822Names + //RFC 5280 specifies that case is not significant in RFC822Names String inName = (((RFC822Name)inputName).getName()).toLowerCase(Locale.ENGLISH); String thisName = name.toLowerCase(Locale.ENGLISH); diff --git a/src/java.base/share/classes/sun/security/x509/URIName.java b/src/java.base/share/classes/sun/security/x509/URIName.java index 4820de87e11..490a35f86b2 100644 --- a/src/java.base/share/classes/sun/security/x509/URIName.java +++ b/src/java.base/share/classes/sun/security/x509/URIName.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 1997, 2014, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 1997, 2018, 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 @@ -213,7 +213,7 @@ public class URIName implements GeneralNameInterface { /** * Compares this name with another, for equality. * - * @return true iff the names are equivalent according to RFC2459. + * @return true iff the names are equivalent according to RFC 5280. */ public boolean equals(Object obj) { if (this == obj) { diff --git a/src/java.base/share/classes/sun/security/x509/X400Address.java b/src/java.base/share/classes/sun/security/x509/X400Address.java index 35839e87825..f0624d93cc3 100644 --- a/src/java.base/share/classes/sun/security/x509/X400Address.java +++ b/src/java.base/share/classes/sun/security/x509/X400Address.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2000, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2000, 2018, 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 @@ -403,7 +403,7 @@ public class X400Address implements GeneralNameInterface { else if (inputName.getType() != NAME_X400) constraintType = NAME_DIFF_TYPE; else - //Narrowing, widening, and match constraints not defined in rfc2459 for X400Address + //Narrowing, widening, and match constraints not defined in RFC 5280 for X400Address throw new UnsupportedOperationException("Narrowing, widening, and match are not supported for X400Address."); return constraintType; } diff --git a/src/java.base/share/classes/sun/security/x509/X509CertImpl.java b/src/java.base/share/classes/sun/security/x509/X509CertImpl.java index 52cbe71e88f..5fff18309d2 100644 --- a/src/java.base/share/classes/sun/security/x509/X509CertImpl.java +++ b/src/java.base/share/classes/sun/security/x509/X509CertImpl.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 1996, 2017, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 1996, 2018, 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 @@ -63,14 +63,8 @@ import sun.security.provider.X509Factory; * direct knowledge of each other. CA certificates are either signed by * themselves, or by some other CA such as a "root" CA. * - *

    RFC 1422 is very informative, though it does not describe much - * of the recent work being done with X.509 certificates. That includes - * a 1996 version (X.509v3) and a variety of enhancements being made to - * facilitate an explosion of personal certificates used as "Internet - * Drivers' Licences", or with SET for credit card transactions. - * - *

    More recent work includes the IETF PKIX Working Group efforts, - * especially RFC2459. + *

    Standards relating to X.509 Public Key Infrastructure for the Internet + * can be referenced in RFC 5280. * * @author Dave Brownell * @author Amit Kapoor From 205598c97fe0d5a10d5fb7f0c36821f4abf7fc41 Mon Sep 17 00:00:00 2001 From: Jesper Wilhelmsson Date: Thu, 20 Dec 2018 02:54:16 +0100 Subject: [PATCH 029/101] Added tag jdk-12+25 for changeset 7496df94b3b7 --- .hgtags | 1 + 1 file changed, 1 insertion(+) diff --git a/.hgtags b/.hgtags index f2d1a58c394..35959b6d547 100644 --- a/.hgtags +++ b/.hgtags @@ -527,3 +527,4 @@ f8fb0c86f2b3d24294d39c5685a628e1beb14ba7 jdk-12+21 732bec44c89e8b93a38296bf690f97b7230c5b6d jdk-12+22 eef755718cb24813031a842bbfc716a6cea18e9a jdk-12+23 7d4397b43fa305806160785a4c7210600d59581a jdk-12+24 +7496df94b3b79f3da53925d2d137317715f11d97 jdk-12+25 From 1cc5cf4c0200e038fd9cd9b3128c1aba4a369ab4 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Erik=20=C3=96sterlund?= Date: Thu, 20 Dec 2018 10:41:45 +0100 Subject: [PATCH 030/101] 8215500: ICRefillVerifierMark does not set the provided verfier as current Reviewed-by: pliden, thartmann --- src/hotspot/share/code/icBuffer.cpp | 14 +++++++------- src/hotspot/share/runtime/thread.cpp | 2 +- src/hotspot/share/runtime/thread.hpp | 11 ++++++----- 3 files changed, 14 insertions(+), 13 deletions(-) diff --git a/src/hotspot/share/code/icBuffer.cpp b/src/hotspot/share/code/icBuffer.cpp index b03554c2bac..39b7ab11e31 100644 --- a/src/hotspot/share/code/icBuffer.cpp +++ b/src/hotspot/share/code/icBuffer.cpp @@ -53,29 +53,29 @@ ICRefillVerifier::ICRefillVerifier() _refill_remembered(false) { Thread* thread = Thread::current(); - assert(thread->missed_ic_stub_refill_mark() == NULL, "nesting not supported"); - thread->set_missed_ic_stub_refill_mark(this); + assert(thread->missed_ic_stub_refill_verifier() == NULL, "nesting not supported"); + thread->set_missed_ic_stub_refill_verifier(this); } ICRefillVerifier::~ICRefillVerifier() { assert(!_refill_requested || _refill_remembered, "Forgot to refill IC stubs after failed IC transition"); - Thread::current()->set_missed_ic_stub_refill_mark(NULL); + Thread::current()->set_missed_ic_stub_refill_verifier(NULL); } ICRefillVerifierMark::ICRefillVerifierMark(ICRefillVerifier* verifier) { Thread* thread = Thread::current(); - assert(thread->missed_ic_stub_refill_mark() == NULL, "nesting not supported"); - thread->set_missed_ic_stub_refill_mark(this); + assert(thread->missed_ic_stub_refill_verifier() == NULL, "nesting not supported"); + thread->set_missed_ic_stub_refill_verifier(verifier); } ICRefillVerifierMark::~ICRefillVerifierMark() { - Thread::current()->set_missed_ic_stub_refill_mark(NULL); + Thread::current()->set_missed_ic_stub_refill_verifier(NULL); } static ICRefillVerifier* current_ic_refill_verifier() { Thread* current = Thread::current(); - ICRefillVerifier* verifier = reinterpret_cast(current->missed_ic_stub_refill_mark()); + ICRefillVerifier* verifier = current->missed_ic_stub_refill_verifier(); assert(verifier != NULL, "need a verifier for safety"); return verifier; } diff --git a/src/hotspot/share/runtime/thread.cpp b/src/hotspot/share/runtime/thread.cpp index e2e43893de2..539810c0fb7 100644 --- a/src/hotspot/share/runtime/thread.cpp +++ b/src/hotspot/share/runtime/thread.cpp @@ -231,7 +231,7 @@ Thread::Thread() { set_active_handles(NULL); set_free_handle_block(NULL); set_last_handle_mark(NULL); - DEBUG_ONLY(_missed_ic_stub_refill_mark = NULL); + DEBUG_ONLY(_missed_ic_stub_refill_verifier = NULL); // This initial value ==> never claimed. _oops_do_parity = 0; diff --git a/src/hotspot/share/runtime/thread.hpp b/src/hotspot/share/runtime/thread.hpp index 8509c918f68..b00f662c495 100644 --- a/src/hotspot/share/runtime/thread.hpp +++ b/src/hotspot/share/runtime/thread.hpp @@ -81,6 +81,7 @@ class jvmtiDeferredLocalVariableSet; class GCTaskQueue; class ThreadClosure; +class ICRefillVerifier; class IdealGraphPrinter; class Metadata; @@ -329,15 +330,15 @@ class Thread: public ThreadShadow { private: #ifdef ASSERT - void* _missed_ic_stub_refill_mark; + ICRefillVerifier* _missed_ic_stub_refill_verifier; public: - void* missed_ic_stub_refill_mark() { - return _missed_ic_stub_refill_mark; + ICRefillVerifier* missed_ic_stub_refill_verifier() { + return _missed_ic_stub_refill_verifier; } - void set_missed_ic_stub_refill_mark(void* mark) { - _missed_ic_stub_refill_mark = mark; + void set_missed_ic_stub_refill_verifier(ICRefillVerifier* verifier) { + _missed_ic_stub_refill_verifier = verifier; } #endif From 83adde986667a2fbfb620bf90f6e27607f660e97 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Erik=20=C3=96sterlund?= Date: Thu, 20 Dec 2018 10:41:45 +0100 Subject: [PATCH 031/101] 8215491: ICStubInterface::finalize finds zombie nmethod with ZGC concurrent class unloading Reviewed-by: dlong, coleenp --- src/hotspot/share/code/nmethod.cpp | 7 +++++++ src/hotspot/share/runtime/sweeper.cpp | 6 ------ 2 files changed, 7 insertions(+), 6 deletions(-) diff --git a/src/hotspot/share/code/nmethod.cpp b/src/hotspot/share/code/nmethod.cpp index bbd416bf828..02c9247bf2a 100644 --- a/src/hotspot/share/code/nmethod.cpp +++ b/src/hotspot/share/code/nmethod.cpp @@ -1283,6 +1283,13 @@ bool nmethod::make_not_entrant_or_zombie(int state) { flush_dependencies(/*delete_immediately*/true); } + // Clear ICStubs to prevent back patching stubs of zombie or flushed + // nmethods during the next safepoint (see ICStub::finalize). + { + CompiledICLocker ml(this); + clear_ic_stubs(); + } + // zombie only - if a JVMTI agent has enabled the CompiledMethodUnload // event and it hasn't already been reported for this nmethod then // report it now. The event may have been reported earlier if the GC diff --git a/src/hotspot/share/runtime/sweeper.cpp b/src/hotspot/share/runtime/sweeper.cpp index 88a94f32dfb..c5e5a7fdfc1 100644 --- a/src/hotspot/share/runtime/sweeper.cpp +++ b/src/hotspot/share/runtime/sweeper.cpp @@ -717,12 +717,6 @@ NMethodSweeper::MethodStateChange NMethodSweeper::process_compiled_method(Compil // stack we can safely convert it to a zombie method OrderAccess::loadload(); // _stack_traversal_mark and _state if (cm->can_convert_to_zombie()) { - // Clear ICStubs to prevent back patching stubs of zombie or flushed - // nmethods during the next safepoint (see ICStub::finalize). - { - CompiledICLocker ml(cm); - cm->clear_ic_stubs(); - } // Code cache state change is tracked in make_zombie() cm->make_zombie(); SWEEP(cm); From 2778c8648819d7a068b0b5e505dc95eb8a6f34c7 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Per=20Lid=C3=A9n?= Date: Thu, 20 Dec 2018 11:43:04 +0100 Subject: [PATCH 032/101] 8215547: ZGC: Fix incorrect match rule for loadBarrierWeakSlowRegNoVec Reviewed-by: eosterlund, neliasso --- src/hotspot/cpu/x86/x86_64.ad | 258 +++++++++--------- .../linux_x86/gc/z/zArguments_linux_x86.cpp | 40 +++ src/hotspot/share/gc/z/zArguments.cpp | 4 +- src/hotspot/share/gc/z/zArguments.hpp | 3 + 4 files changed, 180 insertions(+), 125 deletions(-) create mode 100644 src/hotspot/os_cpu/linux_x86/gc/z/zArguments_linux_x86.cpp diff --git a/src/hotspot/cpu/x86/x86_64.ad b/src/hotspot/cpu/x86/x86_64.ad index a8ae966fa62..ebaf2fe8355 100644 --- a/src/hotspot/cpu/x86/x86_64.ad +++ b/src/hotspot/cpu/x86/x86_64.ad @@ -4265,132 +4265,196 @@ operand cmpOpUCF2() %{ // Operands for bound floating pointer register arguments operand rxmm0() %{ - constraint(ALLOC_IN_RC(xmm0_reg)); match(VecX); - predicate((UseSSE > 0) && (UseAVX<= 2)); format%{%} interface(REG_INTER); + constraint(ALLOC_IN_RC(xmm0_reg)); + match(VecX); + format%{%} + interface(REG_INTER); %} operand rxmm1() %{ - constraint(ALLOC_IN_RC(xmm1_reg)); match(VecX); - predicate((UseSSE > 0) && (UseAVX <= 2)); format%{%} interface(REG_INTER); + constraint(ALLOC_IN_RC(xmm1_reg)); + match(VecX); + format%{%} + interface(REG_INTER); %} operand rxmm2() %{ - constraint(ALLOC_IN_RC(xmm2_reg)); match(VecX); - predicate((UseSSE > 0) && (UseAVX <= 2)); format%{%} interface(REG_INTER); + constraint(ALLOC_IN_RC(xmm2_reg)); + match(VecX); + format%{%} + interface(REG_INTER); %} operand rxmm3() %{ - constraint(ALLOC_IN_RC(xmm3_reg)); match(VecX); - predicate((UseSSE > 0) && (UseAVX <= 2)); format%{%} interface(REG_INTER); + constraint(ALLOC_IN_RC(xmm3_reg)); + match(VecX); + format%{%} + interface(REG_INTER); %} operand rxmm4() %{ - constraint(ALLOC_IN_RC(xmm4_reg)); match(VecX); - predicate((UseSSE > 0) && (UseAVX <= 2)); format%{%} interface(REG_INTER); + constraint(ALLOC_IN_RC(xmm4_reg)); + match(VecX); + format%{%} + interface(REG_INTER); %} operand rxmm5() %{ - constraint(ALLOC_IN_RC(xmm5_reg)); match(VecX); - predicate((UseSSE > 0) && (UseAVX <= 2)); format%{%} interface(REG_INTER); + constraint(ALLOC_IN_RC(xmm5_reg)); + match(VecX); + format%{%} + interface(REG_INTER); %} operand rxmm6() %{ - constraint(ALLOC_IN_RC(xmm6_reg)); match(VecX); - predicate((UseSSE > 0) && (UseAVX <= 2)); format%{%} interface(REG_INTER); + constraint(ALLOC_IN_RC(xmm6_reg)); + match(VecX); + format%{%} + interface(REG_INTER); %} operand rxmm7() %{ - constraint(ALLOC_IN_RC(xmm7_reg)); match(VecX); - predicate((UseSSE > 0) && (UseAVX <= 2)); format%{%} interface(REG_INTER); + constraint(ALLOC_IN_RC(xmm7_reg)); + match(VecX); + format%{%} + interface(REG_INTER); %} operand rxmm8() %{ - constraint(ALLOC_IN_RC(xmm8_reg)); match(VecX); - predicate((UseSSE > 0) && (UseAVX <= 2)); format%{%} interface(REG_INTER); + constraint(ALLOC_IN_RC(xmm8_reg)); + match(VecX); + format%{%} + interface(REG_INTER); %} operand rxmm9() %{ - constraint(ALLOC_IN_RC(xmm9_reg)); match(VecX); - predicate((UseSSE > 0) && (UseAVX <= 2)); format%{%} interface(REG_INTER); + constraint(ALLOC_IN_RC(xmm9_reg)); + match(VecX); + format%{%} + interface(REG_INTER); %} operand rxmm10() %{ - constraint(ALLOC_IN_RC(xmm10_reg)); match(VecX); - predicate((UseSSE > 0) && (UseAVX <= 2)); format%{%} interface(REG_INTER); + constraint(ALLOC_IN_RC(xmm10_reg)); + match(VecX); + format%{%} + interface(REG_INTER); %} operand rxmm11() %{ - constraint(ALLOC_IN_RC(xmm11_reg)); match(VecX); - predicate((UseSSE > 0) && (UseAVX <= 2)); format%{%} interface(REG_INTER); + constraint(ALLOC_IN_RC(xmm11_reg)); + match(VecX); + format%{%} + interface(REG_INTER); %} operand rxmm12() %{ - constraint(ALLOC_IN_RC(xmm12_reg)); match(VecX); - predicate((UseSSE > 0) && (UseAVX <= 2)); format%{%} interface(REG_INTER); + constraint(ALLOC_IN_RC(xmm12_reg)); + match(VecX); + format%{%} + interface(REG_INTER); %} operand rxmm13() %{ - constraint(ALLOC_IN_RC(xmm13_reg)); match(VecX); - predicate((UseSSE > 0) && (UseAVX <= 2)); format%{%} interface(REG_INTER); + constraint(ALLOC_IN_RC(xmm13_reg)); + match(VecX); + format%{%} + interface(REG_INTER); %} operand rxmm14() %{ - constraint(ALLOC_IN_RC(xmm14_reg)); match(VecX); - predicate((UseSSE > 0) && (UseAVX <= 2)); format%{%} interface(REG_INTER); + constraint(ALLOC_IN_RC(xmm14_reg)); + match(VecX); + format%{%} + interface(REG_INTER); %} operand rxmm15() %{ - constraint(ALLOC_IN_RC(xmm15_reg)); match(VecX); - predicate((UseSSE > 0) && (UseAVX <= 2)); format%{%} interface(REG_INTER); + constraint(ALLOC_IN_RC(xmm15_reg)); + match(VecX); + format%{%} + interface(REG_INTER); %} operand rxmm16() %{ - constraint(ALLOC_IN_RC(xmm16_reg)); match(VecX); - predicate(UseAVX == 3); format%{%} interface(REG_INTER); + constraint(ALLOC_IN_RC(xmm16_reg)); + match(VecX); + format%{%} + interface(REG_INTER); %} operand rxmm17() %{ - constraint(ALLOC_IN_RC(xmm17_reg)); match(VecX); - predicate(UseAVX == 3); format%{%} interface(REG_INTER); + constraint(ALLOC_IN_RC(xmm17_reg)); + match(VecX); + format%{%} + interface(REG_INTER); %} operand rxmm18() %{ - constraint(ALLOC_IN_RC(xmm18_reg)); match(VecX); - predicate(UseAVX == 3); format%{%} interface(REG_INTER); + constraint(ALLOC_IN_RC(xmm18_reg)); + match(VecX); + format%{%} + interface(REG_INTER); %} operand rxmm19() %{ - constraint(ALLOC_IN_RC(xmm19_reg)); match(VecX); - predicate(UseAVX == 3); format%{%} interface(REG_INTER); + constraint(ALLOC_IN_RC(xmm19_reg)); + match(VecX); + format%{%} + interface(REG_INTER); %} operand rxmm20() %{ - constraint(ALLOC_IN_RC(xmm20_reg)); match(VecX); - predicate(UseAVX == 3); format%{%} interface(REG_INTER); + constraint(ALLOC_IN_RC(xmm20_reg)); + match(VecX); + format%{%} + interface(REG_INTER); %} operand rxmm21() %{ - constraint(ALLOC_IN_RC(xmm21_reg)); match(VecX); - predicate(UseAVX == 3); format%{%} interface(REG_INTER); + constraint(ALLOC_IN_RC(xmm21_reg)); + match(VecX); + format%{%} + interface(REG_INTER); %} operand rxmm22() %{ - constraint(ALLOC_IN_RC(xmm22_reg)); match(VecX); - predicate(UseAVX == 3); format%{%} interface(REG_INTER); + constraint(ALLOC_IN_RC(xmm22_reg)); + match(VecX); + format%{%} + interface(REG_INTER); %} operand rxmm23() %{ - constraint(ALLOC_IN_RC(xmm23_reg)); match(VecX); - predicate(UseAVX == 3); format%{%} interface(REG_INTER); + constraint(ALLOC_IN_RC(xmm23_reg)); + match(VecX); + format%{%} + interface(REG_INTER); %} operand rxmm24() %{ - constraint(ALLOC_IN_RC(xmm24_reg)); match(VecX); - predicate(UseAVX == 3); format%{%} interface(REG_INTER); + constraint(ALLOC_IN_RC(xmm24_reg)); + match(VecX); + format%{%} + interface(REG_INTER); %} operand rxmm25() %{ - constraint(ALLOC_IN_RC(xmm25_reg)); match(VecX); - predicate(UseAVX == 3); format%{%} interface(REG_INTER); + constraint(ALLOC_IN_RC(xmm25_reg)); + match(VecX); + format%{%} + interface(REG_INTER); %} operand rxmm26() %{ - constraint(ALLOC_IN_RC(xmm26_reg)); match(VecX); - predicate(UseAVX == 3); format%{%} interface(REG_INTER); + constraint(ALLOC_IN_RC(xmm26_reg)); + match(VecX); + format%{%} + interface(REG_INTER); %} operand rxmm27() %{ - constraint(ALLOC_IN_RC(xmm27_reg)); match(VecX); - predicate(UseAVX == 3); format%{%} interface(REG_INTER); + constraint(ALLOC_IN_RC(xmm27_reg)); + match(VecX); + format%{%} + interface(REG_INTER); %} operand rxmm28() %{ - constraint(ALLOC_IN_RC(xmm28_reg)); match(VecX); - predicate(UseAVX == 3); format%{%} interface(REG_INTER); + constraint(ALLOC_IN_RC(xmm28_reg)); + match(VecX); + format%{%} + interface(REG_INTER); %} operand rxmm29() %{ - constraint(ALLOC_IN_RC(xmm29_reg)); match(VecX); - predicate(UseAVX == 3); format%{%} interface(REG_INTER); + constraint(ALLOC_IN_RC(xmm29_reg)); + match(VecX); + format%{%} + interface(REG_INTER); %} operand rxmm30() %{ - constraint(ALLOC_IN_RC(xmm30_reg)); match(VecX); - predicate(UseAVX == 3); format%{%} interface(REG_INTER); + constraint(ALLOC_IN_RC(xmm30_reg)); + match(VecX); + format%{%} + interface(REG_INTER); %} operand rxmm31() %{ - constraint(ALLOC_IN_RC(xmm31_reg)); match(VecX); - predicate(UseAVX == 3); format%{%} interface(REG_INTER); + constraint(ALLOC_IN_RC(xmm31_reg)); + match(VecX); + format%{%} + interface(REG_INTER); %} //----------OPERAND CLASSES---------------------------------------------------- @@ -12651,33 +12715,6 @@ instruct RethrowException() // Execute ZGC load barrier (strong) slow path // -// When running without XMM regs -instruct loadBarrierSlowRegNoVec(rRegP dst, memory mem, rFlagsReg cr) %{ - - match(Set dst (LoadBarrierSlowReg mem)); - predicate(MaxVectorSize < 16); - - effect(DEF dst, KILL cr); - - format %{"LoadBarrierSlowRegNoVec $dst, $mem" %} - ins_encode %{ -#if INCLUDE_ZGC - Register d = $dst$$Register; - ZBarrierSetAssembler* bs = (ZBarrierSetAssembler*)BarrierSet::barrier_set()->barrier_set_assembler(); - - assert(d != r12, "Can't be R12!"); - assert(d != r15, "Can't be R15!"); - assert(d != rsp, "Can't be RSP!"); - - __ lea(d, $mem$$Address); - __ call(RuntimeAddress(bs->load_barrier_slow_stub(d))); -#else - ShouldNotReachHere(); -#endif - %} - ins_pipe(pipe_slow); -%} - // For XMM and YMM enabled processors instruct loadBarrierSlowRegXmmAndYmm(rRegP dst, memory mem, rFlagsReg cr, rxmm0 x0, rxmm1 x1, rxmm2 x2,rxmm3 x3, @@ -12686,7 +12723,7 @@ instruct loadBarrierSlowRegXmmAndYmm(rRegP dst, memory mem, rFlagsReg cr, rxmm12 x12, rxmm13 x13, rxmm14 x14, rxmm15 x15) %{ match(Set dst (LoadBarrierSlowReg mem)); - predicate((UseSSE > 0) && (UseAVX <= 2) && (MaxVectorSize >= 16)); + predicate(UseAVX <= 2); effect(DEF dst, KILL cr, KILL x0, KILL x1, KILL x2, KILL x3, @@ -12694,7 +12731,7 @@ instruct loadBarrierSlowRegXmmAndYmm(rRegP dst, memory mem, rFlagsReg cr, KILL x8, KILL x9, KILL x10, KILL x11, KILL x12, KILL x13, KILL x14, KILL x15); - format %{"LoadBarrierSlowRegXmm $dst, $mem" %} + format %{"LoadBarrierSlowRegXmmAndYmm $dst, $mem" %} ins_encode %{ #if INCLUDE_ZGC Register d = $dst$$Register; @@ -12725,7 +12762,7 @@ instruct loadBarrierSlowRegZmm(rRegP dst, memory mem, rFlagsReg cr, rxmm28 x28, rxmm29 x29, rxmm30 x30, rxmm31 x31) %{ match(Set dst (LoadBarrierSlowReg mem)); - predicate((UseAVX == 3) && (MaxVectorSize >= 16)); + predicate(UseAVX == 3); effect(DEF dst, KILL cr, KILL x0, KILL x1, KILL x2, KILL x3, @@ -12760,33 +12797,6 @@ instruct loadBarrierSlowRegZmm(rRegP dst, memory mem, rFlagsReg cr, // Execute ZGC load barrier (weak) slow path // -// When running without XMM regs -instruct loadBarrierWeakSlowRegNoVec(rRegP dst, memory mem, rFlagsReg cr) %{ - - match(Set dst (LoadBarrierSlowReg mem)); - predicate(MaxVectorSize < 16); - - effect(DEF dst, KILL cr); - - format %{"LoadBarrierSlowRegNoVec $dst, $mem" %} - ins_encode %{ -#if INCLUDE_ZGC - Register d = $dst$$Register; - ZBarrierSetAssembler* bs = (ZBarrierSetAssembler*)BarrierSet::barrier_set()->barrier_set_assembler(); - - assert(d != r12, "Can't be R12!"); - assert(d != r15, "Can't be R15!"); - assert(d != rsp, "Can't be RSP!"); - - __ lea(d, $mem$$Address); - __ call(RuntimeAddress(bs->load_barrier_weak_slow_stub(d))); -#else - ShouldNotReachHere(); -#endif - %} - ins_pipe(pipe_slow); -%} - // For XMM and YMM enabled processors instruct loadBarrierWeakSlowRegXmmAndYmm(rRegP dst, memory mem, rFlagsReg cr, rxmm0 x0, rxmm1 x1, rxmm2 x2,rxmm3 x3, @@ -12795,7 +12805,7 @@ instruct loadBarrierWeakSlowRegXmmAndYmm(rRegP dst, memory mem, rFlagsReg cr, rxmm12 x12, rxmm13 x13, rxmm14 x14, rxmm15 x15) %{ match(Set dst (LoadBarrierWeakSlowReg mem)); - predicate((UseSSE > 0) && (UseAVX <= 2) && (MaxVectorSize >= 16)); + predicate(UseAVX <= 2); effect(DEF dst, KILL cr, KILL x0, KILL x1, KILL x2, KILL x3, @@ -12803,7 +12813,7 @@ instruct loadBarrierWeakSlowRegXmmAndYmm(rRegP dst, memory mem, rFlagsReg cr, KILL x8, KILL x9, KILL x10, KILL x11, KILL x12, KILL x13, KILL x14, KILL x15); - format %{"LoadBarrierWeakSlowRegXmm $dst, $mem" %} + format %{"LoadBarrierWeakSlowRegXmmAndYmm $dst, $mem" %} ins_encode %{ #if INCLUDE_ZGC Register d = $dst$$Register; @@ -12834,7 +12844,7 @@ instruct loadBarrierWeakSlowRegZmm(rRegP dst, memory mem, rFlagsReg cr, rxmm28 x28, rxmm29 x29, rxmm30 x30, rxmm31 x31) %{ match(Set dst (LoadBarrierWeakSlowReg mem)); - predicate((UseAVX == 3) && (MaxVectorSize >= 16)); + predicate(UseAVX == 3); effect(DEF dst, KILL cr, KILL x0, KILL x1, KILL x2, KILL x3, diff --git a/src/hotspot/os_cpu/linux_x86/gc/z/zArguments_linux_x86.cpp b/src/hotspot/os_cpu/linux_x86/gc/z/zArguments_linux_x86.cpp new file mode 100644 index 00000000000..bd0415e63eb --- /dev/null +++ b/src/hotspot/os_cpu/linux_x86/gc/z/zArguments_linux_x86.cpp @@ -0,0 +1,40 @@ +/* + * Copyright (c) 2018, 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. + */ + +#include "precompiled.hpp" +#include "gc/z/zArguments.hpp" +#include "runtime/globals.hpp" +#include "runtime/globals_extension.hpp" +#include "utilities/debug.hpp" + +void ZArguments::initialize_platform() { + // The C2 barrier slow path expects vector registers to be least + // 16 bytes wide, which is the minimum width available on all + // x86-64 systems. However, the user could have speficied a lower + // number on the command-line, in which case we print a warning + // and raise it to 16. + if (MaxVectorSize < 16) { + warning("ZGC requires MaxVectorSize to be at least 16"); + FLAG_SET_DEFAULT(MaxVectorSize, 16); + } +} diff --git a/src/hotspot/share/gc/z/zArguments.cpp b/src/hotspot/share/gc/z/zArguments.cpp index 6df13def6b8..6906ba026ea 100644 --- a/src/hotspot/share/gc/z/zArguments.cpp +++ b/src/hotspot/share/gc/z/zArguments.cpp @@ -19,7 +19,6 @@ * 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. - * */ #include "precompiled.hpp" @@ -91,6 +90,9 @@ void ZArguments::initialize() { // Verification of stacks not (yet) supported, for the same reason // we need fixup_partial_loads DEBUG_ONLY(FLAG_SET_DEFAULT(VerifyStack, false)); + + // Initialize platform specific arguments + initialize_platform(); } CollectedHeap* ZArguments::create_heap() { diff --git a/src/hotspot/share/gc/z/zArguments.hpp b/src/hotspot/share/gc/z/zArguments.hpp index f85b1471a2b..5edb369c9dc 100644 --- a/src/hotspot/share/gc/z/zArguments.hpp +++ b/src/hotspot/share/gc/z/zArguments.hpp @@ -29,6 +29,9 @@ class CollectedHeap; class ZArguments : public GCArguments { +private: + void initialize_platform(); + public: virtual void initialize(); virtual size_t conservative_max_heap_alignment(); From 9f1d035d8d862fc0e827c1df7e34f4b4c062c6a1 Mon Sep 17 00:00:00 2001 From: Erik Joelsson Date: Thu, 20 Dec 2018 05:27:42 -0800 Subject: [PATCH 033/101] 8215635: Pandoc check in Docs.gmk does not work on Windows Reviewed-by: tbell, ihse --- make/Docs.gmk | 2 +- make/autoconf/basics.m4 | 7 +++++++ make/autoconf/spec.gmk.in | 1 + make/launcher/LauncherCommon.gmk | 2 +- 4 files changed, 10 insertions(+), 2 deletions(-) diff --git a/make/Docs.gmk b/make/Docs.gmk index 4ef887fda75..a9fb7e2f2bf 100644 --- a/make/Docs.gmk +++ b/make/Docs.gmk @@ -517,7 +517,7 @@ $(foreach m, $(ALL_MODULES), \ ) \ ) -ifneq ($(PANDOC), ) +ifeq ($(ENABLE_PANDOC), true) # For all markdown files in $module/share/specs directories, convert them to # html, if we have pandoc (otherwise we'll just skip this). diff --git a/make/autoconf/basics.m4 b/make/autoconf/basics.m4 index 02925e9139c..e4fa23acadd 100644 --- a/make/autoconf/basics.m4 +++ b/make/autoconf/basics.m4 @@ -610,7 +610,14 @@ AC_DEFUN_ONCE([BASIC_SETUP_FUNDAMENTAL_TOOLS], BASIC_PATH_PROGS(DF, df) BASIC_PATH_PROGS(CPIO, [cpio bsdcpio]) BASIC_PATH_PROGS(NICE, nice) + BASIC_PATH_PROGS(PANDOC, pandoc) + if test -n "$PANDOC"; then + ENABLE_PANDOC="true" + else + ENABLE_PANDOC="false" + fi + AC_SUBST(ENABLE_PANDOC) ]) ############################################################################### diff --git a/make/autoconf/spec.gmk.in b/make/autoconf/spec.gmk.in index f2b2e56bc71..445c7a49c35 100644 --- a/make/autoconf/spec.gmk.in +++ b/make/autoconf/spec.gmk.in @@ -761,6 +761,7 @@ MSVCR_DLL:=@MSVCR_DLL@ MSVCP_DLL:=@MSVCP_DLL@ UCRT_DLL_DIR:=@UCRT_DLL_DIR@ STLPORT_LIB:=@STLPORT_LIB@ +ENABLE_PANDOC:=@ENABLE_PANDOC@ #################################################### # diff --git a/make/launcher/LauncherCommon.gmk b/make/launcher/LauncherCommon.gmk index 0c77eda7b78..520f6feaa0e 100644 --- a/make/launcher/LauncherCommon.gmk +++ b/make/launcher/LauncherCommon.gmk @@ -203,7 +203,7 @@ ifeq ($(OPENJDK_TARGET_OS_TYPE), unix) ifneq ($(MAN_FILES_MD), ) # If we got markdown files, ignore the troff files - ifeq ($(PANDOC), ) + ifeq ($(ENABLE_PANDOC), false) $(info Warning: pandoc not found. Not generating man pages) else # Create dynamic man pages from markdown using pandoc. We need From 5cdb1c70780c2e7134ef8364909f2d27349d796c Mon Sep 17 00:00:00 2001 From: Lutz Schmidt Date: Thu, 20 Dec 2018 17:29:59 +0100 Subject: [PATCH 034/101] 8215551: Missing case label in nmethod::reloc_string_for() Reviewed-by: kvn, mbaesken --- src/hotspot/share/code/nmethod.cpp | 1 + 1 file changed, 1 insertion(+) diff --git a/src/hotspot/share/code/nmethod.cpp b/src/hotspot/share/code/nmethod.cpp index 02c9247bf2a..3c422fcc08b 100644 --- a/src/hotspot/share/code/nmethod.cpp +++ b/src/hotspot/share/code/nmethod.cpp @@ -2540,6 +2540,7 @@ const char* nmethod::reloc_string_for(u_char* begin, u_char* end) { case relocInfo::section_word_type: return "section_word"; case relocInfo::poll_type: return "poll"; case relocInfo::poll_return_type: return "poll_return"; + case relocInfo::trampoline_stub_type: return "trampoline_stub"; case relocInfo::type_mask: return "type_bit_mask"; default: From d62b658ffcca3fe3ae1baf45552e90807aba853b Mon Sep 17 00:00:00 2001 From: Nick Gasson Date: Thu, 20 Dec 2018 17:07:07 +0000 Subject: [PATCH 035/101] 8215100: AArch64: fix compareTo intrinsic with four-character Latin/Unicode Reviewed-by: aph, dpochepk --- src/hotspot/cpu/aarch64/macroAssembler_aarch64.cpp | 8 +++----- .../compiler/intrinsics/string/TestStringIntrinsics2.java | 8 +++++++- 2 files changed, 10 insertions(+), 6 deletions(-) diff --git a/src/hotspot/cpu/aarch64/macroAssembler_aarch64.cpp b/src/hotspot/cpu/aarch64/macroAssembler_aarch64.cpp index 96106a9f997..43a9ca82560 100644 --- a/src/hotspot/cpu/aarch64/macroAssembler_aarch64.cpp +++ b/src/hotspot/cpu/aarch64/macroAssembler_aarch64.cpp @@ -4896,7 +4896,7 @@ void MacroAssembler::string_compare(Register str1, Register str2, // A very short string cmpw(cnt2, minCharsInWord); - br(Assembler::LT, SHORT_STRING); + br(Assembler::LE, SHORT_STRING); // Compare longwords // load first parts of strings and finish initialization while loading @@ -4920,8 +4920,7 @@ void MacroAssembler::string_compare(Register str1, Register str2, ldr(tmp2, Address(str2)); cmp(cnt2, STUB_THRESHOLD); br(GE, STUB); - subsw(cnt2, cnt2, 4); - br(EQ, TAIL_CHECK); + subw(cnt2, cnt2, 4); eor(vtmpZ, T16B, vtmpZ, vtmpZ); lea(str1, Address(str1, cnt2, Address::uxtw(str1_chr_shift))); lea(str2, Address(str2, cnt2, Address::uxtw(str2_chr_shift))); @@ -4937,8 +4936,7 @@ void MacroAssembler::string_compare(Register str1, Register str2, ldrs(vtmp, Address(str2)); cmp(cnt2, STUB_THRESHOLD); br(GE, STUB); - subsw(cnt2, cnt2, 4); - br(EQ, TAIL_CHECK); + subw(cnt2, cnt2, 4); lea(str1, Address(str1, cnt2, Address::uxtw(str1_chr_shift))); eor(vtmpZ, T16B, vtmpZ, vtmpZ); lea(str2, Address(str2, cnt2, Address::uxtw(str2_chr_shift))); diff --git a/test/hotspot/jtreg/compiler/intrinsics/string/TestStringIntrinsics2.java b/test/hotspot/jtreg/compiler/intrinsics/string/TestStringIntrinsics2.java index 5c4f334afb4..523653b8d0e 100644 --- a/test/hotspot/jtreg/compiler/intrinsics/string/TestStringIntrinsics2.java +++ b/test/hotspot/jtreg/compiler/intrinsics/string/TestStringIntrinsics2.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2016, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2016, 2018, Oracle and/or its affiliates. All rights reserved. * Copyright (c) 2016 SAP SE. All rights reserved. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * @@ -642,6 +642,12 @@ public class TestStringIntrinsics2 { assertEquals(-3, asmStringCompareTo(a2, b2), "TestOther.asmStringCompareTo(very_very_long_strings_2)"); } + + // See bug 8215100 + { + assertEquals(-20, asmStringCompareTo("e.\u0259.", "y.e.")); + assertEquals(20, asmStringCompareTo("y.e.", "e.\u0259.")); + } } From 1a1369d270345a737882ff7071a785a0e4b429c4 Mon Sep 17 00:00:00 2001 From: Thomas Schatzl Date: Thu, 20 Dec 2018 19:44:33 +0100 Subject: [PATCH 036/101] 8215548: G1PeriodicGCSystemLoadThreshold needs to be a double Change G1PeriodicGCSystemLoadThreshold to be a double to allow better granularity in determining idleness Reviewed-by: sjohanss, sangheki --- .../share/gc/g1/g1YoungRemSetSamplingThread.cpp | 4 ++-- src/hotspot/share/gc/g1/g1_globals.hpp | 11 ++++++----- 2 files changed, 8 insertions(+), 7 deletions(-) diff --git a/src/hotspot/share/gc/g1/g1YoungRemSetSamplingThread.cpp b/src/hotspot/share/gc/g1/g1YoungRemSetSamplingThread.cpp index f993ca3ac43..7283dbdc096 100644 --- a/src/hotspot/share/gc/g1/g1YoungRemSetSamplingThread.cpp +++ b/src/hotspot/share/gc/g1/g1YoungRemSetSamplingThread.cpp @@ -71,9 +71,9 @@ bool G1YoungRemSetSamplingThread::should_start_periodic_gc() { // Check if load is lower than max. double recent_load; - if ((G1PeriodicGCSystemLoadThreshold > 0) && + if ((G1PeriodicGCSystemLoadThreshold > 0.0f) && (os::loadavg(&recent_load, 1) == -1 || recent_load > G1PeriodicGCSystemLoadThreshold)) { - log_debug(gc, periodic)("Load %1.2f is higher than threshold " UINTX_FORMAT ". Skipping.", + log_debug(gc, periodic)("Load %1.2f is higher than threshold %1.2f. Skipping.", recent_load, G1PeriodicGCSystemLoadThreshold); return false; } diff --git a/src/hotspot/share/gc/g1/g1_globals.hpp b/src/hotspot/share/gc/g1/g1_globals.hpp index 83f259da7cc..a4ec8770849 100644 --- a/src/hotspot/share/gc/g1/g1_globals.hpp +++ b/src/hotspot/share/gc/g1/g1_globals.hpp @@ -311,10 +311,11 @@ "perform a concurrent GC as periodic GC, otherwise use a STW " \ "Full GC.") \ \ - manageable(uintx, G1PeriodicGCSystemLoadThreshold, 0, \ - "Maximum recent system wide system load as returned by the 1m " \ - "value of getloadavg() at which G1 triggers a periodic GC. A " \ - "load above this value cancels a given periodic GC. A value of " \ - "zero disables this check.") \ + manageable(double, G1PeriodicGCSystemLoadThreshold, 0.0, \ + "Maximum recent system wide load as returned by the 1m value " \ + "of getloadavg() at which G1 triggers a periodic GC. A load " \ + "above this value cancels a given periodic GC. A value of zero " \ + "disables this check.") \ + range(0.0, (double)max_uintx) \ #endif // SHARE_VM_GC_G1_G1_GLOBALS_HPP From cad5fa8f16a0df2494d36ffe3006d200a20834e1 Mon Sep 17 00:00:00 2001 From: Claes Redestad Date: Thu, 20 Dec 2018 19:44:08 +0100 Subject: [PATCH 037/101] 8215555: TieredCompilation C2 threads can excessively block handshakes Reviewed-by: kvn, neliasso, rehn --- src/hotspot/share/runtime/sweeper.cpp | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/hotspot/share/runtime/sweeper.cpp b/src/hotspot/share/runtime/sweeper.cpp index c5e5a7fdfc1..59edaad8d97 100644 --- a/src/hotspot/share/runtime/sweeper.cpp +++ b/src/hotspot/share/runtime/sweeper.cpp @@ -149,7 +149,7 @@ long NMethodSweeper::_time_counter = 0; // Virtual time u long NMethodSweeper::_last_sweep = 0; // Value of _time_counter when the last sweep happened int NMethodSweeper::_seen = 0; // Nof. nmethod we have currently processed in current pass of CodeCache -volatile bool NMethodSweeper::_should_sweep = true; // Indicates if we should invoke the sweeper +volatile bool NMethodSweeper::_should_sweep = false;// Indicates if we should invoke the sweeper volatile bool NMethodSweeper::_force_sweep = false;// Indicates if we should force a sweep volatile int NMethodSweeper::_bytes_changed = 0; // Counts the total nmethod size if the nmethod changed from: // 1) alive -> not_entrant From e049d143b0e8ea5ddc9a0233dee15a2c536c85cb Mon Sep 17 00:00:00 2001 From: Pankaj Bansal Date: Fri, 21 Dec 2018 11:43:04 +0530 Subject: [PATCH 038/101] 8215364: JavaFX crashes on Ubuntu 18.04 with Wayland while using Swing-FX interop Reviewed-by: prr, kcr --- src/java.desktop/unix/native/libawt_xawt/xawt/XToolkit.c | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/src/java.desktop/unix/native/libawt_xawt/xawt/XToolkit.c b/src/java.desktop/unix/native/libawt_xawt/xawt/XToolkit.c index 5814bb6c8a4..d5f1d012355 100644 --- a/src/java.desktop/unix/native/libawt_xawt/xawt/XToolkit.c +++ b/src/java.desktop/unix/native/libawt_xawt/xawt/XToolkit.c @@ -150,6 +150,10 @@ JNIEXPORT jint JNICALL DEF_JNI_OnLoad(JavaVM *vm, void *reserved) { jvm = vm; + + //Set the gtk backend to x11 on all the systems + putenv("GDK_BACKEND=x11"); + return JNI_VERSION_1_2; } From cb553175b7c8248ab89738a7694ea72026027198 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Hannes=20Walln=C3=B6fer?= Date: Fri, 21 Dec 2018 11:02:36 +0100 Subject: [PATCH 039/101] 8215291: Broken links when generating from project without modules Reviewed-by: jjg --- .../javadoc/internal/doclets/formats/html/resources/search.js | 2 +- test/langtools/jdk/javadoc/doclet/testSearch/TestSearch.java | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/formats/html/resources/search.js b/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/formats/html/resources/search.js index 3c535783794..620ba97d003 100644 --- a/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/formats/html/resources/search.js +++ b/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/formats/html/resources/search.js @@ -55,7 +55,7 @@ function getURLPrefix(ui) { return ui.item.m + slash; } else if ((ui.item.category === catTypes && ui.item.p) || ui.item.category === catMembers) { $.each(packageSearchIndex, function(index, item) { - if (ui.item.p == item.l) { + if (item.m && ui.item.p == item.l) { urlPrefix = item.m + slash; } }); diff --git a/test/langtools/jdk/javadoc/doclet/testSearch/TestSearch.java b/test/langtools/jdk/javadoc/doclet/testSearch/TestSearch.java index 1b6b12d0e99..0df1fc27e80 100644 --- a/test/langtools/jdk/javadoc/doclet/testSearch/TestSearch.java +++ b/test/langtools/jdk/javadoc/doclet/testSearch/TestSearch.java @@ -686,7 +686,7 @@ public class TestSearch extends JavadocTester { + " return ui.item.m + slash;\n" + " } else if ((ui.item.category === catTypes && ui.item.p) || ui.item.category === catMembers) {\n" + " $.each(packageSearchIndex, function(index, item) {\n" - + " if (ui.item.p == item.l) {\n" + + " if (item.m && ui.item.p == item.l) {\n" + " urlPrefix = item.m + slash;\n" + " }\n" + " });\n" From 0e2172c7ba7dc1d0f9a04eb721bb03a76cbf60e2 Mon Sep 17 00:00:00 2001 From: Nick Gasson Date: Fri, 21 Dec 2018 17:14:46 +0000 Subject: [PATCH 040/101] 8215202: AArch64: jtreg test test/jdk/sun/nio/cs/FindEncoderBugs.java fails Reviewed-by: aph --- .../cpu/aarch64/macroAssembler_aarch64.cpp | 8 ++--- test/jdk/sun/nio/cs/FindEncoderBugs.java | 29 +++++++++++++++++-- 2 files changed, 31 insertions(+), 6 deletions(-) diff --git a/src/hotspot/cpu/aarch64/macroAssembler_aarch64.cpp b/src/hotspot/cpu/aarch64/macroAssembler_aarch64.cpp index 43a9ca82560..f24f4e46d61 100644 --- a/src/hotspot/cpu/aarch64/macroAssembler_aarch64.cpp +++ b/src/hotspot/cpu/aarch64/macroAssembler_aarch64.cpp @@ -5648,12 +5648,12 @@ void MacroAssembler::encode_iso_array(Register src, Register dst, orr(v5, T16B, Vtmp3, Vtmp4); uzp1(Vtmp1, T16B, Vtmp1, Vtmp2); uzp1(Vtmp3, T16B, Vtmp3, Vtmp4); - stpq(Vtmp1, Vtmp3, dst); uzp2(v5, T16B, v4, v5); // high bytes umov(tmp2, v5, D, 1); fmovd(tmp1, v5); orr(tmp1, tmp1, tmp2); cbnz(tmp1, LOOP_8); + stpq(Vtmp1, Vtmp3, dst); sub(len, len, 32); add(dst, dst, 32); add(src, src, 64); @@ -5671,7 +5671,6 @@ void MacroAssembler::encode_iso_array(Register src, Register dst, prfm(Address(src, SoftwarePrefetchHintDistance)); uzp1(v4, T16B, Vtmp1, Vtmp2); uzp1(v5, T16B, Vtmp3, Vtmp4); - stpq(v4, v5, dst); orr(Vtmp1, T16B, Vtmp1, Vtmp2); orr(Vtmp3, T16B, Vtmp3, Vtmp4); uzp2(Vtmp1, T16B, Vtmp1, Vtmp3); // high bytes @@ -5679,6 +5678,7 @@ void MacroAssembler::encode_iso_array(Register src, Register dst, fmovd(tmp1, Vtmp1); orr(tmp1, tmp1, tmp2); cbnz(tmp1, LOOP_8); + stpq(v4, v5, dst); sub(len, len, 32); add(dst, dst, 32); add(src, src, 64); @@ -5693,9 +5693,9 @@ void MacroAssembler::encode_iso_array(Register src, Register dst, ld1(Vtmp1, T8H, src); uzp1(Vtmp2, T16B, Vtmp1, Vtmp1); // low bytes uzp2(Vtmp3, T16B, Vtmp1, Vtmp1); // high bytes - strd(Vtmp2, dst); fmovd(tmp1, Vtmp3); cbnz(tmp1, NEXT_1); + strd(Vtmp2, dst); sub(len, len, 8); add(dst, dst, 8); @@ -5708,9 +5708,9 @@ void MacroAssembler::encode_iso_array(Register src, Register dst, cbz(len, DONE); BIND(NEXT_1); ldrh(tmp1, Address(post(src, 2))); - strb(tmp1, Address(post(dst, 1))); tst(tmp1, 0xff00); br(NE, SET_RESULT); + strb(tmp1, Address(post(dst, 1))); subs(len, len, 1); br(GT, NEXT_1); diff --git a/test/jdk/sun/nio/cs/FindEncoderBugs.java b/test/jdk/sun/nio/cs/FindEncoderBugs.java index 77bb3ba094d..ba63eb91cc6 100644 --- a/test/jdk/sun/nio/cs/FindEncoderBugs.java +++ b/test/jdk/sun/nio/cs/FindEncoderBugs.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2008, 2017, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2008, 2018, 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 @@ -180,7 +180,7 @@ public class FindEncoderBugs { private final long failed0 = failed; // legend: r=regular d=direct In=Input Ou=Output - static final int maxBufSize = 20; + static final int maxBufSize = 40; static final CharBuffer[] rInBuffers = new CharBuffer[maxBufSize]; static final CharBuffer[] dInBuffers = new CharBuffer[maxBufSize]; @@ -444,6 +444,28 @@ public class FindEncoderBugs { } } + void testISO88591InvalidChar() { + // Several architectures implement the ISO-8859-1 encoder as an + // intrinsic where the vectorised assembly has separate cases + // for different input sizes, so exhaustively test all sizes + // from 0 to maxBufSize to ensure we get coverage + + for (int i = 0; i < CharsetTester.maxBufSize; i++) { + char[] ia = new char[i]; + for (int j = 0; j < i; j++) + ia[j] = randomChar(); + + test(ia); + + // Test break on unrepresentable character + for (int j = 0; j < i; j++) { + char[] iaInvalid = ia.clone(); + iaInvalid[j] = (char)(randomChar() | 0x100); + test(iaInvalid); + } + } + } + void testPrefix(char[] prefix) { if (prefix.length > 0) System.out.printf("Testing prefix %s%n", string(prefix)); @@ -492,6 +514,9 @@ public class FindEncoderBugs { System.out.println("More ISCII testing..."); new CharsetTester(cs).testPrefix(new char[]{'\u094d'}); // Halant new CharsetTester(cs).testPrefix(new char[]{'\u093c'}); // Nukta + } else if (csn.equals("ISO-8859-1")) { + System.out.println("More ISO-8859-1 testing..."); + new CharsetTester(cs).testISO88591InvalidChar(); } } From 9d20ad22c889abf84ebe1f80c2526c3078a50fb2 Mon Sep 17 00:00:00 2001 From: Andrey Petushkov Date: Fri, 21 Dec 2018 18:26:55 +0000 Subject: [PATCH 041/101] 8215879: AArch64: ReservedStackAccess may leave stack guard in inconsistent state Reviewed-by: aph, dholmes --- src/hotspot/os_cpu/linux_aarch64/os_linux_aarch64.cpp | 3 ++- src/hotspot/share/runtime/thread.cpp | 8 +++----- 2 files changed, 5 insertions(+), 6 deletions(-) diff --git a/src/hotspot/os_cpu/linux_aarch64/os_linux_aarch64.cpp b/src/hotspot/os_cpu/linux_aarch64/os_linux_aarch64.cpp index 6d0fdb4969f..3c7b15086e2 100644 --- a/src/hotspot/os_cpu/linux_aarch64/os_linux_aarch64.cpp +++ b/src/hotspot/os_cpu/linux_aarch64/os_linux_aarch64.cpp @@ -370,7 +370,6 @@ JVM_handle_linux_signal(int sig, if (thread->on_local_stack(addr)) { // stack overflow if (thread->in_stack_yellow_reserved_zone(addr)) { - thread->disable_stack_yellow_reserved_zone(); if (thread->thread_state() == _thread_in_Java) { if (thread->in_stack_reserved_zone(addr)) { frame fr; @@ -392,9 +391,11 @@ JVM_handle_linux_signal(int sig, } // Throw a stack overflow exception. Guard pages will be reenabled // while unwinding the stack. + thread->disable_stack_yellow_reserved_zone(); stub = SharedRuntime::continuation_for_implicit_exception(thread, pc, SharedRuntime::STACK_OVERFLOW); } else { // Thread was in the vm or native code. Return and try to finish. + thread->disable_stack_yellow_reserved_zone(); return 1; } } else if (thread->in_stack_red_zone(addr)) { diff --git a/src/hotspot/share/runtime/thread.cpp b/src/hotspot/share/runtime/thread.cpp index 539810c0fb7..e505fdfb4e4 100644 --- a/src/hotspot/share/runtime/thread.cpp +++ b/src/hotspot/share/runtime/thread.cpp @@ -2603,8 +2603,7 @@ void JavaThread::remove_stack_guard_pages() { } void JavaThread::enable_stack_reserved_zone() { - assert(_stack_guard_state != stack_guard_unused, "must be using guard pages."); - assert(_stack_guard_state != stack_guard_enabled, "already enabled"); + assert(_stack_guard_state == stack_guard_reserved_disabled, "inconsistent state"); // The base notation is from the stack's point of view, growing downward. // We need to adjust it to work correctly with guard_memory() @@ -2622,11 +2621,10 @@ void JavaThread::enable_stack_reserved_zone() { } void JavaThread::disable_stack_reserved_zone() { - assert(_stack_guard_state != stack_guard_unused, "must be using guard pages."); - assert(_stack_guard_state != stack_guard_reserved_disabled, "already disabled"); + assert(_stack_guard_state == stack_guard_enabled, "inconsistent state"); // Simply return if called for a thread that does not use guard pages. - if (_stack_guard_state == stack_guard_unused) return; + if (_stack_guard_state != stack_guard_enabled) return; // The base notation is from the stack's point of view, growing downward. // We need to adjust it to work correctly with guard_memory() From d580e2eed289d683a6f6e08648bc9240479b8986 Mon Sep 17 00:00:00 2001 From: Kishor Kharbas Date: Fri, 21 Dec 2018 08:18:59 -0800 Subject: [PATCH 042/101] 8211425: Allocation of old generation of java heap on alternate memory devices - G1 GC 8202286: Allocation of old generation of Java heap on alternate memory devices Enable an experimental feature in HotSpot JVM to allocate old generation of G1 GC on an alternative memory device, such as NV-DIMMs. Reviewed-by: sangheki, sjohanss --- src/hotspot/os/linux/os_linux.cpp | 2 +- .../share/gc/g1/g1Allocator.inline.hpp | 2 +- src/hotspot/share/gc/g1/g1Arguments.cpp | 7 +- src/hotspot/share/gc/g1/g1CardCounts.cpp | 2 +- src/hotspot/share/gc/g1/g1CollectedHeap.cpp | 175 +++--- src/hotspot/share/gc/g1/g1CollectedHeap.hpp | 39 +- .../share/gc/g1/g1CollectedHeap.inline.hpp | 14 +- src/hotspot/share/gc/g1/g1CollectorPolicy.cpp | 10 +- src/hotspot/share/gc/g1/g1CollectorPolicy.hpp | 5 +- src/hotspot/share/gc/g1/g1HeapVerifier.cpp | 8 +- .../gc/g1/g1HeterogeneousCollectorPolicy.cpp | 103 ++++ .../gc/g1/g1HeterogeneousCollectorPolicy.hpp | 48 ++ .../share/gc/g1/g1HeterogeneousHeapPolicy.cpp | 58 ++ .../share/gc/g1/g1HeterogeneousHeapPolicy.hpp | 48 ++ .../g1/g1HeterogeneousHeapYoungGenSizer.cpp | 51 ++ .../g1/g1HeterogeneousHeapYoungGenSizer.hpp | 51 ++ .../share/gc/g1/g1PageBasedVirtualSpace.cpp | 6 + .../share/gc/g1/g1PageBasedVirtualSpace.hpp | 2 + src/hotspot/share/gc/g1/g1Policy.cpp | 27 +- src/hotspot/share/gc/g1/g1Policy.hpp | 17 +- .../share/gc/g1/g1RegionToSpaceMapper.cpp | 148 ++++- .../share/gc/g1/g1RegionToSpaceMapper.hpp | 32 ++ src/hotspot/share/gc/g1/g1VMOperations.cpp | 4 +- src/hotspot/share/gc/g1/g1YoungGenSizer.cpp | 14 +- src/hotspot/share/gc/g1/g1YoungGenSizer.hpp | 15 +- src/hotspot/share/gc/g1/g1_globals.hpp | 10 + src/hotspot/share/gc/g1/heapRegionManager.cpp | 21 +- src/hotspot/share/gc/g1/heapRegionManager.hpp | 64 ++- src/hotspot/share/gc/g1/heapRegionSet.cpp | 15 + src/hotspot/share/gc/g1/heapRegionSet.hpp | 2 + src/hotspot/share/gc/g1/heapRegionType.cpp | 7 +- src/hotspot/share/gc/g1/heapRegionType.hpp | 8 + .../gc/g1/heterogeneousHeapRegionManager.cpp | 523 ++++++++++++++++++ .../gc/g1/heterogeneousHeapRegionManager.hpp | 150 +++++ src/hotspot/share/gc/g1/vmStructs_g1.hpp | 2 +- src/hotspot/share/gc/shared/gcArguments.cpp | 25 + src/hotspot/share/gc/shared/gcArguments.hpp | 1 + src/hotspot/share/prims/whitebox.cpp | 60 ++ src/hotspot/share/runtime/arguments.cpp | 4 + src/hotspot/share/runtime/globals.hpp | 6 + .../jvm/hotspot/gc/g1/G1CollectedHeap.java | 6 +- test/hotspot/jtreg/TEST.groups | 6 +- .../jtreg/gc/nvdimm/TestAllocateOldGenAt.java | 77 +++ .../gc/nvdimm/TestAllocateOldGenAtError.java | 94 ++++ .../nvdimm/TestAllocateOldGenAtMultiple.java | 92 +++ .../nvdimm/TestHumongousObjectsOnNvdimm.java | 117 ++++ .../gc/nvdimm/TestOldObjectsOnNvdimm.java | 118 ++++ .../gc/nvdimm/TestYoungObjectsOnDram.java | 122 ++++ test/lib/sun/hotspot/WhiteBox.java | 4 + 49 files changed, 2259 insertions(+), 163 deletions(-) create mode 100644 src/hotspot/share/gc/g1/g1HeterogeneousCollectorPolicy.cpp create mode 100644 src/hotspot/share/gc/g1/g1HeterogeneousCollectorPolicy.hpp create mode 100644 src/hotspot/share/gc/g1/g1HeterogeneousHeapPolicy.cpp create mode 100644 src/hotspot/share/gc/g1/g1HeterogeneousHeapPolicy.hpp create mode 100644 src/hotspot/share/gc/g1/g1HeterogeneousHeapYoungGenSizer.cpp create mode 100644 src/hotspot/share/gc/g1/g1HeterogeneousHeapYoungGenSizer.hpp create mode 100644 src/hotspot/share/gc/g1/heterogeneousHeapRegionManager.cpp create mode 100644 src/hotspot/share/gc/g1/heterogeneousHeapRegionManager.hpp create mode 100644 test/hotspot/jtreg/gc/nvdimm/TestAllocateOldGenAt.java create mode 100644 test/hotspot/jtreg/gc/nvdimm/TestAllocateOldGenAtError.java create mode 100644 test/hotspot/jtreg/gc/nvdimm/TestAllocateOldGenAtMultiple.java create mode 100644 test/hotspot/jtreg/gc/nvdimm/TestHumongousObjectsOnNvdimm.java create mode 100644 test/hotspot/jtreg/gc/nvdimm/TestOldObjectsOnNvdimm.java create mode 100644 test/hotspot/jtreg/gc/nvdimm/TestYoungObjectsOnDram.java diff --git a/src/hotspot/os/linux/os_linux.cpp b/src/hotspot/os/linux/os_linux.cpp index b5fdbafaa2a..921b8412a5d 100644 --- a/src/hotspot/os/linux/os_linux.cpp +++ b/src/hotspot/os/linux/os_linux.cpp @@ -5073,7 +5073,7 @@ jint os::init_2(void) { // initialize thread priority policy prio_init(); - if (!FLAG_IS_DEFAULT(AllocateHeapAt)) { + if (!FLAG_IS_DEFAULT(AllocateHeapAt) || !FLAG_IS_DEFAULT(AllocateOldGenAt)) { set_coredump_filter(DAX_SHARED_BIT); } diff --git a/src/hotspot/share/gc/g1/g1Allocator.inline.hpp b/src/hotspot/share/gc/g1/g1Allocator.inline.hpp index 7ae2213f337..7df57b5c991 100644 --- a/src/hotspot/share/gc/g1/g1Allocator.inline.hpp +++ b/src/hotspot/share/gc/g1/g1Allocator.inline.hpp @@ -97,7 +97,7 @@ inline void G1ArchiveAllocator::enable_archive_object_check() { } _archive_check_enabled = true; - size_t length = Universe::heap()->max_capacity(); + size_t length = G1CollectedHeap::heap()->max_reserved_capacity(); _closed_archive_region_map.initialize((HeapWord*)Universe::heap()->base(), (HeapWord*)Universe::heap()->base() + length, HeapRegion::GrainBytes); diff --git a/src/hotspot/share/gc/g1/g1Arguments.cpp b/src/hotspot/share/gc/g1/g1Arguments.cpp index 1f5d3beb7c1..712fc854ebb 100644 --- a/src/hotspot/share/gc/g1/g1Arguments.cpp +++ b/src/hotspot/share/gc/g1/g1Arguments.cpp @@ -28,6 +28,7 @@ #include "gc/g1/g1CollectedHeap.inline.hpp" #include "gc/g1/g1CollectorPolicy.hpp" #include "gc/g1/g1HeapVerifier.hpp" +#include "gc/g1/g1HeterogeneousCollectorPolicy.hpp" #include "gc/g1/heapRegion.hpp" #include "gc/shared/gcArguments.inline.hpp" #include "gc/shared/workerPolicy.hpp" @@ -156,5 +157,9 @@ void G1Arguments::initialize() { } CollectedHeap* G1Arguments::create_heap() { - return create_heap_with_policy(); + if (AllocateOldGenAt != NULL) { + return create_heap_with_policy(); + } else { + return create_heap_with_policy(); + } } diff --git a/src/hotspot/share/gc/g1/g1CardCounts.cpp b/src/hotspot/share/gc/g1/g1CardCounts.cpp index 5e4ef3d5bae..64d4a50af2d 100644 --- a/src/hotspot/share/gc/g1/g1CardCounts.cpp +++ b/src/hotspot/share/gc/g1/g1CardCounts.cpp @@ -63,7 +63,7 @@ G1CardCounts::G1CardCounts(G1CollectedHeap *g1h): } void G1CardCounts::initialize(G1RegionToSpaceMapper* mapper) { - assert(_g1h->max_capacity() > 0, "initialization order"); + assert(_g1h->max_reserved_capacity() > 0, "initialization order"); assert(_g1h->capacity() == 0, "initialization order"); if (G1ConcRSHotCardLimit > 0) { diff --git a/src/hotspot/share/gc/g1/g1CollectedHeap.cpp b/src/hotspot/share/gc/g1/g1CollectedHeap.cpp index 6347287d608..e4d97b44eeb 100644 --- a/src/hotspot/share/gc/g1/g1CollectedHeap.cpp +++ b/src/hotspot/share/gc/g1/g1CollectedHeap.cpp @@ -161,12 +161,12 @@ HeapRegion* G1CollectedHeap::new_heap_region(uint hrs_index, // Private methods. -HeapRegion* G1CollectedHeap::new_region(size_t word_size, bool is_old, bool do_expand) { +HeapRegion* G1CollectedHeap::new_region(size_t word_size, HeapRegionType type, bool do_expand) { assert(!is_humongous(word_size) || word_size <= HeapRegion::GrainWords, "the only time we use this to allocate a humongous region is " "when we are allocating a single humongous region"); - HeapRegion* res = _hrm.allocate_free_region(is_old); + HeapRegion* res = _hrm->allocate_free_region(type); if (res == NULL && do_expand && _expand_heap_after_alloc_failure) { // Currently, only attempts to allocate GC alloc regions set @@ -183,7 +183,7 @@ HeapRegion* G1CollectedHeap::new_region(size_t word_size, bool is_old, bool do_e // always expand the heap by an amount aligned to the heap // region size, the free list should in theory not be empty. // In either case allocate_free_region() will check for NULL. - res = _hrm.allocate_free_region(is_old); + res = _hrm->allocate_free_region(type); } else { _expand_heap_after_alloc_failure = false; } @@ -330,16 +330,16 @@ HeapWord* G1CollectedHeap::humongous_obj_allocate(size_t word_size) { // Only one region to allocate, try to use a fast path by directly allocating // from the free lists. Do not try to expand here, we will potentially do that // later. - HeapRegion* hr = new_region(word_size, true /* is_old */, false /* do_expand */); + HeapRegion* hr = new_region(word_size, HeapRegionType::Humongous, false /* do_expand */); if (hr != NULL) { first = hr->hrm_index(); } } else { // Policy: Try only empty regions (i.e. already committed first). Maybe we // are lucky enough to find some. - first = _hrm.find_contiguous_only_empty(obj_regions); + first = _hrm->find_contiguous_only_empty(obj_regions); if (first != G1_NO_HRM_INDEX) { - _hrm.allocate_free_regions_starting_at(first, obj_regions); + _hrm->allocate_free_regions_starting_at(first, obj_regions); } } @@ -347,14 +347,14 @@ HeapWord* G1CollectedHeap::humongous_obj_allocate(size_t word_size) { // Policy: We could not find enough regions for the humongous object in the // free list. Look through the heap to find a mix of free and uncommitted regions. // If so, try expansion. - first = _hrm.find_contiguous_empty_or_unavailable(obj_regions); + first = _hrm->find_contiguous_empty_or_unavailable(obj_regions); if (first != G1_NO_HRM_INDEX) { // We found something. Make sure these regions are committed, i.e. expand // the heap. Alternatively we could do a defragmentation GC. log_debug(gc, ergo, heap)("Attempt heap expansion (humongous allocation request failed). Allocation request: " SIZE_FORMAT "B", word_size * HeapWordSize); - _hrm.expand_at(first, obj_regions, workers()); + _hrm->expand_at(first, obj_regions, workers()); g1_policy()->record_new_heap_size(num_regions()); #ifdef ASSERT @@ -365,7 +365,7 @@ HeapWord* G1CollectedHeap::humongous_obj_allocate(size_t word_size) { assert(is_on_master_free_list(hr), "sanity"); } #endif - _hrm.allocate_free_regions_starting_at(first, obj_regions); + _hrm->allocate_free_regions_starting_at(first, obj_regions); } else { // Policy: Potentially trigger a defragmentation GC. } @@ -554,7 +554,7 @@ void G1CollectedHeap::end_archive_alloc_range(GrowableArray* ranges, bool G1CollectedHeap::check_archive_addresses(MemRegion* ranges, size_t count) { assert(ranges != NULL, "MemRegion array NULL"); assert(count != 0, "No MemRegions provided"); - MemRegion reserved = _hrm.reserved(); + MemRegion reserved = _hrm->reserved(); for (size_t i = 0; i < count; i++) { if (!reserved.contains(ranges[i].start()) || !reserved.contains(ranges[i].last())) { return false; @@ -571,7 +571,7 @@ bool G1CollectedHeap::alloc_archive_regions(MemRegion* ranges, assert(count != 0, "No MemRegions provided"); MutexLockerEx x(Heap_lock); - MemRegion reserved = _hrm.reserved(); + MemRegion reserved = _hrm->reserved(); HeapWord* prev_last_addr = NULL; HeapRegion* prev_last_region = NULL; @@ -605,7 +605,7 @@ bool G1CollectedHeap::alloc_archive_regions(MemRegion* ranges, // range ended, and adjust the start address so we don't try to allocate // the same region again. If the current range is entirely within that // region, skip it, just adjusting the recorded top. - HeapRegion* start_region = _hrm.addr_to_region(start_address); + HeapRegion* start_region = _hrm->addr_to_region(start_address); if ((prev_last_region != NULL) && (start_region == prev_last_region)) { start_address = start_region->end(); if (start_address > last_address) { @@ -615,12 +615,12 @@ bool G1CollectedHeap::alloc_archive_regions(MemRegion* ranges, } start_region->set_top(start_address); curr_range = MemRegion(start_address, last_address + 1); - start_region = _hrm.addr_to_region(start_address); + start_region = _hrm->addr_to_region(start_address); } // Perform the actual region allocation, exiting if it fails. // Then note how much new space we have allocated. - if (!_hrm.allocate_containing_regions(curr_range, &commits, workers())) { + if (!_hrm->allocate_containing_regions(curr_range, &commits, workers())) { return false; } increase_used(word_size * HeapWordSize); @@ -632,8 +632,8 @@ bool G1CollectedHeap::alloc_archive_regions(MemRegion* ranges, // Mark each G1 region touched by the range as archive, add it to // the old set, and set top. - HeapRegion* curr_region = _hrm.addr_to_region(start_address); - HeapRegion* last_region = _hrm.addr_to_region(last_address); + HeapRegion* curr_region = _hrm->addr_to_region(start_address); + HeapRegion* last_region = _hrm->addr_to_region(last_address); prev_last_region = last_region; while (curr_region != NULL) { @@ -650,7 +650,7 @@ bool G1CollectedHeap::alloc_archive_regions(MemRegion* ranges, HeapRegion* next_region; if (curr_region != last_region) { top = curr_region->end(); - next_region = _hrm.next_region_in_heap(curr_region); + next_region = _hrm->next_region_in_heap(curr_region); } else { top = last_address + 1; next_region = NULL; @@ -671,7 +671,7 @@ void G1CollectedHeap::fill_archive_regions(MemRegion* ranges, size_t count) { assert(!is_init_completed(), "Expect to be called at JVM init time"); assert(ranges != NULL, "MemRegion array NULL"); assert(count != 0, "No MemRegions provided"); - MemRegion reserved = _hrm.reserved(); + MemRegion reserved = _hrm->reserved(); HeapWord *prev_last_addr = NULL; HeapRegion* prev_last_region = NULL; @@ -691,8 +691,8 @@ void G1CollectedHeap::fill_archive_regions(MemRegion* ranges, size_t count) { "Ranges not in ascending order: " PTR_FORMAT " <= " PTR_FORMAT , p2i(start_address), p2i(prev_last_addr)); - HeapRegion* start_region = _hrm.addr_to_region(start_address); - HeapRegion* last_region = _hrm.addr_to_region(last_address); + HeapRegion* start_region = _hrm->addr_to_region(start_address); + HeapRegion* last_region = _hrm->addr_to_region(last_address); HeapWord* bottom_address = start_region->bottom(); // Check for a range beginning in the same region in which the @@ -708,7 +708,7 @@ void G1CollectedHeap::fill_archive_regions(MemRegion* ranges, size_t count) { guarantee(curr_region->is_archive(), "Expected archive region at index %u", curr_region->hrm_index()); if (curr_region != last_region) { - curr_region = _hrm.next_region_in_heap(curr_region); + curr_region = _hrm->next_region_in_heap(curr_region); } else { curr_region = NULL; } @@ -757,7 +757,7 @@ void G1CollectedHeap::dealloc_archive_regions(MemRegion* ranges, size_t count, b assert(!is_init_completed(), "Expect to be called at JVM init time"); assert(ranges != NULL, "MemRegion array NULL"); assert(count != 0, "No MemRegions provided"); - MemRegion reserved = _hrm.reserved(); + MemRegion reserved = _hrm->reserved(); HeapWord* prev_last_addr = NULL; HeapRegion* prev_last_region = NULL; size_t size_used = 0; @@ -779,8 +779,8 @@ void G1CollectedHeap::dealloc_archive_regions(MemRegion* ranges, size_t count, b size_used += ranges[i].byte_size(); prev_last_addr = last_address; - HeapRegion* start_region = _hrm.addr_to_region(start_address); - HeapRegion* last_region = _hrm.addr_to_region(last_address); + HeapRegion* start_region = _hrm->addr_to_region(start_address); + HeapRegion* last_region = _hrm->addr_to_region(last_address); // Check for ranges that start in the same G1 region in which the previous // range ended, and adjust the start address so we don't try to free @@ -791,7 +791,7 @@ void G1CollectedHeap::dealloc_archive_regions(MemRegion* ranges, size_t count, b if (start_address > last_address) { continue; } - start_region = _hrm.addr_to_region(start_address); + start_region = _hrm->addr_to_region(start_address); } prev_last_region = last_region; @@ -806,11 +806,11 @@ void G1CollectedHeap::dealloc_archive_regions(MemRegion* ranges, size_t count, b curr_region->set_free(); curr_region->set_top(curr_region->bottom()); if (curr_region != last_region) { - curr_region = _hrm.next_region_in_heap(curr_region); + curr_region = _hrm->next_region_in_heap(curr_region); } else { curr_region = NULL; } - _hrm.shrink_at(curr_index, 1); + _hrm->shrink_at(curr_index, 1); uncommitted_regions++; } @@ -1024,6 +1024,8 @@ void G1CollectedHeap::prepare_heap_for_full_collection() { abandon_collection_set(collection_set()); tear_down_region_sets(false /* free_list_only */); + + hrm()->prepare_for_full_collection_start(); } void G1CollectedHeap::verify_before_full_collection(bool explicit_gc) { @@ -1035,6 +1037,8 @@ void G1CollectedHeap::verify_before_full_collection(bool explicit_gc) { } void G1CollectedHeap::prepare_heap_for_mutators() { + hrm()->prepare_for_full_collection_end(); + // Delete metaspaces for unloaded class loaders and clean up loader_data graph ClassLoaderDataGraph::purge(); MetaspaceUtils::verify_metrics(); @@ -1071,7 +1075,7 @@ void G1CollectedHeap::abort_refinement() { } void G1CollectedHeap::verify_after_full_collection() { - _hrm.verify_optional(); + _hrm->verify_optional(); _verifier->verify_region_sets_optional(); _verifier->verify_after_gc(G1HeapVerifier::G1VerifyFull); // Clear the previous marking bitmap, if needed for bitmap verification. @@ -1325,7 +1329,7 @@ HeapWord* G1CollectedHeap::expand_and_allocate(size_t word_size) { if (expand(expand_bytes, _workers)) { - _hrm.verify_optional(); + _hrm->verify_optional(); _verifier->verify_region_sets_optional(); return attempt_allocation_at_safepoint(word_size, false /* expect_null_mutator_alloc_region */); @@ -1350,7 +1354,7 @@ bool G1CollectedHeap::expand(size_t expand_bytes, WorkGang* pretouch_workers, do uint regions_to_expand = (uint)(aligned_expand_bytes / HeapRegion::GrainBytes); assert(regions_to_expand > 0, "Must expand by at least one region"); - uint expanded_by = _hrm.expand_by(regions_to_expand, pretouch_workers); + uint expanded_by = _hrm->expand_by(regions_to_expand, pretouch_workers); if (expand_time_ms != NULL) { *expand_time_ms = (os::elapsedTime() - expand_heap_start_time_sec) * MILLIUNITS; } @@ -1365,7 +1369,7 @@ bool G1CollectedHeap::expand(size_t expand_bytes, WorkGang* pretouch_workers, do // The expansion of the virtual storage space was unsuccessful. // Let's see if it was because we ran out of swap. if (G1ExitOnExpansionFailure && - _hrm.available() >= regions_to_expand) { + _hrm->available() >= regions_to_expand) { // We had head room... vm_exit_out_of_memory(aligned_expand_bytes, OOM_MMAP_ERROR, "G1 heap expansion"); } @@ -1380,7 +1384,7 @@ void G1CollectedHeap::shrink_helper(size_t shrink_bytes) { HeapRegion::GrainBytes); uint num_regions_to_remove = (uint)(shrink_bytes / HeapRegion::GrainBytes); - uint num_regions_removed = _hrm.shrink_by(num_regions_to_remove); + uint num_regions_removed = _hrm->shrink_by(num_regions_to_remove); size_t shrunk_bytes = num_regions_removed * HeapRegion::GrainBytes; @@ -1408,7 +1412,7 @@ void G1CollectedHeap::shrink(size_t shrink_bytes) { shrink_helper(shrink_bytes); rebuild_region_sets(true /* free_list_only */); - _hrm.verify_optional(); + _hrm->verify_optional(); _verifier->verify_region_sets_optional(); } @@ -1486,7 +1490,7 @@ G1CollectedHeap::G1CollectedHeap(G1CollectorPolicy* collector_policy) : _humongous_set("Humongous Region Set", new HumongousRegionSetChecker()), _bot(NULL), _listener(), - _hrm(), + _hrm(NULL), _allocator(NULL), _verifier(NULL), _summary_bytes_used(0), @@ -1505,7 +1509,7 @@ G1CollectedHeap::G1CollectedHeap(G1CollectorPolicy* collector_policy) : _survivor(), _gc_timer_stw(new (ResourceObj::C_HEAP, mtGC) STWGCTimer()), _gc_tracer_stw(new (ResourceObj::C_HEAP, mtGC) G1NewTracer()), - _g1_policy(new G1Policy(_gc_timer_stw)), + _g1_policy(G1Policy::create_policy(collector_policy, _gc_timer_stw)), _heap_sizing_policy(NULL), _collection_set(this, _g1_policy), _hot_card_cache(NULL), @@ -1632,7 +1636,7 @@ jint G1CollectedHeap::initialize() { guarantee(HeapWordSize == wordSize, "HeapWordSize must equal wordSize"); size_t init_byte_size = collector_policy()->initial_heap_byte_size(); - size_t max_byte_size = collector_policy()->max_heap_byte_size(); + size_t max_byte_size = g1_collector_policy()->heap_reserved_size_bytes(); size_t heap_alignment = collector_policy()->heap_alignment(); // Ensure that the sizes are properly aligned. @@ -1692,12 +1696,17 @@ jint G1CollectedHeap::initialize() { ReservedSpace g1_rs = heap_rs.first_part(max_byte_size); size_t page_size = actual_reserved_page_size(heap_rs); G1RegionToSpaceMapper* heap_storage = - G1RegionToSpaceMapper::create_mapper(g1_rs, - g1_rs.size(), - page_size, - HeapRegion::GrainBytes, - 1, - mtJavaHeap); + G1RegionToSpaceMapper::create_heap_mapper(g1_rs, + g1_rs.size(), + page_size, + HeapRegion::GrainBytes, + 1, + mtJavaHeap); + if(heap_storage == NULL) { + vm_shutdown_during_initialization("Could not initialize G1 heap"); + return JNI_ERR; + } + os::trace_page_sizes("Heap", collector_policy()->min_heap_byte_size(), max_byte_size, @@ -1728,7 +1737,9 @@ jint G1CollectedHeap::initialize() { G1RegionToSpaceMapper* next_bitmap_storage = create_aux_memory_mapper("Next Bitmap", bitmap_size, G1CMBitMap::heap_map_factor()); - _hrm.initialize(heap_storage, prev_bitmap_storage, next_bitmap_storage, bot_storage, cardtable_storage, card_counts_storage); + _hrm = HeapRegionManager::create_manager(this, g1_collector_policy()); + + _hrm->initialize(heap_storage, prev_bitmap_storage, next_bitmap_storage, bot_storage, cardtable_storage, card_counts_storage); _card_table->initialize(cardtable_storage); // Do later initialization work for concurrent refinement. _hot_card_cache->initialize(card_counts_storage); @@ -1743,20 +1754,20 @@ jint G1CollectedHeap::initialize() { guarantee(g1_rs.base() >= (char*)G1CardTable::card_size, "Java heap must not start within the first card."); // Also create a G1 rem set. _g1_rem_set = new G1RemSet(this, _card_table, _hot_card_cache); - _g1_rem_set->initialize(max_capacity(), max_regions()); + _g1_rem_set->initialize(max_reserved_capacity(), max_regions()); size_t max_cards_per_region = ((size_t)1 << (sizeof(CardIdx_t)*BitsPerByte-1)) - 1; guarantee(HeapRegion::CardsPerRegion > 0, "make sure it's initialized"); guarantee(HeapRegion::CardsPerRegion < max_cards_per_region, "too many cards per region"); - FreeRegionList::set_unrealistically_long_length(max_regions() + 1); + FreeRegionList::set_unrealistically_long_length(max_expandable_regions() + 1); _bot = new G1BlockOffsetTable(reserved_region(), bot_storage); { - HeapWord* start = _hrm.reserved().start(); - HeapWord* end = _hrm.reserved().end(); + HeapWord* start = _hrm->reserved().start(); + HeapWord* end = _hrm->reserved().end(); size_t granularity = HeapRegion::GrainBytes; _in_cset_fast_test.initialize(start, end, granularity); @@ -1807,7 +1818,7 @@ jint G1CollectedHeap::initialize() { // Here we allocate the dummy HeapRegion that is required by the // G1AllocRegion class. - HeapRegion* dummy_region = _hrm.get_dummy_region(); + HeapRegion* dummy_region = _hrm->get_dummy_region(); // We'll re-use the same region whether the alloc region will // require BOT updates or not and, if it doesn't, then a non-young @@ -1927,16 +1938,20 @@ CollectorPolicy* G1CollectedHeap::collector_policy() const { return _collector_policy; } +G1CollectorPolicy* G1CollectedHeap::g1_collector_policy() const { + return _collector_policy; +} + SoftRefPolicy* G1CollectedHeap::soft_ref_policy() { return &_soft_ref_policy; } size_t G1CollectedHeap::capacity() const { - return _hrm.length() * HeapRegion::GrainBytes; + return _hrm->length() * HeapRegion::GrainBytes; } size_t G1CollectedHeap::unused_committed_regions_in_bytes() const { - return _hrm.total_free_bytes(); + return _hrm->total_free_bytes(); } void G1CollectedHeap::iterate_hcc_closure(CardTableEntryClosure* cl, uint worker_i) { @@ -2002,6 +2017,18 @@ bool G1CollectedHeap::should_do_concurrent_full_gc(GCCause::Cause cause) { } } +bool G1CollectedHeap::should_upgrade_to_full_gc(GCCause::Cause cause) { + if(g1_policy()->force_upgrade_to_full()) { + return true; + } else if (should_do_concurrent_full_gc(_gc_cause)) { + return false; + } else if (has_regions_left_for_allocation()) { + return false; + } else { + return true; + } +} + #ifndef PRODUCT void G1CollectedHeap::allocate_dummy_regions() { // Let's fill up most of the region @@ -2152,7 +2179,7 @@ void G1CollectedHeap::collect(GCCause::Cause cause) { } bool G1CollectedHeap::is_in(const void* p) const { - if (_hrm.reserved().contains(p)) { + if (_hrm->reserved().contains(p)) { // Given that we know that p is in the reserved space, // heap_region_containing() should successfully // return the containing region. @@ -2166,7 +2193,7 @@ bool G1CollectedHeap::is_in(const void* p) const { #ifdef ASSERT bool G1CollectedHeap::is_in_exact(const void* p) const { bool contains = reserved_region().contains(p); - bool available = _hrm.is_available(addr_to_region((HeapWord*)p)); + bool available = _hrm->is_available(addr_to_region((HeapWord*)p)); if (contains && available) { return true; } else { @@ -2197,18 +2224,18 @@ void G1CollectedHeap::object_iterate(ObjectClosure* cl) { } void G1CollectedHeap::heap_region_iterate(HeapRegionClosure* cl) const { - _hrm.iterate(cl); + _hrm->iterate(cl); } void G1CollectedHeap::heap_region_par_iterate_from_worker_offset(HeapRegionClosure* cl, HeapRegionClaimer *hrclaimer, uint worker_id) const { - _hrm.par_iterate(cl, hrclaimer, hrclaimer->offset_for_worker(worker_id)); + _hrm->par_iterate(cl, hrclaimer, hrclaimer->offset_for_worker(worker_id)); } void G1CollectedHeap::heap_region_par_iterate_from_start(HeapRegionClosure* cl, HeapRegionClaimer *hrclaimer) const { - _hrm.par_iterate(cl, hrclaimer, 0); + _hrm->par_iterate(cl, hrclaimer, 0); } void G1CollectedHeap::collection_set_iterate(HeapRegionClosure* cl) { @@ -2257,7 +2284,11 @@ size_t G1CollectedHeap::unsafe_max_tlab_alloc(Thread* ignored) const { } size_t G1CollectedHeap::max_capacity() const { - return _hrm.reserved().byte_size(); + return _hrm->max_expandable_length() * HeapRegion::GrainBytes; +} + +size_t G1CollectedHeap::max_reserved_capacity() const { + return _hrm->max_length() * HeapRegion::GrainBytes; } jlong G1CollectedHeap::millis_since_last_gc() { @@ -2347,8 +2378,8 @@ void G1CollectedHeap::print_on(outputStream* st) const { st->print(" total " SIZE_FORMAT "K, used " SIZE_FORMAT "K", capacity()/K, used_unlocked()/K); st->print(" [" PTR_FORMAT ", " PTR_FORMAT ")", - p2i(_hrm.reserved().start()), - p2i(_hrm.reserved().end())); + p2i(_hrm->reserved().start()), + p2i(_hrm->reserved().end())); st->cr(); st->print(" region size " SIZE_FORMAT "K, ", HeapRegion::GrainBytes / K); uint young_regions = young_regions_count(); @@ -3131,7 +3162,7 @@ G1CollectedHeap::do_collection_pause_at_safepoint(double target_pause_time_ms) { // output from the concurrent mark thread interfering with this // logging output either. - _hrm.verify_optional(); + _hrm->verify_optional(); _verifier->verify_region_sets_optional(); TASKQUEUE_STATS_ONLY(print_taskqueue_stats()); @@ -3947,7 +3978,7 @@ void G1CollectedHeap::free_region(HeapRegion* hr, bool locked) { assert(!hr->is_free(), "the region should not be free"); assert(!hr->is_empty(), "the region should not be empty"); - assert(_hrm.is_available(hr->hrm_index()), "region should be committed"); + assert(_hrm->is_available(hr->hrm_index()), "region should be committed"); assert(free_list != NULL, "pre-condition"); if (G1VerifyBitmaps) { @@ -3988,7 +4019,7 @@ void G1CollectedHeap::prepend_to_freelist(FreeRegionList* list) { assert(list != NULL, "list can't be null"); if (!list->is_empty()) { MutexLockerEx x(FreeList_lock, Mutex::_no_safepoint_check_flag); - _hrm.insert_list_into_free_list(list); + _hrm->insert_list_into_free_list(list); } } @@ -4521,7 +4552,7 @@ void G1CollectedHeap::tear_down_region_sets(bool free_list_only) { // this is that during a full GC string deduplication needs to know if // a collected region was young or old when the full GC was initiated. } - _hrm.remove_all_free_regions(); + _hrm->remove_all_free_regions(); } void G1CollectedHeap::increase_used(size_t bytes) { @@ -4596,7 +4627,7 @@ void G1CollectedHeap::rebuild_region_sets(bool free_list_only) { _survivor.clear(); } - RebuildRegionSetsClosure cl(free_list_only, &_old_set, &_hrm); + RebuildRegionSetsClosure cl(free_list_only, &_old_set, _hrm); heap_region_iterate(&cl); if (!free_list_only) { @@ -4623,7 +4654,7 @@ HeapRegion* G1CollectedHeap::new_mutator_alloc_region(size_t word_size, bool should_allocate = g1_policy()->should_allocate_mutator_region(); if (force || should_allocate) { HeapRegion* new_alloc_region = new_region(word_size, - false /* is_old */, + HeapRegionType::Eden, false /* do_expand */); if (new_alloc_region != NULL) { set_region_short_lived_locked(new_alloc_region); @@ -4667,13 +4698,19 @@ HeapRegion* G1CollectedHeap::new_gc_alloc_region(size_t word_size, InCSetState d return NULL; } - const bool is_survivor = dest.is_young(); + HeapRegionType type; + if (dest.is_young()) { + type = HeapRegionType::Survivor; + } else { + type = HeapRegionType::Old; + } HeapRegion* new_alloc_region = new_region(word_size, - !is_survivor, + type, true /* do_expand */); + if (new_alloc_region != NULL) { - if (is_survivor) { + if (type.is_survivor()) { new_alloc_region->set_survivor(); _survivor.add(new_alloc_region); _verifier->check_bitmaps("Survivor Region Allocation", new_alloc_region); @@ -4705,14 +4742,14 @@ void G1CollectedHeap::retire_gc_alloc_region(HeapRegion* alloc_region, HeapRegion* G1CollectedHeap::alloc_highest_free_region() { bool expanded = false; - uint index = _hrm.find_highest_free(&expanded); + uint index = _hrm->find_highest_free(&expanded); if (index != G1_NO_HRM_INDEX) { if (expanded) { log_debug(gc, ergo, heap)("Attempt heap expansion (requested address range outside heap bounds). region size: " SIZE_FORMAT "B", HeapRegion::GrainWords * HeapWordSize); } - _hrm.allocate_free_regions_starting_at(index, 1); + _hrm->allocate_free_regions_starting_at(index, 1); return region_at(index); } return NULL; diff --git a/src/hotspot/share/gc/g1/g1CollectedHeap.hpp b/src/hotspot/share/gc/g1/g1CollectedHeap.hpp index 72e1b5f9988..6eb00bc23ba 100644 --- a/src/hotspot/share/gc/g1/g1CollectedHeap.hpp +++ b/src/hotspot/share/gc/g1/g1CollectedHeap.hpp @@ -45,6 +45,7 @@ #include "gc/g1/g1YCTypes.hpp" #include "gc/g1/heapRegionManager.hpp" #include "gc/g1/heapRegionSet.hpp" +#include "gc/g1/heterogeneousHeapRegionManager.hpp" #include "gc/shared/barrierSet.hpp" #include "gc/shared/collectedHeap.hpp" #include "gc/shared/gcHeapSummary.hpp" @@ -194,7 +195,7 @@ private: G1RegionMappingChangedListener _listener; // The sequence of all heap regions in the heap. - HeapRegionManager _hrm; + HeapRegionManager* _hrm; // Manages all allocations with regions except humongous object allocations. G1Allocator* _allocator; @@ -267,6 +268,9 @@ private: // (e) cause == _wb_conc_mark bool should_do_concurrent_full_gc(GCCause::Cause cause); + // Return true if should upgrade to full gc after an incremental one. + bool should_upgrade_to_full_gc(GCCause::Cause cause); + // indicates whether we are in young or mixed GC mode G1CollectorState _collector_state; @@ -369,9 +373,9 @@ private: // Try to allocate a single non-humongous HeapRegion sufficient for // an allocation of the given word_size. If do_expand is true, // attempt to expand the heap if necessary to satisfy the allocation - // request. If the region is to be used as an old region or for a - // humongous object, set is_old to true. If not, to false. - HeapRegion* new_region(size_t word_size, bool is_old, bool do_expand); + // request. 'type' takes the type of region to be allocated. (Use constants + // Old, Eden, Humongous, Survivor defined in HeapRegionType.) + HeapRegion* new_region(size_t word_size, HeapRegionType type, bool do_expand); // Initialize a contiguous set of free regions of length num_regions // and starting at index first so that they appear as a single @@ -957,10 +961,13 @@ public: // The current policy object for the collector. G1Policy* g1_policy() const { return _g1_policy; } + HeapRegionManager* hrm() const { return _hrm; } + const G1CollectionSet* collection_set() const { return &_collection_set; } G1CollectionSet* collection_set() { return &_collection_set; } virtual CollectorPolicy* collector_policy() const; + virtual G1CollectorPolicy* g1_collector_policy() const; virtual SoftRefPolicy* soft_ref_policy(); @@ -1009,7 +1016,7 @@ public: // But G1CollectedHeap doesn't yet support this. virtual bool is_maximal_no_gc() const { - return _hrm.available() == 0; + return _hrm->available() == 0; } // Returns whether there are any regions left in the heap for allocation. @@ -1018,19 +1025,22 @@ public: } // The current number of regions in the heap. - uint num_regions() const { return _hrm.length(); } + uint num_regions() const { return _hrm->length(); } // The max number of regions in the heap. - uint max_regions() const { return _hrm.max_length(); } + uint max_regions() const { return _hrm->max_length(); } + + // Max number of regions that can be comitted. + uint max_expandable_regions() const { return _hrm->max_expandable_length(); } // The number of regions that are completely free. - uint num_free_regions() const { return _hrm.num_free_regions(); } + uint num_free_regions() const { return _hrm->num_free_regions(); } // The number of regions that can be allocated into. - uint num_free_or_available_regions() const { return num_free_regions() + _hrm.available(); } + uint num_free_or_available_regions() const { return num_free_regions() + _hrm->available(); } MemoryUsage get_auxiliary_data_memory_usage() const { - return _hrm.get_auxiliary_data_memory_usage(); + return _hrm->get_auxiliary_data_memory_usage(); } // The number of regions that are not completely free. @@ -1038,7 +1048,7 @@ public: #ifdef ASSERT bool is_on_master_free_list(HeapRegion* hr) { - return _hrm.is_free(hr); + return _hrm->is_free(hr); } #endif // ASSERT @@ -1095,13 +1105,13 @@ public: // Return "TRUE" iff the given object address is in the reserved // region of g1. bool is_in_g1_reserved(const void* p) const { - return _hrm.reserved().contains(p); + return _hrm->reserved().contains(p); } // Returns a MemRegion that corresponds to the space that has been // reserved for the heap MemRegion g1_reserved() const { - return _hrm.reserved(); + return _hrm->reserved(); } virtual bool is_in_closed_subset(const void* p) const; @@ -1227,6 +1237,9 @@ public: // Print the maximum heap capacity. virtual size_t max_capacity() const; + // Return the size of reserved memory. Returns different value than max_capacity() when AllocateOldGenAt is used. + virtual size_t max_reserved_capacity() const; + virtual jlong millis_since_last_gc(); diff --git a/src/hotspot/share/gc/g1/g1CollectedHeap.inline.hpp b/src/hotspot/share/gc/g1/g1CollectedHeap.inline.hpp index bd950a02402..c800f0d4236 100644 --- a/src/hotspot/share/gc/g1/g1CollectedHeap.inline.hpp +++ b/src/hotspot/share/gc/g1/g1CollectedHeap.inline.hpp @@ -57,13 +57,13 @@ size_t G1CollectedHeap::desired_plab_sz(InCSetState dest) { // Inline functions for G1CollectedHeap // Return the region with the given index. It assumes the index is valid. -inline HeapRegion* G1CollectedHeap::region_at(uint index) const { return _hrm.at(index); } +inline HeapRegion* G1CollectedHeap::region_at(uint index) const { return _hrm->at(index); } // Return the region with the given index, or NULL if unmapped. It assumes the index is valid. -inline HeapRegion* G1CollectedHeap::region_at_or_null(uint index) const { return _hrm.at_or_null(index); } +inline HeapRegion* G1CollectedHeap::region_at_or_null(uint index) const { return _hrm->at_or_null(index); } inline HeapRegion* G1CollectedHeap::next_region_in_humongous(HeapRegion* hr) const { - return _hrm.next_region_in_humongous(hr); + return _hrm->next_region_in_humongous(hr); } inline uint G1CollectedHeap::addr_to_region(HeapWord* addr) const { @@ -74,7 +74,7 @@ inline uint G1CollectedHeap::addr_to_region(HeapWord* addr) const { } inline HeapWord* G1CollectedHeap::bottom_addr_for_region(uint index) const { - return _hrm.reserved().start() + index * HeapRegion::GrainWords; + return _hrm->reserved().start() + index * HeapRegion::GrainWords; } template @@ -83,7 +83,7 @@ inline HeapRegion* G1CollectedHeap::heap_region_containing(const T addr) const { assert(is_in_g1_reserved((const void*) addr), "Address " PTR_FORMAT " is outside of the heap ranging from [" PTR_FORMAT " to " PTR_FORMAT ")", p2i((void*)addr), p2i(g1_reserved().start()), p2i(g1_reserved().end())); - return _hrm.addr_to_region((HeapWord*) addr); + return _hrm->addr_to_region((HeapWord*) addr); } template @@ -266,12 +266,12 @@ inline bool G1CollectedHeap::is_obj_dead_full(const oop obj) const { } inline void G1CollectedHeap::set_humongous_reclaim_candidate(uint region, bool value) { - assert(_hrm.at(region)->is_starts_humongous(), "Must start a humongous object"); + assert(_hrm->at(region)->is_starts_humongous(), "Must start a humongous object"); _humongous_reclaim_candidates.set_candidate(region, value); } inline bool G1CollectedHeap::is_humongous_reclaim_candidate(uint region) { - assert(_hrm.at(region)->is_starts_humongous(), "Must start a humongous object"); + assert(_hrm->at(region)->is_starts_humongous(), "Must start a humongous object"); return _humongous_reclaim_candidates.is_candidate(region); } diff --git a/src/hotspot/share/gc/g1/g1CollectorPolicy.cpp b/src/hotspot/share/gc/g1/g1CollectorPolicy.cpp index c0288ed133b..5f27d29b3f0 100644 --- a/src/hotspot/share/gc/g1/g1CollectorPolicy.cpp +++ b/src/hotspot/share/gc/g1/g1CollectorPolicy.cpp @@ -1,5 +1,5 @@ /* - * Copyright (c) 2001, 2016, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2001, 2018, 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 @@ -55,3 +55,11 @@ void G1CollectorPolicy::initialize_alignments() { size_t page_size = UseLargePages ? os::large_page_size() : os::vm_page_size(); _heap_alignment = MAX3(card_table_alignment, _space_alignment, page_size); } + +size_t G1CollectorPolicy::heap_reserved_size_bytes() const { + return _max_heap_byte_size; +} + +bool G1CollectorPolicy::is_hetero_heap() const { + return false; +} diff --git a/src/hotspot/share/gc/g1/g1CollectorPolicy.hpp b/src/hotspot/share/gc/g1/g1CollectorPolicy.hpp index 6e58448a342..94a00844f68 100644 --- a/src/hotspot/share/gc/g1/g1CollectorPolicy.hpp +++ b/src/hotspot/share/gc/g1/g1CollectorPolicy.hpp @@ -1,5 +1,5 @@ /* - * Copyright (c) 2001, 2016, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2001, 2018, 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 @@ -38,6 +38,7 @@ protected: public: G1CollectorPolicy(); + virtual size_t heap_reserved_size_bytes() const; + virtual bool is_hetero_heap() const; }; - #endif // SHARE_VM_GC_G1_G1COLLECTORPOLICY_HPP diff --git a/src/hotspot/share/gc/g1/g1HeapVerifier.cpp b/src/hotspot/share/gc/g1/g1HeapVerifier.cpp index 52c8b5c02db..e7b09f32a18 100644 --- a/src/hotspot/share/gc/g1/g1HeapVerifier.cpp +++ b/src/hotspot/share/gc/g1/g1HeapVerifier.cpp @@ -603,14 +603,14 @@ void G1HeapVerifier::verify_region_sets() { assert_heap_locked_or_at_safepoint(true /* should_be_vm_thread */); // First, check the explicit lists. - _g1h->_hrm.verify(); + _g1h->_hrm->verify(); // Finally, make sure that the region accounting in the lists is // consistent with what we see in the heap. - VerifyRegionListsClosure cl(&_g1h->_old_set, &_g1h->_archive_set, &_g1h->_humongous_set, &_g1h->_hrm); + VerifyRegionListsClosure cl(&_g1h->_old_set, &_g1h->_archive_set, &_g1h->_humongous_set, _g1h->_hrm); _g1h->heap_region_iterate(&cl); - cl.verify_counts(&_g1h->_old_set, &_g1h->_archive_set, &_g1h->_humongous_set, &_g1h->_hrm); + cl.verify_counts(&_g1h->_old_set, &_g1h->_archive_set, &_g1h->_humongous_set, _g1h->_hrm); } void G1HeapVerifier::prepare_for_verify() { @@ -851,7 +851,7 @@ class G1CheckCSetFastTableClosure : public HeapRegionClosure { bool G1HeapVerifier::check_cset_fast_test() { G1CheckCSetFastTableClosure cl; - _g1h->_hrm.iterate(&cl); + _g1h->_hrm->iterate(&cl); return !cl.failures(); } #endif // PRODUCT diff --git a/src/hotspot/share/gc/g1/g1HeterogeneousCollectorPolicy.cpp b/src/hotspot/share/gc/g1/g1HeterogeneousCollectorPolicy.cpp new file mode 100644 index 00000000000..c9db9986328 --- /dev/null +++ b/src/hotspot/share/gc/g1/g1HeterogeneousCollectorPolicy.cpp @@ -0,0 +1,103 @@ +/* + * Copyright (c) 2018, 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. + * + */ + +#include "precompiled.hpp" +#include "gc/g1/g1HeterogeneousCollectorPolicy.hpp" +#include "logging/log.hpp" +#include "runtime/globals_extension.hpp" +#include "runtime/os.hpp" +#include "utilities/formatBuffer.hpp" + +const double G1HeterogeneousCollectorPolicy::MaxRamFractionForYoung = 0.8; +size_t G1HeterogeneousCollectorPolicy::MaxMemoryForYoung; + +static size_t calculate_reasonable_max_memory_for_young(FormatBuffer<100> &calc_str, double max_ram_fraction_for_young) { + julong phys_mem; + // If MaxRam is specified, we use that as maximum physical memory available. + if (FLAG_IS_DEFAULT(MaxRAM)) { + phys_mem = os::physical_memory(); + calc_str.append("Physical_Memory"); + } else { + phys_mem = (julong)MaxRAM; + calc_str.append("MaxRAM"); + } + + julong reasonable_max = phys_mem; + + // If either MaxRAMFraction or MaxRAMPercentage is specified, we use them to calculate + // reasonable max size of young generation. + if (!FLAG_IS_DEFAULT(MaxRAMFraction)) { + reasonable_max = (julong)(phys_mem / MaxRAMFraction); + calc_str.append(" / MaxRAMFraction"); + } else if (!FLAG_IS_DEFAULT(MaxRAMPercentage)) { + reasonable_max = (julong)((phys_mem * MaxRAMPercentage) / 100); + calc_str.append(" * MaxRAMPercentage / 100"); + } else { + // We use our own fraction to calculate max size of young generation. + reasonable_max = phys_mem * max_ram_fraction_for_young; + calc_str.append(" * %0.2f", max_ram_fraction_for_young); + } + + return (size_t)reasonable_max; +} + +void G1HeterogeneousCollectorPolicy::initialize_flags() { + + FormatBuffer<100> calc_str(""); + + MaxMemoryForYoung = calculate_reasonable_max_memory_for_young(calc_str, MaxRamFractionForYoung); + + if (MaxNewSize > MaxMemoryForYoung) { + if (FLAG_IS_CMDLINE(MaxNewSize)) { + log_warning(gc, ergo)("Setting MaxNewSize to " SIZE_FORMAT " based on dram available (calculation = align(%s))", + MaxMemoryForYoung, calc_str.buffer()); + } else { + log_info(gc, ergo)("Setting MaxNewSize to " SIZE_FORMAT " based on dram available (calculation = align(%s)). " + "Dram usage can be lowered by setting MaxNewSize to a lower value", MaxMemoryForYoung, calc_str.buffer()); + } + MaxNewSize = MaxMemoryForYoung; + } + if (NewSize > MaxMemoryForYoung) { + if (FLAG_IS_CMDLINE(NewSize)) { + log_warning(gc, ergo)("Setting NewSize to " SIZE_FORMAT " based on dram available (calculation = align(%s))", + MaxMemoryForYoung, calc_str.buffer()); + } + NewSize = MaxMemoryForYoung; + } + + // After setting new size flags, call base class initialize_flags() + G1CollectorPolicy::initialize_flags(); +} + +size_t G1HeterogeneousCollectorPolicy::reasonable_max_memory_for_young() { + return MaxMemoryForYoung; +} + +size_t G1HeterogeneousCollectorPolicy::heap_reserved_size_bytes() const { + return 2 * _max_heap_byte_size; +} + +bool G1HeterogeneousCollectorPolicy::is_hetero_heap() const { + return true; +} diff --git a/src/hotspot/share/gc/g1/g1HeterogeneousCollectorPolicy.hpp b/src/hotspot/share/gc/g1/g1HeterogeneousCollectorPolicy.hpp new file mode 100644 index 00000000000..9e4f097ebb2 --- /dev/null +++ b/src/hotspot/share/gc/g1/g1HeterogeneousCollectorPolicy.hpp @@ -0,0 +1,48 @@ +/* + * Copyright (c) 2018, 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. + * + */ + +#ifndef SHARE_VM_GC_G1_G1HETEROGENEOUSCOLLECTORPOLICY_HPP +#define SHARE_VM_GC_G1_G1HETEROGENEOUSCOLLECTORPOLICY_HPP + +#include "gc/g1/g1CollectorPolicy.hpp" +#include "gc/g1/g1HeterogeneousHeapYoungGenSizer.hpp" + +class G1HeterogeneousCollectorPolicy : public G1CollectorPolicy { +private: + // Max fraction of dram to use for young generation when MaxRAMFraction and + // MaxRAMPercentage are not specified on commandline. + static const double MaxRamFractionForYoung; + static size_t MaxMemoryForYoung; + +protected: + virtual void initialize_flags(); + +public: + G1HeterogeneousCollectorPolicy() {} + virtual size_t heap_reserved_size_bytes() const; + virtual bool is_hetero_heap() const; + static size_t reasonable_max_memory_for_young(); +}; + +#endif // SHARE_VM_GC_G1_G1HETEROGENEOUSCOLLECTORPOLICY_HPP diff --git a/src/hotspot/share/gc/g1/g1HeterogeneousHeapPolicy.cpp b/src/hotspot/share/gc/g1/g1HeterogeneousHeapPolicy.cpp new file mode 100644 index 00000000000..65b75937704 --- /dev/null +++ b/src/hotspot/share/gc/g1/g1HeterogeneousHeapPolicy.cpp @@ -0,0 +1,58 @@ +/* + * Copyright (c) 2018, 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. + * + */ + +#include "precompiled.hpp" +#include "gc/g1/g1CollectedHeap.hpp" +#include "gc/g1/g1HeterogeneousHeapPolicy.hpp" +#include "gc/g1/g1Policy.hpp" +#include "gc/g1/heterogeneousHeapRegionManager.hpp" + +G1HeterogeneousHeapPolicy::G1HeterogeneousHeapPolicy(G1CollectorPolicy* policy, STWGCTimer* gc_timer) : + G1Policy(policy, gc_timer), _manager(NULL) {} + +// We call the super class init(), after which we provision young_list_target_length() regions in dram. +void G1HeterogeneousHeapPolicy::init(G1CollectedHeap* g1h, G1CollectionSet* collection_set) { + G1Policy::init(g1h, collection_set); + _manager = HeterogeneousHeapRegionManager::manager(); + _manager->adjust_dram_regions((uint)young_list_target_length(), G1CollectedHeap::heap()->workers()); +} + +// After a collection pause, young list target length is updated. So we need to make sure we have enough regions in dram for young gen. +void G1HeterogeneousHeapPolicy::record_collection_pause_end(double pause_time_ms, size_t cards_scanned, size_t heap_used_bytes_before_gc) { + G1Policy::record_collection_pause_end(pause_time_ms, cards_scanned, heap_used_bytes_before_gc); + _manager->adjust_dram_regions((uint)young_list_target_length(), G1CollectedHeap::heap()->workers()); +} + +// After a full collection, young list target length is updated. So we need to make sure we have enough regions in dram for young gen. +void G1HeterogeneousHeapPolicy::record_full_collection_end() { + G1Policy::record_full_collection_end(); + _manager->adjust_dram_regions((uint)young_list_target_length(), G1CollectedHeap::heap()->workers()); +} + +bool G1HeterogeneousHeapPolicy::force_upgrade_to_full() { + if (_manager->has_borrowed_regions()) { + return true; + } + return false; +} diff --git a/src/hotspot/share/gc/g1/g1HeterogeneousHeapPolicy.hpp b/src/hotspot/share/gc/g1/g1HeterogeneousHeapPolicy.hpp new file mode 100644 index 00000000000..37402f77741 --- /dev/null +++ b/src/hotspot/share/gc/g1/g1HeterogeneousHeapPolicy.hpp @@ -0,0 +1,48 @@ +/* + * Copyright (c) 2018, 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. + * + */ + +#ifndef SHARE_VM_GC_G1_G1HETEROGENEOUSHEAPPOLICY_HPP +#define SHARE_VM_GC_G1_G1HETEROGENEOUSHEAPPOLICY_HPP + +#include "gc/g1/g1CollectorPolicy.hpp" +#include "gc/g1/g1Policy.hpp" +#include "gc/g1/heterogeneousHeapRegionManager.hpp" + +class G1HeterogeneousHeapPolicy : public G1Policy { + // Stash a pointer to the hrm. + HeterogeneousHeapRegionManager* _manager; + +public: + G1HeterogeneousHeapPolicy(G1CollectorPolicy* policy, STWGCTimer* gc_timer); + + // initialize policy + virtual void init(G1CollectedHeap* g1h, G1CollectionSet* collection_set); + // Record end of an evacuation pause. + virtual void record_collection_pause_end(double pause_time_ms, size_t cards_scanned, size_t heap_used_bytes_before_gc); + // Record the end of full collection. + virtual void record_full_collection_end(); + + virtual bool force_upgrade_to_full(); +}; +#endif // SHARE_VM_GC_G1_G1HETEROGENEOUSHEAPPOLICY_HPP diff --git a/src/hotspot/share/gc/g1/g1HeterogeneousHeapYoungGenSizer.cpp b/src/hotspot/share/gc/g1/g1HeterogeneousHeapYoungGenSizer.cpp new file mode 100644 index 00000000000..e516815c1ad --- /dev/null +++ b/src/hotspot/share/gc/g1/g1HeterogeneousHeapYoungGenSizer.cpp @@ -0,0 +1,51 @@ +/* + * Copyright (c) 2018, 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. + * + */ + +#include "precompiled.hpp" +#include "gc/g1/g1HeterogeneousCollectorPolicy.hpp" +#include "gc/g1/g1HeterogeneousHeapYoungGenSizer.hpp" +#include "gc/g1/heapRegion.hpp" + +G1HeterogeneousHeapYoungGenSizer::G1HeterogeneousHeapYoungGenSizer() : G1YoungGenSizer() { + // will be used later when min and max young size is calculated. + _max_young_length = (uint)(G1HeterogeneousCollectorPolicy::reasonable_max_memory_for_young() / HeapRegion::GrainBytes); +} + +// Since heap is sized potentially to larger value accounting for dram + nvdimm, we need to limit +// max young gen size to the available dram. +// Call parent class method first and then adjust sizes based on available dram +void G1HeterogeneousHeapYoungGenSizer::adjust_max_new_size(uint number_of_heap_regions) { + G1YoungGenSizer::adjust_max_new_size(number_of_heap_regions); + adjust_lengths_based_on_dram_memory(); +} + +void G1HeterogeneousHeapYoungGenSizer::heap_size_changed(uint new_number_of_heap_regions) { + G1YoungGenSizer::heap_size_changed(new_number_of_heap_regions); + adjust_lengths_based_on_dram_memory(); +} + +void G1HeterogeneousHeapYoungGenSizer::adjust_lengths_based_on_dram_memory() { + _min_desired_young_length = MIN2(_min_desired_young_length, _max_young_length); + _max_desired_young_length = MIN2(_max_desired_young_length, _max_young_length); +} diff --git a/src/hotspot/share/gc/g1/g1HeterogeneousHeapYoungGenSizer.hpp b/src/hotspot/share/gc/g1/g1HeterogeneousHeapYoungGenSizer.hpp new file mode 100644 index 00000000000..329b124584b --- /dev/null +++ b/src/hotspot/share/gc/g1/g1HeterogeneousHeapYoungGenSizer.hpp @@ -0,0 +1,51 @@ +/* + * Copyright (c) 2018, 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. + * + */ + +#ifndef SHARE_VM_GC_G1_G1HETEROGENEOUSHEAPYOUNGGENSIZER_HPP +#define SHARE_VM_GC_G1_G1HETEROGENEOUSHEAPYOUNGGENSIZER_HPP + +#include "gc/g1/g1YoungGenSizer.hpp" + +// This class prevents the size of young generation of G1 heap to exceed dram +// memory available. If set on command line, MaxRAM and MaxRAMFraction/MaxRAMPercentage +// are used to determine the maximum size that young generation can grow. +// Else we set the maximum size to 80% of dram available in the system. + +class G1HeterogeneousHeapYoungGenSizer : public G1YoungGenSizer { +private: + // maximum no of regions that young generation can grow to. Calculated in constructor. + uint _max_young_length; + void adjust_lengths_based_on_dram_memory(); + +public: + G1HeterogeneousHeapYoungGenSizer(); + + // Calculate the maximum length of the young gen given the number of regions + // depending on the sizing algorithm. + virtual void adjust_max_new_size(uint number_of_heap_regions); + + virtual void heap_size_changed(uint new_number_of_heap_regions); +}; + +#endif // SHARE_VM_GC_G1_G1HETEROGENEOUSHEAPYOUNGGENSIZER_HPP diff --git a/src/hotspot/share/gc/g1/g1PageBasedVirtualSpace.cpp b/src/hotspot/share/gc/g1/g1PageBasedVirtualSpace.cpp index 9120f21e50c..fc30400f28e 100644 --- a/src/hotspot/share/gc/g1/g1PageBasedVirtualSpace.cpp +++ b/src/hotspot/share/gc/g1/g1PageBasedVirtualSpace.cpp @@ -100,6 +100,12 @@ size_t G1PageBasedVirtualSpace::uncommitted_size() const { return reserved_size() - committed_size(); } +void G1PageBasedVirtualSpace::commit_and_set_special() { + commit_internal(addr_to_page_index(_low_boundary), addr_to_page_index(_high_boundary)); + _special = true; + _dirty.initialize(reserved_size()/_page_size); +} + size_t G1PageBasedVirtualSpace::addr_to_page_index(char* addr) const { return (addr - _low_boundary) / _page_size; } diff --git a/src/hotspot/share/gc/g1/g1PageBasedVirtualSpace.hpp b/src/hotspot/share/gc/g1/g1PageBasedVirtualSpace.hpp index 538467992ac..5792ca56f52 100644 --- a/src/hotspot/share/gc/g1/g1PageBasedVirtualSpace.hpp +++ b/src/hotspot/share/gc/g1/g1PageBasedVirtualSpace.hpp @@ -136,6 +136,8 @@ class G1PageBasedVirtualSpace { // Memory left to use/expand in this virtual space. size_t uncommitted_size() const; + void commit_and_set_special(); + bool contains(const void* p) const; MemRegion reserved() { diff --git a/src/hotspot/share/gc/g1/g1Policy.cpp b/src/hotspot/share/gc/g1/g1Policy.cpp index 36ec52dd24a..31887f9e4da 100644 --- a/src/hotspot/share/gc/g1/g1Policy.cpp +++ b/src/hotspot/share/gc/g1/g1Policy.cpp @@ -29,6 +29,7 @@ #include "gc/g1/g1ConcurrentMark.hpp" #include "gc/g1/g1ConcurrentMarkThread.inline.hpp" #include "gc/g1/g1ConcurrentRefine.hpp" +#include "gc/g1/g1HeterogeneousHeapPolicy.hpp" #include "gc/g1/g1HotCardCache.hpp" #include "gc/g1/g1IHOPControl.hpp" #include "gc/g1/g1GCPhaseTimes.hpp" @@ -46,7 +47,7 @@ #include "utilities/growableArray.hpp" #include "utilities/pair.hpp" -G1Policy::G1Policy(STWGCTimer* gc_timer) : +G1Policy::G1Policy(G1CollectorPolicy* policy, STWGCTimer* gc_timer) : _predictor(G1ConfidencePercent / 100.0), _analytics(new G1Analytics(&_predictor)), _remset_tracker(), @@ -62,7 +63,7 @@ G1Policy::G1Policy(STWGCTimer* gc_timer) : _survivor_surv_rate_group(new SurvRateGroup()), _reserve_factor((double) G1ReservePercent / 100.0), _reserve_regions(0), - _young_gen_sizer(), + _young_gen_sizer(G1YoungGenSizer::create_gen_sizer(policy)), _free_regions_at_end_of_collection(0), _max_rs_lengths(0), _rs_lengths_prediction(0), @@ -83,6 +84,15 @@ G1Policy::G1Policy(STWGCTimer* gc_timer) : G1Policy::~G1Policy() { delete _ihop_control; + delete _young_gen_sizer; +} + +G1Policy* G1Policy::create_policy(G1CollectorPolicy* policy, STWGCTimer* gc_timer_stw) { + if (policy->is_hetero_heap()) { + return new G1HeterogeneousHeapPolicy(policy, gc_timer_stw); + } else { + return new G1Policy(policy, gc_timer_stw); + } } G1CollectorState* G1Policy::collector_state() const { return _g1h->collector_state(); } @@ -94,9 +104,9 @@ void G1Policy::init(G1CollectedHeap* g1h, G1CollectionSet* collection_set) { assert(Heap_lock->owned_by_self(), "Locking discipline."); if (!adaptive_young_list_length()) { - _young_list_fixed_length = _young_gen_sizer.min_desired_young_length(); + _young_list_fixed_length = _young_gen_sizer->min_desired_young_length(); } - _young_gen_sizer.adjust_max_new_size(_g1h->max_regions()); + _young_gen_sizer->adjust_max_new_size(_g1h->max_expandable_regions()); _free_regions_at_end_of_collection = _g1h->num_free_regions(); @@ -176,7 +186,7 @@ void G1Policy::record_new_heap_size(uint new_number_of_regions) { // smaller than 1.0) we'll get 1. _reserve_regions = (uint) ceil(reserve_regions_d); - _young_gen_sizer.heap_size_changed(new_number_of_regions); + _young_gen_sizer->heap_size_changed(new_number_of_regions); _ihop_control->update_target_occupancy(new_number_of_regions * HeapRegion::GrainBytes); } @@ -195,14 +205,14 @@ uint G1Policy::calculate_young_list_desired_min_length(uint base_min_length) con } desired_min_length += base_min_length; // make sure we don't go below any user-defined minimum bound - return MAX2(_young_gen_sizer.min_desired_young_length(), desired_min_length); + return MAX2(_young_gen_sizer->min_desired_young_length(), desired_min_length); } uint G1Policy::calculate_young_list_desired_max_length() const { // Here, we might want to also take into account any additional // constraints (i.e., user-defined minimum bound). Currently, we // effectively don't set this bound. - return _young_gen_sizer.max_desired_young_length(); + return _young_gen_sizer->max_desired_young_length(); } uint G1Policy::update_young_list_max_and_target_length() { @@ -218,6 +228,7 @@ uint G1Policy::update_young_list_max_and_target_length(size_t rs_lengths) { uint G1Policy::update_young_list_target_length(size_t rs_lengths) { YoungTargetLengths young_lengths = young_list_target_lengths(rs_lengths); _young_list_target_length = young_lengths.first; + return young_lengths.second; } @@ -900,7 +911,7 @@ bool G1Policy::can_expand_young_list() const { } bool G1Policy::adaptive_young_list_length() const { - return _young_gen_sizer.adaptive_young_list_length(); + return _young_gen_sizer->adaptive_young_list_length(); } size_t G1Policy::desired_survivor_size(uint max_regions) const { diff --git a/src/hotspot/share/gc/g1/g1Policy.hpp b/src/hotspot/share/gc/g1/g1Policy.hpp index e4f752c5446..3c75f3b502d 100644 --- a/src/hotspot/share/gc/g1/g1Policy.hpp +++ b/src/hotspot/share/gc/g1/g1Policy.hpp @@ -25,6 +25,7 @@ #ifndef SHARE_VM_GC_G1_G1POLICY_HPP #define SHARE_VM_GC_G1_G1POLICY_HPP +#include "gc/g1/g1CollectorPolicy.hpp" #include "gc/g1/g1CollectorState.hpp" #include "gc/g1/g1GCPhaseTimes.hpp" #include "gc/g1/g1InCSetState.hpp" @@ -91,7 +92,7 @@ class G1Policy: public CHeapObj { // for the first time during initialization. uint _reserve_regions; - G1YoungGenSizer _young_gen_sizer; + G1YoungGenSizer* _young_gen_sizer; uint _free_regions_at_end_of_collection; @@ -282,10 +283,12 @@ private: void abort_time_to_mixed_tracking(); public: - G1Policy(STWGCTimer* gc_timer); + G1Policy(G1CollectorPolicy* policy, STWGCTimer* gc_timer); virtual ~G1Policy(); + static G1Policy* create_policy(G1CollectorPolicy* policy, STWGCTimer* gc_timer_stw); + G1CollectorState* collector_state() const; G1GCPhaseTimes* phase_times() const { return _phase_times; } @@ -298,7 +301,7 @@ public: // This should be called after the heap is resized. void record_new_heap_size(uint new_number_of_regions); - void init(G1CollectedHeap* g1h, G1CollectionSet* collection_set); + virtual void init(G1CollectedHeap* g1h, G1CollectionSet* collection_set); void note_gc_start(); @@ -308,11 +311,11 @@ public: // Record the start and end of an evacuation pause. void record_collection_pause_start(double start_time_sec); - void record_collection_pause_end(double pause_time_ms, size_t cards_scanned, size_t heap_used_bytes_before_gc); + virtual void record_collection_pause_end(double pause_time_ms, size_t cards_scanned, size_t heap_used_bytes_before_gc); // Record the start and end of a full collection. void record_full_collection_start(); - void record_full_collection_end(); + virtual void record_full_collection_end(); // Must currently be called while the world is stopped. void record_concurrent_mark_init_end(double mark_init_elapsed_time_ms); @@ -432,6 +435,10 @@ public: void update_max_gc_locker_expansion(); void update_survivors_policy(); + + virtual bool force_upgrade_to_full() { + return false; + } }; #endif // SHARE_VM_GC_G1_G1POLICY_HPP diff --git a/src/hotspot/share/gc/g1/g1RegionToSpaceMapper.cpp b/src/hotspot/share/gc/g1/g1RegionToSpaceMapper.cpp index dfbb67f42a0..a5fe219f994 100644 --- a/src/hotspot/share/gc/g1/g1RegionToSpaceMapper.cpp +++ b/src/hotspot/share/gc/g1/g1RegionToSpaceMapper.cpp @@ -1,5 +1,5 @@ /* - * Copyright (c) 2001, 2017, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2001, 2018, 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 @@ -25,11 +25,15 @@ #include "precompiled.hpp" #include "gc/g1/g1BiasedArray.hpp" #include "gc/g1/g1RegionToSpaceMapper.hpp" +#include "logging/log.hpp" #include "memory/allocation.inline.hpp" #include "memory/virtualspace.hpp" +#include "runtime/java.hpp" +#include "runtime/os.inline.hpp" #include "services/memTracker.hpp" #include "utilities/align.hpp" #include "utilities/bitMap.inline.hpp" +#include "utilities/formatBuffer.hpp" G1RegionToSpaceMapper::G1RegionToSpaceMapper(ReservedSpace rs, size_t used_size, @@ -170,16 +174,156 @@ void G1RegionToSpaceMapper::fire_on_commit(uint start_idx, size_t num_regions, b } } +static bool map_nvdimm_space(ReservedSpace rs) { + assert(AllocateOldGenAt != NULL, ""); + int _backing_fd = os::create_file_for_heap(AllocateOldGenAt); + if (_backing_fd == -1) { + log_error(gc, init)("Could not create file for Old generation at location %s", AllocateOldGenAt); + return false; + } + // commit this memory in nv-dimm + char* ret = os::attempt_reserve_memory_at(rs.size(), rs.base(), _backing_fd); + + if (ret != rs.base()) { + if (ret != NULL) { + os::unmap_memory(rs.base(), rs.size()); + } + log_error(gc, init)("Error in mapping Old Gen to given AllocateOldGenAt = %s", AllocateOldGenAt); + os::close(_backing_fd); + return false; + } + + os::close(_backing_fd); + return true; +} + +G1RegionToHeteroSpaceMapper::G1RegionToHeteroSpaceMapper(ReservedSpace rs, + size_t actual_size, + size_t page_size, + size_t alloc_granularity, + size_t commit_factor, + MemoryType type) : + G1RegionToSpaceMapper(rs, actual_size, page_size, alloc_granularity, commit_factor, type), + _rs(rs), + _num_committed_dram(0), + _num_committed_nvdimm(0), + _page_size(page_size), + _commit_factor(commit_factor), + _type(type) { + assert(actual_size == 2 * MaxHeapSize, "For 2-way heterogenuous heap, reserved space is two times MaxHeapSize"); +} + +bool G1RegionToHeteroSpaceMapper::initialize() { + // Since we need to re-map the reserved space - 'Xmx' to nv-dimm and 'Xmx' to dram, we need to release the reserved memory first. + // Because on some OSes (e.g. Windows) you cannot do a file mapping on memory reserved with regular mapping. + os::release_memory(_rs.base(), _rs.size()); + // First half of size Xmx is for nv-dimm. + ReservedSpace rs_nvdimm = _rs.first_part(MaxHeapSize); + assert(rs_nvdimm.base() == _rs.base(), "We should get the same base address"); + + // Second half of reserved memory is mapped to dram. + ReservedSpace rs_dram = _rs.last_part(MaxHeapSize); + + assert(rs_dram.size() == rs_nvdimm.size() && rs_nvdimm.size() == MaxHeapSize, "They all should be same"); + + // Reserve dram memory + char* base = os::attempt_reserve_memory_at(rs_dram.size(), rs_dram.base()); + if (base != rs_dram.base()) { + if (base != NULL) { + os::release_memory(base, rs_dram.size()); + } + log_error(gc, init)("Error in re-mapping memory on dram during G1 heterogenous memory initialization"); + return false; + } + + // We reserve and commit this entire space to NV-DIMM. + if (!map_nvdimm_space(rs_nvdimm)) { + log_error(gc, init)("Error in re-mapping memory to nv-dimm during G1 heterogenous memory initialization"); + return false; + } + + if (_region_granularity >= (_page_size * _commit_factor)) { + _dram_mapper = new G1RegionsLargerThanCommitSizeMapper(rs_dram, rs_dram.size(), _page_size, _region_granularity, _commit_factor, _type); + } else { + _dram_mapper = new G1RegionsSmallerThanCommitSizeMapper(rs_dram, rs_dram.size(), _page_size, _region_granularity, _commit_factor, _type); + } + + _start_index_of_nvdimm = 0; + _start_index_of_dram = (uint)(rs_nvdimm.size() / _region_granularity); + return true; +} + +void G1RegionToHeteroSpaceMapper::commit_regions(uint start_idx, size_t num_regions, WorkGang* pretouch_gang) { + uint end_idx = (start_idx + (uint)num_regions - 1); + + uint num_dram = end_idx >= _start_index_of_dram ? MIN2((end_idx - _start_index_of_dram + 1), (uint)num_regions) : 0; + uint num_nvdimm = (uint)num_regions - num_dram; + + if (num_nvdimm > 0) { + // We do not need to commit nv-dimm regions, since they are committed in the beginning. + _num_committed_nvdimm += num_nvdimm; + } + if (num_dram > 0) { + _dram_mapper->commit_regions(start_idx > _start_index_of_dram ? (start_idx - _start_index_of_dram) : 0, num_dram, pretouch_gang); + _num_committed_dram += num_dram; + } +} + +void G1RegionToHeteroSpaceMapper::uncommit_regions(uint start_idx, size_t num_regions) { + uint end_idx = (start_idx + (uint)num_regions - 1); + uint num_dram = end_idx >= _start_index_of_dram ? MIN2((end_idx - _start_index_of_dram + 1), (uint)num_regions) : 0; + uint num_nvdimm = (uint)num_regions - num_dram; + + if (num_nvdimm > 0) { + // We do not uncommit memory for nv-dimm regions. + _num_committed_nvdimm -= num_nvdimm; + } + + if (num_dram > 0) { + _dram_mapper->uncommit_regions(start_idx > _start_index_of_dram ? (start_idx - _start_index_of_dram) : 0, num_dram); + _num_committed_dram -= num_dram; + } +} + +uint G1RegionToHeteroSpaceMapper::num_committed_dram() const { + return _num_committed_dram; +} + +uint G1RegionToHeteroSpaceMapper::num_committed_nvdimm() const { + return _num_committed_nvdimm; +} + +G1RegionToSpaceMapper* G1RegionToSpaceMapper::create_heap_mapper(ReservedSpace rs, + size_t actual_size, + size_t page_size, + size_t region_granularity, + size_t commit_factor, + MemoryType type) { + if (AllocateOldGenAt != NULL) { + G1RegionToHeteroSpaceMapper* mapper = new G1RegionToHeteroSpaceMapper(rs, actual_size, page_size, region_granularity, commit_factor, type); + if (!mapper->initialize()) { + delete mapper; + return NULL; + } + return (G1RegionToSpaceMapper*)mapper; + } else { + return create_mapper(rs, actual_size, page_size, region_granularity, commit_factor, type); + } +} + G1RegionToSpaceMapper* G1RegionToSpaceMapper::create_mapper(ReservedSpace rs, size_t actual_size, size_t page_size, size_t region_granularity, size_t commit_factor, MemoryType type) { - if (region_granularity >= (page_size * commit_factor)) { return new G1RegionsLargerThanCommitSizeMapper(rs, actual_size, page_size, region_granularity, commit_factor, type); } else { return new G1RegionsSmallerThanCommitSizeMapper(rs, actual_size, page_size, region_granularity, commit_factor, type); } } + +void G1RegionToSpaceMapper::commit_and_set_special() { + _storage.commit_and_set_special(); +} diff --git a/src/hotspot/share/gc/g1/g1RegionToSpaceMapper.hpp b/src/hotspot/share/gc/g1/g1RegionToSpaceMapper.hpp index 0a5227c770b..53bb04f3fb6 100644 --- a/src/hotspot/share/gc/g1/g1RegionToSpaceMapper.hpp +++ b/src/hotspot/share/gc/g1/g1RegionToSpaceMapper.hpp @@ -70,6 +70,7 @@ class G1RegionToSpaceMapper : public CHeapObj { return _commit_map.at(idx); } + void commit_and_set_special(); virtual void commit_regions(uint start_idx, size_t num_regions = 1, WorkGang* pretouch_workers = NULL) = 0; virtual void uncommit_regions(uint start_idx, size_t num_regions = 1) = 0; @@ -87,6 +88,37 @@ class G1RegionToSpaceMapper : public CHeapObj { size_t region_granularity, size_t byte_translation_factor, MemoryType type); + + static G1RegionToSpaceMapper* create_heap_mapper(ReservedSpace rs, + size_t actual_size, + size_t page_size, + size_t region_granularity, + size_t byte_translation_factor, + MemoryType type); }; +// G1RegionToSpaceMapper implementation where +// part of space is mapped to dram and part to nv-dimm +class G1RegionToHeteroSpaceMapper : public G1RegionToSpaceMapper { +private: + size_t _pages_per_region; + ReservedSpace _rs; + G1RegionToSpaceMapper* _dram_mapper; + uint _num_committed_dram; + uint _num_committed_nvdimm; + uint _start_index_of_nvdimm; + uint _start_index_of_dram; + size_t _page_size; + size_t _commit_factor; + MemoryType _type; + +public: + G1RegionToHeteroSpaceMapper(ReservedSpace rs, size_t used_size, size_t page_size, size_t region_granularity, size_t commit_factor, MemoryType type); + bool initialize(); + uint num_committed_dram() const; + uint num_committed_nvdimm() const; + + virtual void commit_regions(uint start_idx, size_t num_regions = 1, WorkGang* pretouch_workers = NULL); + virtual void uncommit_regions(uint start_idx, size_t num_regions = 1); +}; #endif // SHARE_VM_GC_G1_G1REGIONTOSPACEMAPPER_HPP diff --git a/src/hotspot/share/gc/g1/g1VMOperations.cpp b/src/hotspot/share/gc/g1/g1VMOperations.cpp index 6b521181c31..9c29a16d7c9 100644 --- a/src/hotspot/share/gc/g1/g1VMOperations.cpp +++ b/src/hotspot/share/gc/g1/g1VMOperations.cpp @@ -138,8 +138,8 @@ void VM_G1CollectForAllocation::doit() { // kind of GC. _result = g1h->satisfy_failed_allocation(_word_size, &_pause_succeeded); } else { - bool should_upgrade_to_full = !g1h->should_do_concurrent_full_gc(_gc_cause) && - !g1h->has_regions_left_for_allocation(); + bool should_upgrade_to_full = g1h->should_upgrade_to_full_gc(_gc_cause); + if (should_upgrade_to_full) { // There has been a request to perform a GC to free some space. We have no // information on how much memory has been asked for. In case there are diff --git a/src/hotspot/share/gc/g1/g1YoungGenSizer.cpp b/src/hotspot/share/gc/g1/g1YoungGenSizer.cpp index 46bcdb58199..954d8bb7efe 100644 --- a/src/hotspot/share/gc/g1/g1YoungGenSizer.cpp +++ b/src/hotspot/share/gc/g1/g1YoungGenSizer.cpp @@ -1,5 +1,5 @@ /* - * Copyright (c) 2016, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2016, 2018, 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 @@ -23,12 +23,14 @@ */ #include "precompiled.hpp" +#include "gc/g1/g1CollectorPolicy.hpp" +#include "gc/g1/g1HeterogeneousHeapYoungGenSizer.hpp" #include "gc/g1/g1YoungGenSizer.hpp" #include "gc/g1/heapRegion.hpp" #include "logging/log.hpp" G1YoungGenSizer::G1YoungGenSizer() : _sizer_kind(SizerDefaults), - _min_desired_young_length(0), _max_desired_young_length(0), _adaptive_size(true) { + _adaptive_size(true), _min_desired_young_length(0), _max_desired_young_length(0) { if (FLAG_IS_CMDLINE(NewRatio)) { if (FLAG_IS_CMDLINE(NewSize) || FLAG_IS_CMDLINE(MaxNewSize)) { @@ -127,3 +129,11 @@ void G1YoungGenSizer::heap_size_changed(uint new_number_of_heap_regions) { recalculate_min_max_young_length(new_number_of_heap_regions, &_min_desired_young_length, &_max_desired_young_length); } + +G1YoungGenSizer* G1YoungGenSizer::create_gen_sizer(G1CollectorPolicy* policy) { + if (policy->is_hetero_heap()) { + return new G1HeterogeneousHeapYoungGenSizer(); + } else { + return new G1YoungGenSizer(); + } +} diff --git a/src/hotspot/share/gc/g1/g1YoungGenSizer.hpp b/src/hotspot/share/gc/g1/g1YoungGenSizer.hpp index ddd26afbbef..0be87271a43 100644 --- a/src/hotspot/share/gc/g1/g1YoungGenSizer.hpp +++ b/src/hotspot/share/gc/g1/g1YoungGenSizer.hpp @@ -25,6 +25,7 @@ #ifndef SHARE_VM_GC_G1_G1YOUNGGENSIZER_HPP #define SHARE_VM_GC_G1_G1YOUNGGENSIZER_HPP +#include "gc/g1/g1CollectorPolicy.hpp" #include "utilities/globalDefinitions.hpp" // There are three command line options related to the young gen size: @@ -63,7 +64,7 @@ // // NewSize and MaxNewSize override NewRatio. So, NewRatio is ignored if it is // combined with either NewSize or MaxNewSize. (A warning message is printed.) -class G1YoungGenSizer { +class G1YoungGenSizer : public CHeapObj { private: enum SizerKind { SizerDefaults, @@ -73,8 +74,6 @@ private: SizerNewRatio }; SizerKind _sizer_kind; - uint _min_desired_young_length; - uint _max_desired_young_length; // False when using a fixed young generation size due to command-line options, // true otherwise. @@ -87,13 +86,17 @@ private: // given the number of heap regions depending on the kind of sizing algorithm. void recalculate_min_max_young_length(uint number_of_heap_regions, uint* min_young_length, uint* max_young_length); +protected: + uint _min_desired_young_length; + uint _max_desired_young_length; + public: G1YoungGenSizer(); // Calculate the maximum length of the young gen given the number of regions // depending on the sizing algorithm. - void adjust_max_new_size(uint number_of_heap_regions); + virtual void adjust_max_new_size(uint number_of_heap_regions); - void heap_size_changed(uint new_number_of_heap_regions); + virtual void heap_size_changed(uint new_number_of_heap_regions); uint min_desired_young_length() const { return _min_desired_young_length; } @@ -104,6 +107,8 @@ public: bool adaptive_young_list_length() const { return _adaptive_size; } + + static G1YoungGenSizer* create_gen_sizer(G1CollectorPolicy* policy); }; #endif // SHARE_VM_GC_G1_G1YOUNGGENSIZER_HPP diff --git a/src/hotspot/share/gc/g1/g1_globals.hpp b/src/hotspot/share/gc/g1/g1_globals.hpp index a4ec8770849..d98e69404a1 100644 --- a/src/hotspot/share/gc/g1/g1_globals.hpp +++ b/src/hotspot/share/gc/g1/g1_globals.hpp @@ -317,5 +317,15 @@ "above this value cancels a given periodic GC. A value of zero " \ "disables this check.") \ range(0.0, (double)max_uintx) \ + \ + experimental(uintx, G1YoungExpansionBufferPercent, 10, \ + "When heterogenous heap is enabled by AllocateOldGenAt " \ + "option, after every GC, young gen is re-sized which " \ + "involves system calls to commit/uncommit memory. To " \ + "reduce these calls, we keep a buffer of extra regions to " \ + "absorb small changes in young gen length. This flag takes " \ + "the buffer size as an percentage of young gen length") \ + range(0, 100) \ + #endif // SHARE_VM_GC_G1_G1_GLOBALS_HPP diff --git a/src/hotspot/share/gc/g1/heapRegionManager.cpp b/src/hotspot/share/gc/g1/heapRegionManager.cpp index 178e0cac6a8..fde04893f19 100644 --- a/src/hotspot/share/gc/g1/heapRegionManager.cpp +++ b/src/hotspot/share/gc/g1/heapRegionManager.cpp @@ -28,6 +28,8 @@ #include "gc/g1/heapRegion.hpp" #include "gc/g1/heapRegionManager.inline.hpp" #include "gc/g1/heapRegionSet.inline.hpp" +#include "gc/g1/heterogeneousHeapRegionManager.hpp" +#include "gc/shared/collectorPolicy.hpp" #include "memory/allocation.hpp" #include "utilities/bitMap.inline.hpp" @@ -54,18 +56,25 @@ public: }; HeapRegionManager::HeapRegionManager() : - _regions(), _heap_mapper(NULL), - _prev_bitmap_mapper(NULL), - _next_bitmap_mapper(NULL), _bot_mapper(NULL), _cardtable_mapper(NULL), _card_counts_mapper(NULL), - _free_list("Free list", new MasterFreeRegionListChecker()), _available_map(mtGC), _num_committed(0), - _allocated_heapregions_length(0) + _allocated_heapregions_length(0), + _regions(), _heap_mapper(NULL), + _prev_bitmap_mapper(NULL), + _next_bitmap_mapper(NULL), + _free_list("Free list", new MasterFreeRegionListChecker()) { } +HeapRegionManager* HeapRegionManager::create_manager(G1CollectedHeap* heap, G1CollectorPolicy* policy) { + if (policy->is_hetero_heap()) { + return new HeterogeneousHeapRegionManager((uint)(policy->max_heap_byte_size() / HeapRegion::GrainBytes) /*heap size as num of regions*/); + } + return new HeapRegionManager(); +} + void HeapRegionManager::initialize(G1RegionToSpaceMapper* heap_storage, G1RegionToSpaceMapper* prev_bitmap, G1RegionToSpaceMapper* next_bitmap, @@ -514,7 +523,7 @@ void HeapRegionManager::verify_optional() { #endif // PRODUCT HeapRegionClaimer::HeapRegionClaimer(uint n_workers) : - _n_workers(n_workers), _n_regions(G1CollectedHeap::heap()->_hrm._allocated_heapregions_length), _claims(NULL) { + _n_workers(n_workers), _n_regions(G1CollectedHeap::heap()->_hrm->_allocated_heapregions_length), _claims(NULL) { assert(n_workers > 0, "Need at least one worker."); uint* new_claims = NEW_C_HEAP_ARRAY(uint, _n_regions, mtGC); memset(new_claims, Unclaimed, sizeof(*_claims) * _n_regions); diff --git a/src/hotspot/share/gc/g1/heapRegionManager.hpp b/src/hotspot/share/gc/g1/heapRegionManager.hpp index 90578ac1703..8c2389ad461 100644 --- a/src/hotspot/share/gc/g1/heapRegionManager.hpp +++ b/src/hotspot/share/gc/g1/heapRegionManager.hpp @@ -26,8 +26,10 @@ #define SHARE_VM_GC_G1_HEAPREGIONMANAGER_HPP #include "gc/g1/g1BiasedArray.hpp" +#include "gc/g1/g1CollectorPolicy.hpp" #include "gc/g1/g1RegionToSpaceMapper.hpp" #include "gc/g1/heapRegionSet.hpp" +#include "gc/shared/collectorPolicy.hpp" #include "services/memoryUsage.hpp" class HeapRegion; @@ -71,17 +73,10 @@ class HeapRegionManager: public CHeapObj { friend class VMStructs; friend class HeapRegionClaimer; - G1HeapRegionTable _regions; - - G1RegionToSpaceMapper* _heap_mapper; - G1RegionToSpaceMapper* _prev_bitmap_mapper; - G1RegionToSpaceMapper* _next_bitmap_mapper; G1RegionToSpaceMapper* _bot_mapper; G1RegionToSpaceMapper* _cardtable_mapper; G1RegionToSpaceMapper* _card_counts_mapper; - FreeRegionList _free_list; - // Each bit in this bitmap indicates that the corresponding region is available // for allocation. CHeapBitMap _available_map; @@ -95,11 +90,8 @@ class HeapRegionManager: public CHeapObj { HeapWord* heap_bottom() const { return _regions.bottom_address_mapped(); } HeapWord* heap_end() const {return _regions.end_address_mapped(); } - void make_regions_available(uint index, uint num_regions = 1, WorkGang* pretouch_gang = NULL); - // Pass down commit calls to the VirtualSpace. void commit_regions(uint index, size_t num_regions = 1, WorkGang* pretouch_gang = NULL); - void uncommit_regions(uint index, size_t num_regions = 1); // Notify other data structures about change in the heap layout. void update_committed_space(HeapWord* old_end, HeapWord* new_end); @@ -117,6 +109,16 @@ class HeapRegionManager: public CHeapObj { // the heap. Returns the length of the sequence found. If this value is zero, no // sequence could be found, otherwise res_idx contains the start index of this range. uint find_empty_from_idx_reverse(uint start_idx, uint* res_idx) const; + +protected: + G1HeapRegionTable _regions; + G1RegionToSpaceMapper* _heap_mapper; + G1RegionToSpaceMapper* _prev_bitmap_mapper; + G1RegionToSpaceMapper* _next_bitmap_mapper; + FreeRegionList _free_list; + + void make_regions_available(uint index, uint num_regions = 1, WorkGang* pretouch_gang = NULL); + void uncommit_regions(uint index, size_t num_regions = 1); // Allocate a new HeapRegion for the given index. HeapRegion* new_heap_region(uint hrm_index); #ifdef ASSERT @@ -127,18 +129,25 @@ public: // Empty constructor, we'll initialize it with the initialize() method. HeapRegionManager(); - void initialize(G1RegionToSpaceMapper* heap_storage, - G1RegionToSpaceMapper* prev_bitmap, - G1RegionToSpaceMapper* next_bitmap, - G1RegionToSpaceMapper* bot, - G1RegionToSpaceMapper* cardtable, - G1RegionToSpaceMapper* card_counts); + static HeapRegionManager* create_manager(G1CollectedHeap* heap, G1CollectorPolicy* policy); + + virtual void initialize(G1RegionToSpaceMapper* heap_storage, + G1RegionToSpaceMapper* prev_bitmap, + G1RegionToSpaceMapper* next_bitmap, + G1RegionToSpaceMapper* bot, + G1RegionToSpaceMapper* cardtable, + G1RegionToSpaceMapper* card_counts); + + // Prepare heap regions before and after full collection. + // Nothing to be done in this class. + virtual void prepare_for_full_collection_start() {} + virtual void prepare_for_full_collection_end() {} // Return the "dummy" region used for G1AllocRegion. This is currently a hardwired // new HeapRegion that owns HeapRegion at index 0. Since at the moment we commit // the heap from the lowest address, this region (and its associated data // structures) are available and we do not need to check further. - HeapRegion* get_dummy_region() { return new_heap_region(0); } + virtual HeapRegion* get_dummy_region() { return new_heap_region(0); } // Return the HeapRegion at the given index. Assume that the index // is valid. @@ -167,8 +176,8 @@ public: _free_list.add_ordered(list); } - HeapRegion* allocate_free_region(bool is_old) { - HeapRegion* hr = _free_list.remove_region(is_old); + virtual HeapRegion* allocate_free_region(HeapRegionType type) { + HeapRegion* hr = _free_list.remove_region(!type.is_young()); if (hr != NULL) { assert(hr->next() == NULL, "Single region should not have next"); @@ -202,6 +211,9 @@ public: // Return the maximum number of regions in the heap. uint max_length() const { return (uint)_regions.length(); } + // Return maximum number of regions that heap can expand to. + virtual uint max_expandable_length() const { return (uint)_regions.length(); } + MemoryUsage get_auxiliary_data_memory_usage() const; MemRegion reserved() const { return MemRegion(heap_bottom(), heap_end()); } @@ -210,26 +222,26 @@ public: // HeapRegions, or re-use existing ones. Returns the number of regions the // sequence was expanded by. If a HeapRegion allocation fails, the resulting // number of regions might be smaller than what's desired. - uint expand_by(uint num_regions, WorkGang* pretouch_workers); + virtual uint expand_by(uint num_regions, WorkGang* pretouch_workers); // Makes sure that the regions from start to start+num_regions-1 are available // for allocation. Returns the number of regions that were committed to achieve // this. - uint expand_at(uint start, uint num_regions, WorkGang* pretouch_workers); + virtual uint expand_at(uint start, uint num_regions, WorkGang* pretouch_workers); // Find a contiguous set of empty regions of length num. Returns the start index of // that set, or G1_NO_HRM_INDEX. - uint find_contiguous_only_empty(size_t num) { return find_contiguous(num, true); } + virtual uint find_contiguous_only_empty(size_t num) { return find_contiguous(num, true); } // Find a contiguous set of empty or unavailable regions of length num. Returns the // start index of that set, or G1_NO_HRM_INDEX. - uint find_contiguous_empty_or_unavailable(size_t num) { return find_contiguous(num, false); } + virtual uint find_contiguous_empty_or_unavailable(size_t num) { return find_contiguous(num, false); } HeapRegion* next_region_in_heap(const HeapRegion* r) const; // Find the highest free or uncommitted region in the reserved heap, // and if uncommitted, commit it. If none are available, return G1_NO_HRM_INDEX. // Set the 'expanded' boolean true if a new region was committed. - uint find_highest_free(bool* expanded); + virtual uint find_highest_free(bool* expanded); // Allocate the regions that contain the address range specified, committing the // regions if necessary. Return false if any of the regions is already committed @@ -244,13 +256,13 @@ public: // Uncommit up to num_regions_to_remove regions that are completely free. // Return the actual number of uncommitted regions. - uint shrink_by(uint num_regions_to_remove); + virtual uint shrink_by(uint num_regions_to_remove); // Uncommit a number of regions starting at the specified index, which must be available, // empty, and free. void shrink_at(uint index, size_t num_regions); - void verify(); + virtual void verify(); // Do some sanity checking. void verify_optional() PRODUCT_RETURN; diff --git a/src/hotspot/share/gc/g1/heapRegionSet.cpp b/src/hotspot/share/gc/g1/heapRegionSet.cpp index c296a53400c..873da3330cc 100644 --- a/src/hotspot/share/gc/g1/heapRegionSet.cpp +++ b/src/hotspot/share/gc/g1/heapRegionSet.cpp @@ -234,6 +234,21 @@ void FreeRegionList::remove_starting_at(HeapRegion* first, uint num_regions) { verify_optional(); } +uint FreeRegionList::num_of_regions_in_range(uint start, uint end) const { + HeapRegion* cur = _head; + uint num = 0; + while (cur != NULL) { + uint index = cur->hrm_index(); + if (index > end) { + break; + } else if (index >= start) { + num++; + } + cur = cur->next(); + } + return num; +} + void FreeRegionList::verify() { // See comment in HeapRegionSetBase::verify() about MT safety and // verification. diff --git a/src/hotspot/share/gc/g1/heapRegionSet.hpp b/src/hotspot/share/gc/g1/heapRegionSet.hpp index b693918df46..ff4e35872fc 100644 --- a/src/hotspot/share/gc/g1/heapRegionSet.hpp +++ b/src/hotspot/share/gc/g1/heapRegionSet.hpp @@ -194,6 +194,8 @@ public: void remove_starting_at(HeapRegion* first, uint num_regions); virtual void verify(); + + uint num_of_regions_in_range(uint start, uint end) const; }; // Iterator class that provides a convenient way to iterate over the diff --git a/src/hotspot/share/gc/g1/heapRegionType.cpp b/src/hotspot/share/gc/g1/heapRegionType.cpp index 755e94ba923..35f11950eb9 100644 --- a/src/hotspot/share/gc/g1/heapRegionType.cpp +++ b/src/hotspot/share/gc/g1/heapRegionType.cpp @@ -1,5 +1,5 @@ /* - * Copyright (c) 2014, 2017, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2014, 2018, 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 @@ -26,6 +26,11 @@ #include "gc/g1/g1HeapRegionTraceType.hpp" #include "gc/g1/heapRegionType.hpp" +const HeapRegionType HeapRegionType::Eden = HeapRegionType(EdenTag); +const HeapRegionType HeapRegionType::Survivor = HeapRegionType(SurvTag); +const HeapRegionType HeapRegionType::Old = HeapRegionType(OldTag); +const HeapRegionType HeapRegionType::Humongous = HeapRegionType(StartsHumongousTag); + bool HeapRegionType::is_valid(Tag tag) { switch (tag) { case FreeTag: diff --git a/src/hotspot/share/gc/g1/heapRegionType.hpp b/src/hotspot/share/gc/g1/heapRegionType.hpp index ed6f92f4768..5d15d98b7c5 100644 --- a/src/hotspot/share/gc/g1/heapRegionType.hpp +++ b/src/hotspot/share/gc/g1/heapRegionType.hpp @@ -117,6 +117,9 @@ private: _tag = tag; } + // Private constructor used static constants + HeapRegionType(Tag t) : _tag(t) { hrt_assert_is_valid(_tag); } + public: // Queries @@ -186,6 +189,11 @@ public: G1HeapRegionTraceType::Type get_trace_type(); HeapRegionType() : _tag(FreeTag) { hrt_assert_is_valid(_tag); } + + static const HeapRegionType Eden; + static const HeapRegionType Survivor; + static const HeapRegionType Old; + static const HeapRegionType Humongous; }; #endif // SHARE_VM_GC_G1_HEAPREGIONTYPE_HPP diff --git a/src/hotspot/share/gc/g1/heterogeneousHeapRegionManager.cpp b/src/hotspot/share/gc/g1/heterogeneousHeapRegionManager.cpp new file mode 100644 index 00000000000..56261b6984c --- /dev/null +++ b/src/hotspot/share/gc/g1/heterogeneousHeapRegionManager.cpp @@ -0,0 +1,523 @@ +/* + * Copyright (c) 2018, 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. + * + */ + +#include "precompiled.hpp" +#include "gc/g1/g1CollectedHeap.inline.hpp" +#include "gc/g1/g1ConcurrentRefine.hpp" +#include "gc/g1/heapRegion.hpp" +#include "gc/g1/heapRegionManager.inline.hpp" +#include "gc/g1/heapRegionSet.inline.hpp" +#include "gc/g1/heterogeneousHeapRegionManager.hpp" +#include "memory/allocation.hpp" + + +HeterogeneousHeapRegionManager* HeterogeneousHeapRegionManager::manager() { + G1CollectedHeap* g1h = G1CollectedHeap::heap(); + assert(g1h != NULL, "Uninitialized access to HeterogeneousHeapRegionManager::manager()"); + + HeapRegionManager* hrm = g1h->hrm(); + assert(hrm != NULL, "Uninitialized access to HeterogeneousHeapRegionManager::manager()"); + return (HeterogeneousHeapRegionManager*)hrm; +} + +void HeterogeneousHeapRegionManager::initialize(G1RegionToSpaceMapper* heap_storage, + G1RegionToSpaceMapper* prev_bitmap, + G1RegionToSpaceMapper* next_bitmap, + G1RegionToSpaceMapper* bot, + G1RegionToSpaceMapper* cardtable, + G1RegionToSpaceMapper* card_counts) { + HeapRegionManager::initialize(heap_storage, prev_bitmap, next_bitmap, bot, cardtable, card_counts); + + // We commit bitmap for all regions during initialization and mark the bitmap space as special. + // This allows regions to be un-committed while concurrent-marking threads are accessing the bitmap concurrently. + _prev_bitmap_mapper->commit_and_set_special(); + _next_bitmap_mapper->commit_and_set_special(); +} + +// expand_by() is called to grow the heap. We grow into nvdimm now. +// Dram regions are committed later as needed during mutator region allocation or +// when young list target length is determined after gc cycle. +uint HeterogeneousHeapRegionManager::expand_by(uint num_regions, WorkGang* pretouch_workers) { + uint num_regions_possible = total_regions_committed() >= max_expandable_length() ? 0 : max_expandable_length() - total_regions_committed(); + uint num_expanded = expand_nvdimm(MIN2(num_regions, num_regions_possible), pretouch_workers); + return num_expanded; +} + +// Expands heap starting from 'start' index. The question is should we expand from one memory (e.g. nvdimm) to another (e.g. dram). +// Looking at the code, expand_at() is called for humongous allocation where 'start' is in nv-dimm. +// So we only allocate regions in the same kind of memory as 'start'. +uint HeterogeneousHeapRegionManager::expand_at(uint start, uint num_regions, WorkGang* pretouch_workers) { + if (num_regions == 0) { + return 0; + } + uint target_num_regions = MIN2(num_regions, max_expandable_length() - total_regions_committed()); + uint end = is_in_nvdimm(start) ? end_index_of_nvdimm() : end_index_of_dram(); + + uint num_expanded = expand_in_range(start, end, target_num_regions, pretouch_workers); + assert(total_regions_committed() <= max_expandable_length(), "must be"); + return num_expanded; +} + +// This function ensures that there are 'expected_num_regions' committed regions in dram. +// If new regions are committed, it un-commits that many regions from nv-dimm. +// If there are already more regions committed in dram, extra regions are un-committed. +void HeterogeneousHeapRegionManager::adjust_dram_regions(uint expected_num_regions, WorkGang* pretouch_workers) { + + // Release back the extra regions allocated in evacuation failure scenario. + if(_no_borrowed_regions > 0) { + _no_borrowed_regions -= shrink_dram(_no_borrowed_regions); + _no_borrowed_regions -= shrink_nvdimm(_no_borrowed_regions); + } + + if(expected_num_regions > free_list_dram_length()) { + // If we are going to expand DRAM, we expand a little more so that we can absorb small variations in Young gen sizing. + uint targeted_dram_regions = expected_num_regions * (1 + (double)G1YoungExpansionBufferPercent / 100); + uint to_be_made_available = targeted_dram_regions - free_list_dram_length(); + +#ifdef ASSERT + uint total_committed_before = total_regions_committed(); +#endif + uint can_be_made_available = shrink_nvdimm(to_be_made_available); + uint ret = expand_dram(can_be_made_available, pretouch_workers); +#ifdef ASSERT + assert(ret == can_be_made_available, "should be equal"); + assert(total_committed_before == total_regions_committed(), "invariant not met"); +#endif + } else { + uint to_be_released = free_list_dram_length() - expected_num_regions; + // if number of extra DRAM regions is small, do not shrink. + if (to_be_released < expected_num_regions * G1YoungExpansionBufferPercent / 100) { + return; + } + +#ifdef ASSERT + uint total_committed_before = total_regions_committed(); +#endif + uint ret = shrink_dram(to_be_released); + assert(ret == to_be_released, "Should be able to shrink by given amount"); + ret = expand_nvdimm(to_be_released, pretouch_workers); +#ifdef ASSERT + assert(ret == to_be_released, "Should be able to expand by given amount"); + assert(total_committed_before == total_regions_committed(), "invariant not met"); +#endif + } +} + +uint HeterogeneousHeapRegionManager::total_regions_committed() const { + return num_committed_dram() + num_committed_nvdimm(); +} + +uint HeterogeneousHeapRegionManager::num_committed_dram() const { + // This class does not keep count of committed regions in dram and nv-dimm. + // G1RegionToHeteroSpaceMapper keeps this information. + return static_cast(_heap_mapper)->num_committed_dram(); +} + +uint HeterogeneousHeapRegionManager::num_committed_nvdimm() const { + // See comment for num_committed_dram() + return static_cast(_heap_mapper)->num_committed_nvdimm(); +} + +// Return maximum number of regions that heap can expand to. +uint HeterogeneousHeapRegionManager::max_expandable_length() const { + return _max_regions; +} + +uint HeterogeneousHeapRegionManager::find_unavailable_in_range(uint start_idx, uint end_idx, uint* res_idx) const { + guarantee(res_idx != NULL, "checking"); + guarantee(start_idx <= (max_length() + 1), "checking"); + + uint num_regions = 0; + + uint cur = start_idx; + while (cur <= end_idx && is_available(cur)) { + cur++; + } + if (cur == end_idx + 1) { + return num_regions; + } + *res_idx = cur; + while (cur <= end_idx && !is_available(cur)) { + cur++; + } + num_regions = cur - *res_idx; + +#ifdef ASSERT + for (uint i = *res_idx; i < (*res_idx + num_regions); i++) { + assert(!is_available(i), "just checking"); + } + assert(cur == end_idx + 1 || num_regions == 0 || is_available(cur), + "The region at the current position %u must be available or at the end", cur); +#endif + return num_regions; +} + +uint HeterogeneousHeapRegionManager::expand_dram(uint num_regions, WorkGang* pretouch_workers) { + return expand_in_range(start_index_of_dram(), end_index_of_dram(), num_regions, pretouch_workers); +} + +uint HeterogeneousHeapRegionManager::expand_nvdimm(uint num_regions, WorkGang* pretouch_workers) { + return expand_in_range(start_index_of_nvdimm(), end_index_of_nvdimm(), num_regions, pretouch_workers); +} + +// Follows same logic as expand_at() form HeapRegionManager. +uint HeterogeneousHeapRegionManager::expand_in_range(uint start, uint end, uint num_regions, WorkGang* pretouch_gang) { + + uint so_far = 0; + uint chunk_start = 0; + uint num_last_found = 0; + while (so_far < num_regions && + (num_last_found = find_unavailable_in_range(start, end, &chunk_start)) > 0) { + uint to_commit = MIN2(num_regions - so_far, num_last_found); + make_regions_available(chunk_start, to_commit, pretouch_gang); + so_far += to_commit; + start = chunk_start + to_commit + 1; + } + + return so_far; +} + +// Shrink in the range of indexes which are reserved for dram. +uint HeterogeneousHeapRegionManager::shrink_dram(uint num_regions, bool update_free_list) { + return shrink_in_range(start_index_of_dram(), end_index_of_dram(), num_regions, update_free_list); +} + +// Shrink in the range of indexes which are reserved for nv-dimm. +uint HeterogeneousHeapRegionManager::shrink_nvdimm(uint num_regions, bool update_free_list) { + return shrink_in_range(start_index_of_nvdimm(), end_index_of_nvdimm(), num_regions, update_free_list); +} + +// Find empty regions in given range, un-commit them and return the count. +uint HeterogeneousHeapRegionManager::shrink_in_range(uint start, uint end, uint num_regions, bool update_free_list) { + + if (num_regions == 0) { + return 0; + } + uint so_far = 0; + uint idx_last_found = 0; + uint num_last_found; + while (so_far < num_regions && + (num_last_found = find_empty_in_range_reverse(start, end, &idx_last_found)) > 0) { + uint to_uncommit = MIN2(num_regions - so_far, num_last_found); + if(update_free_list) { + _free_list.remove_starting_at(at(idx_last_found + num_last_found - to_uncommit), to_uncommit); + } + uncommit_regions(idx_last_found + num_last_found - to_uncommit, to_uncommit); + so_far += to_uncommit; + end = idx_last_found; + } + return so_far; +} + +uint HeterogeneousHeapRegionManager::find_empty_in_range_reverse(uint start_idx, uint end_idx, uint* res_idx) { + guarantee(res_idx != NULL, "checking"); + guarantee(start_idx < max_length(), "checking"); + guarantee(end_idx < max_length(), "checking"); + if(start_idx > end_idx) { + return 0; + } + + uint num_regions_found = 0; + + jlong cur = end_idx; + while (cur >= start_idx && !(is_available(cur) && at(cur)->is_empty())) { + cur--; + } + if (cur == start_idx - 1) { + return num_regions_found; + } + jlong old_cur = cur; + // cur indexes the first empty region + while (cur >= start_idx && is_available(cur) && at(cur)->is_empty()) { + cur--; + } + *res_idx = cur + 1; + num_regions_found = old_cur - cur; + +#ifdef ASSERT + for (uint i = *res_idx; i < (*res_idx + num_regions_found); i++) { + assert(at(i)->is_empty(), "just checking"); + } +#endif + return num_regions_found; +} + +HeapRegion* HeterogeneousHeapRegionManager::allocate_free_region(HeapRegionType type) { + + // We want to prevent mutators from proceeding when we have borrowed regions from the last collection. This + // will force a full collection to remedy the situation. + // Free region requests from GC threads can proceed. + if(type.is_eden() || type.is_humongous()) { + if(has_borrowed_regions()) { + return NULL; + } + } + + // old and humongous regions are allocated from nv-dimm; eden and survivor regions are allocated from dram + // assumption: dram regions take higher indexes + bool from_nvdimm = (type.is_old() || type.is_humongous()) ? true : false; + bool from_head = from_nvdimm; + HeapRegion* hr = _free_list.remove_region(from_head); + + if (hr != NULL && ( (from_nvdimm && !is_in_nvdimm(hr->hrm_index())) || (!from_nvdimm && !is_in_dram(hr->hrm_index())) ) ) { + _free_list.add_ordered(hr); + hr = NULL; + } + +#ifdef ASSERT + uint total_committed_before = total_regions_committed(); +#endif + + if (hr == NULL) { + if (!from_nvdimm) { + uint ret = shrink_nvdimm(1); + if (ret == 1) { + ret = expand_dram(1, NULL); + assert(ret == 1, "We should be able to commit one region"); + hr = _free_list.remove_region(from_head); + } + } + else { /*is_old*/ + uint ret = shrink_dram(1); + if (ret == 1) { + ret = expand_nvdimm(1, NULL); + assert(ret == 1, "We should be able to commit one region"); + hr = _free_list.remove_region(from_head); + } + } + } +#ifdef ASSERT + assert(total_committed_before == total_regions_committed(), "invariant not met"); +#endif + + // When an old region is requested (which happens during collection pause) and we can't find any empty region + // in the set of available regions (which is an evacuation failure scenario), we borrow (or pre-allocate) an unavailable region + // from nv-dimm. This region is used to evacuate surviving objects from eden, survivor or old. + if(hr == NULL && type.is_old()) { + hr = borrow_old_region_for_gc(); + } + + if (hr != NULL) { + assert(hr->next() == NULL, "Single region should not have next"); + assert(is_available(hr->hrm_index()), "Must be committed"); + } + return hr; +} + +uint HeterogeneousHeapRegionManager::find_contiguous_only_empty(size_t num) { + if (has_borrowed_regions()) { + return G1_NO_HRM_INDEX; + } + return find_contiguous(start_index_of_nvdimm(), end_index_of_nvdimm(), num, true); +} + +uint HeterogeneousHeapRegionManager::find_contiguous_empty_or_unavailable(size_t num) { + if (has_borrowed_regions()) { + return G1_NO_HRM_INDEX; + } + return find_contiguous(start_index_of_nvdimm(), end_index_of_nvdimm(), num, false); +} + +uint HeterogeneousHeapRegionManager::find_contiguous(size_t start, size_t end, size_t num, bool empty_only) { + uint found = 0; + size_t length_found = 0; + uint cur = (uint)start; + uint length_unavailable = 0; + + while (length_found < num && cur <= end) { + HeapRegion* hr = _regions.get_by_index(cur); + if ((!empty_only && !is_available(cur)) || (is_available(cur) && hr != NULL && hr->is_empty())) { + // This region is a potential candidate for allocation into. + if (!is_available(cur)) { + if(shrink_dram(1) == 1) { + uint ret = expand_in_range(cur, cur, 1, NULL); + assert(ret == 1, "We should be able to expand at this index"); + } else { + length_unavailable++; + } + } + length_found++; + } + else { + // This region is not a candidate. The next region is the next possible one. + found = cur + 1; + length_found = 0; + } + cur++; + } + + if (length_found == num) { + for (uint i = found; i < (found + num); i++) { + HeapRegion* hr = _regions.get_by_index(i); + // sanity check + guarantee((!empty_only && !is_available(i)) || (is_available(i) && hr != NULL && hr->is_empty()), + "Found region sequence starting at " UINT32_FORMAT ", length " SIZE_FORMAT + " that is not empty at " UINT32_FORMAT ". Hr is " PTR_FORMAT, found, num, i, p2i(hr)); + } + if (!empty_only && length_unavailable > (max_expandable_length() - total_regions_committed())) { + // if 'length_unavailable' number of regions will be made available, we will exceed max regions. + return G1_NO_HRM_INDEX; + } + return found; + } + else { + return G1_NO_HRM_INDEX; + } +} + +uint HeterogeneousHeapRegionManager::find_highest_free(bool* expanded) { + // Loop downwards from the highest dram region index, looking for an + // entry which is either free or not yet committed. If not yet + // committed, expand_at that index. + uint curr = end_index_of_dram(); + while (true) { + HeapRegion *hr = _regions.get_by_index(curr); + if (hr == NULL && !(total_regions_committed() < _max_regions)) { + uint res = shrink_nvdimm(1); + if (res == 1) { + res = expand_in_range(curr, curr, 1, NULL); + assert(res == 1, "We should be able to expand since shrink was successful"); + *expanded = true; + return curr; + } + } + else { + if (hr->is_free()) { + *expanded = false; + return curr; + } + } + if (curr == start_index_of_dram()) { + return G1_NO_HRM_INDEX; + } + curr--; + } +} + +// We need to override this since region 0 which serves are dummy region in base class may not be available here. +// This is a corner condition when either number of regions is small. When adaptive sizing is used, initial heap size +// could be just one region. This region is commited in dram to be used for young generation, leaving region 0 (which is in nvdimm) +// unavailable. +HeapRegion* HeterogeneousHeapRegionManager::get_dummy_region() { + uint curr = 0; + + while (curr < _regions.length()) { + if (is_available(curr)) { + return new_heap_region(curr); + } + curr++; + } + assert(false, "We should always find a region available for dummy region"); + return NULL; +} + +// First shrink in dram, then in nv-dimm. +uint HeterogeneousHeapRegionManager::shrink_by(uint num_regions) { + // This call is made at end of full collection. Before making this call the region sets are tore down (tear_down_region_sets()). + // So shrink() calls below do not need to remove uncomitted regions from free list. + uint ret = shrink_dram(num_regions, false /* update_free_list */); + ret += shrink_nvdimm(num_regions - ret, false /* update_free_list */); + return ret; +} + +void HeterogeneousHeapRegionManager::verify() { + HeapRegionManager::verify(); +} + +uint HeterogeneousHeapRegionManager::free_list_dram_length() const { + return _free_list.num_of_regions_in_range(start_index_of_dram(), end_index_of_dram()); +} + +uint HeterogeneousHeapRegionManager::free_list_nvdimm_length() const { + return _free_list.num_of_regions_in_range(start_index_of_nvdimm(), end_index_of_nvdimm()); +} + +bool HeterogeneousHeapRegionManager::is_in_nvdimm(uint index) const { + return index >= start_index_of_nvdimm() && index <= end_index_of_nvdimm(); +} + +bool HeterogeneousHeapRegionManager::is_in_dram(uint index) const { + return index >= start_index_of_dram() && index <= end_index_of_dram(); +} + +// We have to make sure full collection copies all surviving objects to NV-DIMM. +// We might not have enough regions in nvdimm_set, so we need to make more regions on NV-DIMM available for full collection. +// Note: by doing this we are breaking the in-variant that total number of committed regions is equal to current heap size. +// After full collection ends, we will re-establish this in-variant by freeing DRAM regions. +void HeterogeneousHeapRegionManager::prepare_for_full_collection_start() { + _total_commited_before_full_gc = total_regions_committed() - _no_borrowed_regions; + _no_borrowed_regions = 0; + expand_nvdimm(num_committed_dram(), NULL); + remove_all_free_regions(); +} + +// We need to bring back the total committed regions to before full collection start. +// Unless we are close to OOM, all regular (not pinned) regions in DRAM should be free. +// We shrink all free regions in DRAM and if needed from NV-DIMM (when there are pinned DRAM regions) +// If we can't bring back committed regions count to _total_commited_before_full_gc, we keep the extra count in _no_borrowed_regions. +// When this GC finishes, new regions won't be allocated since has_borrowed_regions() is true. VM will be forced to re-try GC +// with clear soft references followed by OOM error in worst case. +void HeterogeneousHeapRegionManager::prepare_for_full_collection_end() { + uint shrink_size = total_regions_committed() - _total_commited_before_full_gc; + uint so_far = 0; + uint idx_last_found = 0; + uint num_last_found; + uint end = (uint)_regions.length() - 1; + while (so_far < shrink_size && + (num_last_found = find_empty_in_range_reverse(0, end, &idx_last_found)) > 0) { + uint to_uncommit = MIN2(shrink_size - so_far, num_last_found); + uncommit_regions(idx_last_found + num_last_found - to_uncommit, to_uncommit); + so_far += to_uncommit; + end = idx_last_found; + } + // See comment above the function. + _no_borrowed_regions = shrink_size - so_far; +} + +uint HeterogeneousHeapRegionManager::start_index_of_dram() const { return _max_regions;} + +uint HeterogeneousHeapRegionManager::end_index_of_dram() const { return 2*_max_regions - 1; } + +uint HeterogeneousHeapRegionManager::start_index_of_nvdimm() const { return 0; } + +uint HeterogeneousHeapRegionManager::end_index_of_nvdimm() const { return _max_regions - 1; } + +// This function is called when there are no free nv-dimm regions. +// It borrows a region from the set of unavailable regions in nv-dimm for GC purpose. +HeapRegion* HeterogeneousHeapRegionManager::borrow_old_region_for_gc() { + assert(free_list_nvdimm_length() == 0, "this function should be called only when there are no nv-dimm regions in free list"); + + uint ret = expand_nvdimm(1, NULL); + if(ret != 1) { + return NULL; + } + HeapRegion* hr = _free_list.remove_region(true /*from_head*/); + assert(is_in_nvdimm(hr->hrm_index()), "allocated region should be in nv-dimm"); + _no_borrowed_regions++; + return hr; +} + +bool HeterogeneousHeapRegionManager::has_borrowed_regions() const { + return _no_borrowed_regions > 0; +} diff --git a/src/hotspot/share/gc/g1/heterogeneousHeapRegionManager.hpp b/src/hotspot/share/gc/g1/heterogeneousHeapRegionManager.hpp new file mode 100644 index 00000000000..cc4321e2a0c --- /dev/null +++ b/src/hotspot/share/gc/g1/heterogeneousHeapRegionManager.hpp @@ -0,0 +1,150 @@ +/* + * Copyright (c) 2018, 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. + * + */ + +#ifndef SHARE_VM_GC_G1_HETEROGENEOUSHEAPREGIONMANAGER_HPP +#define SHARE_VM_GC_G1_HETEROGENEOUSHEAPREGIONMANAGER_HPP + +#include "gc/g1/heapRegionManager.hpp" + +// This class manages heap regions on heterogenous memory comprising of dram and nv-dimm. +// Regions in dram (dram_set) are used for young objects and archive regions (CDS). +// Regions in nv-dimm (nvdimm_set) are used for old objects and humongous objects. +// At any point there are some regions committed on dram and some on nv-dimm with the following guarantees: +// 1. The total number of regions committed in dram and nv-dimm equals the current size of heap. +// 2. Consequently, total number of regions committed is less than or equal to Xmx. +// 3. To maintain the guarantee stated by 1., whenever one set grows (new regions committed), the other set shrinks (regions un-committed). +// 3a. If more dram regions are needed (young generation expansion), corresponding number of regions in nv-dimm are un-committed. +// 3b. When old generation or humongous set grows, and new regions need to be committed to nv-dimm, corresponding number of regions +// are un-committed in dram. +class HeterogeneousHeapRegionManager : public HeapRegionManager { + const uint _max_regions; + uint _max_dram_regions; + uint _max_nvdimm_regions; + uint _start_index_of_nvdimm; + uint _total_commited_before_full_gc; + uint _no_borrowed_regions; + + uint total_regions_committed() const; + uint num_committed_dram() const; + uint num_committed_nvdimm() const; + + // Similar to find_unavailable_from_idx() function from base class, difference is this function searches in range [start, end]. + uint find_unavailable_in_range(uint start_idx, uint end_idx, uint* res_idx) const; + + // Expand into dram. Maintains the invariant that total number of committed regions is less than current heap size. + uint expand_dram(uint num_regions, WorkGang* pretouch_workers); + + // Expand into nv-dimm. + uint expand_nvdimm(uint num_regions, WorkGang* pretouch_workers); + + // Expand by finding unavailable regions in [start, end] range. + uint expand_in_range(uint start, uint end, uint num_regions, WorkGang* pretouch_workers); + + // Shrink dram set of regions. + uint shrink_dram(uint num_regions, bool update_free_list = true); + + // Shrink nv-dimm set of regions. + uint shrink_nvdimm(uint num_regions, bool update_free_list = true); + + // Shrink regions from [start, end] range. + uint shrink_in_range(uint start, uint end, uint num_regions, bool update_free_list = true); + + // Similar to find_empty_from_idx_reverse() in base class. Only here it searches in a range. + uint find_empty_in_range_reverse(uint start_idx, uint end_idx, uint* res_idx); + + // Similar to find_contiguous() in base class, with [start, end] range + uint find_contiguous(size_t start, size_t end, size_t num, bool empty_only); + + // This function is called when there are no free nv-dimm regions. + // It borrows a region from the set of unavailable regions in nv-dimm for GC purpose. + HeapRegion* borrow_old_region_for_gc(); + + uint free_list_dram_length() const; + uint free_list_nvdimm_length() const; + + // is region with given index in nv-dimm? + bool is_in_nvdimm(uint index) const; + bool is_in_dram(uint index) const; + +public: + + // Empty constructor, we'll initialize it with the initialize() method. + HeterogeneousHeapRegionManager(uint num_regions) : _max_regions(num_regions), _max_dram_regions(0), + _max_nvdimm_regions(0), _start_index_of_nvdimm(0), + _total_commited_before_full_gc(0), _no_borrowed_regions(0) + {} + + static HeterogeneousHeapRegionManager* manager(); + + virtual void initialize(G1RegionToSpaceMapper* heap_storage, + G1RegionToSpaceMapper* prev_bitmap, + G1RegionToSpaceMapper* next_bitmap, + G1RegionToSpaceMapper* bot, + G1RegionToSpaceMapper* cardtable, + G1RegionToSpaceMapper* card_counts); + + uint start_index_of_nvdimm() const; + uint start_index_of_dram() const; + uint end_index_of_nvdimm() const; + uint end_index_of_dram() const; + + // Override. + HeapRegion* get_dummy_region(); + + // Adjust dram_set to provision 'expected_num_regions' regions. + void adjust_dram_regions(uint expected_num_regions, WorkGang* pretouch_workers); + + // Prepare heap regions before and after full collection. + void prepare_for_full_collection_start(); + void prepare_for_full_collection_end(); + + virtual HeapRegion* allocate_free_region(HeapRegionType type); + + // Return maximum number of regions that heap can expand to. + uint max_expandable_length() const; + + // Override. Expand in nv-dimm. + uint expand_by(uint num_regions, WorkGang* pretouch_workers); + + // Override. + uint expand_at(uint start, uint num_regions, WorkGang* pretouch_workers); + + // Override. This function is called for humongous allocation, so we need to find empty regions in nv-dimm. + uint find_contiguous_only_empty(size_t num); + + // Override. This function is called for humongous allocation, so we need to find empty or unavailable regions in nv-dimm. + uint find_contiguous_empty_or_unavailable(size_t num); + + // Overrides base class implementation to find highest free region in dram. + uint find_highest_free(bool* expanded); + + // Override. This fuction is called to shrink the heap, we shrink in dram first then in nv-dimm. + uint shrink_by(uint num_regions_to_remove); + + bool has_borrowed_regions() const; + + void verify(); +}; + +#endif // SHARE_VM_GC_G1_HETEROGENEOUSHEAPREGIONMANAGER_HPP diff --git a/src/hotspot/share/gc/g1/vmStructs_g1.hpp b/src/hotspot/share/gc/g1/vmStructs_g1.hpp index 7f1055ef0db..961143a8cfc 100644 --- a/src/hotspot/share/gc/g1/vmStructs_g1.hpp +++ b/src/hotspot/share/gc/g1/vmStructs_g1.hpp @@ -53,7 +53,7 @@ nonstatic_field(HeapRegionManager, _num_committed, uint) \ \ nonstatic_field(G1CollectedHeap, _summary_bytes_used, size_t) \ - nonstatic_field(G1CollectedHeap, _hrm, HeapRegionManager) \ + nonstatic_field(G1CollectedHeap, _hrm, HeapRegionManager*) \ nonstatic_field(G1CollectedHeap, _g1mm, G1MonitoringSupport*) \ nonstatic_field(G1CollectedHeap, _old_set, HeapRegionSetBase) \ nonstatic_field(G1CollectedHeap, _archive_set, HeapRegionSetBase) \ diff --git a/src/hotspot/share/gc/shared/gcArguments.cpp b/src/hotspot/share/gc/shared/gcArguments.cpp index 9d8690d12f7..75a35b191ad 100644 --- a/src/hotspot/share/gc/shared/gcArguments.cpp +++ b/src/hotspot/share/gc/shared/gcArguments.cpp @@ -28,6 +28,7 @@ #include "runtime/arguments.hpp" #include "runtime/globals.hpp" #include "runtime/globals_extension.hpp" +#include "utilities/defaultStream.hpp" #include "utilities/macros.hpp" void GCArguments::initialize() { @@ -53,4 +54,28 @@ void GCArguments::initialize() { // If class unloading is disabled, also disable concurrent class unloading. FLAG_SET_CMDLINE(bool, ClassUnloadingWithConcurrentMark, false); } + + if (!FLAG_IS_DEFAULT(AllocateOldGenAt)) { + // CompressedOops not supported when AllocateOldGenAt is set. + FLAG_SET_DEFAULT(UseCompressedOops, false); + FLAG_SET_DEFAULT(UseCompressedClassPointers, false); + // When AllocateOldGenAt is set, we cannot use largepages for entire heap memory. + // Only young gen which is allocated in dram can use large pages, but we currently don't support that. + FLAG_SET_DEFAULT(UseLargePages, false); + } +} + +bool GCArguments::check_args_consistency() { + bool status = true; + if (!FLAG_IS_DEFAULT(AllocateHeapAt) && !FLAG_IS_DEFAULT(AllocateOldGenAt)) { + jio_fprintf(defaultStream::error_stream(), + "AllocateHeapAt and AllocateOldGenAt cannot be used together.\n"); + status = false; + } + if (!FLAG_IS_DEFAULT(AllocateOldGenAt) && (UseSerialGC || UseConcMarkSweepGC || UseEpsilonGC || UseZGC)) { + jio_fprintf(defaultStream::error_stream(), + "AllocateOldGenAt is not supported for selected GC.\n"); + status = false; + } + return status; } diff --git a/src/hotspot/share/gc/shared/gcArguments.hpp b/src/hotspot/share/gc/shared/gcArguments.hpp index afda9b36e9c..2d10dada9db 100644 --- a/src/hotspot/share/gc/shared/gcArguments.hpp +++ b/src/hotspot/share/gc/shared/gcArguments.hpp @@ -39,6 +39,7 @@ public: virtual void initialize(); virtual size_t conservative_max_heap_alignment() = 0; virtual CollectedHeap* create_heap() = 0; + static bool check_args_consistency(); }; #endif // SHARE_GC_SHARED_GCARGUMENTS_HPP diff --git a/src/hotspot/share/prims/whitebox.cpp b/src/hotspot/share/prims/whitebox.cpp index 22eac057106..03223ad0b57 100644 --- a/src/hotspot/share/prims/whitebox.cpp +++ b/src/hotspot/share/prims/whitebox.cpp @@ -82,6 +82,7 @@ #include "gc/g1/g1ConcurrentMark.hpp" #include "gc/g1/g1ConcurrentMarkThread.hpp" #include "gc/g1/heapRegionRemSet.hpp" +#include "gc/g1/heterogeneousHeapRegionManager.hpp" #endif // INCLUDE_G1GC #if INCLUDE_PARALLELGC #include "gc/parallel/parallelScavengeHeap.inline.hpp" @@ -499,6 +500,61 @@ WB_END #endif // INCLUDE_G1GC +#if INCLUDE_G1GC +WB_ENTRY(jlong, WB_DramReservedStart(JNIEnv* env, jobject o)) + if (UseG1GC) { + G1CollectedHeap* g1h = G1CollectedHeap::heap(); + if (g1h->g1_collector_policy()->is_hetero_heap()) { + uint start_region = HeterogeneousHeapRegionManager::manager()->start_index_of_dram(); + return (jlong)(g1h->base() + start_region * HeapRegion::GrainBytes); + } else { + return (jlong)g1h->base(); + } + } + THROW_MSG_0(vmSymbols::java_lang_UnsupportedOperationException(), "WB_DramReservedStart: enabled only for G1"); +WB_END + +WB_ENTRY(jlong, WB_DramReservedEnd(JNIEnv* env, jobject o)) + if (UseG1GC) { + G1CollectedHeap* g1h = G1CollectedHeap::heap(); + if (g1h->g1_collector_policy()->is_hetero_heap()) { + uint end_region = HeterogeneousHeapRegionManager::manager()->end_index_of_dram(); + return (jlong)(g1h->base() + (end_region + 1) * HeapRegion::GrainBytes - 1); + } else { + return (jlong)g1h->base() + g1h->collector_policy()->max_heap_byte_size(); + } + } + THROW_MSG_0(vmSymbols::java_lang_UnsupportedOperationException(), "WB_DramReservedEnd: enabled only for G1"); +WB_END + +WB_ENTRY(jlong, WB_NvdimmReservedStart(JNIEnv* env, jobject o)) + if (UseG1GC) { + G1CollectedHeap* g1h = G1CollectedHeap::heap(); + if (g1h->g1_collector_policy()->is_hetero_heap()) { + uint start_region = HeterogeneousHeapRegionManager::manager()->start_index_of_nvdimm(); + return (jlong)(g1h->base() + start_region * HeapRegion::GrainBytes); + } else { + THROW_MSG_0(vmSymbols::java_lang_UnsupportedOperationException(), "WB_NvdimmReservedStart: Old gen is not allocated on NV-DIMM using AllocateOldGenAt flag"); + } + } + THROW_MSG_0(vmSymbols::java_lang_UnsupportedOperationException(), "WB_NvdimmReservedStart: enabled only for G1"); +WB_END + +WB_ENTRY(jlong, WB_NvdimmReservedEnd(JNIEnv* env, jobject o)) + if (UseG1GC) { + G1CollectedHeap* g1h = G1CollectedHeap::heap(); + if (g1h->g1_collector_policy()->is_hetero_heap()) { + uint end_region = HeterogeneousHeapRegionManager::manager()->start_index_of_nvdimm(); + return (jlong)(g1h->base() + (end_region + 1) * HeapRegion::GrainBytes - 1); + } else { + THROW_MSG_0(vmSymbols::java_lang_UnsupportedOperationException(), "WB_NvdimmReservedEnd: Old gen is not allocated on NV-DIMM using AllocateOldGenAt flag"); + } + } + THROW_MSG_0(vmSymbols::java_lang_UnsupportedOperationException(), "WB_NvdimmReservedEnd: enabled only for G1"); +WB_END + +#endif // INCLUDE_G1GC + #if INCLUDE_PARALLELGC WB_ENTRY(jlong, WB_PSVirtualSpaceAlignment(JNIEnv* env, jobject o)) @@ -2053,6 +2109,10 @@ static JNINativeMethod methods[] = { {CC"g1AuxiliaryMemoryUsage", CC"()Ljava/lang/management/MemoryUsage;", (void*)&WB_G1AuxiliaryMemoryUsage }, {CC"g1GetMixedGCInfo", CC"(I)[J", (void*)&WB_G1GetMixedGCInfo }, + {CC"dramReservedStart", CC"()J", (void*)&WB_DramReservedStart }, + {CC"dramReservedEnd", CC"()J", (void*)&WB_DramReservedEnd }, + {CC"nvdimmReservedStart", CC"()J", (void*)&WB_NvdimmReservedStart }, + {CC"nvdimmReservedEnd", CC"()J", (void*)&WB_NvdimmReservedEnd }, #endif // INCLUDE_G1GC #if INCLUDE_PARALLELGC {CC"psVirtualSpaceAlignment",CC"()J", (void*)&WB_PSVirtualSpaceAlignment}, diff --git a/src/hotspot/share/runtime/arguments.cpp b/src/hotspot/share/runtime/arguments.cpp index 0025c316726..3941c473273 100644 --- a/src/hotspot/share/runtime/arguments.cpp +++ b/src/hotspot/share/runtime/arguments.cpp @@ -2062,6 +2062,9 @@ bool Arguments::check_vm_args_consistency() { log_warning(arguments) ("NUMA support for Heap depends on the file system when AllocateHeapAt option is used.\n"); } } + + status = status && GCArguments::check_args_consistency(); + return status; } @@ -2953,6 +2956,7 @@ jint Arguments::parse_each_vm_init_arg(const JavaVMInitArgs* args, bool* patch_m } #endif // LINUX fix_appclasspath(); + return JNI_OK; } diff --git a/src/hotspot/share/runtime/globals.hpp b/src/hotspot/share/runtime/globals.hpp index 8a9a592e95a..dc788da28f0 100644 --- a/src/hotspot/share/runtime/globals.hpp +++ b/src/hotspot/share/runtime/globals.hpp @@ -2575,6 +2575,12 @@ define_pd_global(uint64_t,MaxRAM, 1ULL*G); "Path to the directoy where a temporary file will be created " \ "to use as the backing store for Java Heap.") \ \ + experimental(ccstr, AllocateOldGenAt, NULL, \ + "Path to the directoy where a temporary file will be " \ + "created to use as the backing store for old generation." \ + "File of size Xmx is pre-allocated for performance reason, so" \ + "we need that much space available") \ + \ develop(bool, VerifyMetaspace, false, \ "Verify metaspace on chunk movements.") \ \ diff --git a/src/jdk.hotspot.agent/share/classes/sun/jvm/hotspot/gc/g1/G1CollectedHeap.java b/src/jdk.hotspot.agent/share/classes/sun/jvm/hotspot/gc/g1/G1CollectedHeap.java index 636c1764cb6..65e7caac7bf 100644 --- a/src/jdk.hotspot.agent/share/classes/sun/jvm/hotspot/gc/g1/G1CollectedHeap.java +++ b/src/jdk.hotspot.agent/share/classes/sun/jvm/hotspot/gc/g1/G1CollectedHeap.java @@ -47,7 +47,7 @@ import sun.jvm.hotspot.tools.HeapSummary; public class G1CollectedHeap extends CollectedHeap { // HeapRegionManager _hrm; - static private long hrmFieldOffset; + static private AddressField hrmField; // MemRegion _g1_reserved; static private long g1ReservedFieldOffset; // size_t _summary_bytes_used; @@ -72,7 +72,7 @@ public class G1CollectedHeap extends CollectedHeap { static private synchronized void initialize(TypeDataBase db) { Type type = db.lookupType("G1CollectedHeap"); - hrmFieldOffset = type.getField("_hrm").getOffset(); + hrmField = type.getAddressField("_hrm"); summaryBytesUsedField = type.getCIntegerField("_summary_bytes_used"); g1mmField = type.getAddressField("_g1mm"); oldSetFieldOffset = type.getField("_old_set").getOffset(); @@ -93,7 +93,7 @@ public class G1CollectedHeap extends CollectedHeap { } public HeapRegionManager hrm() { - Address hrmAddr = addr.addOffsetTo(hrmFieldOffset); + Address hrmAddr = hrmField.getValue(addr); return (HeapRegionManager) VMObjectFactory.newObject(HeapRegionManager.class, hrmAddr); } diff --git a/test/hotspot/jtreg/TEST.groups b/test/hotspot/jtreg/TEST.groups index c9c7db8fdd2..18dcd52be3c 100644 --- a/test/hotspot/jtreg/TEST.groups +++ b/test/hotspot/jtreg/TEST.groups @@ -40,7 +40,8 @@ hotspot_compiler_all_gcs = \ -:tier1_compiler_not_cms hotspot_gc = \ - gc + gc \ + -gc/nvdimm hotspot_runtime = \ runtime @@ -190,7 +191,8 @@ tier1_gc_2 = \ -gc/survivorAlignment/TestPromotionFromSurvivorToTenuredAfterMinorGC.java \ -gc/cms/TestMBeanCMS.java \ -gc/metaspace/CompressedClassSpaceSizeInJmapHeap.java \ - -gc/shenandoah + -gc/shenandoah \ + -gc/nvdimm gc_epsilon = \ gc/epsilon/ \ diff --git a/test/hotspot/jtreg/gc/nvdimm/TestAllocateOldGenAt.java b/test/hotspot/jtreg/gc/nvdimm/TestAllocateOldGenAt.java new file mode 100644 index 00000000000..d8149d5ec1b --- /dev/null +++ b/test/hotspot/jtreg/gc/nvdimm/TestAllocateOldGenAt.java @@ -0,0 +1,77 @@ +/* + * Copyright (c) 2018, 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 TestAllocateOldGenAt.java + * @key gc + * @summary Test to check allocation of Java Heap with AllocateOldGenAt option + * @requires vm.gc=="null" + * @library /test/lib + * @modules java.base/jdk.internal.misc + */ + +import jdk.test.lib.JDKToolFinder; +import jdk.test.lib.process.ProcessTools; +import jdk.test.lib.process.OutputAnalyzer; +import java.util.ArrayList; +import java.util.Collections; + +public class TestAllocateOldGenAt { + private static ArrayList commonOpts; + + public static void main(String args[]) throws Exception { + commonOpts = new ArrayList(); + + String testVmOptsStr = System.getProperty("test.java.opts"); + if (!testVmOptsStr.isEmpty()) { + String[] testVmOpts = testVmOptsStr.split(" "); + Collections.addAll(commonOpts, testVmOpts); + } + String test_dir = System.getProperty("test.dir", "."); + Collections.addAll(commonOpts, new String[] {"-XX:+UnlockExperimentalVMOptions", + "-XX:AllocateOldGenAt=" + test_dir, + "-Xmx32m", + "-Xms32m", + "-version"}); + + runTest("-XX:+UseG1GC"); + } + + private static void runTest(String... extraFlags) throws Exception { + ArrayList testOpts = new ArrayList(); + Collections.addAll(testOpts, commonOpts.toArray(new String[commonOpts.size()])); + Collections.addAll(testOpts, extraFlags); + + System.out.print("Testing:\n" + JDKToolFinder.getJDKTool("java")); + for (int i = 0; i < testOpts.size(); i += 1) { + System.out.print(" " + testOpts.get(i)); + } + System.out.println(); + + ProcessBuilder pb = + ProcessTools.createJavaProcessBuilder(testOpts.toArray(new String[testOpts.size()])); + OutputAnalyzer output = new OutputAnalyzer(pb.start()); + + output.shouldHaveExitValue(0); + + } +} diff --git a/test/hotspot/jtreg/gc/nvdimm/TestAllocateOldGenAtError.java b/test/hotspot/jtreg/gc/nvdimm/TestAllocateOldGenAtError.java new file mode 100644 index 00000000000..63cd39bf1ec --- /dev/null +++ b/test/hotspot/jtreg/gc/nvdimm/TestAllocateOldGenAtError.java @@ -0,0 +1,94 @@ +/* + * Copyright (c) 2018, 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 TestAllocateOldGenAtError.java + * @key gc + * @summary Test to check correct handling of non-existent directory passed to AllocateOldGenAt option + * @requires vm.gc=="null" + * @library /test/lib + * @modules java.base/jdk.internal.misc + */ + +import java.io.File; +import jdk.test.lib.JDKToolFinder; +import jdk.test.lib.process.ProcessTools; +import jdk.test.lib.process.OutputAnalyzer; +import java.util.ArrayList; +import java.util.Collections; +import java.util.UUID; + +public class TestAllocateOldGenAtError { + private static ArrayList commonOpts; + + public static void main(String args[]) throws Exception { + commonOpts = new ArrayList(); + + String testVmOptsStr = System.getProperty("test.java.opts"); + if (!testVmOptsStr.isEmpty()) { + String[] testVmOpts = testVmOptsStr.split(" "); + Collections.addAll(commonOpts, testVmOpts); + } + String test_dir = System.getProperty("test.dir", "."); + + File f = null; + do { + f = new File(test_dir, UUID.randomUUID().toString()); + } while(f.exists()); + + Collections.addAll(commonOpts, new String[] {"-XX:+UnlockExperimentalVMOptions", + "-XX:AllocateOldGenAt=" + f.getName(), + "-Xlog:gc+heap=info", + "-Xmx32m", + "-Xms32m", + "-version"}); + + testG1(); + } + + private static void testG1() throws Exception { + System.out.println("Testing G1 GC"); + + OutputAnalyzer output = runTest("-XX:+UseG1GC"); + + output.shouldContain("Could not initialize G1 heap"); + output.shouldContain("Error occurred during initialization of VM"); + output.shouldNotHaveExitValue(0); + } + + private static OutputAnalyzer runTest(String... extraFlags) throws Exception { + ArrayList testOpts = new ArrayList(); + Collections.addAll(testOpts, commonOpts.toArray(new String[commonOpts.size()])); + Collections.addAll(testOpts, extraFlags); + + System.out.print("Testing:\n" + JDKToolFinder.getJDKTool("java")); + for (int i = 0; i < testOpts.size(); i += 1) { + System.out.print(" " + testOpts.get(i)); + } + System.out.println(); + + ProcessBuilder pb = + ProcessTools.createJavaProcessBuilder(testOpts.toArray(new String[testOpts.size()])); + OutputAnalyzer output = new OutputAnalyzer(pb.start()); + return output; + } +} diff --git a/test/hotspot/jtreg/gc/nvdimm/TestAllocateOldGenAtMultiple.java b/test/hotspot/jtreg/gc/nvdimm/TestAllocateOldGenAtMultiple.java new file mode 100644 index 00000000000..819d4e6bf42 --- /dev/null +++ b/test/hotspot/jtreg/gc/nvdimm/TestAllocateOldGenAtMultiple.java @@ -0,0 +1,92 @@ +/* + * Copyright (c) 2018, 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 TestAllocateOldGenAtMultiple.java + * @key gc + * @summary Test to check allocation of Java Heap with AllocateOldGenAt option. Has multiple sub-tests to cover different code paths. + * @requires vm.gc=="null" + * @library /test/lib + * @modules java.base/jdk.internal.misc + * @requires vm.bits == "64" + * @run main TestAllocateOldGenAtMultiple -XX:+UseG1GC + */ + +import jdk.test.lib.JDKToolFinder; +import jdk.test.lib.process.ProcessTools; +import jdk.test.lib.process.OutputAnalyzer; +import java.util.ArrayList; +import java.util.Collections; + +public class TestAllocateOldGenAtMultiple { + public static void main(String args[]) throws Exception { + ArrayList vmOpts = new ArrayList(); + String[] testVmOpts = null; + + String test_dir = System.getProperty("test.dir", "."); + + String testVmOptsStr = System.getProperty("test.java.opts"); + if (!testVmOptsStr.isEmpty()) { + testVmOpts = testVmOptsStr.split(" "); + } + + // Extra options for each of the sub-tests + String[] extraOptsList = new String[] { + "-Xmx32m -Xms32m -XX:+UseCompressedOops", // 1. With compressedoops enabled. + "-Xmx32m -Xms32m -XX:-UseCompressedOops", // 2. With compressedoops disabled. + "-Xmx32m -Xms32m -XX:HeapBaseMinAddress=3g", // 3. With user specified HeapBaseMinAddress. + "-Xmx4g -Xms4g", // 4. With larger heap size (UnscaledNarrowOop not possible). + "-Xmx4g -Xms4g -XX:+UseLargePages", // 5. Set UseLargePages. + "-Xmx4g -Xms4g -XX:+UseNUMA" // 6. Set UseNUMA. + }; + + for(String extraOpts : extraOptsList) { + vmOpts.clear(); + if(testVmOpts != null) { + Collections.addAll(vmOpts, testVmOpts); + } + // Add extra options specific to the sub-test. + String[] extraOptsArray = extraOpts.split(" "); + if(extraOptsArray != null) { + Collections.addAll(vmOpts, extraOptsArray); + } + // Add common options + Collections.addAll(vmOpts, new String[] {"-XX:+UnlockExperimentalVMOptions", + "-XX:AllocateOldGenAt=" + test_dir, + "-version"}); + + System.out.print("Testing:\n" + JDKToolFinder.getJDKTool("java")); + for (int i = 0; i < vmOpts.size(); i += 1) { + System.out.print(" " + vmOpts.get(i)); + } + System.out.println(); + + ProcessBuilder pb = + ProcessTools.createJavaProcessBuilder(vmOpts.toArray(new String[vmOpts.size()])); + OutputAnalyzer output = new OutputAnalyzer(pb.start()); + + System.out.println("Output:\n" + output.getOutput()); + + output.shouldHaveExitValue(0); + } + } +} diff --git a/test/hotspot/jtreg/gc/nvdimm/TestHumongousObjectsOnNvdimm.java b/test/hotspot/jtreg/gc/nvdimm/TestHumongousObjectsOnNvdimm.java new file mode 100644 index 00000000000..24fc1d56dbf --- /dev/null +++ b/test/hotspot/jtreg/gc/nvdimm/TestHumongousObjectsOnNvdimm.java @@ -0,0 +1,117 @@ +/* + * Copyright (c) 2018, 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 TestHumongousObjectsOnNvdimm + * @summary Check that humongous objects reside in nv-dimm + * @library /test/lib / + * @requires vm.gc=="null" + * @build sun.hotspot.WhiteBox + * @run driver ClassFileInstaller sun.hotspot.WhiteBox + * @run main TestHumongousObjectsOnNvdimm -Xbootclasspath/a:. -XX:+UnlockDiagnosticVMOptions + * -XX:+WhiteBoxAPI + */ + +import jdk.test.lib.process.OutputAnalyzer; +import jdk.test.lib.process.ProcessTools; +import jdk.test.lib.Asserts; +import sun.hotspot.WhiteBox; + +import java.util.ArrayList; +import java.util.List; +import java.util.Collections; +import gc.testlibrary.Helpers; + +/** + * Test spawns HumongousObjectTest in a separate VM and expects that it + * completes without a RuntimeException. + */ +public class TestHumongousObjectsOnNvdimm { + + private static ArrayList testOpts; + + public static void main(String args[]) throws Exception { + testOpts = new ArrayList(); + + String[] common_options = new String[] { + "-Xbootclasspath/a:.", + "-XX:+UnlockExperimentalVMOptions", + "-XX:+UnlockDiagnosticVMOptions", + "-XX:+WhiteBoxAPI", + "-XX:AllocateOldGenAt="+System.getProperty("test.dir", "."), + "-Xms10M", "-Xmx10M", + "-XX:G1HeapRegionSize=1m" + }; + + String testVmOptsStr = System.getProperty("test.java.opts"); + if (!testVmOptsStr.isEmpty()) { + String[] testVmOpts = testVmOptsStr.split(" "); + Collections.addAll(testOpts, testVmOpts); + } + Collections.addAll(testOpts, common_options); + + // Test with G1 GC + runTest("-XX:+UseG1GC"); + } + + private static void runTest(String... extraFlags) throws Exception { + Collections.addAll(testOpts, extraFlags); + testOpts.add(HumongousObjectTest.class.getName()); + System.out.println(testOpts); + + ProcessBuilder pb = ProcessTools.createJavaProcessBuilder(false, + testOpts.toArray(new String[testOpts.size()])); + + OutputAnalyzer output = new OutputAnalyzer(pb.start()); + output.shouldHaveExitValue(0); + } +} + +/** + * This class tests that a humongous object resides in NVDIMM. + */ +class HumongousObjectTest { + private static final WhiteBox WB = WhiteBox.getWhiteBox(); + + private static void validateObject(Object o) { + Asserts.assertTrue(WB.isObjectInOldGen(o), + "Object is supposed to be in OldGen"); + + long obj_addr = WB.getObjectAddress(o); + long nvdimm_heap_start = WB.nvdimmReservedStart(); + long nvdimm_heap_end = WB.nvdimmReservedEnd(); + + Asserts.assertTrue(WB.g1BelongsToHumongousRegion(obj_addr), "Object address should be in Humongous set"); + Asserts.assertTrue(obj_addr >= nvdimm_heap_start && obj_addr < nvdimm_heap_end, + "Humongous object does not reside in NVDIMM"); + } + + public static void main(String args[]) throws Exception { + // allocate an humongous object + int byteArrayMemoryOverhead = Helpers.detectByteArrayAllocationOverhead(); + int MinByteArrayHumongousSize = (WB.g1RegionSize() / 2) - byteArrayMemoryOverhead + 1; + byte[] obj = new byte[MinByteArrayHumongousSize]; + + validateObject(obj); + } +} diff --git a/test/hotspot/jtreg/gc/nvdimm/TestOldObjectsOnNvdimm.java b/test/hotspot/jtreg/gc/nvdimm/TestOldObjectsOnNvdimm.java new file mode 100644 index 00000000000..9a7489ab6b0 --- /dev/null +++ b/test/hotspot/jtreg/gc/nvdimm/TestOldObjectsOnNvdimm.java @@ -0,0 +1,118 @@ +/* + * Copyright (c) 2018, 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 TestOldObjectsOnNvdimm + * @summary Check that objects in old generation reside in dram. + * @requires vm.gc=="null" + * @library /test/lib + * @build sun.hotspot.WhiteBox + * @run driver ClassFileInstaller sun.hotspot.WhiteBox + * @run main TestOldObjectsOnNvdimm -Xbootclasspath/a:. -XX:+UnlockDiagnosticVMOptions + * -XX:+WhiteBoxAPI + */ + +import jdk.test.lib.process.OutputAnalyzer; +import jdk.test.lib.process.ProcessTools; +import jdk.test.lib.Asserts; +import sun.hotspot.WhiteBox; + +import java.util.ArrayList; +import java.util.List; +import java.util.Collections; + +/* + * Test spawns OldObjectTest in a separate VM and expects that it + * completes without a RuntimeException. + */ +public class TestOldObjectsOnNvdimm { + + public static final int ALLOCATION_SIZE = 100; + private static ArrayList testOpts; + + public static void main(String args[]) throws Exception { + testOpts = new ArrayList(); + + String[] common_options = new String[] { + "-Xbootclasspath/a:.", + "-XX:+UnlockExperimentalVMOptions", + "-XX:+UnlockDiagnosticVMOptions", + "-XX:+WhiteBoxAPI", + "-XX:AllocateOldGenAt="+System.getProperty("test.dir", "."), + "-Xms10M", "-Xmx10M", + "-XX:MaxTenuringThreshold=1" // Promote objects to Old Gen + }; + + String testVmOptsStr = System.getProperty("test.java.opts"); + if (!testVmOptsStr.isEmpty()) { + String[] testVmOpts = testVmOptsStr.split(" "); + Collections.addAll(testOpts, testVmOpts); + } + Collections.addAll(testOpts, common_options); + + // Test with G1 GC + runTest("-XX:+UseG1GC"); + } + + private static void runTest(String... extraFlags) throws Exception { + Collections.addAll(testOpts, extraFlags); + testOpts.add(OldObjectTest.class.getName()); + System.out.println(testOpts); + + ProcessBuilder pb = ProcessTools.createJavaProcessBuilder(false, + testOpts.toArray(new String[testOpts.size()])); + + OutputAnalyzer output = new OutputAnalyzer(pb.start()); + System.out.println(output.getStdout()); + output.shouldHaveExitValue(0); + } +} + +/* + * This class tests that object is in Old generation after tenuring and resides in NVDIMM. + * The necessary condition for this test is running in VM with the following flags: + * -XX:AllocateOldGenAt=, -XX:MaxTenuringThreshold=1 + */ +class OldObjectTest { + private static final WhiteBox WB = WhiteBox.getWhiteBox(); + + private static void validateOldObject(Object o) { + Asserts.assertTrue(WB.isObjectInOldGen(o), + "Object is supposed to be in OldGen"); + + long oldObj_addr = WB.getObjectAddress(o); + long nvdimm_heap_start = WB.nvdimmReservedStart(); + long nvdimm_heap_end = WB.nvdimmReservedEnd(); + + Asserts.assertTrue(oldObj_addr >= nvdimm_heap_start && oldObj_addr <= nvdimm_heap_end, + "Old object does not reside in NVDIMM"); + } + + public static void main(String args[]) throws Exception { + // allocate an object and perform Young GCs to promote it to Old + byte[] oldObj = new byte[TestOldObjectsOnNvdimm.ALLOCATION_SIZE]; + WB.youngGC(); + WB.youngGC(); + validateOldObject(oldObj); + } +} diff --git a/test/hotspot/jtreg/gc/nvdimm/TestYoungObjectsOnDram.java b/test/hotspot/jtreg/gc/nvdimm/TestYoungObjectsOnDram.java new file mode 100644 index 00000000000..2f84d441e59 --- /dev/null +++ b/test/hotspot/jtreg/gc/nvdimm/TestYoungObjectsOnDram.java @@ -0,0 +1,122 @@ +/* + * Copyright (c) 2018, 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 TestYoungObjectsOnDram + * @summary Check that objects in young generation reside in dram. + * @requires vm.gc=="null" + * @library /test/lib + * @build sun.hotspot.WhiteBox + * @run driver ClassFileInstaller sun.hotspot.WhiteBox + * @run main TestYoungObjectsOnDram -Xbootclasspath/a:. -XX:+UnlockDiagnosticVMOptions + * -XX:+WhiteBoxAPI + */ + +import jdk.test.lib.process.OutputAnalyzer; +import jdk.test.lib.process.ProcessTools; +import jdk.test.lib.Asserts; +import sun.hotspot.WhiteBox; + +import java.util.ArrayList; +import java.util.List; +import java.util.Collections; + +/** + * Test spawns YoungObjectTest in a separate VM and expects that it + * completes without a RuntimeException. + */ +public class TestYoungObjectsOnDram { + + public static final int ALLOCATION_SIZE = 100; + private static ArrayList testOpts; + + public static void main(String args[]) throws Exception { + testOpts = new ArrayList(); + + String[] common_options = new String[] { + "-Xbootclasspath/a:.", + "-XX:+UnlockExperimentalVMOptions", + "-XX:+UnlockDiagnosticVMOptions", + "-XX:+WhiteBoxAPI", + "-XX:AllocateOldGenAt="+System.getProperty("test.dir", "."), + "-XX:SurvivorRatio=1", // Survivor-to-eden ratio is 1:1 + "-Xms10M", "-Xmx10M", + "-XX:InitialTenuringThreshold=15" // avoid promotion of objects to Old Gen + }; + + String testVmOptsStr = System.getProperty("test.java.opts"); + if (!testVmOptsStr.isEmpty()) { + String[] testVmOpts = testVmOptsStr.split(" "); + Collections.addAll(testOpts, testVmOpts); + } + Collections.addAll(testOpts, common_options); + + // Test with G1 GC + runTest("-XX:+UseG1GC"); + } + + private static void runTest(String... extraFlags) throws Exception { + Collections.addAll(testOpts, extraFlags); + testOpts.add(YoungObjectTest.class.getName()); + System.out.println(testOpts); + + ProcessBuilder pb = ProcessTools.createJavaProcessBuilder(false, + testOpts.toArray(new String[testOpts.size()])); + + OutputAnalyzer output = new OutputAnalyzer(pb.start()); + System.out.println(output.getStdout()); + output.shouldHaveExitValue(0); + } +} + +/** + * This class tests that newly created object is in Young generation and resides in DRAM. + * The necessary condition for this test is running in VM with the following flags: + * -XX:AllocateOldGenAt=, -XX:InitialTenuringThreshold=15, -XX:SurvivorRatio=1 + */ +class YoungObjectTest { + private static final WhiteBox WB = WhiteBox.getWhiteBox(); + + private static void validateYoungObject(Object o) { + Asserts.assertTrue(!WB.isObjectInOldGen(o), + "Object is supposed to be in YoungGen"); + + long youngObj_addr = WB.getObjectAddress(o); + long dram_heap_start = WB.dramReservedStart(); + long dram_heap_end = WB.dramReservedEnd(); + + Asserts.assertTrue(youngObj_addr >= dram_heap_start && youngObj_addr <= dram_heap_end, + "Young object does not reside in DRAM"); + } + + public static void main(String args[]) throws Exception { + // allocate an object + byte[] youngObj = new byte[TestYoungObjectsOnDram.ALLOCATION_SIZE]; + validateYoungObject(youngObj); + + // Start a Young GC and check that object is still in DRAM. + // We have used -XX:InitialTenuringThreshold=15 to invoke this test + WB.youngGC(); + validateYoungObject(youngObj); + } +} diff --git a/test/lib/sun/hotspot/WhiteBox.java b/test/lib/sun/hotspot/WhiteBox.java index 1a8101304a5..0b93a859143 100644 --- a/test/lib/sun/hotspot/WhiteBox.java +++ b/test/lib/sun/hotspot/WhiteBox.java @@ -182,6 +182,10 @@ public class WhiteBox { public native long g1NumMaxRegions(); public native long g1NumFreeRegions(); public native int g1RegionSize(); + public native long dramReservedStart(); + public native long dramReservedEnd(); + public native long nvdimmReservedStart(); + public native long nvdimmReservedEnd(); public native MemoryUsage g1AuxiliaryMemoryUsage(); private native Object[] parseCommandLine0(String commandline, char delim, DiagnosticCommand[] args); public Object[] parseCommandLine(String commandline, char delim, DiagnosticCommand[] args) { From 28f71c2c72da38d95219d918e6f6678b7396a98b Mon Sep 17 00:00:00 2001 From: Kishor Kharbas Date: Fri, 21 Dec 2018 08:23:55 -0800 Subject: [PATCH 043/101] 8211424: Allocation of old generation of java heap on alternate memory devices - Parallel GC 8202286: Allocation of old generation of Java heap on alternate memory devices Enable an experimental feature in HotSpot JVM to allocate old generation of Parallel GC on an alternative memory device, such as NV-DIMMs. Reviewed-by: sangheki, sjohanss --- .../gc/parallel/adjoiningGenerations.cpp | 38 ++- .../gc/parallel/adjoiningGenerations.hpp | 26 +- .../adjoiningGenerationsForHeteroHeap.cpp | 260 ++++++++++++++++++ .../adjoiningGenerationsForHeteroHeap.hpp | 85 ++++++ .../gc/parallel/adjoiningVirtualSpaces.hpp | 16 +- .../share/gc/parallel/generationSizer.cpp | 10 +- .../share/gc/parallel/generationSizer.hpp | 7 +- .../parallel/heterogeneousGenerationSizer.cpp | 106 +++++++ .../parallel/heterogeneousGenerationSizer.hpp | 46 ++++ .../share/gc/parallel/parallelArguments.cpp | 7 +- .../gc/parallel/parallelScavengeHeap.cpp | 7 +- .../gc/parallel/parallelScavengeHeap.hpp | 2 + .../gc/parallel/psFileBackedVirtualspace.cpp | 87 ++++++ .../gc/parallel/psFileBackedVirtualspace.hpp | 46 ++++ src/hotspot/share/gc/parallel/psOldGen.cpp | 10 +- .../share/gc/parallel/psParallelCompact.cpp | 5 +- src/hotspot/share/prims/whitebox.cpp | 52 +++- .../jtreg/gc/nvdimm/TestAllocateOldGenAt.java | 2 + .../gc/nvdimm/TestAllocateOldGenAtError.java | 14 + .../gc/nvdimm/TestOldObjectsOnNvdimm.java | 3 + .../gc/nvdimm/TestYoungObjectsOnDram.java | 3 + 21 files changed, 785 insertions(+), 47 deletions(-) create mode 100644 src/hotspot/share/gc/parallel/adjoiningGenerationsForHeteroHeap.cpp create mode 100644 src/hotspot/share/gc/parallel/adjoiningGenerationsForHeteroHeap.hpp create mode 100644 src/hotspot/share/gc/parallel/heterogeneousGenerationSizer.cpp create mode 100644 src/hotspot/share/gc/parallel/heterogeneousGenerationSizer.hpp create mode 100644 src/hotspot/share/gc/parallel/psFileBackedVirtualspace.cpp create mode 100644 src/hotspot/share/gc/parallel/psFileBackedVirtualspace.hpp diff --git a/src/hotspot/share/gc/parallel/adjoiningGenerations.cpp b/src/hotspot/share/gc/parallel/adjoiningGenerations.cpp index ab45b902502..b41108520a4 100644 --- a/src/hotspot/share/gc/parallel/adjoiningGenerations.cpp +++ b/src/hotspot/share/gc/parallel/adjoiningGenerations.cpp @@ -24,6 +24,7 @@ #include "precompiled.hpp" #include "gc/parallel/adjoiningGenerations.hpp" +#include "gc/parallel/adjoiningGenerationsForHeteroHeap.hpp" #include "gc/parallel/adjoiningVirtualSpaces.hpp" #include "gc/parallel/generationSizer.hpp" #include "gc/parallel/parallelScavengeHeap.hpp" @@ -40,8 +41,8 @@ AdjoiningGenerations::AdjoiningGenerations(ReservedSpace old_young_rs, GenerationSizer* policy, size_t alignment) : - _virtual_spaces(old_young_rs, policy->min_old_size(), - policy->min_young_size(), alignment) { + _virtual_spaces(new AdjoiningVirtualSpaces(old_young_rs, policy->min_old_size(), + policy->min_young_size(), alignment)) { size_t init_low_byte_size = policy->initial_old_size(); size_t min_low_byte_size = policy->min_old_size(); size_t max_low_byte_size = policy->max_old_size(); @@ -61,21 +62,21 @@ AdjoiningGenerations::AdjoiningGenerations(ReservedSpace old_young_rs, // generation. // Does the actual creation of the virtual spaces - _virtual_spaces.initialize(max_low_byte_size, - init_low_byte_size, - init_high_byte_size); + _virtual_spaces->initialize(max_low_byte_size, + init_low_byte_size, + init_high_byte_size); // Place the young gen at the high end. Passes in the virtual space. - _young_gen = new ASPSYoungGen(_virtual_spaces.high(), - _virtual_spaces.high()->committed_size(), + _young_gen = new ASPSYoungGen(_virtual_spaces->high(), + _virtual_spaces->high()->committed_size(), min_high_byte_size, - _virtual_spaces.high_byte_size_limit()); + _virtual_spaces->high_byte_size_limit()); // Place the old gen at the low end. Passes in the virtual space. - _old_gen = new ASPSOldGen(_virtual_spaces.low(), - _virtual_spaces.low()->committed_size(), + _old_gen = new ASPSOldGen(_virtual_spaces->low(), + _virtual_spaces->low()->committed_size(), min_low_byte_size, - _virtual_spaces.low_byte_size_limit(), + _virtual_spaces->low_byte_size_limit(), "old", 1); young_gen()->initialize_work(); @@ -92,8 +93,9 @@ AdjoiningGenerations::AdjoiningGenerations(ReservedSpace old_young_rs, } else { // Layout the reserved space for the generations. + // If OldGen is allocated on nv-dimm, we need to split the reservation (this is required for windows). ReservedSpace old_rs = - virtual_spaces()->reserved_space().first_part(max_low_byte_size); + virtual_spaces()->reserved_space().first_part(max_low_byte_size, policy->is_hetero_heap() /* split */); ReservedSpace heap_rs = virtual_spaces()->reserved_space().last_part(max_low_byte_size); ReservedSpace young_rs = heap_rs.first_part(max_high_byte_size); @@ -117,6 +119,8 @@ AdjoiningGenerations::AdjoiningGenerations(ReservedSpace old_young_rs, } } +AdjoiningGenerations::AdjoiningGenerations() { } + size_t AdjoiningGenerations::reserved_byte_size() { return virtual_spaces()->reserved_space().size(); } @@ -279,3 +283,13 @@ void AdjoiningGenerations::adjust_boundary_for_young_gen_needs(size_t eden_size, } } } + +AdjoiningGenerations* AdjoiningGenerations::create_adjoining_generations(ReservedSpace old_young_rs, + GenerationSizer* policy, + size_t alignment) { + if (policy->is_hetero_heap() && UseAdaptiveGCBoundary) { + return new AdjoiningGenerationsForHeteroHeap(old_young_rs, policy, alignment); + } else { + return new AdjoiningGenerations(old_young_rs, policy, alignment); + } +} diff --git a/src/hotspot/share/gc/parallel/adjoiningGenerations.hpp b/src/hotspot/share/gc/parallel/adjoiningGenerations.hpp index 3c0067a0909..7212918e92d 100644 --- a/src/hotspot/share/gc/parallel/adjoiningGenerations.hpp +++ b/src/hotspot/share/gc/parallel/adjoiningGenerations.hpp @@ -1,5 +1,5 @@ /* - * Copyright (c) 2003, 2015, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2003, 2018, 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 @@ -43,27 +43,29 @@ class AdjoiningGenerations : public CHeapObj { friend class VMStructs; private: - // The young generation and old generation, respectively - PSYoungGen* _young_gen; - PSOldGen* _old_gen; - - // The spaces used by the two generations. - AdjoiningVirtualSpaces _virtual_spaces; - // Move boundary up to expand old gen. Checks are made to // determine if the move can be done with specified limits. void request_old_gen_expansion(size_t desired_change_in_bytes); // Move boundary down to expand young gen. bool request_young_gen_expansion(size_t desired_change_in_bytes); + protected: + // The young generation and old generation, respectively + PSYoungGen* _young_gen; + PSOldGen* _old_gen; + + // The spaces used by the two generations. + AdjoiningVirtualSpaces* _virtual_spaces; + public: AdjoiningGenerations(ReservedSpace rs, GenerationSizer* policy, size_t alignment); + AdjoiningGenerations(); // Accessors PSYoungGen* young_gen() { return _young_gen; } PSOldGen* old_gen() { return _old_gen; } - AdjoiningVirtualSpaces* virtual_spaces() { return &_virtual_spaces; } + AdjoiningVirtualSpaces* virtual_spaces() { return _virtual_spaces; } // Additional space is needed in the old generation. Check // the available space and attempt to move the boundary if more space @@ -74,7 +76,9 @@ class AdjoiningGenerations : public CHeapObj { // Return the total byte size of the reserved space // for the adjoining generations. - size_t reserved_byte_size(); -}; + virtual size_t reserved_byte_size(); + // Return new AdjoiningGenerations instance based on collector policy (specifically - whether heap is heterogeneous). + static AdjoiningGenerations* create_adjoining_generations(ReservedSpace rs, GenerationSizer* policy, size_t alignment); +}; #endif // SHARE_VM_GC_PARALLEL_ADJOININGGENERATIONS_HPP diff --git a/src/hotspot/share/gc/parallel/adjoiningGenerationsForHeteroHeap.cpp b/src/hotspot/share/gc/parallel/adjoiningGenerationsForHeteroHeap.cpp new file mode 100644 index 00000000000..c2fe5e83abd --- /dev/null +++ b/src/hotspot/share/gc/parallel/adjoiningGenerationsForHeteroHeap.cpp @@ -0,0 +1,260 @@ +/* + * Copyright (c) 2018, 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. + * + */ + +#include "precompiled.hpp" +#include "gc/parallel/adjoiningGenerationsForHeteroHeap.hpp" +#include "gc/parallel/adjoiningVirtualSpaces.hpp" +#include "gc/parallel/generationSizer.hpp" +#include "gc/parallel/parallelScavengeHeap.hpp" +#include "gc/parallel/psFileBackedVirtualspace.hpp" +#include "logging/log.hpp" +#include "logging/logStream.hpp" +#include "memory/resourceArea.hpp" +#include "utilities/align.hpp" +#include "utilities/ostream.hpp" + +// Create two virtual spaces (HeteroVirtualSpaces), low() on nv-dimm memory, high() on dram. +// create ASPSOldGen and ASPSYoungGen the same way as in base class + +AdjoiningGenerationsForHeteroHeap::AdjoiningGenerationsForHeteroHeap(ReservedSpace old_young_rs, GenerationSizer* policy, size_t alignment) : + _total_size_limit(policy->max_heap_byte_size()) { + size_t init_old_byte_size = policy->initial_old_size(); + size_t min_old_byte_size = policy->min_old_size(); + size_t max_old_byte_size = policy->max_old_size(); + size_t init_young_byte_size = policy->initial_young_size(); + size_t min_young_byte_size = policy->min_young_size(); + size_t max_young_byte_size = policy->max_young_size(); + // create HeteroVirtualSpaces which is composed of non-overlapping virtual spaces. + HeteroVirtualSpaces* hetero_virtual_spaces = new HeteroVirtualSpaces(old_young_rs, min_old_byte_size, + min_young_byte_size, _total_size_limit, alignment); + + assert(min_old_byte_size <= init_old_byte_size && + init_old_byte_size <= max_old_byte_size, "Parameter check"); + assert(min_young_byte_size <= init_young_byte_size && + init_young_byte_size <= max_young_byte_size, "Parameter check"); + + assert(UseAdaptiveGCBoundary, "Should be used only when UseAdaptiveGCBoundary is true"); + + // Initialize the virtual spaces. Then pass a virtual space to each generation + // for initialization of the generation. + + // Does the actual creation of the virtual spaces + hetero_virtual_spaces->initialize(max_old_byte_size, init_old_byte_size, init_young_byte_size); + + _young_gen = new ASPSYoungGen(hetero_virtual_spaces->high(), + hetero_virtual_spaces->high()->committed_size() /* intial_size */, + min_young_byte_size, + hetero_virtual_spaces->max_young_size()); + + _old_gen = new ASPSOldGen(hetero_virtual_spaces->low(), + hetero_virtual_spaces->low()->committed_size() /* intial_size */, + min_old_byte_size, + hetero_virtual_spaces->max_old_size(), "old", 1); + + young_gen()->initialize_work(); + assert(young_gen()->reserved().byte_size() <= young_gen()->gen_size_limit(), "Consistency check"); + assert(old_young_rs.size() >= young_gen()->gen_size_limit(), "Consistency check"); + + old_gen()->initialize_work("old", 1); + assert(old_gen()->reserved().byte_size() <= old_gen()->gen_size_limit(), "Consistency check"); + assert(old_young_rs.size() >= old_gen()->gen_size_limit(), "Consistency check"); + + _virtual_spaces = hetero_virtual_spaces; +} + +size_t AdjoiningGenerationsForHeteroHeap::required_reserved_memory(GenerationSizer* policy) { + // This is the size that young gen can grow to, when AdaptiveGCBoundary is true. + size_t max_yg_size = policy->max_heap_byte_size() - policy->min_old_size(); + // This is the size that old gen can grow to, when AdaptiveGCBoundary is true. + size_t max_old_size = policy->max_heap_byte_size() - policy->min_young_size(); + + return max_yg_size + max_old_size; +} + +// We override this function since size of reservedspace here is more than heap size and +// callers expect this function to return heap size. +size_t AdjoiningGenerationsForHeteroHeap::reserved_byte_size() { + return total_size_limit(); +} + +AdjoiningGenerationsForHeteroHeap::HeteroVirtualSpaces::HeteroVirtualSpaces(ReservedSpace rs, size_t min_old_byte_size, size_t min_yg_byte_size, size_t max_total_size, size_t alignment) : + AdjoiningVirtualSpaces(rs, min_old_byte_size, min_yg_byte_size, alignment), + _max_total_size(max_total_size), + _min_old_byte_size(min_old_byte_size), _min_young_byte_size(min_yg_byte_size), + _max_old_byte_size(_max_total_size - _min_young_byte_size), + _max_young_byte_size(_max_total_size - _min_old_byte_size) { +} + +void AdjoiningGenerationsForHeteroHeap::HeteroVirtualSpaces::initialize(size_t initial_old_reserved_size, size_t init_old_byte_size, + size_t init_young_byte_size) { + + // This is the reserved space exclusively for old generation. + ReservedSpace low_rs = _reserved_space.first_part(_max_old_byte_size, true); + // Intially we only assign 'initial_old_reserved_size' of the reserved space to old virtual space. + low_rs = low_rs.first_part(initial_old_reserved_size); + + // This is the reserved space exclusively for young generation. + ReservedSpace high_rs = _reserved_space.last_part(_max_old_byte_size).first_part(_max_young_byte_size); + + // Carve out 'initial_young_reserved_size' of reserved space. + size_t initial_young_reserved_size = _max_total_size - initial_old_reserved_size; + high_rs = high_rs.last_part(_max_young_byte_size - initial_young_reserved_size); + + _low = new PSFileBackedVirtualSpace(low_rs, alignment(), AllocateOldGenAt); + if (!static_cast (_low)->initialize()) { + vm_exit_during_initialization("Could not map space for old generation at given AllocateOldGenAt path"); + } + + if (!_low->expand_by(init_old_byte_size)) { + vm_exit_during_initialization("Could not reserve enough space for object heap"); + } + + _high = new PSVirtualSpaceHighToLow(high_rs, alignment()); + if (!_high->expand_by(init_young_byte_size)) { + vm_exit_during_initialization("Could not reserve enough space for object heap"); + } +} + +// Since the virtual spaces are non-overlapping, there is no boundary as such. +// We replicate the same behavior and maintain the same invariants as base class 'AdjoiningVirtualSpaces' by +// increasing old generation size and decreasing young generation size by same amount. +bool AdjoiningGenerationsForHeteroHeap::HeteroVirtualSpaces::adjust_boundary_up(size_t change_in_bytes) { + assert(UseAdaptiveSizePolicy && UseAdaptiveGCBoundary, "runtime check"); + DEBUG_ONLY(size_t total_size_before = young_vs()->reserved_size() + old_vs()->reserved_size()); + + size_t bytes_needed = change_in_bytes; + size_t uncommitted_in_old = MIN2(old_vs()->uncommitted_size(), bytes_needed); + bool old_expanded = false; + + // 1. Try to expand old within its reserved space. + if (uncommitted_in_old != 0) { + if (!old_vs()->expand_by(uncommitted_in_old)) { + return false; + } + old_expanded = true; + bytes_needed -= uncommitted_in_old; + if (bytes_needed == 0) { + return true; + } + } + + size_t bytes_to_add_in_old = 0; + + // 2. Get uncommitted memory from Young virtualspace. + size_t young_uncommitted = MIN2(young_vs()->uncommitted_size(), bytes_needed); + if (young_uncommitted > 0) { + young_vs()->set_reserved(young_vs()->reserved_low_addr() + young_uncommitted, + young_vs()->reserved_high_addr(), + young_vs()->special()); + bytes_needed -= young_uncommitted; + bytes_to_add_in_old = young_uncommitted; + } + + // 3. Get committed memory from Young virtualspace + if (bytes_needed > 0) { + size_t shrink_size = align_down(bytes_needed, young_vs()->alignment()); + bool ret = young_vs()->shrink_by(shrink_size); + assert(ret, "We should be able to shrink young space"); + young_vs()->set_reserved(young_vs()->reserved_low_addr() + shrink_size, + young_vs()->reserved_high_addr(), + young_vs()->special()); + + bytes_to_add_in_old += shrink_size; + } + + // 4. Increase size of old space + old_vs()->set_reserved(old_vs()->reserved_low_addr(), + old_vs()->reserved_high_addr() + bytes_to_add_in_old, + old_vs()->special()); + if (!old_vs()->expand_by(bytes_to_add_in_old) && !old_expanded) { + return false; + } + + DEBUG_ONLY(size_t total_size_after = young_vs()->reserved_size() + old_vs()->reserved_size()); + assert(total_size_after == total_size_before, "should be equal"); + + return true; +} + +// Read comment for adjust_boundary_up() +// Increase young generation size and decrease old generation size by same amount. +bool AdjoiningGenerationsForHeteroHeap::HeteroVirtualSpaces::adjust_boundary_down(size_t change_in_bytes) { + assert(UseAdaptiveSizePolicy && UseAdaptiveGCBoundary, "runtime check"); + DEBUG_ONLY(size_t total_size_before = young_vs()->reserved_size() + old_vs()->reserved_size()); + + size_t bytes_needed = change_in_bytes; + size_t uncommitted_in_young = MIN2(young_vs()->uncommitted_size(), bytes_needed); + bool young_expanded = false; + + // 1. Try to expand old within its reserved space. + if (uncommitted_in_young > 0) { + if (!young_vs()->expand_by(uncommitted_in_young)) { + return false; + } + young_expanded = true; + bytes_needed -= uncommitted_in_young; + if (bytes_needed == 0) { + return true; + } + } + + size_t bytes_to_add_in_young = 0; + + // 2. Get uncommitted memory from Old virtualspace. + size_t old_uncommitted = MIN2(old_vs()->uncommitted_size(), bytes_needed); + if (old_uncommitted > 0) { + old_vs()->set_reserved(old_vs()->reserved_low_addr(), + old_vs()->reserved_high_addr() - old_uncommitted, + old_vs()->special()); + bytes_needed -= old_uncommitted; + bytes_to_add_in_young = old_uncommitted; + } + + // 3. Get committed memory from Old virtualspace + if (bytes_needed > 0) { + size_t shrink_size = align_down(bytes_needed, old_vs()->alignment()); + bool ret = old_vs()->shrink_by(shrink_size); + assert(ret, "We should be able to shrink young space"); + old_vs()->set_reserved(old_vs()->reserved_low_addr(), + old_vs()->reserved_high_addr() - shrink_size, + old_vs()->special()); + + bytes_to_add_in_young += shrink_size; + } + + assert(bytes_to_add_in_young <= change_in_bytes, "should not be more than requested size"); + // 4. Increase size of young space + young_vs()->set_reserved(young_vs()->reserved_low_addr() - bytes_to_add_in_young, + young_vs()->reserved_high_addr(), + young_vs()->special()); + if (!young_vs()->expand_by(bytes_to_add_in_young) && !young_expanded) { + return false; + } + + DEBUG_ONLY(size_t total_size_after = young_vs()->reserved_size() + old_vs()->reserved_size()); + assert(total_size_after == total_size_before, "should be equal"); + + return true; +} + diff --git a/src/hotspot/share/gc/parallel/adjoiningGenerationsForHeteroHeap.hpp b/src/hotspot/share/gc/parallel/adjoiningGenerationsForHeteroHeap.hpp new file mode 100644 index 00000000000..4e0accba451 --- /dev/null +++ b/src/hotspot/share/gc/parallel/adjoiningGenerationsForHeteroHeap.hpp @@ -0,0 +1,85 @@ +/* + * Copyright (c) 2018, 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. + * + */ + +#ifndef SHARE_VM_GC_PARALLEL_ADJOININGGENERATIONSFORHETEROHEAP_HPP +#define SHARE_VM_GC_PARALLEL_ADJOININGGENERATIONSFORHETEROHEAP_HPP + +#include "gc/parallel/adjoiningGenerations.hpp" + +class AdjoiningGenerationsForHeteroHeap : public AdjoiningGenerations { + friend class VMStructs; +private: + // Maximum total size of the generations. This is equal to the heap size specified by user. + // When adjusting young and old generation sizes, we need ensure that sum of the generation sizes does not exceed this. + size_t _total_size_limit; + + size_t total_size_limit() const { + return _total_size_limit; + } + + // HeteroVirtualSpaces creates non-overlapping virtual spaces. Here _low and _high do not share a reserved space, i.e. there is no boundary + // separating the two virtual spaces. + class HeteroVirtualSpaces : public AdjoiningVirtualSpaces { + size_t _max_total_size; + size_t _min_old_byte_size; + size_t _min_young_byte_size; + size_t _max_old_byte_size; + size_t _max_young_byte_size; + + // Internally we access the virtual spaces using these methods. It increases readability, since we were not really + // dealing with adjoining virtual spaces separated by a boundary as is the case in base class. + // Externally they are accessed using low() and high() methods of base class. + PSVirtualSpace* young_vs() { return high(); } + PSVirtualSpace* old_vs() { return low(); } + + public: + HeteroVirtualSpaces(ReservedSpace rs, + size_t min_old_byte_size, + size_t min_young_byte_size, size_t max_total_size, + size_t alignment); + + // Increase old generation size and decrease young generation size by same amount + bool adjust_boundary_up(size_t size_in_bytes); + // Increase young generation size and decrease old generation size by same amount + bool adjust_boundary_down(size_t size_in_bytes); + + size_t max_young_size() const { return _max_young_byte_size; } + size_t max_old_size() const { return _max_old_byte_size; } + + void initialize(size_t initial_old_reserved_size, size_t init_low_byte_size, + size_t init_high_byte_size); + }; + +public: + AdjoiningGenerationsForHeteroHeap(ReservedSpace rs, GenerationSizer* policy, size_t alignment); + + // Given the size policy, calculate the total amount of memory that needs to be reserved. + // We need to reserve more memory than Xmx, since we use non-overlapping virtual spaces for the young and old generations. + static size_t required_reserved_memory(GenerationSizer* policy); + + // Return the total byte size of the reserved space + size_t reserved_byte_size(); +}; +#endif // SHARE_VM_GC_PARALLEL_ADJOININGGENERATIONSFORHETEROHEAP_HPP + diff --git a/src/hotspot/share/gc/parallel/adjoiningVirtualSpaces.hpp b/src/hotspot/share/gc/parallel/adjoiningVirtualSpaces.hpp index 7078ab9db38..89fb5e5b018 100644 --- a/src/hotspot/share/gc/parallel/adjoiningVirtualSpaces.hpp +++ b/src/hotspot/share/gc/parallel/adjoiningVirtualSpaces.hpp @@ -1,5 +1,5 @@ /* - * Copyright (c) 2003, 2015, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2003, 2018, 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 @@ -59,7 +59,8 @@ // moved up consistently. AdjoiningVirtualSpaces provide the // interfaces for moving the this boundary. -class AdjoiningVirtualSpaces { +class AdjoiningVirtualSpaces : public CHeapObj { +protected: // space at the high end and the low end, respectively PSVirtualSpace* _high; PSVirtualSpace* _low; @@ -84,17 +85,17 @@ class AdjoiningVirtualSpaces { size_t alignment); // accessors - PSVirtualSpace* high() { return _high; } - PSVirtualSpace* low() { return _low; } + virtual PSVirtualSpace* high() { return _high; } + virtual PSVirtualSpace* low() { return _low; } ReservedSpace reserved_space() { return _reserved_space; } size_t min_low_byte_size() { return _min_low_byte_size; } size_t min_high_byte_size() { return _min_high_byte_size; } size_t alignment() const { return _alignment; } // move boundary between the two spaces up - bool adjust_boundary_up(size_t size_in_bytes); + virtual bool adjust_boundary_up(size_t size_in_bytes); // and down - bool adjust_boundary_down(size_t size_in_bytes); + virtual bool adjust_boundary_down(size_t size_in_bytes); // Maximum byte size for the high space. size_t high_byte_size_limit() { @@ -107,9 +108,8 @@ class AdjoiningVirtualSpaces { // Sets the boundaries for the virtual spaces and commits and // initial size; - void initialize(size_t max_low_byte_size, + virtual void initialize(size_t max_low_byte_size, size_t init_low_byte_size, size_t init_high_byte_size); }; - #endif // SHARE_VM_GC_PARALLEL_ADJOININGVIRTUALSPACES_HPP diff --git a/src/hotspot/share/gc/parallel/generationSizer.cpp b/src/hotspot/share/gc/parallel/generationSizer.cpp index 1d4bb96f9cc..f82095900c0 100644 --- a/src/hotspot/share/gc/parallel/generationSizer.cpp +++ b/src/hotspot/share/gc/parallel/generationSizer.cpp @@ -1,5 +1,5 @@ /* - * Copyright (c) 2001, 2015, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2001, 2018, 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 @@ -67,3 +67,11 @@ void GenerationSizer::initialize_size_info() { } GenCollectorPolicy::initialize_size_info(); } + +bool GenerationSizer::is_hetero_heap() const { + return false; +} + +size_t GenerationSizer::heap_reserved_size_bytes() const { + return _max_heap_byte_size; +} diff --git a/src/hotspot/share/gc/parallel/generationSizer.hpp b/src/hotspot/share/gc/parallel/generationSizer.hpp index 539629b1c96..80ee613901c 100644 --- a/src/hotspot/share/gc/parallel/generationSizer.hpp +++ b/src/hotspot/share/gc/parallel/generationSizer.hpp @@ -1,5 +1,5 @@ /* - * Copyright (c) 2001, 2015, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2001, 2018, 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 @@ -32,7 +32,6 @@ class GenerationSizer : public GenCollectorPolicy { private: - // The alignment used for boundary between young gen and old gen static size_t default_gen_alignment() { return 64 * K * HeapWordSize; } @@ -41,5 +40,9 @@ class GenerationSizer : public GenCollectorPolicy { void initialize_alignments(); void initialize_flags(); void initialize_size_info(); + + public: + virtual size_t heap_reserved_size_bytes() const; + virtual bool is_hetero_heap() const; }; #endif // SHARE_VM_GC_PARALLEL_GENERATIONSIZER_HPP diff --git a/src/hotspot/share/gc/parallel/heterogeneousGenerationSizer.cpp b/src/hotspot/share/gc/parallel/heterogeneousGenerationSizer.cpp new file mode 100644 index 00000000000..7e7987cdd1e --- /dev/null +++ b/src/hotspot/share/gc/parallel/heterogeneousGenerationSizer.cpp @@ -0,0 +1,106 @@ +/* + * Copyright (c) 2018, 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. + * + */ + +#include "precompiled.hpp" +#include "gc/parallel/heterogeneousGenerationSizer.hpp" +#include "gc/shared/collectorPolicy.hpp" +#include "logging/log.hpp" +#include "runtime/globals_extension.hpp" +#include "runtime/os.hpp" +#include "utilities/align.hpp" +#include "utilities/formatBuffer.hpp" +#include "utilities/globalDefinitions.hpp" + +const double HeterogeneousGenerationSizer::MaxRamFractionForYoung = 0.8; + +// Check the available dram memory to limit NewSize and MaxNewSize before +// calling base class initialize_flags(). +void HeterogeneousGenerationSizer::initialize_flags() { + FormatBuffer<100> calc_str(""); + + julong phys_mem; + // If MaxRam is specified, we use that as maximum physical memory available. + if (FLAG_IS_DEFAULT(MaxRAM)) { + phys_mem = os::physical_memory(); + calc_str.append("Physical_Memory"); + } else { + phys_mem = (julong)MaxRAM; + calc_str.append("MaxRAM"); + } + + julong reasonable_max = phys_mem; + + // If either MaxRAMFraction or MaxRAMPercentage is specified, we use them to calculate + // reasonable max size of young generation. + if (!FLAG_IS_DEFAULT(MaxRAMFraction)) { + reasonable_max = (julong)(phys_mem / MaxRAMFraction); + calc_str.append(" / MaxRAMFraction"); + } else if (!FLAG_IS_DEFAULT(MaxRAMPercentage)) { + reasonable_max = (julong)((phys_mem * MaxRAMPercentage) / 100); + calc_str.append(" * MaxRAMPercentage / 100"); + } else { + // We use our own fraction to calculate max size of young generation. + reasonable_max = phys_mem * MaxRamFractionForYoung; + calc_str.append(" * %0.2f", MaxRamFractionForYoung); + } + reasonable_max = align_up(reasonable_max, _gen_alignment); + + if (MaxNewSize > reasonable_max) { + if (FLAG_IS_CMDLINE(MaxNewSize)) { + log_warning(gc, ergo)("Setting MaxNewSize to " SIZE_FORMAT " based on dram available (calculation = align(%s))", + (size_t)reasonable_max, calc_str.buffer()); + } else { + log_info(gc, ergo)("Setting MaxNewSize to " SIZE_FORMAT " based on dram available (calculation = align(%s)). " + "Dram usage can be lowered by setting MaxNewSize to a lower value", (size_t)reasonable_max, calc_str.buffer()); + } + MaxNewSize = reasonable_max; + } + if (NewSize > reasonable_max) { + if (FLAG_IS_CMDLINE(NewSize)) { + log_warning(gc, ergo)("Setting NewSize to " SIZE_FORMAT " based on dram available (calculation = align(%s))", + (size_t)reasonable_max, calc_str.buffer()); + } + NewSize = reasonable_max; + } + + // After setting new size flags, call base class initialize_flags() + GenerationSizer::initialize_flags(); +} + +bool HeterogeneousGenerationSizer::is_hetero_heap() const { + return true; +} + +size_t HeterogeneousGenerationSizer::heap_reserved_size_bytes() const { + if (UseAdaptiveGCBoundary) { + // This is the size that young gen can grow to, when UseAdaptiveGCBoundary is true. + size_t max_yg_size = _max_heap_byte_size - _min_old_size; + // This is the size that old gen can grow to, when UseAdaptiveGCBoundary is true. + size_t max_old_size = _max_heap_byte_size - _min_young_size; + + return max_yg_size + max_old_size; + } else { + return _max_heap_byte_size; + } +} diff --git a/src/hotspot/share/gc/parallel/heterogeneousGenerationSizer.hpp b/src/hotspot/share/gc/parallel/heterogeneousGenerationSizer.hpp new file mode 100644 index 00000000000..809e00e5c18 --- /dev/null +++ b/src/hotspot/share/gc/parallel/heterogeneousGenerationSizer.hpp @@ -0,0 +1,46 @@ +/* + * Copyright (c) 2018, 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. + * + */ + +#ifndef SHARE_VM_GC_PARALLEL_HETEROGENEOUSGENERATIONSIZER_HPP +#define SHARE_VM_GC_PARALLEL_HETEROGENEOUSGENERATIONSIZER_HPP + +#include "gc/parallel/generationSizer.hpp" + +// There is a nice batch of tested generation sizing code in +// GenCollectorPolicy. Lets reuse it! + +class HeterogeneousGenerationSizer : public GenerationSizer { +private: + // Max fraction of dram to use for young generation when MaxRAMFraction and + // MaxRAMPercentage are not specified on commandline. + static const double MaxRamFractionForYoung; + +protected: + virtual void initialize_flags(); + +public: + virtual size_t heap_reserved_size_bytes() const; + virtual bool is_hetero_heap() const; +}; +#endif // SHARE_VM_GC_PARALLEL_HETEROGENEOUSGENERATIONSIZER_HPP diff --git a/src/hotspot/share/gc/parallel/parallelArguments.cpp b/src/hotspot/share/gc/parallel/parallelArguments.cpp index 6caacb68bb1..a844de08bec 100644 --- a/src/hotspot/share/gc/parallel/parallelArguments.cpp +++ b/src/hotspot/share/gc/parallel/parallelArguments.cpp @@ -24,6 +24,7 @@ */ #include "precompiled.hpp" +#include "gc/parallel/heterogeneousGenerationSizer.hpp" #include "gc/parallel/parallelArguments.hpp" #include "gc/parallel/parallelScavengeHeap.hpp" #include "gc/shared/adaptiveSizePolicy.hpp" @@ -93,5 +94,9 @@ void ParallelArguments::initialize() { } CollectedHeap* ParallelArguments::create_heap() { - return create_heap_with_policy(); + if (AllocateOldGenAt != NULL) { + return create_heap_with_policy(); + } else { + return create_heap_with_policy(); + } } diff --git a/src/hotspot/share/gc/parallel/parallelScavengeHeap.cpp b/src/hotspot/share/gc/parallel/parallelScavengeHeap.cpp index 7957d304099..c68a14d7fe6 100644 --- a/src/hotspot/share/gc/parallel/parallelScavengeHeap.cpp +++ b/src/hotspot/share/gc/parallel/parallelScavengeHeap.cpp @@ -25,6 +25,7 @@ #include "precompiled.hpp" #include "code/codeCache.hpp" #include "gc/parallel/adjoiningGenerations.hpp" +#include "gc/parallel/adjoiningGenerationsForHeteroHeap.hpp" #include "gc/parallel/adjoiningVirtualSpaces.hpp" #include "gc/parallel/gcTaskManager.hpp" #include "gc/parallel/generationSizer.hpp" @@ -58,7 +59,7 @@ PSGCAdaptivePolicyCounters* ParallelScavengeHeap::_gc_policy_counters = NULL; GCTaskManager* ParallelScavengeHeap::_gc_task_manager = NULL; jint ParallelScavengeHeap::initialize() { - const size_t heap_size = _collector_policy->max_heap_byte_size(); + size_t heap_size = _collector_policy->heap_reserved_size_bytes(); ReservedSpace heap_rs = Universe::reserve_heap(heap_size, _collector_policy->heap_alignment()); @@ -86,7 +87,7 @@ jint ParallelScavengeHeap::initialize() { double max_gc_pause_sec = ((double) MaxGCPauseMillis)/1000.0; double max_gc_minor_pause_sec = ((double) MaxGCMinorPauseMillis)/1000.0; - _gens = new AdjoiningGenerations(heap_rs, _collector_policy, generation_alignment()); + _gens = AdjoiningGenerations::create_adjoining_generations(heap_rs, _collector_policy, generation_alignment()); _old_gen = _gens->old_gen(); _young_gen = _gens->young_gen(); @@ -104,7 +105,7 @@ jint ParallelScavengeHeap::initialize() { GCTimeRatio ); - assert(!UseAdaptiveGCBoundary || + assert(_collector_policy->is_hetero_heap() || !UseAdaptiveGCBoundary || (old_gen()->virtual_space()->high_boundary() == young_gen()->virtual_space()->low_boundary()), "Boundaries must meet"); diff --git a/src/hotspot/share/gc/parallel/parallelScavengeHeap.hpp b/src/hotspot/share/gc/parallel/parallelScavengeHeap.hpp index 0f715fd291e..bddf4f2b84c 100644 --- a/src/hotspot/share/gc/parallel/parallelScavengeHeap.hpp +++ b/src/hotspot/share/gc/parallel/parallelScavengeHeap.hpp @@ -111,6 +111,8 @@ class ParallelScavengeHeap : public CollectedHeap { virtual CollectorPolicy* collector_policy() const { return _collector_policy; } + virtual GenerationSizer* ps_collector_policy() const { return _collector_policy; } + virtual SoftRefPolicy* soft_ref_policy() { return &_soft_ref_policy; } virtual GrowableArray memory_managers(); diff --git a/src/hotspot/share/gc/parallel/psFileBackedVirtualspace.cpp b/src/hotspot/share/gc/parallel/psFileBackedVirtualspace.cpp new file mode 100644 index 00000000000..83172401530 --- /dev/null +++ b/src/hotspot/share/gc/parallel/psFileBackedVirtualspace.cpp @@ -0,0 +1,87 @@ +/* + * Copyright (c) 2018, 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. + * + */ + +#include "precompiled.hpp" +#include "gc/parallel/psFileBackedVirtualspace.hpp" +#include "memory/virtualspace.hpp" +#include "runtime/os.inline.hpp" + +PSFileBackedVirtualSpace::PSFileBackedVirtualSpace(ReservedSpace rs, size_t alignment, const char* path) : PSVirtualSpace(rs, alignment), + _file_path(path), _fd(-1), _mapping_succeeded(false) { + assert(!rs.special(), "ReservedSpace passed to PSFileBackedVirtualSpace cannot be special"); +} + +bool PSFileBackedVirtualSpace::initialize() { + _fd = os::create_file_for_heap(_file_path); + if (_fd == -1) { + return false; + } + // We map the reserved space to a file at initialization. + char* ret = os::replace_existing_mapping_with_file_mapping(reserved_low_addr(), reserved_size(), _fd); + if (ret != reserved_low_addr()) { + os::close(_fd); + return false; + } + // _mapping_succeeded is false if we return before this point. + // expand calls later check value of this flag and return error if it is false. + _mapping_succeeded = true; + _special = true; + os::close(_fd); + return true; +} + +PSFileBackedVirtualSpace::PSFileBackedVirtualSpace(ReservedSpace rs, const char* path) { + PSFileBackedVirtualSpace(rs, os::vm_page_size(), path); +} + +bool PSFileBackedVirtualSpace::expand_by(size_t bytes) { + assert(special(), "Since entire space is committed at initialization, _special should always be true for PSFileBackedVirtualSpace"); + + // if mapping did not succeed during intialization return false + if (!_mapping_succeeded) { + return false; + } + return PSVirtualSpace::expand_by(bytes); + +} + +bool PSFileBackedVirtualSpace::shrink_by(size_t bytes) { + assert(special(), "Since entire space is committed at initialization, _special should always be true for PSFileBackedVirtualSpace"); + return PSVirtualSpace::shrink_by(bytes); +} + +size_t PSFileBackedVirtualSpace::expand_into(PSVirtualSpace* space, size_t bytes) { + // not supported. Since doing this will change page mapping which will lead to large TLB penalties. + assert(false, "expand_into() should not be called for PSFileBackedVirtualSpace"); + return 0; +} + +void PSFileBackedVirtualSpace::release() { + os::close(_fd); + _fd = -1; + _file_path = NULL; + + PSVirtualSpace::release(); +} + diff --git a/src/hotspot/share/gc/parallel/psFileBackedVirtualspace.hpp b/src/hotspot/share/gc/parallel/psFileBackedVirtualspace.hpp new file mode 100644 index 00000000000..ae25141cb46 --- /dev/null +++ b/src/hotspot/share/gc/parallel/psFileBackedVirtualspace.hpp @@ -0,0 +1,46 @@ +/* + * Copyright (c) 2018, 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. + * + */ + +#ifndef SHARE_VM_GC_PARALLEL_PSFILEBACKEDVIRTUALSPACE_HPP +#define SHARE_VM_GC_PARALLEL_PSFILEBACKEDVIRTUALSPACE_HPP + +#include "gc/parallel/psVirtualspace.hpp" + +class PSFileBackedVirtualSpace : public PSVirtualSpace { +private: + const char* _file_path; + int _fd; + bool _mapping_succeeded; +public: + PSFileBackedVirtualSpace(ReservedSpace rs, size_t alignment, const char* file_path); + PSFileBackedVirtualSpace(ReservedSpace rs, const char* file_path); + + bool initialize(); + bool expand_by(size_t bytes); + bool shrink_by(size_t bytes); + size_t expand_into(PSVirtualSpace* space, size_t bytes); + void release(); +}; +#endif // SHARE_VM_GC_PARALLEL_PSFILEBACKEDVIRTUALSPACE_HPP + diff --git a/src/hotspot/share/gc/parallel/psOldGen.cpp b/src/hotspot/share/gc/parallel/psOldGen.cpp index 35844b14b62..4b6afbb4eaf 100644 --- a/src/hotspot/share/gc/parallel/psOldGen.cpp +++ b/src/hotspot/share/gc/parallel/psOldGen.cpp @@ -27,6 +27,7 @@ #include "gc/parallel/parallelScavengeHeap.hpp" #include "gc/parallel/psAdaptiveSizePolicy.hpp" #include "gc/parallel/psCardTable.hpp" +#include "gc/parallel/psFileBackedVirtualspace.hpp" #include "gc/parallel/psMarkSweepDecorator.hpp" #include "gc/parallel/psOldGen.hpp" #include "gc/shared/cardTableBarrierSet.hpp" @@ -71,7 +72,14 @@ void PSOldGen::initialize(ReservedSpace rs, size_t alignment, void PSOldGen::initialize_virtual_space(ReservedSpace rs, size_t alignment) { - _virtual_space = new PSVirtualSpace(rs, alignment); + if(ParallelScavengeHeap::heap()->ps_collector_policy()->is_hetero_heap()) { + _virtual_space = new PSFileBackedVirtualSpace(rs, alignment, AllocateOldGenAt); + if (!(static_cast (_virtual_space))->initialize()) { + vm_exit_during_initialization("Could not map space for PSOldGen at given AllocateOldGenAt path"); + } + } else { + _virtual_space = new PSVirtualSpace(rs, alignment); + } if (!_virtual_space->expand_by(_init_gen_size)) { vm_exit_during_initialization("Could not reserve enough space for " "object heap"); diff --git a/src/hotspot/share/gc/parallel/psParallelCompact.cpp b/src/hotspot/share/gc/parallel/psParallelCompact.cpp index e3817797557..64ce4650711 100644 --- a/src/hotspot/share/gc/parallel/psParallelCompact.cpp +++ b/src/hotspot/share/gc/parallel/psParallelCompact.cpp @@ -1995,7 +1995,10 @@ bool PSParallelCompact::absorb_live_data_from_eden(PSAdaptiveSizePolicy* size_po assert(young_gen->virtual_space()->alignment() == old_gen->virtual_space()->alignment(), "alignments do not match"); - if (!(UseAdaptiveSizePolicy && UseAdaptiveGCBoundary)) { + // We also return false when it's a heterogenous heap because old generation cannot absorb data from eden + // when it is allocated on different memory (example, nv-dimm) than young. + if (!(UseAdaptiveSizePolicy && UseAdaptiveGCBoundary) || + ParallelScavengeHeap::heap()->ps_collector_policy()->is_hetero_heap()) { return false; } diff --git a/src/hotspot/share/prims/whitebox.cpp b/src/hotspot/share/prims/whitebox.cpp index 03223ad0b57..f6a7f33bf43 100644 --- a/src/hotspot/share/prims/whitebox.cpp +++ b/src/hotspot/share/prims/whitebox.cpp @@ -500,7 +500,7 @@ WB_END #endif // INCLUDE_G1GC -#if INCLUDE_G1GC +#if INCLUDE_G1GC || INCLUDE_PARALLELGC WB_ENTRY(jlong, WB_DramReservedStart(JNIEnv* env, jobject o)) if (UseG1GC) { G1CollectedHeap* g1h = G1CollectedHeap::heap(); @@ -511,7 +511,16 @@ WB_ENTRY(jlong, WB_DramReservedStart(JNIEnv* env, jobject o)) return (jlong)g1h->base(); } } - THROW_MSG_0(vmSymbols::java_lang_UnsupportedOperationException(), "WB_DramReservedStart: enabled only for G1"); + if (UseParallelGC) { + ParallelScavengeHeap* ps_heap = ParallelScavengeHeap::heap(); + if (AllocateOldGenAt != NULL) { + MemRegion reserved = ps_heap->young_gen()->reserved(); + return (jlong)reserved.start(); + } else { + return (jlong)ps_heap->base(); + } + } + THROW_MSG_0(vmSymbols::java_lang_UnsupportedOperationException(), "WB_DramReservedStart: enabled only for G1 and Parallel GC"); WB_END WB_ENTRY(jlong, WB_DramReservedEnd(JNIEnv* env, jobject o)) @@ -524,7 +533,16 @@ WB_ENTRY(jlong, WB_DramReservedEnd(JNIEnv* env, jobject o)) return (jlong)g1h->base() + g1h->collector_policy()->max_heap_byte_size(); } } - THROW_MSG_0(vmSymbols::java_lang_UnsupportedOperationException(), "WB_DramReservedEnd: enabled only for G1"); + if (UseParallelGC) { + ParallelScavengeHeap* ps_heap = ParallelScavengeHeap::heap(); + if (AllocateOldGenAt != NULL) { + MemRegion reserved = ps_heap->young_gen()->reserved(); + return (jlong)reserved.end(); + } else { + return (jlong)ps_heap->reserved_region().end(); + } + } + THROW_MSG_0(vmSymbols::java_lang_UnsupportedOperationException(), "WB_DramReservedEnd: enabled only for G1 and Parallel GC"); WB_END WB_ENTRY(jlong, WB_NvdimmReservedStart(JNIEnv* env, jobject o)) @@ -537,7 +555,16 @@ WB_ENTRY(jlong, WB_NvdimmReservedStart(JNIEnv* env, jobject o)) THROW_MSG_0(vmSymbols::java_lang_UnsupportedOperationException(), "WB_NvdimmReservedStart: Old gen is not allocated on NV-DIMM using AllocateOldGenAt flag"); } } - THROW_MSG_0(vmSymbols::java_lang_UnsupportedOperationException(), "WB_NvdimmReservedStart: enabled only for G1"); + if (UseParallelGC) { + ParallelScavengeHeap* ps_heap = ParallelScavengeHeap::heap(); + if (AllocateOldGenAt != NULL) { + MemRegion reserved = ps_heap->old_gen()->reserved(); + return (jlong)reserved.start(); + } else { + THROW_MSG_0(vmSymbols::java_lang_UnsupportedOperationException(), "WB_NvdimmReservedStart: Old gen is not allocated on NV-DIMM using AllocateOldGenAt flag"); + } + } + THROW_MSG_0(vmSymbols::java_lang_UnsupportedOperationException(), "WB_NvdimmReservedStart: enabled only for G1 and Parallel GC"); WB_END WB_ENTRY(jlong, WB_NvdimmReservedEnd(JNIEnv* env, jobject o)) @@ -550,10 +577,19 @@ WB_ENTRY(jlong, WB_NvdimmReservedEnd(JNIEnv* env, jobject o)) THROW_MSG_0(vmSymbols::java_lang_UnsupportedOperationException(), "WB_NvdimmReservedEnd: Old gen is not allocated on NV-DIMM using AllocateOldGenAt flag"); } } - THROW_MSG_0(vmSymbols::java_lang_UnsupportedOperationException(), "WB_NvdimmReservedEnd: enabled only for G1"); + if (UseParallelGC) { + ParallelScavengeHeap* ps_heap = ParallelScavengeHeap::heap(); + if (AllocateOldGenAt != NULL) { + MemRegion reserved = ps_heap->old_gen()->reserved(); + return (jlong)reserved.end(); + } else { + THROW_MSG_0(vmSymbols::java_lang_UnsupportedOperationException(), "WB_NvdimmReservedEnd: Old gen is not allocated on NV-DIMM using AllocateOldGenAt flag"); + } + } + THROW_MSG_0(vmSymbols::java_lang_UnsupportedOperationException(), "WB_NvdimmReservedEnd: enabled only for G1 and Parallel GC"); WB_END -#endif // INCLUDE_G1GC +#endif // INCLUDE_G1GC || INCLUDE_PARALLELGC #if INCLUDE_PARALLELGC @@ -2109,11 +2145,13 @@ static JNINativeMethod methods[] = { {CC"g1AuxiliaryMemoryUsage", CC"()Ljava/lang/management/MemoryUsage;", (void*)&WB_G1AuxiliaryMemoryUsage }, {CC"g1GetMixedGCInfo", CC"(I)[J", (void*)&WB_G1GetMixedGCInfo }, +#endif // INCLUDE_G1GC +#if INCLUDE_G1GC || INCLUDE_PARALLELGC {CC"dramReservedStart", CC"()J", (void*)&WB_DramReservedStart }, {CC"dramReservedEnd", CC"()J", (void*)&WB_DramReservedEnd }, {CC"nvdimmReservedStart", CC"()J", (void*)&WB_NvdimmReservedStart }, {CC"nvdimmReservedEnd", CC"()J", (void*)&WB_NvdimmReservedEnd }, -#endif // INCLUDE_G1GC +#endif // INCLUDE_G1GC || INCLUDE_PARALLELGC #if INCLUDE_PARALLELGC {CC"psVirtualSpaceAlignment",CC"()J", (void*)&WB_PSVirtualSpaceAlignment}, {CC"psHeapGenerationAlignment",CC"()J", (void*)&WB_PSHeapGenerationAlignment}, diff --git a/test/hotspot/jtreg/gc/nvdimm/TestAllocateOldGenAt.java b/test/hotspot/jtreg/gc/nvdimm/TestAllocateOldGenAt.java index d8149d5ec1b..5eea1efa81c 100644 --- a/test/hotspot/jtreg/gc/nvdimm/TestAllocateOldGenAt.java +++ b/test/hotspot/jtreg/gc/nvdimm/TestAllocateOldGenAt.java @@ -54,6 +54,8 @@ public class TestAllocateOldGenAt { "-version"}); runTest("-XX:+UseG1GC"); + runTest("-XX:+UseParallelOldGC -XX:-UseAdaptiveGCBoundary"); + runTest("-XX:+UseParallelOldGC -XX:+UseAdaptiveGCBoundary"); } private static void runTest(String... extraFlags) throws Exception { diff --git a/test/hotspot/jtreg/gc/nvdimm/TestAllocateOldGenAtError.java b/test/hotspot/jtreg/gc/nvdimm/TestAllocateOldGenAtError.java index 63cd39bf1ec..4f0c4ae91ab 100644 --- a/test/hotspot/jtreg/gc/nvdimm/TestAllocateOldGenAtError.java +++ b/test/hotspot/jtreg/gc/nvdimm/TestAllocateOldGenAtError.java @@ -63,6 +63,7 @@ public class TestAllocateOldGenAtError { "-version"}); testG1(); + testParallelOld(); } private static void testG1() throws Exception { @@ -73,6 +74,19 @@ public class TestAllocateOldGenAtError { output.shouldContain("Could not initialize G1 heap"); output.shouldContain("Error occurred during initialization of VM"); output.shouldNotHaveExitValue(0); + + } + + private static void testParallelOld() throws Exception { + System.out.println("Testing ParallelOld GC with UseAdaptiveGCBoundary disabled"); + OutputAnalyzer output = runTest("-XX:+UseParallelOldGC -XX:-UseAdaptiveGCBoundary"); + output.shouldContain("Error occurred during initialization of VM"); + output.shouldNotHaveExitValue(0); + + System.out.println("Testing ParallelOld GC with UseAdaptiveGCBoundary enabled"); + output = runTest("-XX:+UseParallelOldGC -XX:+UseAdaptiveGCBoundary"); + output.shouldContain("Error occurred during initialization of VM"); + output.shouldNotHaveExitValue(0); } private static OutputAnalyzer runTest(String... extraFlags) throws Exception { diff --git a/test/hotspot/jtreg/gc/nvdimm/TestOldObjectsOnNvdimm.java b/test/hotspot/jtreg/gc/nvdimm/TestOldObjectsOnNvdimm.java index 9a7489ab6b0..8a14980c669 100644 --- a/test/hotspot/jtreg/gc/nvdimm/TestOldObjectsOnNvdimm.java +++ b/test/hotspot/jtreg/gc/nvdimm/TestOldObjectsOnNvdimm.java @@ -72,6 +72,9 @@ public class TestOldObjectsOnNvdimm { // Test with G1 GC runTest("-XX:+UseG1GC"); + // Test with ParallelOld GC + runTest("-XX:+UseParallelOldGC -XX:-UseAdaptiveGCBoundary"); + runTest("-XX:+UseParallelOldGC -XX:+UseAdaptiveGCBoundary"); } private static void runTest(String... extraFlags) throws Exception { diff --git a/test/hotspot/jtreg/gc/nvdimm/TestYoungObjectsOnDram.java b/test/hotspot/jtreg/gc/nvdimm/TestYoungObjectsOnDram.java index 2f84d441e59..4ea0dbca9ee 100644 --- a/test/hotspot/jtreg/gc/nvdimm/TestYoungObjectsOnDram.java +++ b/test/hotspot/jtreg/gc/nvdimm/TestYoungObjectsOnDram.java @@ -73,6 +73,9 @@ public class TestYoungObjectsOnDram { // Test with G1 GC runTest("-XX:+UseG1GC"); + // Test with ParallelOld GC + runTest("-XX:+UseParallelOldGC -XX:-UseAdaptiveGCBoundary"); + runTest("-XX:+UseParallelOldGC -XX:+UseAdaptiveGCBoundary"); } private static void runTest(String... extraFlags) throws Exception { From 65b77b109b4cb3fd0672aa44a4251d16d3c93122 Mon Sep 17 00:00:00 2001 From: Stefan Johansson Date: Sat, 22 Dec 2018 15:46:54 +0100 Subject: [PATCH 044/101] 8215897: Build broken on zero after JDK-8211424 Reviewed-by: tschatzl --- src/hotspot/share/prims/whitebox.cpp | 16 ++++++++++++++++ 1 file changed, 16 insertions(+) diff --git a/src/hotspot/share/prims/whitebox.cpp b/src/hotspot/share/prims/whitebox.cpp index f6a7f33bf43..342bd0f68af 100644 --- a/src/hotspot/share/prims/whitebox.cpp +++ b/src/hotspot/share/prims/whitebox.cpp @@ -502,6 +502,7 @@ WB_END #if INCLUDE_G1GC || INCLUDE_PARALLELGC WB_ENTRY(jlong, WB_DramReservedStart(JNIEnv* env, jobject o)) +#if INCLUDE_G1GC if (UseG1GC) { G1CollectedHeap* g1h = G1CollectedHeap::heap(); if (g1h->g1_collector_policy()->is_hetero_heap()) { @@ -511,6 +512,8 @@ WB_ENTRY(jlong, WB_DramReservedStart(JNIEnv* env, jobject o)) return (jlong)g1h->base(); } } +#endif // INCLUDE_G1GC +#if INCLUDE_PARALLELGC if (UseParallelGC) { ParallelScavengeHeap* ps_heap = ParallelScavengeHeap::heap(); if (AllocateOldGenAt != NULL) { @@ -520,10 +523,12 @@ WB_ENTRY(jlong, WB_DramReservedStart(JNIEnv* env, jobject o)) return (jlong)ps_heap->base(); } } +#endif // INCLUDE_PARALLELGC THROW_MSG_0(vmSymbols::java_lang_UnsupportedOperationException(), "WB_DramReservedStart: enabled only for G1 and Parallel GC"); WB_END WB_ENTRY(jlong, WB_DramReservedEnd(JNIEnv* env, jobject o)) +#if INCLUDE_G1GC if (UseG1GC) { G1CollectedHeap* g1h = G1CollectedHeap::heap(); if (g1h->g1_collector_policy()->is_hetero_heap()) { @@ -533,6 +538,8 @@ WB_ENTRY(jlong, WB_DramReservedEnd(JNIEnv* env, jobject o)) return (jlong)g1h->base() + g1h->collector_policy()->max_heap_byte_size(); } } +#endif // INCLUDE_G1GC +#if INCLUDE_PARALLELGC if (UseParallelGC) { ParallelScavengeHeap* ps_heap = ParallelScavengeHeap::heap(); if (AllocateOldGenAt != NULL) { @@ -542,10 +549,12 @@ WB_ENTRY(jlong, WB_DramReservedEnd(JNIEnv* env, jobject o)) return (jlong)ps_heap->reserved_region().end(); } } +#endif // INCLUDE_PARALLELGC THROW_MSG_0(vmSymbols::java_lang_UnsupportedOperationException(), "WB_DramReservedEnd: enabled only for G1 and Parallel GC"); WB_END WB_ENTRY(jlong, WB_NvdimmReservedStart(JNIEnv* env, jobject o)) +#if INCLUDE_G1GC if (UseG1GC) { G1CollectedHeap* g1h = G1CollectedHeap::heap(); if (g1h->g1_collector_policy()->is_hetero_heap()) { @@ -555,6 +564,8 @@ WB_ENTRY(jlong, WB_NvdimmReservedStart(JNIEnv* env, jobject o)) THROW_MSG_0(vmSymbols::java_lang_UnsupportedOperationException(), "WB_NvdimmReservedStart: Old gen is not allocated on NV-DIMM using AllocateOldGenAt flag"); } } +#endif // INCLUDE_G1GC +#if INCLUDE_PARALLELGC if (UseParallelGC) { ParallelScavengeHeap* ps_heap = ParallelScavengeHeap::heap(); if (AllocateOldGenAt != NULL) { @@ -564,10 +575,12 @@ WB_ENTRY(jlong, WB_NvdimmReservedStart(JNIEnv* env, jobject o)) THROW_MSG_0(vmSymbols::java_lang_UnsupportedOperationException(), "WB_NvdimmReservedStart: Old gen is not allocated on NV-DIMM using AllocateOldGenAt flag"); } } +#endif // INCLUDE_PARALLELGC THROW_MSG_0(vmSymbols::java_lang_UnsupportedOperationException(), "WB_NvdimmReservedStart: enabled only for G1 and Parallel GC"); WB_END WB_ENTRY(jlong, WB_NvdimmReservedEnd(JNIEnv* env, jobject o)) +#if INCLUDE_G1GC if (UseG1GC) { G1CollectedHeap* g1h = G1CollectedHeap::heap(); if (g1h->g1_collector_policy()->is_hetero_heap()) { @@ -577,6 +590,8 @@ WB_ENTRY(jlong, WB_NvdimmReservedEnd(JNIEnv* env, jobject o)) THROW_MSG_0(vmSymbols::java_lang_UnsupportedOperationException(), "WB_NvdimmReservedEnd: Old gen is not allocated on NV-DIMM using AllocateOldGenAt flag"); } } +#endif // INCLUDE_G1GC +#if INCLUDE_PARALLELGC if (UseParallelGC) { ParallelScavengeHeap* ps_heap = ParallelScavengeHeap::heap(); if (AllocateOldGenAt != NULL) { @@ -586,6 +601,7 @@ WB_ENTRY(jlong, WB_NvdimmReservedEnd(JNIEnv* env, jobject o)) THROW_MSG_0(vmSymbols::java_lang_UnsupportedOperationException(), "WB_NvdimmReservedEnd: Old gen is not allocated on NV-DIMM using AllocateOldGenAt flag"); } } +#endif // INCLUDE_PARALLELGC THROW_MSG_0(vmSymbols::java_lang_UnsupportedOperationException(), "WB_NvdimmReservedEnd: enabled only for G1 and Parallel GC"); WB_END From b91fa3a6c8efb3525d141210a9c26d7e0569b72f Mon Sep 17 00:00:00 2001 From: Stefan Johansson Date: Sat, 22 Dec 2018 15:47:10 +0100 Subject: [PATCH 045/101] 8215898: Build broken on 32-bit after JDK-8211425 Reviewed-by: tschatzl --- src/hotspot/share/gc/shared/gcArguments.cpp | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/src/hotspot/share/gc/shared/gcArguments.cpp b/src/hotspot/share/gc/shared/gcArguments.cpp index 75a35b191ad..86d387f3549 100644 --- a/src/hotspot/share/gc/shared/gcArguments.cpp +++ b/src/hotspot/share/gc/shared/gcArguments.cpp @@ -57,8 +57,8 @@ void GCArguments::initialize() { if (!FLAG_IS_DEFAULT(AllocateOldGenAt)) { // CompressedOops not supported when AllocateOldGenAt is set. - FLAG_SET_DEFAULT(UseCompressedOops, false); - FLAG_SET_DEFAULT(UseCompressedClassPointers, false); + LP64_ONLY(FLAG_SET_DEFAULT(UseCompressedOops, false)); + LP64_ONLY(FLAG_SET_DEFAULT(UseCompressedClassPointers, false)); // When AllocateOldGenAt is set, we cannot use largepages for entire heap memory. // Only young gen which is allocated in dram can use large pages, but we currently don't support that. FLAG_SET_DEFAULT(UseLargePages, false); From 6774e567aabc2ab8137fd4b12ba580342d21d36b Mon Sep 17 00:00:00 2001 From: Deepak kejriwal Date: Wed, 26 Dec 2018 17:09:19 +0530 Subject: [PATCH 046/101] 8214567: Use {@systemProperty} for definitions of system properties 8214569: Use {@systemProperty} for definitions of system properties Reviewed-by: lancea, mchung, alanb, naoto --- .../share/classes/java/lang/ClassLoader.java | 12 ++++++------ src/java.base/share/classes/java/net/URL.java | 2 +- .../classes/java/time/zone/ZoneRulesProvider.java | 2 +- src/java.base/share/classes/java/util/Currency.java | 2 +- .../classes/java/util/PropertyResourceBundle.java | 2 +- .../share/classes/java/util/jar/Pack200.java | 4 ++-- .../classes/java/util/spi/LocaleServiceProvider.java | 2 +- .../share/classes/java/util/logging/LogManager.java | 4 ++-- .../classes/java/util/logging/SimpleFormatter.java | 2 +- .../share/classes/java/rmi/server/ObjID.java | 2 +- .../classes/java/rmi/server/RMIClassLoader.java | 4 ++-- .../classes/java/rmi/server/RMISocketFactory.java | 2 +- .../javax/rmi/ssl/SslRMIClientSocketFactory.java | 8 ++++---- 13 files changed, 24 insertions(+), 24 deletions(-) diff --git a/src/java.base/share/classes/java/lang/ClassLoader.java b/src/java.base/share/classes/java/lang/ClassLoader.java index 71d4898450b..60e5a2f4aef 100644 --- a/src/java.base/share/classes/java/lang/ClassLoader.java +++ b/src/java.base/share/classes/java/lang/ClassLoader.java @@ -1864,12 +1864,12 @@ public abstract class ClassLoader { *

    The default system class loader is an implementation-dependent * instance of this class. * - *

    If the system property "{@code java.system.class.loader}" is defined - * when this method is first invoked then the value of that property is - * taken to be the name of a class that will be returned as the system - * class loader. The class is loaded using the default system class loader - * and must define a public constructor that takes a single parameter of - * type {@code ClassLoader} which is used as the delegation parent. An + *

    If the system property "{@systemProperty java.system.class.loader}" + * is defined when this method is first invoked then the value of that + * property is taken to be the name of a class that will be returned as the + * system class loader. The class is loaded using the default system class + * loader and must define a public constructor that takes a single parameter + * of type {@code ClassLoader} which is used as the delegation parent. An * instance is then created using this constructor with the default system * class loader as the parameter. The resulting class loader is defined * to be the system class loader. During construction, the class loader diff --git a/src/java.base/share/classes/java/net/URL.java b/src/java.base/share/classes/java/net/URL.java index 9d4abcc3901..6d2c351aa90 100644 --- a/src/java.base/share/classes/java/net/URL.java +++ b/src/java.base/share/classes/java/net/URL.java @@ -304,7 +304,7 @@ public final class URL implements java.io.Serializable { * or all providers have been exhausted. *

  2. If the previous step fails to find a protocol handler, the * constructor reads the value of the system property: - *
    {@code + *
    {@systemProperty * java.protocol.handler.pkgs * }
    * If the value of that system property is not {@code null}, diff --git a/src/java.base/share/classes/java/time/zone/ZoneRulesProvider.java b/src/java.base/share/classes/java/time/zone/ZoneRulesProvider.java index 03e618731fc..fdae3a6cd07 100644 --- a/src/java.base/share/classes/java/time/zone/ZoneRulesProvider.java +++ b/src/java.base/share/classes/java/time/zone/ZoneRulesProvider.java @@ -99,7 +99,7 @@ import java.util.Collections; *

    * The Java virtual machine has a default provider that provides zone rules * for the time-zones defined by IANA Time Zone Database (TZDB). If the system - * property {@code java.time.zone.DefaultZoneRulesProvider} is defined then + * property {@systemProperty java.time.zone.DefaultZoneRulesProvider} is defined then * it is taken to be the fully-qualified name of a concrete ZoneRulesProvider * class to be loaded as the default provider, using the system class loader. * If this system property is not defined, a system-default provider will be diff --git a/src/java.base/share/classes/java/util/Currency.java b/src/java.base/share/classes/java/util/Currency.java index 415334ab973..945579a92d0 100644 --- a/src/java.base/share/classes/java/util/Currency.java +++ b/src/java.base/share/classes/java/util/Currency.java @@ -60,7 +60,7 @@ import sun.util.logging.PlatformLogger; * the getInstance methods. *

    * Users can supersede the Java runtime currency data by means of the system - * property {@code java.util.currency.data}. If this system property is + * property {@systemProperty java.util.currency.data}. If this system property is * defined then its value is the location of a properties file, the contents of * which are key/value pairs of the ISO 3166 country codes and the ISO 4217 * currency data respectively. The value part consists of three ISO 4217 values diff --git a/src/java.base/share/classes/java/util/PropertyResourceBundle.java b/src/java.base/share/classes/java/util/PropertyResourceBundle.java index a7231fbef99..dfe6ef986a7 100644 --- a/src/java.base/share/classes/java/util/PropertyResourceBundle.java +++ b/src/java.base/share/classes/java/util/PropertyResourceBundle.java @@ -115,7 +115,7 @@ import sun.util.ResourceBundleEnumeration; * input stream, then the {@code PropertyResourceBundle} instance resets to the state * before the exception, re-reads the input stream in {@code ISO-8859-1}, and * continues reading. If the system property - * {@code java.util.PropertyResourceBundle.encoding} is set to either + * {@systemProperty java.util.PropertyResourceBundle.encoding} is set to either * "ISO-8859-1" or "UTF-8", the input stream is solely read in that encoding, * and throws the exception if it encounters an invalid sequence. * If "ISO-8859-1" is specified, characters that cannot be represented in diff --git a/src/java.base/share/classes/java/util/jar/Pack200.java b/src/java.base/share/classes/java/util/jar/Pack200.java index 571035caeb9..029408e6fb8 100644 --- a/src/java.base/share/classes/java/util/jar/Pack200.java +++ b/src/java.base/share/classes/java/util/jar/Pack200.java @@ -112,7 +112,7 @@ public abstract class Pack200 { /** * Obtain new instance of a class that implements Packer. *