Skip to content

Commit 784f1c1

Browse files
c-clearyAlekseiEfimov
authored andcommittedApr 13, 2021
8264200: java/nio/channels/DatagramChannel/SRTest.java fails intermittently
Reviewed-by: dfuchs, rriggs, msheppar
1 parent a4f644e commit 784f1c1

File tree

1 file changed

+125
-94
lines changed

1 file changed

+125
-94
lines changed
 
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2001, 2010, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2001, 2021, Oracle and/or its affiliates. All rights reserved.
33
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
44
*
55
* This code is free software; you can redistribute it and/or modify it
@@ -23,185 +23,216 @@
2323

2424
/* @test
2525
* @summary Test DatagramChannel's send and receive methods
26-
* @author Mike McCloskey
26+
* @run testng/othervm/timeout=20 SRTest
2727
*/
2828

2929
import java.io.*;
3030
import java.net.*;
3131
import java.nio.*;
3232
import java.nio.channels.*;
33-
import java.nio.charset.*;
33+
import java.util.concurrent.CompletableFuture;
34+
import java.util.concurrent.CompletionException;
35+
import java.util.concurrent.ExecutorService;
36+
import java.util.concurrent.Executors;
37+
import java.util.stream.Stream;
38+
import static java.nio.charset.StandardCharsets.US_ASCII;
3439

40+
import org.testng.annotations.*;
3541

3642
public class SRTest {
3743

44+
ExecutorService executorService;
3845
static PrintStream log = System.err;
3946

40-
public static void main(String[] args) throws Exception {
41-
test();
47+
static final String DATA_STRING = "hello";
48+
49+
@BeforeClass
50+
public void beforeClass() {
51+
executorService = Executors.newCachedThreadPool();
4252
}
4353

44-
static void test() throws Exception {
45-
ClassicReader classicReader;
46-
NioReader nioReader;
54+
@AfterClass
55+
public void afterClass() {
56+
executorService.shutdown();
57+
}
4758

48-
classicReader = new ClassicReader();
49-
invoke(classicReader, new ClassicWriter(classicReader.port()));
59+
@Test
60+
public void classicReaderClassicWriter() throws Exception {
61+
try (ClassicReader cr = new ClassicReader();
62+
ClassicWriter cw = new ClassicWriter(cr.port())) {
63+
invoke(executorService, cr, cw);
64+
}
5065
log.println("Classic RW: OK");
66+
}
5167

52-
classicReader = new ClassicReader();
53-
invoke(classicReader, new NioWriter(classicReader.port()));
68+
@Test
69+
public void classicReaderNioWriter() throws Exception {
70+
try (ClassicReader cr = new ClassicReader();
71+
NioWriter nw = new NioWriter(cr.port())) {
72+
invoke(executorService, cr, nw);
73+
}
5474
log.println("Classic R, Nio W: OK");
75+
}
5576

56-
nioReader = new NioReader();
57-
invoke(nioReader, new ClassicWriter(nioReader.port()));
77+
@Test
78+
public void nioReaderClassicWriter() throws Exception {
79+
try (NioReader nr = new NioReader();
80+
ClassicWriter cw = new ClassicWriter(nr.port())) {
81+
invoke(executorService, nr, cw);
82+
}
5883
log.println("Classic W, Nio R: OK");
84+
}
5985

60-
nioReader = new NioReader();
61-
invoke(nioReader, new NioWriter(nioReader.port()));
86+
@Test
87+
public void nioReaderNioWriter() throws Exception {
88+
try (NioReader nr = new NioReader();
89+
NioWriter nw = new NioWriter(nr.port())) {
90+
invoke(executorService, nr, nw);
91+
}
6292
log.println("Nio RW: OK");
6393
}
6494

65-
static void invoke(Sprintable reader, Sprintable writer) throws Exception {
66-
Thread readerThread = new Thread(reader);
67-
readerThread.start();
68-
Thread.sleep(50);
69-
70-
Thread writerThread = new Thread(writer);
71-
writerThread.start();
72-
73-
writerThread.join();
74-
readerThread.join();
75-
76-
reader.throwException();
77-
writer.throwException();
95+
private static void invoke(ExecutorService e, Runnable reader, Runnable writer) {
96+
CompletableFuture<Void> f1 = CompletableFuture.runAsync(writer, e);
97+
CompletableFuture<Void> f2 = CompletableFuture.runAsync(reader, e);
98+
wait(f1, f2);
7899
}
79100

80-
public interface Sprintable extends Runnable {
81-
public void throwException() throws Exception;
101+
// Exit with CompletionException if any passed futures complete exceptionally
102+
private static void wait(CompletableFuture<?>... futures) throws CompletionException {
103+
CompletableFuture<?> future = CompletableFuture.allOf(futures);
104+
Stream.of(futures)
105+
.forEach(f -> f.exceptionally(ex -> {
106+
future.completeExceptionally(ex);
107+
return null;
108+
}));
109+
future.join();
82110
}
83111

84-
public static class ClassicWriter implements Sprintable {
85-
final int port;
86-
Exception e = null;
87-
88-
ClassicWriter(int port) {
89-
this.port = port;
90-
}
112+
public static class ClassicWriter implements Runnable, AutoCloseable {
113+
final DatagramSocket ds;
114+
final int dstPort;
91115

92-
public void throwException() throws Exception {
93-
if (e != null)
94-
throw e;
116+
ClassicWriter(int dstPort) throws SocketException {
117+
this.dstPort = dstPort;
118+
this.ds = new DatagramSocket();
95119
}
96120

97121
public void run() {
98122
try {
99-
DatagramSocket ds = new DatagramSocket();
100-
String dataString = "hello";
101-
byte[] data = dataString.getBytes();
102-
InetAddress address = InetAddress.getLocalHost();
123+
byte[] data = DATA_STRING.getBytes(US_ASCII);
124+
InetAddress address = InetAddress.getLoopbackAddress();
103125
DatagramPacket dp = new DatagramPacket(data, data.length,
104-
address, port);
126+
address, dstPort);
105127
ds.send(dp);
106-
Thread.sleep(50);
107-
ds.send(dp);
108-
} catch (Exception ex) {
109-
e = ex;
128+
} catch (Exception e) {
129+
log.println("ClassicWriter [" + ds.getLocalAddress() + "]");
130+
throw new RuntimeException("ClassicWriter threw exception: " + e);
131+
} finally {
132+
log.println("ClassicWriter finished");
110133
}
111134
}
112-
}
113-
114-
public static class NioWriter implements Sprintable {
115-
final int port;
116-
Exception e = null;
117135

118-
NioWriter(int port) {
119-
this.port = port;
136+
@Override
137+
public void close() throws IOException {
138+
ds.close();
120139
}
140+
}
141+
142+
public static class NioWriter implements Runnable, AutoCloseable {
143+
final DatagramChannel dc;
144+
final int dstPort;
121145

122-
public void throwException() throws Exception {
123-
if (e != null)
124-
throw e;
146+
NioWriter(int dstPort) throws IOException {
147+
this.dc = DatagramChannel.open();
148+
this.dstPort = dstPort;
125149
}
126150

127151
public void run() {
128152
try {
129-
DatagramChannel dc = DatagramChannel.open();
130153
ByteBuffer bb = ByteBuffer.allocateDirect(256);
131-
bb.put("hello".getBytes());
154+
bb.put(DATA_STRING.getBytes(US_ASCII));
132155
bb.flip();
133-
InetAddress address = InetAddress.getLocalHost();
134-
InetSocketAddress isa = new InetSocketAddress(address, port);
135-
dc.send(bb, isa);
136-
Thread.sleep(50);
156+
InetAddress address = InetAddress.getLoopbackAddress();
157+
InetSocketAddress isa = new InetSocketAddress(address, dstPort);
137158
dc.send(bb, isa);
138159
} catch (Exception ex) {
139-
e = ex;
160+
log.println("NioWriter [" + dc.socket().getLocalAddress() + "]");
161+
throw new RuntimeException("NioWriter threw exception: " + ex);
162+
} finally {
163+
log.println("NioWriter finished");
140164
}
141165
}
166+
167+
@Override
168+
public void close() throws IOException {
169+
dc.close();
170+
}
142171
}
143172

144-
public static class ClassicReader implements Sprintable {
173+
public static class ClassicReader implements Runnable, AutoCloseable {
145174
final DatagramSocket ds;
146-
Exception e = null;
147175

148176
ClassicReader() throws IOException {
149-
this.ds = new DatagramSocket();
177+
InetSocketAddress address = new InetSocketAddress(InetAddress.getLoopbackAddress(), 0);
178+
this.ds = new DatagramSocket(address);
150179
}
151180

152181
int port() {
153182
return ds.getLocalPort();
154183
}
155184

156-
public void throwException() throws Exception {
157-
if (e != null)
158-
throw e;
159-
}
160-
161185
public void run() {
162186
try {
163187
byte[] buf = new byte[256];
164188
DatagramPacket dp = new DatagramPacket(buf, buf.length);
165189
ds.receive(dp);
166-
String received = new String(dp.getData());
167-
log.println(received);
168-
ds.close();
190+
String received = new String(dp.getData(), dp.getOffset(), dp.getLength(), US_ASCII);
191+
log.println("ClassicReader received: " + received);
169192
} catch (Exception ex) {
170-
e = ex;
193+
log.println("ClassicReader [" + ds.getLocalAddress() +"]");
194+
throw new RuntimeException("ClassicReader threw exception: " + ex);
195+
} finally {
196+
log.println("ClassicReader finished");
171197
}
172198
}
199+
200+
@Override
201+
public void close() throws IOException {
202+
ds.close();
203+
}
173204
}
174205

175-
public static class NioReader implements Sprintable {
206+
public static class NioReader implements Runnable, AutoCloseable {
176207
final DatagramChannel dc;
177-
Exception e = null;
178208

179209
NioReader() throws IOException {
180-
this.dc = DatagramChannel.open().bind(new InetSocketAddress(0));
210+
InetSocketAddress address = new InetSocketAddress(InetAddress.getLoopbackAddress(), 0);
211+
this.dc = DatagramChannel.open().bind(address);
181212
}
182213

183214
int port() {
184215
return dc.socket().getLocalPort();
185216
}
186217

187-
public void throwException() throws Exception {
188-
if (e != null)
189-
throw e;
190-
}
191-
192218
public void run() {
193219
try {
194220
ByteBuffer bb = ByteBuffer.allocateDirect(100);
195-
SocketAddress sa = dc.receive(bb);
221+
dc.receive(bb);
196222
bb.flip();
197-
CharBuffer cb = Charset.forName("US-ASCII").
198-
newDecoder().decode(bb);
199-
log.println("From: "+sa+ " said " +cb);
200-
dc.close();
223+
CharBuffer cb = US_ASCII.newDecoder().decode(bb);
224+
log.println("NioReader received: " + cb);
201225
} catch (Exception ex) {
202-
e = ex;
226+
log.println("NioReader [" + dc.socket().getLocalAddress() +"]");
227+
throw new RuntimeException("NioReader threw exception: " + ex);
228+
} finally {
229+
log.println("NioReader finished");
203230
}
204231
}
205-
}
206232

233+
@Override
234+
public void close() throws IOException {
235+
dc.close();
236+
}
237+
}
207238
}

0 commit comments

Comments
 (0)
Please sign in to comment.