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
This commit is contained in:
Patrick Concannon 2020-04-07 16:21:01 +01:00
parent e53ae5ae63
commit 378aef32ab
6 changed files with 560 additions and 0 deletions

View File

@ -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()) if (!isBound())
bind(new InetSocketAddress(0)); bind(new InetSocketAddress(0));
@ -772,6 +775,9 @@ public class DatagramSocket implements java.io.Closeable {
packetPort); packetPort);
} }
} }
if (packetPort == 0) {
throw new SocketException("Can't send to port 0");
}
} else { } else {
// we're connected // we're connected
if (packetAddress == null) { if (packetAddress == null) {

View File

@ -41,6 +41,7 @@ import java.net.NetworkInterface;
import java.net.PortUnreachableException; import java.net.PortUnreachableException;
import java.net.ProtocolFamily; import java.net.ProtocolFamily;
import java.net.SocketAddress; import java.net.SocketAddress;
import java.net.SocketException;
import java.net.SocketOption; import java.net.SocketOption;
import java.net.SocketTimeoutException; import java.net.SocketTimeoutException;
import java.net.StandardProtocolFamily; import java.net.StandardProtocolFamily;
@ -811,6 +812,8 @@ class DatagramChannelImpl
} }
if (ia.isLinkLocalAddress()) if (ia.isLinkLocalAddress())
isa = IPAddressUtil.toScopedAddress(isa); isa = IPAddressUtil.toScopedAddress(isa);
if (isa.getPort() == 0)
throw new SocketException("Can't send to port 0");
n = send(fd, src, isa); n = send(fd, src, isa);
if (blocking) { if (blocking) {
while (IOStatus.okayToRetry(n) && isOpen()) { while (IOStatus.okayToRetry(n) && isOpen()) {
@ -1226,6 +1229,8 @@ class DatagramChannelImpl
ensureOpen(); ensureOpen();
if (check && state == ST_CONNECTED) if (check && state == ST_CONNECTED)
throw new AlreadyConnectedException(); throw new AlreadyConnectedException();
if (isa.getPort() == 0)
throw new SocketException("Can't connect to port 0");
// ensure that the socket is bound // ensure that the socket is bound
if (localAddress == null) { if (localAddress == null) {

View File

@ -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<SocketException> SE = SocketException.class;
private static final Class<UncheckedIOException> UCIOE =
UncheckedIOException.class;
private static final Class<AccessControlException> 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();
}
}

View File

@ -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<SocketException> SE = SocketException.class;
private static final Class<AccessControlException> 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();
}
}

View File

@ -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<Object[]> channels;
private static final Class<SocketException> SE = SocketException.class;
private static final Class<AccessControlException> 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();
}
}
}

View File

@ -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<Object[]> channels;
private InetSocketAddress loopbackZeroAddr, wildcardZeroAddr;
private DatagramChannel datagramChannel, datagramChannelIPv4,
datagramChannelIPv6;
private static final Class<SocketException> SE = SocketException.class;
private static final Class<AccessControlException> 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();
}
}
}