jdk-24/test/jdk/java/foreign/channels/TestSocketChannels.java
Maurizio Cimadamore 73baadceb6 8295044: Implementation of Foreign Function and Memory API (Second Preview)
Co-authored-by: Jorn Vernee <jvernee@openjdk.org>
Co-authored-by: Per Minborg <pminborg@openjdk.org>
Co-authored-by: Maurizio Cimadamore <mcimadamore@openjdk.org>
Reviewed-by: jvernee, pminborg, psandoz, alanb, sundar
2022-12-05 13:49:53 +00:00

243 lines
10 KiB
Java

/*
* Copyright (c) 2021, 2022, 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
* @enablePreview
* @library /test/lib
* @modules java.base/sun.nio.ch
* @key randomness
* @run testng/othervm TestSocketChannels
*/
import java.lang.foreign.Arena;
import java.net.InetAddress;
import java.net.InetSocketAddress;
import java.nio.ByteBuffer;
import java.nio.channels.ServerSocketChannel;
import java.nio.channels.SocketChannel;
import java.util.Arrays;
import java.util.List;
import java.util.concurrent.atomic.AtomicReference;
import java.util.function.Supplier;
import java.util.stream.Stream;
import java.lang.foreign.MemorySegment;
import org.testng.annotations.*;
import static java.lang.foreign.ValueLayout.JAVA_BYTE;
import static org.testng.Assert.*;
/**
* Tests consisting of buffer views with synchronous NIO network channels.
*/
public class TestSocketChannels extends AbstractChannelsTest {
static final Class<IllegalStateException> ISE = IllegalStateException.class;
static final Class<WrongThreadException> WTE = WrongThreadException.class;
@Test(dataProvider = "closeableArenas")
public void testBasicIOWithClosedSegment(Supplier<Arena> arenaSupplier)
throws Exception
{
try (var channel = SocketChannel.open();
var server = ServerSocketChannel.open();
var connectedChannel = connectChannels(server, channel)) {
Arena drop = arenaSupplier.get();
ByteBuffer bb = segmentBufferOfSize(drop.scope(), 16);
drop.close();
assertMessage(expectThrows(ISE, () -> channel.read(bb)), "Already closed");
assertMessage(expectThrows(ISE, () -> channel.read(new ByteBuffer[] {bb})), "Already closed");
assertMessage(expectThrows(ISE, () -> channel.read(new ByteBuffer[] {bb}, 0, 1)), "Already closed");
assertMessage(expectThrows(ISE, () -> channel.write(bb)), "Already closed");
assertMessage(expectThrows(ISE, () -> channel.write(new ByteBuffer[] {bb})), "Already closed");
assertMessage(expectThrows(ISE, () -> channel.write(new ByteBuffer[] {bb}, 0 ,1)), "Already closed");
}
}
@Test(dataProvider = "closeableArenas")
public void testScatterGatherWithClosedSegment(Supplier<Arena> arenaSupplier)
throws Exception
{
try (var channel = SocketChannel.open();
var server = ServerSocketChannel.open();
var connectedChannel = connectChannels(server, channel)) {
Arena drop = arenaSupplier.get();
ByteBuffer[] buffers = segmentBuffersOfSize(8, drop.scope(), 16);
drop.close();
assertMessage(expectThrows(ISE, () -> channel.write(buffers)), "Already closed");
assertMessage(expectThrows(ISE, () -> channel.read(buffers)), "Already closed");
assertMessage(expectThrows(ISE, () -> channel.write(buffers, 0 ,8)), "Already closed");
assertMessage(expectThrows(ISE, () -> channel.read(buffers, 0, 8)), "Already closed");
}
}
@Test(dataProvider = "closeableArenas")
public void testBasicIO(Supplier<Arena> arenaSupplier)
throws Exception
{
Arena drop;
try (var sc1 = SocketChannel.open();
var ssc = ServerSocketChannel.open();
var sc2 = connectChannels(ssc, sc1);
var scp = drop = arenaSupplier.get()) {
MemorySegment segment1 = MemorySegment.allocateNative(10, 1, drop.scope());
MemorySegment segment2 = MemorySegment.allocateNative(10, 1, drop.scope());
for (int i = 0; i < 10; i++) {
segment1.set(JAVA_BYTE, i, (byte) i);
}
ByteBuffer bb1 = segment1.asByteBuffer();
ByteBuffer bb2 = segment2.asByteBuffer();
assertEquals(sc1.write(bb1), 10);
assertEquals(sc2.read(bb2), 10);
assertEquals(bb2.flip(), ByteBuffer.wrap(new byte[]{0, 1, 2, 3, 4, 5, 6, 7, 8, 9}));
}
}
@Test
public void testBasicHeapIOWithGlobalSession() throws Exception {
try (var sc1 = SocketChannel.open();
var ssc = ServerSocketChannel.open();
var sc2 = connectChannels(ssc, sc1)) {
var segment1 = MemorySegment.ofArray(new byte[10]);
var segment2 = MemorySegment.ofArray(new byte[10]);
for (int i = 0; i < 10; i++) {
segment1.set(JAVA_BYTE, i, (byte) i);
}
ByteBuffer bb1 = segment1.asByteBuffer();
ByteBuffer bb2 = segment2.asByteBuffer();
assertEquals(sc1.write(bb1), 10);
assertEquals(sc2.read(bb2), 10);
assertEquals(bb2.flip(), ByteBuffer.wrap(new byte[]{0, 1, 2, 3, 4, 5, 6, 7, 8, 9}));
}
}
@Test(dataProvider = "confinedArenas")
public void testIOOnConfinedFromAnotherThread(Supplier<Arena> arenaSupplier)
throws Exception
{
try (var channel = SocketChannel.open();
var server = ServerSocketChannel.open();
var connected = connectChannels(server, channel);
var drop = arenaSupplier.get()) {
var segment = MemorySegment.allocateNative(10, 1, drop.scope());
ByteBuffer bb = segment.asByteBuffer();
List<ThrowingRunnable> ioOps = List.of(
() -> channel.write(bb),
() -> channel.read(bb),
() -> channel.write(new ByteBuffer[] {bb}),
() -> channel.read(new ByteBuffer[] {bb}),
() -> channel.write(new ByteBuffer[] {bb}, 0, 1),
() -> channel.read(new ByteBuffer[] {bb}, 0, 1)
);
for (var ioOp : ioOps) {
AtomicReference<Exception> exception = new AtomicReference<>();
Runnable task = () -> exception.set(expectThrows(WTE, ioOp));
var t = new Thread(task);
t.start();
t.join();
assertMessage(exception.get(), "Attempted access outside owning thread");
}
}
}
@Test(dataProvider = "closeableArenas")
public void testScatterGatherIO(Supplier<Arena> arenaSupplier)
throws Exception
{
Arena drop;
try (var sc1 = SocketChannel.open();
var ssc = ServerSocketChannel.open();
var sc2 = connectChannels(ssc, sc1);
var scp = drop = arenaSupplier.get()) {
var writeBuffers = mixedBuffersOfSize(32, drop.scope(), 64);
var readBuffers = mixedBuffersOfSize(32, drop.scope(), 64);
long expectedCount = remaining(writeBuffers);
assertEquals(writeNBytes(sc1, writeBuffers, 0, 32, expectedCount), expectedCount);
assertEquals(readNBytes(sc2, readBuffers, 0, 32, expectedCount), expectedCount);
assertEquals(flip(readBuffers), clear(writeBuffers));
}
}
@Test(dataProvider = "closeableArenas")
public void testBasicIOWithDifferentSessions(Supplier<Arena> arenaSupplier)
throws Exception
{
try (var sc1 = SocketChannel.open();
var ssc = ServerSocketChannel.open();
var sc2 = connectChannels(ssc, sc1);
var drop1 = arenaSupplier.get();
var drop2 = arenaSupplier.get()) {
var writeBuffers = Stream.of(mixedBuffersOfSize(16, drop1.scope(), 64), mixedBuffersOfSize(16, drop2.scope(), 64))
.flatMap(Arrays::stream)
.toArray(ByteBuffer[]::new);
var readBuffers = Stream.of(mixedBuffersOfSize(16, drop1.scope(), 64), mixedBuffersOfSize(16, drop2.scope(), 64))
.flatMap(Arrays::stream)
.toArray(ByteBuffer[]::new);
long expectedCount = remaining(writeBuffers);
assertEquals(writeNBytes(sc1, writeBuffers, 0, 32, expectedCount), expectedCount);
assertEquals(readNBytes(sc2, readBuffers, 0, 32, expectedCount), expectedCount);
assertEquals(flip(readBuffers), clear(writeBuffers));
}
}
static SocketChannel connectChannels(ServerSocketChannel ssc, SocketChannel sc)
throws Exception
{
ssc.bind(new InetSocketAddress(InetAddress.getLoopbackAddress(), 0));
sc.connect(ssc.getLocalAddress());
return ssc.accept();
}
static long writeNBytes(SocketChannel channel,
ByteBuffer[] buffers, int offset, int len,
long bytes)
throws Exception
{
long total = 0L;
do {
long n = channel.write(buffers, offset, len);
assertTrue(n > 0, "got:" + n);
total += n;
} while (total < bytes);
return total;
}
static long readNBytes(SocketChannel channel,
ByteBuffer[] buffers, int offset, int len,
long bytes)
throws Exception
{
long total = 0L;
do {
long n = channel.read(buffers, offset, len);
assertTrue(n > 0, "got:" + n);
total += n;
} while (total < bytes);
return total;
}
}