From 378aef32ab650eb883f86c76ea06a8c6ac1333ad Mon Sep 17 00:00:00 2001 From: Patrick Concannon Date: Tue, 7 Apr 2020 16:21:01 +0100 Subject: [PATCH] 8240533: Inconsistent Exceptions are thrown by DatagramSocket and DatagramChannel when sending a DatagramPacket to port 0 Fix adds checks for port == 0 to the send and connect methods in DatagramSocket and DatagramChannelImpl Reviewed-by: alanb, chegar, dfuchs, lancea --- .../classes/java/net/DatagramSocket.java | 6 + .../sun/nio/ch/DatagramChannelImpl.java | 5 + .../net/DatagramSocket/ConnectPortZero.java | 133 ++++++++++++++++ .../java/net/DatagramSocket/SendPortZero.java | 146 ++++++++++++++++++ .../DatagramChannel/ConnectPortZero.java | 133 ++++++++++++++++ .../DatagramChannel/SendPortZero.java | 137 ++++++++++++++++ 6 files changed, 560 insertions(+) create mode 100644 test/jdk/java/net/DatagramSocket/ConnectPortZero.java create mode 100644 test/jdk/java/net/DatagramSocket/SendPortZero.java create mode 100644 test/jdk/java/nio/channels/DatagramChannel/ConnectPortZero.java create mode 100644 test/jdk/java/nio/channels/DatagramChannel/SendPortZero.java diff --git a/src/java.base/share/classes/java/net/DatagramSocket.java b/src/java.base/share/classes/java/net/DatagramSocket.java index d0235a18229..6cbaeacce50 100644 --- a/src/java.base/share/classes/java/net/DatagramSocket.java +++ b/src/java.base/share/classes/java/net/DatagramSocket.java @@ -188,6 +188,9 @@ public class DatagramSocket implements java.io.Closeable { } } + if (port == 0) { + throw new SocketException("Can't connect to port 0"); + } if (!isBound()) bind(new InetSocketAddress(0)); @@ -772,6 +775,9 @@ public class DatagramSocket implements java.io.Closeable { packetPort); } } + if (packetPort == 0) { + throw new SocketException("Can't send to port 0"); + } } else { // we're connected if (packetAddress == null) { diff --git a/src/java.base/share/classes/sun/nio/ch/DatagramChannelImpl.java b/src/java.base/share/classes/sun/nio/ch/DatagramChannelImpl.java index cdde41271e9..1494cbc2dc7 100644 --- a/src/java.base/share/classes/sun/nio/ch/DatagramChannelImpl.java +++ b/src/java.base/share/classes/sun/nio/ch/DatagramChannelImpl.java @@ -41,6 +41,7 @@ import java.net.NetworkInterface; import java.net.PortUnreachableException; import java.net.ProtocolFamily; import java.net.SocketAddress; +import java.net.SocketException; import java.net.SocketOption; import java.net.SocketTimeoutException; import java.net.StandardProtocolFamily; @@ -811,6 +812,8 @@ class DatagramChannelImpl } if (ia.isLinkLocalAddress()) isa = IPAddressUtil.toScopedAddress(isa); + if (isa.getPort() == 0) + throw new SocketException("Can't send to port 0"); n = send(fd, src, isa); if (blocking) { while (IOStatus.okayToRetry(n) && isOpen()) { @@ -1226,6 +1229,8 @@ class DatagramChannelImpl ensureOpen(); if (check && state == ST_CONNECTED) throw new AlreadyConnectedException(); + if (isa.getPort() == 0) + throw new SocketException("Can't connect to port 0"); // ensure that the socket is bound if (localAddress == null) { diff --git a/test/jdk/java/net/DatagramSocket/ConnectPortZero.java b/test/jdk/java/net/DatagramSocket/ConnectPortZero.java new file mode 100644 index 00000000000..a1488c7961a --- /dev/null +++ b/test/jdk/java/net/DatagramSocket/ConnectPortZero.java @@ -0,0 +1,133 @@ +/* + * Copyright (c) 2020, Oracle and/or its affiliates. All rights reserved. + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. + * + * This code is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License version 2 only, as + * published by the Free Software Foundation. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ + +import org.testng.annotations.AfterTest; +import org.testng.annotations.BeforeTest; +import org.testng.annotations.DataProvider; +import org.testng.annotations.Test; + +import java.io.IOException; +import java.io.UncheckedIOException; +import java.net.DatagramSocket; +import java.net.InetAddress; +import java.net.InetSocketAddress; +import java.net.MulticastSocket; +import java.net.SocketException; +import java.net.SocketPermission; +import java.nio.channels.DatagramChannel; +import java.security.AccessControlException; +import java.security.Permission; +import java.security.PermissionCollection; +import java.security.Permissions; +import java.security.Policy; +import java.security.ProtectionDomain; + +import static org.testng.Assert.assertEquals; +import static org.testng.Assert.assertThrows; +import static org.testng.Assert.expectThrows; + +/* + * @test + * @bug 8240533 + * @summary Check that DatagramSocket, MulticastSocket and DatagramSocketAdaptor + * throw expected Exception when connecting to port 0 + * @run testng/othervm ConnectPortZero + */ + +public class ConnectPortZero{ + private InetAddress loopbackAddr, wildcardAddr; + private DatagramSocket datagramSocket, datagramSocketAdaptor; + private MulticastSocket multicastSocket; + + private static final Class SE = SocketException.class; + private static final Class UCIOE = + UncheckedIOException.class; + private static final Class ACE = + AccessControlException.class; + + @BeforeTest + public void setUp() throws IOException { + loopbackAddr = InetAddress.getLoopbackAddress(); + wildcardAddr = new InetSocketAddress(0).getAddress(); + + datagramSocket = new DatagramSocket(); + multicastSocket = new MulticastSocket(); + datagramSocketAdaptor = DatagramChannel.open().socket(); + } + + @DataProvider(name = "data") + public Object[][] variants() { + return new Object[][]{ + { datagramSocket, loopbackAddr }, + { datagramSocketAdaptor, loopbackAddr }, + { multicastSocket, loopbackAddr }, + { datagramSocket, wildcardAddr }, + { datagramSocketAdaptor, wildcardAddr }, + { multicastSocket, wildcardAddr } + }; + } + + @Test(dataProvider = "data") + public void testConnect(DatagramSocket ds, InetAddress addr) { + Throwable t = expectThrows(UCIOE, () -> ds.connect(addr, 0)); + assertEquals(t.getCause().getClass(), SE); + + assertThrows(SE, () -> ds + .connect(new InetSocketAddress(addr, 0))); + } + + + // Check that 0 port check doesn't override security manager check + @Test(dataProvider = "data") + public void testConnectWithSecurityManager(DatagramSocket ds, + InetAddress addr) { + Policy defaultPolicy = Policy.getPolicy(); + try { + Policy.setPolicy(new NoSendPolicy()); + System.setSecurityManager(new SecurityManager()); + + assertThrows(ACE, () -> ds.connect(addr, 0)); + assertThrows(ACE, () -> + ds.connect(new InetSocketAddress(addr, 0))); + } finally { + System.setSecurityManager(null); + Policy.setPolicy(defaultPolicy); + } + } + + static class NoSendPolicy extends Policy { + final PermissionCollection perms = new Permissions(); + { perms.add(new SocketPermission("*:0", "connect")); } + + public boolean implies(ProtectionDomain domain, Permission perm) { + return !perms.implies(perm); + } + } + + @AfterTest + public void tearDown() { + datagramSocket.close(); + multicastSocket.close(); + datagramSocketAdaptor.close(); + } +} diff --git a/test/jdk/java/net/DatagramSocket/SendPortZero.java b/test/jdk/java/net/DatagramSocket/SendPortZero.java new file mode 100644 index 00000000000..55f884149ba --- /dev/null +++ b/test/jdk/java/net/DatagramSocket/SendPortZero.java @@ -0,0 +1,146 @@ +/* + * Copyright (c) 2020, Oracle and/or its affiliates. All rights reserved. + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. + * + * This code is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License version 2 only, as + * published by the Free Software Foundation. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ + +import org.testng.annotations.AfterTest; +import org.testng.annotations.BeforeTest; +import org.testng.annotations.DataProvider; +import org.testng.annotations.Test; + +import java.io.IOException; +import java.net.DatagramPacket; +import java.net.DatagramSocket; +import java.net.InetAddress; +import java.net.MulticastSocket; +import java.net.SocketException; +import java.net.SocketPermission; +import java.nio.channels.DatagramChannel; +import java.security.AccessControlException; +import java.security.Permission; +import java.security.PermissionCollection; +import java.security.Permissions; +import java.security.Policy; +import java.security.ProtectionDomain; + +import static org.testng.Assert.assertThrows; + +/* + * @test + * @bug 8236105 8240533 + * @summary Check that DatagramSocket and MulticastSocket throw expected + * Exception when sending a DatagramPacket with port 0 + * @run testng/othervm SendPortZero + */ + +public class SendPortZero { + private InetAddress loopbackAddr, wildcardAddr; + private DatagramSocket datagramSocket, datagramSocketAdaptor; + private MulticastSocket multicastSocket; + private DatagramPacket loopbackZeroPkt, wildcardZeroPkt, wildcardValidPkt; + + private static final Class SE = SocketException.class; + private static final Class ACE = + AccessControlException.class; + + @BeforeTest + public void setUp() throws IOException { + datagramSocket = new DatagramSocket(); + multicastSocket = new MulticastSocket(); + datagramSocketAdaptor = DatagramChannel.open().socket(); + + byte[] buf = "test".getBytes(); + + // Addresses + loopbackAddr = InetAddress.getLoopbackAddress(); + //wildcardAddr = new InetSocketAddress(0).getAddress(); + + // Packets + // loopback w/port 0 + loopbackZeroPkt = new DatagramPacket(buf, 0, buf.length); + loopbackZeroPkt.setAddress(loopbackAddr); + loopbackZeroPkt.setPort(0); + + /* + //Commented until JDK-8236852 is fixed + + // wildcard w/port 0 + wildcardZeroPkt = new DatagramPacket(buf, 0, buf.length); + wildcardZeroPkt.setAddress(wildcardAddr); + wildcardZeroPkt.setPort(0); + + //Commented until JDK-8236807 is fixed + + // wildcard addr w/valid port + wildcardValidPkt = new DatagramPacket(buf, 0, buf.length); + var addr = socket.getAddress(); + wildcardValidPkt.setAddress(addr); + wildcardValidPkt.setPort(socket.getLocalPort()); + */ + } + + @DataProvider(name = "data") + public Object[][] variants() { + return new Object[][]{ + { datagramSocket, loopbackZeroPkt }, + { datagramSocketAdaptor, loopbackZeroPkt }, + { multicastSocket, loopbackZeroPkt } + }; + } + + @Test(dataProvider = "data") + public void testSend(DatagramSocket ds, DatagramPacket pkt) { + assertThrows(SE, () -> ds.send(pkt)); + } + + // Check that 0 port check doesn't override security manager check + @Test(dataProvider = "data") + public void testSendWithSecurityManager(DatagramSocket ds, + DatagramPacket pkt) { + Policy defaultPolicy = Policy.getPolicy(); + try { + Policy.setPolicy(new NoSendPolicy()); + System.setSecurityManager(new SecurityManager()); + + assertThrows(ACE, () -> ds.send(pkt)); + } finally { + System.setSecurityManager(null); + Policy.setPolicy(defaultPolicy); + } + } + + static class NoSendPolicy extends Policy { + final PermissionCollection perms = new Permissions(); + { perms.add( + new SocketPermission("*:0", "connect")); } + + public boolean implies(ProtectionDomain domain, Permission perm) { + return !perms.implies(perm); + } + } + + @AfterTest + public void tearDown() { + datagramSocket.close(); + multicastSocket.close(); + datagramSocketAdaptor.close(); + } +} diff --git a/test/jdk/java/nio/channels/DatagramChannel/ConnectPortZero.java b/test/jdk/java/nio/channels/DatagramChannel/ConnectPortZero.java new file mode 100644 index 00000000000..79d243e988b --- /dev/null +++ b/test/jdk/java/nio/channels/DatagramChannel/ConnectPortZero.java @@ -0,0 +1,133 @@ +/* + * Copyright (c) 2020, Oracle and/or its affiliates. All rights reserved. + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. + * + * This code is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License version 2 only, as + * published by the Free Software Foundation. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ + +import org.testng.annotations.AfterTest; +import org.testng.annotations.BeforeTest; +import org.testng.annotations.DataProvider; +import org.testng.annotations.Test; + +import java.io.IOException; +import java.net.InetAddress; +import java.net.InetSocketAddress; +import java.net.SocketException; +import java.net.SocketPermission; +import java.nio.channels.DatagramChannel; +import java.security.AccessControlException; +import java.security.Permission; +import java.security.PermissionCollection; +import java.security.Permissions; +import java.security.Policy; +import java.security.ProtectionDomain; +import java.util.ArrayList; +import java.util.List; + +import static jdk.test.lib.net.IPSupport.hasIPv6; +import static jdk.test.lib.net.IPSupport.hasIPv4; +import static java.net.StandardProtocolFamily.INET; +import static java.net.StandardProtocolFamily.INET6; +import static org.testng.Assert.assertThrows; + +/* + * @test + * @bug 8240533 + * @library /test/lib + * @build jdk.test.lib.net.IPSupport + * @summary Check that DatagramChannel throws expected Exception when connecting + * to port 0 + * @run testng/othervm ConnectPortZero + * @run testng/othervm -Djava.net.preferIPv4Stack=true ConnectPortZero + */ + +public class ConnectPortZero { + private InetSocketAddress loopbackZeroAddr, wildcardZeroAddr; + private DatagramChannel datagramChannel, datagramChannelIPv4, + datagramChannelIPv6; + private List channels; + + private static final Class SE = SocketException.class; + private static final Class ACE = + AccessControlException.class; + + @BeforeTest + public void setUp() throws IOException { + wildcardZeroAddr = new InetSocketAddress(0); + loopbackZeroAddr = new + InetSocketAddress(InetAddress.getLoopbackAddress(), 0); + + channels = new ArrayList<>(); + + datagramChannel = DatagramChannel.open(); + channels.add(new Object[]{datagramChannel}); + if (hasIPv4()) { + datagramChannelIPv4 = DatagramChannel.open(INET); + channels.add(new Object[]{datagramChannelIPv4}); + } + if (hasIPv6()) { + datagramChannelIPv6 = DatagramChannel.open(INET6); + channels.add(new Object[]{datagramChannelIPv6}); + } + } + + @DataProvider(name = "data") + public Object[][] variants() { + return channels.toArray(Object[][]::new); + } + + @Test(dataProvider = "data") + public void testChannelConnect(DatagramChannel dc) { + assertThrows(SE, () -> dc.connect(loopbackZeroAddr)); + assertThrows(SE, () -> dc.connect(wildcardZeroAddr)); + } + + @Test(dataProvider = "data") + public void testSendWithSecurityManager(DatagramChannel dc) { + Policy defaultPolicy = Policy.getPolicy(); + try { + Policy.setPolicy(new SendPortZero.NoSendPolicy()); + System.setSecurityManager(new SecurityManager()); + + assertThrows(ACE, () -> dc.connect(loopbackZeroAddr)); + assertThrows(ACE, () -> dc.connect(wildcardZeroAddr)); + } finally { + System.setSecurityManager(null); + Policy.setPolicy(defaultPolicy); + } + } + + static class NoSendPolicy extends Policy { + final PermissionCollection perms = new Permissions(); + { perms.add( + new SocketPermission("*:0", "connect")); } + + public boolean implies(ProtectionDomain domain, Permission perm) { + return !perms.implies(perm); + } + } + + @AfterTest + public void tearDown() throws IOException { + for(Object[] ch : channels) { + ((DatagramChannel)ch[0]).close(); + } + } +} diff --git a/test/jdk/java/nio/channels/DatagramChannel/SendPortZero.java b/test/jdk/java/nio/channels/DatagramChannel/SendPortZero.java new file mode 100644 index 00000000000..30c6c36d218 --- /dev/null +++ b/test/jdk/java/nio/channels/DatagramChannel/SendPortZero.java @@ -0,0 +1,137 @@ +/* + * Copyright (c) 2020, Oracle and/or its affiliates. All rights reserved. + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. + * + * This code is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License version 2 only, as + * published by the Free Software Foundation. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ + +import org.testng.annotations.AfterTest; +import org.testng.annotations.BeforeTest; +import org.testng.annotations.DataProvider; +import org.testng.annotations.Test; + +import java.io.IOException; +import java.net.InetAddress; +import java.net.InetSocketAddress; +import java.net.SocketException; +import java.net.SocketPermission; +import java.nio.ByteBuffer; +import java.nio.channels.DatagramChannel; +import java.security.AccessControlException; +import java.security.Permission; +import java.security.PermissionCollection; +import java.security.Permissions; +import java.security.Policy; +import java.security.ProtectionDomain; +import java.util.ArrayList; +import java.util.List; + +import static java.net.StandardProtocolFamily.INET; +import static java.net.StandardProtocolFamily.INET6; +import static jdk.test.lib.net.IPSupport.hasIPv4; +import static jdk.test.lib.net.IPSupport.hasIPv6; +import static org.testng.Assert.assertThrows; + +/* + * @test + * @bug 8236105 8240533 + * @library /test/lib + * @build jdk.test.lib.net.IPSupport + * @summary Check that DatagramChannel throws expected Exception when sending to + * port 0 + * @run testng/othervm SendPortZero + * @run testng/othervm -Djava.net.preferIPv4Stack=true SendPortZero + */ + +public class SendPortZero { + private ByteBuffer buf; + private List channels; + private InetSocketAddress loopbackZeroAddr, wildcardZeroAddr; + private DatagramChannel datagramChannel, datagramChannelIPv4, + datagramChannelIPv6; + + private static final Class SE = SocketException.class; + private static final Class ACE = + AccessControlException.class; + + @BeforeTest + public void setUp() throws IOException { + buf = ByteBuffer.wrap("test".getBytes()); + + wildcardZeroAddr = new InetSocketAddress(0); + loopbackZeroAddr = new + InetSocketAddress(InetAddress.getLoopbackAddress(), 0); + + channels = new ArrayList<>(); + + datagramChannel = DatagramChannel.open(); + channels.add(new Object[]{datagramChannel}); + if (hasIPv4()) { + datagramChannelIPv4 = DatagramChannel.open(INET); + channels.add(new Object[]{datagramChannelIPv4}); + } + if (hasIPv6()) { + datagramChannelIPv6 = DatagramChannel.open(INET6); + channels.add(new Object[]{datagramChannelIPv6}); + } + } + + @DataProvider(name = "data") + public Object[][] variants() { + return channels.toArray(Object[][]::new); + } + + @Test(dataProvider = "data") + public void testChannelSend(DatagramChannel dc) { + assertThrows(SE, () -> dc.send(buf, loopbackZeroAddr)); + assertThrows(SE, () -> dc.send(buf, wildcardZeroAddr)); + } + + @Test(dataProvider = "data") + public void testSendWithSecurityManager(DatagramChannel dc) { + Policy defaultPolicy = Policy.getPolicy(); + try { + Policy.setPolicy(new NoSendPolicy()); + System.setSecurityManager(new SecurityManager()); + + assertThrows(ACE, () -> dc.send(buf, loopbackZeroAddr)); + assertThrows(ACE, () -> dc.send(buf, wildcardZeroAddr)); + } finally { + System.setSecurityManager(null); + Policy.setPolicy(defaultPolicy); + } + } + + static class NoSendPolicy extends Policy { + final PermissionCollection perms = new Permissions(); + { perms.add( + new SocketPermission("*:0", "connect")); } + + public boolean implies(ProtectionDomain domain, Permission perm) { + return !perms.implies(perm); + } + } + + @AfterTest + public void tearDown() throws IOException { + for(Object[] ch : channels) { + ((DatagramChannel)ch[0]).close(); + } + } +}