8242885: PlainDatagramSocketImpl doesn’t allow for the sending of IPv6 datagrams on macOS with sizes between 65508-65527 bytes

This fix changes the current max size for IPv6 datagrams on macOS from it's current size of 65507, which is the IPv4 limit, to 65527, the actual limit for IPv6 on macOS

Reviewed-by: alanb, dfuchs, vtewari
This commit is contained in:
Patrick Concannon 2020-07-31 12:42:32 +01:00
parent 2dda9965cd
commit 7fd5cb6117
5 changed files with 283 additions and 170 deletions

View File

@ -48,6 +48,11 @@
#endif
#endif // __linux__
#ifdef __APPLE__
#define IPV4_SNDBUF_LIMIT 65507
#define IPV6_SNDBUF_LIMIT 65527
#endif // __APPLE__
#ifndef IPTOS_TOS_MASK
#define IPTOS_TOS_MASK 0x1e
#endif
@ -895,7 +900,7 @@ Java_java_net_PlainDatagramSocketImpl_datagramSocketCreate(JNIEnv *env,
}
#ifdef __APPLE__
arg = 65507;
arg = (domain == AF_INET6) ? IPV6_SNDBUF_LIMIT : IPV4_SNDBUF_LIMIT;
if (setsockopt(fd, SOL_SOCKET, SO_SNDBUF,
(char *)&arg, sizeof(arg)) < 0) {
getErrorString(errno, tmpbuf, sizeof(tmpbuf));

View File

@ -0,0 +1,117 @@
/*
* Copyright (c) 2020, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
*
* This code is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License version 2 only, as
* published by the Free Software Foundation.
*
* This code is distributed in the hope that it will be useful, but WITHOUT
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
* version 2 for more details (a copy is included in the LICENSE file that
* accompanied this code).
*
* You should have received a copy of the GNU General Public License version
* 2 along with this work; if not, write to the Free Software Foundation,
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
*
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
* or visit www.oracle.com if you need additional information or have any
* questions.
*/
/*
* @test
* @bug 8242885
* @summary This test verifies that on macOS, the send buffer size is configured
* by default so that none of our implementations of the UDP protocol
* will fail with a "packet too large" exception when trying to send a
* packet of the maximum possible size allowed by the protocol.
* However, an exception is expected if the packet size exceeds that
* limit.
* @library /test/lib
* @build jdk.test.lib.net.IPSupport
* @requires os.family == "mac"
* @run testng/othervm SendReceiveMaxSize
* @run testng/othervm -Djava.net.preferIPv4Stack=true SendReceiveMaxSize
* @run testng/othervm -Djava.net.preferIPv6Addresses=true SendReceiveMaxSize
* @run testng/othervm -Djdk.net.usePlainDatagramSocketImpl SendReceiveMaxSize
* @run testng/othervm -Djdk.net.usePlainDatagramSocketImpl -Djava.net.preferIPv4Stack=true SendReceiveMaxSize
* @run testng/othervm -Djdk.net.usePlainDatagramSocketImpl -Djava.net.preferIPv6Addresses=true SendReceiveMaxSize
*/
import jdk.test.lib.net.IPSupport;
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.Inet6Address;
import java.net.InetAddress;
import java.net.InetSocketAddress;
import java.net.MulticastSocket;
import java.nio.channels.DatagramChannel;
import static org.testng.Assert.expectThrows;
public class SendReceiveMaxSize {
private int BUF_LIMIT;
private InetAddress HOST_ADDR;
private final static int IPV4_SNDBUF = 65507;
private final static int IPV6_SNDBUF = 65527;
private final static Class<IOException> IOE = IOException.class;
public interface DatagramSocketSupplier {
DatagramSocket open() throws IOException;
}
static DatagramSocketSupplier supplier(DatagramSocketSupplier supplier) { return supplier; }
@BeforeTest
public void setUp() throws IOException {
IPSupport.throwSkippedExceptionIfNonOperational();
HOST_ADDR = InetAddress.getLocalHost();
BUF_LIMIT = (HOST_ADDR instanceof Inet6Address) ? IPV6_SNDBUF : IPV4_SNDBUF;
}
@DataProvider
public Object[][] invariants() {
var ds = supplier(() -> new DatagramSocket());
var ms = supplier(() -> new MulticastSocket());
var dsa = supplier(() -> DatagramChannel.open().socket());
return new Object[][]{
{ "DatagramSocket", BUF_LIMIT - 1, ds, null },
{ "DatagramSocket", BUF_LIMIT, ds, null },
{ "DatagramSocket", BUF_LIMIT + 1, ds, IOE },
{ "MulticastSocket", BUF_LIMIT - 1, ms, null },
{ "MulticastSocket", BUF_LIMIT, ms, null },
{ "MulticastSocket", BUF_LIMIT + 1, ms, IOE },
{ "DatagramSocketAdaptor", BUF_LIMIT - 1, dsa, null },
{ "DatagramSocketAdaptor", BUF_LIMIT, dsa, null },
{ "DatagramSocketAdaptor", BUF_LIMIT + 1, dsa, IOE },
};
}
@Test(dataProvider = "invariants")
public void testSendReceiveMaxSize(String name, int capacity,
DatagramSocketSupplier supplier,
Class<? extends Exception> exception) throws IOException {
try (var receiver = new DatagramSocket(new InetSocketAddress(HOST_ADDR, 0))) {
var port = receiver.getLocalPort();
var addr = new InetSocketAddress(HOST_ADDR, port);
try (var sender = supplier.open()) {
var sendPkt = new DatagramPacket(new byte[capacity], capacity, addr);
if (exception != null) {
Exception ex = expectThrows(IOE, () -> sender.send(sendPkt));
System.out.println(name + " got expected exception: " + ex);
} else {
sender.send(sendPkt);
var receivePkt = new DatagramPacket(new byte[capacity], capacity);
receiver.receive(receivePkt);
}
}
}
}
}

View File

@ -30,8 +30,11 @@
* @run testng SetGetSendBufferSize
* @run testng/othervm -Djava.net.preferIPv4Stack=true SetGetSendBufferSize
* @run testng/othervm -Djdk.net.usePlainDatagramSocketImpl SetGetSendBufferSize
* @run testng/othervm -Djdk.net.usePlainDatagramSocketImpl -Djava.net.preferIPv4Stack=true SetGetSendBufferSize
*/
import jdk.test.lib.Platform;
import jdk.test.lib.net.IPSupport;
import org.testng.annotations.DataProvider;
import org.testng.annotations.Test;
@ -41,8 +44,6 @@ import java.net.MulticastSocket;
import java.net.SocketException;
import java.nio.channels.DatagramChannel;
import jdk.test.lib.Platform;
import static org.testng.Assert.assertEquals;
import static org.testng.Assert.assertTrue;
import static org.testng.Assert.expectThrows;
@ -91,9 +92,12 @@ public class SetGetSendBufferSize {
@Test(dataProvider = "invariants")
public void testInitialSendBufferSize(String name, DatagramSocketSupplier supplier) throws IOException {
if(Platform.isOSX()) {
if (Platform.isOSX()) {
try (var socket = supplier.open()){
assertTrue(socket.getSendBufferSize() >= 65507, name);
if (IPSupport.hasIPv6() && !IPSupport.preferIPv4Stack()) {
assertEquals(socket.getSendBufferSize(), 65527, name);
}
}
}
}

View File

@ -1,166 +0,0 @@
/*
* Copyright (c) 2020, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
*
* This code is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License version 2 only, as
* published by the Free Software Foundation.
*
* This code is distributed in the hope that it will be useful, but WITHOUT
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
* version 2 for more details (a copy is included in the LICENSE file that
* accompanied this code).
*
* You should have received a copy of the GNU General Public License version
* 2 along with this work; if not, write to the Free Software Foundation,
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
*
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
* or visit www.oracle.com if you need additional information or have any
* questions.
*/
/*
* @test
* @bug 8239355
* @summary Check that new SO_SNDBUF limit on macOS is adhered to
* @library /test/lib
* @build jdk.test.lib.net.IPSupport
* @requires os.family == "mac"
* @run testng/othervm MinSendBufferSize
* @run testng/othervm -Djava.net.preferIPv4Stack=true MinSendBufferSize
*/
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.DatagramSocket;
import java.net.InetAddress;
import java.net.ProtocolFamily;
import java.nio.ByteBuffer;
import java.nio.channels.DatagramChannel;
import java.util.ArrayList;
import java.util.List;
import static java.net.StandardSocketOptions.SO_SNDBUF;
import static jdk.test.lib.net.IPSupport.hasIPv4;
import static jdk.test.lib.net.IPSupport.hasIPv6;
import static jdk.test.lib.net.IPSupport.preferIPv4Stack;
import static java.net.StandardProtocolFamily.INET;
import static java.net.StandardProtocolFamily.INET6;
import static org.testng.Assert.assertTrue;
import static org.testng.Assert.expectThrows;
public class MinSendBufferSize {
private int EXPECTED_SNDBUF;
private DatagramChannel datagramChannel, datagramChannelIPv4,
datagramChannelIPv6;
private final static int IPV4_SNDBUF = 65507;
private final static int IPV6_SNDBUF = 65527;
private final static Class<IOException> IOE = IOException.class;
@BeforeTest
public void setUp() throws IOException {
datagramChannel = DatagramChannel.open();
if (hasIPv4())
datagramChannelIPv4 = DatagramChannel.open(INET);
if (hasIPv6())
datagramChannelIPv6 = DatagramChannel.open(INET6);
EXPECTED_SNDBUF = hasIPv6() && !preferIPv4Stack()
? IPV6_SNDBUF : IPV4_SNDBUF;
}
private void populateDataProvider(List<Object[]> testcases,
DatagramChannel datagramChannel,
int payloadSize,
ProtocolFamily family) {
testcases.add(new Object[]{datagramChannel, payloadSize - 1,
family, null});
testcases.add(new Object[]{datagramChannel, payloadSize,
family, null});
testcases.add(new Object[]{datagramChannel, payloadSize + 1,
family, IOE});
}
@DataProvider(name = "testGetOptionProvider")
public Object[][] providerIO() {
var testcases = new ArrayList<Object[]>();
testcases.add(new Object[]{datagramChannel, EXPECTED_SNDBUF});
if (hasIPv4())
testcases.add(new Object[]{datagramChannelIPv4, IPV4_SNDBUF});
if (hasIPv6())
testcases.add(new Object[]{datagramChannelIPv6, IPV6_SNDBUF});
return testcases.toArray(Object[][]::new);
}
@DataProvider(name = "testSendPayloadProvider")
public Object[][] providerIO_Payload() {
var testcases = new ArrayList<Object[]>();
if (hasIPv4())
populateDataProvider(testcases, datagramChannel,
IPV4_SNDBUF, INET);
if (hasIPv6() && !preferIPv4Stack())
populateDataProvider(testcases, datagramChannel,
IPV6_SNDBUF, INET6);
if (hasIPv4())
populateDataProvider(testcases, datagramChannelIPv4,
IPV4_SNDBUF, INET);
if (hasIPv6())
populateDataProvider(testcases, datagramChannelIPv6,
IPV6_SNDBUF, INET6);
return testcases.toArray(Object[][]::new);
}
@Test(dataProvider = "testGetOptionProvider")
public void testGetOption(DatagramChannel channel, int sendBuf)
throws IOException {
assertTrue(channel.getOption(SO_SNDBUF) >= sendBuf);
}
@Test(dataProvider = "testSendPayloadProvider")
public void testSend(DatagramChannel channel, int sendBuf,
ProtocolFamily family,
Class<? extends Throwable> exception)
throws IOException {
InetAddress targetAddress;
assert family != null;
if (family == INET) {
targetAddress = InetAddress.getByName("127.0.0.1");
} else {
targetAddress = InetAddress.getByName("::1");
}
try (var receiver = new DatagramSocket(0, targetAddress)) {
var buf = ByteBuffer.allocate(sendBuf);
var addr = receiver.getLocalSocketAddress();
if (exception != null) {
expectThrows(exception, () -> channel.send(buf, addr));
} else {
channel.send(buf, addr);
}
}
}
@AfterTest
public void tearDown() throws IOException {
datagramChannel.close();
if (hasIPv4())
datagramChannelIPv4.close();
if (hasIPv6())
datagramChannelIPv6.close();
}
}

View File

@ -0,0 +1,153 @@
/*
* Copyright (c) 2020, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
*
* This code is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License version 2 only, as
* published by the Free Software Foundation.
*
* This code is distributed in the hope that it will be useful, but WITHOUT
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
* version 2 for more details (a copy is included in the LICENSE file that
* accompanied this code).
*
* You should have received a copy of the GNU General Public License version
* 2 along with this work; if not, write to the Free Software Foundation,
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
*
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
* or visit www.oracle.com if you need additional information or have any
* questions.
*/
/*
* @test
* @bug 8239355 8242885
* @summary Check that it is possible to send and receive datagrams of
* maximum size on macOS.
* @library /test/lib
* @build jdk.test.lib.net.IPSupport
* @requires os.family == "mac"
* @run testng/othervm SendReceiveMaxSize
* @run testng/othervm -Djava.net.preferIPv4Stack=true SendReceiveMaxSize
* @run testng/othervm -Djdk.net.usePlainDatagramSocketImpl SendReceiveMaxSize
* @run testng/othervm -Djdk.net.usePlainDatagramSocketImpl -Djava.net.preferIPv4Stack=true SendReceiveMaxSize
*/
import jdk.test.lib.NetworkConfiguration;
import jdk.test.lib.net.IPSupport;
import org.testng.annotations.BeforeTest;
import org.testng.annotations.DataProvider;
import org.testng.annotations.Test;
import java.io.IOException;
import java.net.Inet4Address;
import java.net.Inet6Address;
import java.net.InetAddress;
import java.net.InetSocketAddress;
import java.nio.ByteBuffer;
import java.nio.channels.DatagramChannel;
import java.util.ArrayList;
import java.util.function.Predicate;
import static java.net.StandardProtocolFamily.INET;
import static java.net.StandardProtocolFamily.INET6;
import static java.net.StandardSocketOptions.SO_SNDBUF;
import static jdk.test.lib.net.IPSupport.hasIPv4;
import static jdk.test.lib.net.IPSupport.hasIPv6;
import static jdk.test.lib.net.IPSupport.preferIPv4Stack;
import static org.testng.Assert.assertEquals;
import static org.testng.Assert.assertThrows;
import static org.testng.Assert.assertTrue;
public class SendReceiveMaxSize {
private final static int IPV4_SNDBUF = 65507;
private final static int IPV6_SNDBUF = 65527;
private final static Class<IOException> IOE = IOException.class;
public interface DatagramChannelSupplier {
DatagramChannel open() throws IOException;
}
static DatagramChannelSupplier supplier(DatagramChannelSupplier supplier) { return supplier; }
@BeforeTest
public void setUp() {
IPSupport.throwSkippedExceptionIfNonOperational();
}
@DataProvider
public Object[][] invariants() throws IOException {
var testcases = new ArrayList<Object[]>();
var nc = NetworkConfiguration.probe();
if (hasIPv4()) {
InetAddress IPv4Addr = nc.ip4Addresses()
.filter(Predicate.not(InetAddress::isLoopbackAddress))
.findFirst()
.orElse((Inet4Address) InetAddress.getByName("127.0.0.1"));
testcases.add(new Object[]{
supplier(() -> DatagramChannel.open()),
IPV4_SNDBUF,
IPv4Addr
});
testcases.add(new Object[]{
supplier(() -> DatagramChannel.open(INET)),
IPV4_SNDBUF,
IPv4Addr
});
}
if (!preferIPv4Stack() && hasIPv6()) {
InetAddress IPv6Addr = nc.ip6Addresses()
.filter(Predicate.not(InetAddress::isLoopbackAddress))
.findFirst()
.orElse((Inet6Address) InetAddress.getByName("::1"));
testcases.add(new Object[]{
supplier(() -> DatagramChannel.open()),
IPV6_SNDBUF,
IPv6Addr
});
testcases.add(new Object[]{
supplier(() -> DatagramChannel.open(INET6)),
IPV6_SNDBUF,
IPv6Addr
});
}
return testcases.toArray(Object[][]::new);
}
@Test(dataProvider = "invariants")
public void testGetOption(DatagramChannelSupplier supplier, int capacity, InetAddress host)
throws IOException {
try (var dc = supplier.open()) {
assertTrue(dc.getOption(SO_SNDBUF) >= capacity);
}
}
@Test(dataProvider = "invariants")
public void testSendReceiveMaxSize(DatagramChannelSupplier supplier, int capacity, InetAddress host)
throws IOException {
try (var receiver = DatagramChannel.open()) {
receiver.bind(new InetSocketAddress(host, 0));
var port = receiver.socket().getLocalPort();
var addr = new InetSocketAddress(host, port);
try (var sender = supplier.open()) {
sender.bind(null);
var sendBuf = ByteBuffer.allocate(capacity);
sender.send(sendBuf, addr);
var receiveBuf = ByteBuffer.allocate(capacity);
receiver.receive(receiveBuf);
assertEquals(sendBuf, receiveBuf);
sendBuf = ByteBuffer.allocate(capacity - 1);
sender.send(sendBuf, addr);
receiveBuf = ByteBuffer.allocate(capacity - 1);
receiver.receive(receiveBuf);
assertTrue(sendBuf.compareTo(receiveBuf) == 0);
var failSendBuf = ByteBuffer.allocate(capacity + 1);
assertThrows(IOE, () -> sender.send(failSendBuf, addr));
}
}
}
}