diff --git a/src/java.base/unix/native/libnet/PlainDatagramSocketImpl.c b/src/java.base/unix/native/libnet/PlainDatagramSocketImpl.c index a5d4d80c0ab..dc4abafcd68 100644 --- a/src/java.base/unix/native/libnet/PlainDatagramSocketImpl.c +++ b/src/java.base/unix/native/libnet/PlainDatagramSocketImpl.c @@ -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)); diff --git a/test/jdk/java/net/DatagramSocket/SendReceiveMaxSize.java b/test/jdk/java/net/DatagramSocket/SendReceiveMaxSize.java new file mode 100644 index 00000000000..aa7e74b2968 --- /dev/null +++ b/test/jdk/java/net/DatagramSocket/SendReceiveMaxSize.java @@ -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 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 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); + } + } + } + } +} diff --git a/test/jdk/java/net/DatagramSocket/SetGetSendBufferSize.java b/test/jdk/java/net/DatagramSocket/SetGetSendBufferSize.java index dda6fc45f89..661ffffdf82 100644 --- a/test/jdk/java/net/DatagramSocket/SetGetSendBufferSize.java +++ b/test/jdk/java/net/DatagramSocket/SetGetSendBufferSize.java @@ -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); + } } } } diff --git a/test/jdk/java/nio/channels/DatagramChannel/MinSendBufferSize.java b/test/jdk/java/nio/channels/DatagramChannel/MinSendBufferSize.java deleted file mode 100644 index 13401d6badd..00000000000 --- a/test/jdk/java/nio/channels/DatagramChannel/MinSendBufferSize.java +++ /dev/null @@ -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 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 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(); - - 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(); - - 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 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(); - } -} diff --git a/test/jdk/java/nio/channels/DatagramChannel/SendReceiveMaxSize.java b/test/jdk/java/nio/channels/DatagramChannel/SendReceiveMaxSize.java new file mode 100644 index 00000000000..09b38d393a0 --- /dev/null +++ b/test/jdk/java/nio/channels/DatagramChannel/SendReceiveMaxSize.java @@ -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 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(); + 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)); + } + } + } +}