mirror of
https://github.com/openjdk/jdk.git
synced 2025-08-28 15:24:43 +02:00
8200458: (se) Readiness information previously recorded in the ready set not preserved
Reviewed-by: bpb, chegar
This commit is contained in:
parent
d185d65b69
commit
97d7cfb14e
8 changed files with 193 additions and 128 deletions
|
@ -191,7 +191,7 @@ class EPollSelectorImpl extends SelectorImpl {
|
||||||
if (ski != null) {
|
if (ski != null) {
|
||||||
int rOps = EPoll.getEvents(event);
|
int rOps = EPoll.getEvents(event);
|
||||||
if (selectedKeys.contains(ski)) {
|
if (selectedKeys.contains(ski)) {
|
||||||
if (ski.translateAndSetReadyOps(rOps)) {
|
if (ski.translateAndUpdateReadyOps(rOps)) {
|
||||||
numKeysUpdated++;
|
numKeysUpdated++;
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
|
|
|
@ -216,15 +216,12 @@ class KQueueSelectorImpl extends SelectorImpl {
|
||||||
}
|
}
|
||||||
|
|
||||||
if (selectedKeys.contains(ski)) {
|
if (selectedKeys.contains(ski)) {
|
||||||
|
if (ski.translateAndUpdateReadyOps(rOps)) {
|
||||||
// file descriptor may be polled more than once per poll
|
// file descriptor may be polled more than once per poll
|
||||||
if (ski.lastPolled != pollCount) {
|
if (ski.lastPolled != pollCount) {
|
||||||
if (ski.translateAndSetReadyOps(rOps)) {
|
|
||||||
numKeysUpdated++;
|
numKeysUpdated++;
|
||||||
ski.lastPolled = pollCount;
|
ski.lastPolled = pollCount;
|
||||||
}
|
}
|
||||||
} else {
|
|
||||||
// ready ops have already been set on this update
|
|
||||||
ski.translateAndUpdateReadyOps(rOps);
|
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
ski.translateAndSetReadyOps(rOps);
|
ski.translateAndSetReadyOps(rOps);
|
||||||
|
|
|
@ -184,7 +184,7 @@ class DevPollSelectorImpl
|
||||||
if (ski != null) {
|
if (ski != null) {
|
||||||
int rOps = pollWrapper.getReventOps(i);
|
int rOps = pollWrapper.getReventOps(i);
|
||||||
if (selectedKeys.contains(ski)) {
|
if (selectedKeys.contains(ski)) {
|
||||||
if (ski.translateAndSetReadyOps(rOps)) {
|
if (ski.translateAndUpdateReadyOps(rOps)) {
|
||||||
numKeysUpdated++;
|
numKeysUpdated++;
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
|
|
|
@ -188,7 +188,7 @@ class EventPortSelectorImpl
|
||||||
if (ski != null) {
|
if (ski != null) {
|
||||||
int rOps = getEventOps(i);
|
int rOps = getEventOps(i);
|
||||||
if (selectedKeys.contains(ski)) {
|
if (selectedKeys.contains(ski)) {
|
||||||
if (ski.translateAndSetReadyOps(rOps)) {
|
if (ski.translateAndUpdateReadyOps(rOps)) {
|
||||||
numKeysUpdated++;
|
numKeysUpdated++;
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
|
|
|
@ -174,7 +174,7 @@ class PollSelectorImpl extends SelectorImpl {
|
||||||
assert ski.getFDVal() == getDescriptor(i);
|
assert ski.getFDVal() == getDescriptor(i);
|
||||||
if (ski.isValid()) {
|
if (ski.isValid()) {
|
||||||
if (selectedKeys.contains(ski)) {
|
if (selectedKeys.contains(ski)) {
|
||||||
if (ski.translateAndSetReadyOps(rOps)) {
|
if (ski.translateAndUpdateReadyOps(rOps)) {
|
||||||
numKeysUpdated++;
|
numKeysUpdated++;
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
|
|
|
@ -98,7 +98,6 @@ class WindowsSelectorImpl extends SelectorImpl {
|
||||||
private static final class MapEntry {
|
private static final class MapEntry {
|
||||||
final SelectionKeyImpl ski;
|
final SelectionKeyImpl ski;
|
||||||
long updateCount = 0;
|
long updateCount = 0;
|
||||||
long clearedCount = 0;
|
|
||||||
MapEntry(SelectionKeyImpl ski) {
|
MapEntry(SelectionKeyImpl ski) {
|
||||||
this.ski = ski;
|
this.ski = ski;
|
||||||
}
|
}
|
||||||
|
@ -368,12 +367,10 @@ class WindowsSelectorImpl extends SelectorImpl {
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Note, clearedCount is used to determine if the readyOps have
|
* updateCount is used to tell if a key has been counted as updated
|
||||||
* been reset in this select operation. updateCount is used to
|
* in this select operation.
|
||||||
* tell if a key has been counted as updated in this select
|
|
||||||
* operation.
|
|
||||||
*
|
*
|
||||||
* me.updateCount <= me.clearedCount <= updateCount
|
* me.updateCount <= updateCount
|
||||||
*/
|
*/
|
||||||
private int processFDSet(long updateCount, int[] fds, int rOps,
|
private int processFDSet(long updateCount, int[] fds, int rOps,
|
||||||
boolean isExceptFds)
|
boolean isExceptFds)
|
||||||
|
@ -405,37 +402,19 @@ class WindowsSelectorImpl extends SelectorImpl {
|
||||||
}
|
}
|
||||||
|
|
||||||
if (selectedKeys.contains(sk)) { // Key in selected set
|
if (selectedKeys.contains(sk)) { // Key in selected set
|
||||||
if (me.clearedCount != updateCount) {
|
if (sk.translateAndUpdateReadyOps(rOps)) {
|
||||||
if (sk.translateAndSetReadyOps(rOps) &&
|
if (me.updateCount != updateCount) {
|
||||||
(me.updateCount != updateCount)) {
|
|
||||||
me.updateCount = updateCount;
|
|
||||||
numKeysUpdated++;
|
|
||||||
}
|
|
||||||
} else { // The readyOps have been set; now add
|
|
||||||
if (sk.translateAndUpdateReadyOps(rOps) &&
|
|
||||||
(me.updateCount != updateCount)) {
|
|
||||||
me.updateCount = updateCount;
|
me.updateCount = updateCount;
|
||||||
numKeysUpdated++;
|
numKeysUpdated++;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
me.clearedCount = updateCount;
|
|
||||||
} else { // Key is not in selected set yet
|
} else { // Key is not in selected set yet
|
||||||
if (me.clearedCount != updateCount) {
|
|
||||||
sk.translateAndSetReadyOps(rOps);
|
sk.translateAndSetReadyOps(rOps);
|
||||||
if ((sk.nioReadyOps() & sk.nioInterestOps()) != 0) {
|
if ((sk.nioReadyOps() & sk.nioInterestOps()) != 0) {
|
||||||
selectedKeys.add(sk);
|
selectedKeys.add(sk);
|
||||||
me.updateCount = updateCount;
|
me.updateCount = updateCount;
|
||||||
numKeysUpdated++;
|
numKeysUpdated++;
|
||||||
}
|
}
|
||||||
} else { // The readyOps have been set; now add
|
|
||||||
sk.translateAndUpdateReadyOps(rOps);
|
|
||||||
if ((sk.nioReadyOps() & sk.nioInterestOps()) != 0) {
|
|
||||||
selectedKeys.add(sk);
|
|
||||||
me.updateCount = updateCount;
|
|
||||||
numKeysUpdated++;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
me.clearedCount = updateCount;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return numKeysUpdated;
|
return numKeysUpdated;
|
||||||
|
|
176
test/jdk/java/nio/channels/Selector/UpdateReadyOps.java
Normal file
176
test/jdk/java/nio/channels/Selector/UpdateReadyOps.java
Normal file
|
@ -0,0 +1,176 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) 2018, 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
|
||||||
|
* @run testng UpdateReadyOps
|
||||||
|
* @summary Test that the ready set from a selection operation is bitwise-disjoined
|
||||||
|
* into a key's ready set when the key is already in the selected-key set
|
||||||
|
*/
|
||||||
|
|
||||||
|
import java.io.Closeable;
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.net.InetAddress;
|
||||||
|
import java.net.InetSocketAddress;
|
||||||
|
import java.nio.ByteBuffer;
|
||||||
|
import java.nio.channels.SelectionKey;
|
||||||
|
import java.nio.channels.Selector;
|
||||||
|
import java.nio.channels.ServerSocketChannel;
|
||||||
|
import java.nio.channels.SocketChannel;
|
||||||
|
|
||||||
|
import org.testng.annotations.Test;
|
||||||
|
import static org.testng.Assert.*;
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public class UpdateReadyOps {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Test that OP_WRITE is preserved when updating the ready set of a key in
|
||||||
|
* the selected-key set to add OP_READ.
|
||||||
|
*/
|
||||||
|
public void testOpWritePreserved() throws Exception {
|
||||||
|
try (ConnectionPair pair = new ConnectionPair();
|
||||||
|
Selector sel = Selector.open()) {
|
||||||
|
|
||||||
|
SocketChannel sc1 = pair.channel1();
|
||||||
|
SocketChannel sc2 = pair.channel2();
|
||||||
|
|
||||||
|
sc1.configureBlocking(false);
|
||||||
|
SelectionKey key = sc1.register(sel, SelectionKey.OP_WRITE);
|
||||||
|
|
||||||
|
int updated = sel.select();
|
||||||
|
assertTrue(updated == 1);
|
||||||
|
assertTrue(sel.selectedKeys().contains(key));
|
||||||
|
assertFalse(key.isReadable());
|
||||||
|
assertTrue(key.isWritable());
|
||||||
|
|
||||||
|
// select again, should be no updates
|
||||||
|
updated = sel.select();
|
||||||
|
assertTrue(updated == 0);
|
||||||
|
assertTrue(sel.selectedKeys().contains(key));
|
||||||
|
assertFalse(key.isReadable());
|
||||||
|
assertTrue(key.isWritable());
|
||||||
|
|
||||||
|
// write some bytes
|
||||||
|
sc2.write(helloMessage());
|
||||||
|
|
||||||
|
// change interest ops to OP_READ, do a selection operation, and
|
||||||
|
// check that the ready set becomes OP_READ|OP_WRITE.
|
||||||
|
|
||||||
|
key.interestOps(SelectionKey.OP_READ);
|
||||||
|
updated = sel.select();
|
||||||
|
assertTrue(updated == 1);
|
||||||
|
assertTrue(sel.selectedKeys().size() == 1);
|
||||||
|
assertTrue(key.isReadable());
|
||||||
|
assertTrue(key.isWritable());
|
||||||
|
assertTrue(key.readyOps() == (SelectionKey.OP_READ|SelectionKey.OP_WRITE));
|
||||||
|
|
||||||
|
// select again, should be no updates
|
||||||
|
updated = sel.select();
|
||||||
|
assertTrue(updated == 0);
|
||||||
|
assertTrue(sel.selectedKeys().size() == 1);
|
||||||
|
assertTrue(key.isReadable());
|
||||||
|
assertTrue(key.isWritable());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Test that OP_READ is preserved when updating the ready set of a key in
|
||||||
|
* the selected-key set to add OP_WRITE.
|
||||||
|
*/
|
||||||
|
public void testOpReadPreserved() throws Exception {
|
||||||
|
try (ConnectionPair pair = new ConnectionPair();
|
||||||
|
Selector sel = Selector.open()) {
|
||||||
|
|
||||||
|
SocketChannel sc1 = pair.channel1();
|
||||||
|
SocketChannel sc2 = pair.channel2();
|
||||||
|
|
||||||
|
sc1.configureBlocking(false);
|
||||||
|
SelectionKey key = sc1.register(sel, SelectionKey.OP_READ);
|
||||||
|
|
||||||
|
// write some bytes
|
||||||
|
sc2.write(helloMessage());
|
||||||
|
|
||||||
|
int updated = sel.select();
|
||||||
|
assertTrue(updated == 1);
|
||||||
|
assertTrue(sel.selectedKeys().size() == 1);
|
||||||
|
assertTrue(sel.selectedKeys().contains(key));
|
||||||
|
assertTrue(key.isReadable());
|
||||||
|
assertFalse(key.isWritable());
|
||||||
|
|
||||||
|
// select again, should be no updates
|
||||||
|
updated = sel.select();
|
||||||
|
assertTrue(updated == 0);
|
||||||
|
assertTrue(sel.selectedKeys().contains(key));
|
||||||
|
assertTrue(key.isReadable());
|
||||||
|
assertFalse(key.isWritable());
|
||||||
|
|
||||||
|
key.interestOps(SelectionKey.OP_WRITE);
|
||||||
|
updated = sel.select();
|
||||||
|
assertTrue(updated == 1);
|
||||||
|
assertTrue(sel.selectedKeys().size() == 1);
|
||||||
|
assertTrue(sel.selectedKeys().contains(key));
|
||||||
|
assertTrue(key.isReadable());
|
||||||
|
assertTrue(key.isWritable());
|
||||||
|
assertTrue(key.readyOps() == (SelectionKey.OP_READ|SelectionKey.OP_WRITE));
|
||||||
|
|
||||||
|
// select again, should be no updates
|
||||||
|
updated = sel.select();
|
||||||
|
assertTrue(updated == 0);
|
||||||
|
assertTrue(sel.selectedKeys().size() == 1);
|
||||||
|
assertTrue(key.isReadable());
|
||||||
|
assertTrue(key.isWritable());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static class ConnectionPair implements Closeable {
|
||||||
|
|
||||||
|
private final SocketChannel sc1;
|
||||||
|
private final SocketChannel sc2;
|
||||||
|
|
||||||
|
ConnectionPair() throws IOException {
|
||||||
|
InetAddress lb = InetAddress.getLoopbackAddress();
|
||||||
|
try (ServerSocketChannel ssc = ServerSocketChannel.open()) {
|
||||||
|
ssc.bind(new InetSocketAddress(lb, 0));
|
||||||
|
this.sc1 = SocketChannel.open(ssc.getLocalAddress());
|
||||||
|
this.sc2 = ssc.accept();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
SocketChannel channel1() {
|
||||||
|
return sc1;
|
||||||
|
}
|
||||||
|
|
||||||
|
SocketChannel channel2() {
|
||||||
|
return sc2;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void close() throws IOException {
|
||||||
|
sc1.close();
|
||||||
|
sc2.close();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static ByteBuffer helloMessage() throws Exception {
|
||||||
|
return ByteBuffer.wrap("hello".getBytes("UTF-8"));
|
||||||
|
}
|
||||||
|
}
|
|
@ -1,87 +0,0 @@
|
||||||
/*
|
|
||||||
* Copyright (c) 2002, 2018, 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 4737146 4750573
|
|
||||||
* @summary Test if isConnectable returns true after connected
|
|
||||||
* @library .. /test/lib
|
|
||||||
* @build jdk.test.lib.Utils TestServers
|
|
||||||
* @run main IsConnectable
|
|
||||||
*/
|
|
||||||
|
|
||||||
import java.net.*;
|
|
||||||
import java.nio.channels.*;
|
|
||||||
import java.nio.channels.spi.SelectorProvider;
|
|
||||||
import java.util.*;
|
|
||||||
|
|
||||||
public class IsConnectable {
|
|
||||||
|
|
||||||
static void test(TestServers.DayTimeServer daytimeServer) throws Exception {
|
|
||||||
InetSocketAddress isa
|
|
||||||
= new InetSocketAddress(daytimeServer.getAddress(),
|
|
||||||
daytimeServer.getPort());
|
|
||||||
SocketChannel sc = SocketChannel.open();
|
|
||||||
sc.configureBlocking(false);
|
|
||||||
final boolean immediatelyConnected = sc.connect(isa);
|
|
||||||
|
|
||||||
Selector selector = SelectorProvider.provider().openSelector();
|
|
||||||
try {
|
|
||||||
SelectionKey key = sc.register(selector, SelectionKey.OP_CONNECT);
|
|
||||||
int keysAdded = selector.select();
|
|
||||||
if (keysAdded > 0) {
|
|
||||||
boolean result = sc.finishConnect();
|
|
||||||
if (result) {
|
|
||||||
keysAdded = selector.select(5000);
|
|
||||||
// 4750573: keysAdded should not be incremented when op is dropped
|
|
||||||
// from a key already in the selected key set
|
|
||||||
if (keysAdded > 0)
|
|
||||||
throw new Exception("Test failed: 4750573 detected");
|
|
||||||
Set<SelectionKey> sel = selector.selectedKeys();
|
|
||||||
Iterator<SelectionKey> i = sel.iterator();
|
|
||||||
SelectionKey sk = i.next();
|
|
||||||
// 4737146: isConnectable should be false while connected
|
|
||||||
if (sk.isConnectable())
|
|
||||||
throw new Exception("Test failed: 4737146 detected");
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
if (!immediatelyConnected) {
|
|
||||||
throw new Exception("Select failed");
|
|
||||||
} else {
|
|
||||||
System.out.println("IsConnectable couldn't be fully tested for "
|
|
||||||
+ System.getProperty("os.name"));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} finally {
|
|
||||||
sc.close();
|
|
||||||
selector.close();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public static void main(String[] args) throws Exception {
|
|
||||||
try (TestServers.DayTimeServer daytimeServer
|
|
||||||
= TestServers.DayTimeServer.startNewServer(100)) {
|
|
||||||
test(daytimeServer);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
Loading…
Add table
Add a link
Reference in a new issue