8278270: ServerSocket is not thread safe
Reviewed-by: dfuchs
This commit is contained in:
parent
e535cb3fba
commit
24877ac078
@ -78,17 +78,17 @@ import sun.net.PlatformSocketImpl;
|
||||
*/
|
||||
public class ServerSocket implements java.io.Closeable {
|
||||
/**
|
||||
* Various states of this socket.
|
||||
* The underlying SocketImpl
|
||||
*/
|
||||
private boolean created = false;
|
||||
private boolean bound = false;
|
||||
private boolean closed = false;
|
||||
private Object closeLock = new Object();
|
||||
private final SocketImpl impl;
|
||||
|
||||
/**
|
||||
* The implementation of this Socket.
|
||||
* Various states of this socket, need stateLock to change.
|
||||
*/
|
||||
private SocketImpl impl;
|
||||
private volatile boolean created; // impl.create(boolean) called
|
||||
private volatile boolean bound;
|
||||
private volatile boolean closed;
|
||||
private final Object stateLock = new Object();
|
||||
|
||||
/**
|
||||
* Creates a server socket with a user-specified {@code SocketImpl}.
|
||||
@ -124,7 +124,7 @@ public class ServerSocket implements java.io.Closeable {
|
||||
* @revised 1.4
|
||||
*/
|
||||
public ServerSocket() throws IOException {
|
||||
setImpl();
|
||||
this.impl = createImpl();
|
||||
}
|
||||
|
||||
/**
|
||||
@ -264,18 +264,15 @@ public class ServerSocket implements java.io.Closeable {
|
||||
* @since 1.1
|
||||
*/
|
||||
public ServerSocket(int port, int backlog, InetAddress bindAddr) throws IOException {
|
||||
setImpl();
|
||||
if (port < 0 || port > 0xFFFF)
|
||||
throw new IllegalArgumentException(
|
||||
"Port value out of range: " + port);
|
||||
throw new IllegalArgumentException("Port value out of range: " + port);
|
||||
if (backlog < 1)
|
||||
backlog = 50;
|
||||
|
||||
this.impl = createImpl();
|
||||
try {
|
||||
bind(new InetSocketAddress(bindAddr, port), backlog);
|
||||
} catch(SecurityException e) {
|
||||
close();
|
||||
throw e;
|
||||
} catch(IOException e) {
|
||||
} catch (IOException | SecurityException e) {
|
||||
close();
|
||||
throw e;
|
||||
}
|
||||
@ -289,35 +286,36 @@ public class ServerSocket implements java.io.Closeable {
|
||||
* @throws SocketException if creation fails.
|
||||
* @since 1.4
|
||||
*/
|
||||
SocketImpl getImpl() throws SocketException {
|
||||
if (!created)
|
||||
createImpl();
|
||||
private SocketImpl getImpl() throws SocketException {
|
||||
if (!created) {
|
||||
synchronized (stateLock) {
|
||||
if (!created) {
|
||||
if (closed) {
|
||||
throw new SocketException("Socket is closed");
|
||||
}
|
||||
try {
|
||||
impl.create(true);
|
||||
} catch (SocketException e) {
|
||||
throw e;
|
||||
} catch (IOException e) {
|
||||
throw new SocketException(e.getMessage());
|
||||
}
|
||||
created = true;
|
||||
}
|
||||
}
|
||||
}
|
||||
return impl;
|
||||
}
|
||||
|
||||
private void setImpl() {
|
||||
/**
|
||||
* Create a SocketImpl for a server socket.
|
||||
*/
|
||||
private static SocketImpl createImpl() {
|
||||
SocketImplFactory factory = ServerSocket.factory;
|
||||
if (factory != null) {
|
||||
impl = factory.createSocketImpl();
|
||||
return factory.createSocketImpl();
|
||||
} else {
|
||||
impl = SocketImpl.createPlatformSocketImpl(true);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Creates the socket implementation.
|
||||
*
|
||||
* @throws SocketException if creation fails
|
||||
* @since 1.4
|
||||
*/
|
||||
void createImpl() throws SocketException {
|
||||
if (impl == null)
|
||||
setImpl();
|
||||
try {
|
||||
impl.create(true);
|
||||
created = true;
|
||||
} catch (IOException e) {
|
||||
throw new SocketException(e.getMessage());
|
||||
return SocketImpl.createPlatformSocketImpl(true);
|
||||
}
|
||||
}
|
||||
|
||||
@ -380,20 +378,20 @@ public class ServerSocket implements java.io.Closeable {
|
||||
throw new SocketException("Unresolved address");
|
||||
if (backlog < 1)
|
||||
backlog = 50;
|
||||
try {
|
||||
|
||||
@SuppressWarnings("removal")
|
||||
SecurityManager security = System.getSecurityManager();
|
||||
if (security != null)
|
||||
security.checkListen(epoint.getPort());
|
||||
|
||||
synchronized (stateLock) {
|
||||
if (closed)
|
||||
throw new SocketException("Socket is closed");
|
||||
if (bound)
|
||||
throw new SocketException("Already bound");
|
||||
getImpl().bind(epoint.getAddress(), epoint.getPort());
|
||||
getImpl().listen(backlog);
|
||||
bound = true;
|
||||
} catch(SecurityException e) {
|
||||
bound = false;
|
||||
throw e;
|
||||
} catch(IOException e) {
|
||||
bound = false;
|
||||
throw e;
|
||||
}
|
||||
}
|
||||
|
||||
@ -711,13 +709,19 @@ public class ServerSocket implements java.io.Closeable {
|
||||
* @revised 1.4
|
||||
*/
|
||||
public void close() throws IOException {
|
||||
synchronized(closeLock) {
|
||||
if (isClosed())
|
||||
return;
|
||||
if (created)
|
||||
synchronized (stateLock) {
|
||||
if (!closed) {
|
||||
try {
|
||||
// close underlying socket if created
|
||||
if (created) {
|
||||
impl.close();
|
||||
}
|
||||
} finally {
|
||||
closed = true;
|
||||
}
|
||||
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
@ -760,10 +764,8 @@ public class ServerSocket implements java.io.Closeable {
|
||||
* @since 1.4
|
||||
*/
|
||||
public boolean isClosed() {
|
||||
synchronized(closeLock) {
|
||||
return closed;
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Enable/disable {@link SocketOptions#SO_TIMEOUT SO_TIMEOUT} with the
|
||||
@ -783,7 +785,7 @@ public class ServerSocket implements java.io.Closeable {
|
||||
* @since 1.1
|
||||
* @see #getSoTimeout()
|
||||
*/
|
||||
public synchronized void setSoTimeout(int timeout) throws SocketException {
|
||||
public void setSoTimeout(int timeout) throws SocketException {
|
||||
if (isClosed())
|
||||
throw new SocketException("Socket is closed");
|
||||
if (timeout < 0)
|
||||
@ -799,7 +801,7 @@ public class ServerSocket implements java.io.Closeable {
|
||||
* @since 1.1
|
||||
* @see #setSoTimeout(int)
|
||||
*/
|
||||
public synchronized int getSoTimeout() throws IOException {
|
||||
public int getSoTimeout() throws IOException {
|
||||
if (isClosed())
|
||||
throw new SocketException("Socket is closed");
|
||||
Object o = getImpl().getOption(SocketOptions.SO_TIMEOUT);
|
||||
@ -984,7 +986,7 @@ public class ServerSocket implements java.io.Closeable {
|
||||
* @since 1.4
|
||||
* @see #getReceiveBufferSize
|
||||
*/
|
||||
public synchronized void setReceiveBufferSize (int size) throws SocketException {
|
||||
public void setReceiveBufferSize (int size) throws SocketException {
|
||||
if (!(size > 0)) {
|
||||
throw new IllegalArgumentException("negative receive size");
|
||||
}
|
||||
@ -1007,8 +1009,7 @@ public class ServerSocket implements java.io.Closeable {
|
||||
* @see #setReceiveBufferSize(int)
|
||||
* @since 1.4
|
||||
*/
|
||||
public synchronized int getReceiveBufferSize()
|
||||
throws SocketException{
|
||||
public int getReceiveBufferSize() throws SocketException {
|
||||
if (isClosed())
|
||||
throw new SocketException("Socket is closed");
|
||||
int result = 0;
|
||||
|
Loading…
Reference in New Issue
Block a user