1 /* 2 * Copyright (c) 2008, 2013, Oracle and/or its affiliates. All rights reserved. 3 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. 4 * 5 * This code is free software; you can redistribute it and/or modify it 6 * under the terms of the GNU General Public License version 2 only, as 7 * published by the Free Software Foundation. Oracle designates this 8 * particular file as subject to the "Classpath" exception as provided 9 * by Oracle in the LICENSE file that accompanied this code. 10 * 11 * This code is distributed in the hope that it will be useful, but WITHOUT 12 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or 13 * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License 14 * version 2 for more details (a copy is included in the LICENSE file that 15 * accompanied this code). 16 * 17 * You should have received a copy of the GNU General Public License version 18 * 2 along with this work; if not, write to the Free Software Foundation, 19 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. 20 * 21 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA 22 * or visit www.oracle.com if you need additional information or have any 23 * questions. 24 */ 25 26 package sun.nio.ch; 27 28 import java.nio.channels.*; 29 import java.util.concurrent.*; 30 import java.io.IOException; 31 import java.io.FileDescriptor; 32 import java.net.InetSocketAddress; 33 import java.util.concurrent.atomic.AtomicBoolean; 34 import java.security.AccessControlContext; 35 import java.security.AccessController; 36 import java.security.PrivilegedAction; 37 38 import dalvik.annotation.optimization.ReachabilitySensitive; 39 import dalvik.system.CloseGuard; 40 41 /** 42 * Unix implementation of AsynchronousServerSocketChannel 43 */ 44 45 class UnixAsynchronousServerSocketChannelImpl 46 extends AsynchronousServerSocketChannelImpl 47 implements Port.PollableChannel 48 { 49 private static final NativeDispatcher nd = new SocketDispatcher(); 50 51 private final Port port; 52 private final int fdVal; 53 54 // flag to indicate an accept is outstanding 55 private final AtomicBoolean accepting = new AtomicBoolean(); enableAccept()56 private void enableAccept() { 57 accepting.set(false); 58 } 59 60 // used to ensure that the context for an asynchronous accept is visible 61 // the pooled thread that handles the I/O event 62 private final Object updateLock = new Object(); 63 64 // pending accept 65 private boolean acceptPending; 66 private CompletionHandler<AsynchronousSocketChannel,Object> acceptHandler; 67 private Object acceptAttachment; 68 private PendingFuture<AsynchronousSocketChannel,Object> acceptFuture; 69 70 // context for permission check when security manager set 71 private AccessControlContext acceptAcc; 72 73 // Android-added: CloseGuard support. 74 @ReachabilitySensitive 75 private final CloseGuard guard = CloseGuard.get(); 76 77 UnixAsynchronousServerSocketChannelImpl(Port port)78 UnixAsynchronousServerSocketChannelImpl(Port port) 79 throws IOException 80 { 81 super(port); 82 83 try { 84 IOUtil.configureBlocking(fd, false); 85 } catch (IOException x) { 86 nd.close(fd); // prevent leak 87 throw x; 88 } 89 this.port = port; 90 this.fdVal = IOUtil.fdVal(fd); 91 92 // add mapping from file descriptor to this channel 93 port.register(fdVal, this); 94 // Android-added: CloseGuard support. 95 guard.open("close"); 96 } 97 98 @Override implClose()99 void implClose() throws IOException { 100 // Android-added: CloseGuard support. 101 guard.close(); 102 // remove the mapping 103 port.unregister(fdVal); 104 105 // close file descriptor 106 nd.close(fd); 107 108 // if there is a pending accept then complete it 109 CompletionHandler<AsynchronousSocketChannel,Object> handler; 110 Object att; 111 PendingFuture<AsynchronousSocketChannel,Object> future; 112 synchronized (updateLock) { 113 if (!acceptPending) 114 return; // no pending accept 115 acceptPending = false; 116 handler = acceptHandler; 117 att = acceptAttachment; 118 future = acceptFuture; 119 } 120 121 // discard the stack trace as otherwise it may appear that implClose 122 // has thrown the exception. 123 AsynchronousCloseException x = new AsynchronousCloseException(); 124 x.setStackTrace(new StackTraceElement[0]); 125 if (handler == null) { 126 future.setFailure(x); 127 } else { 128 // invoke by submitting task rather than directly 129 Invoker.invokeIndirectly(this, handler, att, null, x); 130 } 131 } 132 133 // Android-added: CloseGuard support. finalize()134 protected void finalize() throws Throwable { 135 try { 136 if (guard != null) { 137 guard.warnIfOpen(); 138 } 139 close(); 140 } finally { 141 super.finalize(); 142 } 143 } 144 145 @Override group()146 public AsynchronousChannelGroupImpl group() { 147 return port; 148 } 149 150 /** 151 * Invoked by event handling thread when listener socket is polled 152 */ 153 @Override onEvent(int events, boolean mayInvokeDirect)154 public void onEvent(int events, boolean mayInvokeDirect) { 155 synchronized (updateLock) { 156 if (!acceptPending) 157 return; // may have been grabbed by asynchronous close 158 acceptPending = false; 159 } 160 161 // attempt to accept connection 162 FileDescriptor newfd = new FileDescriptor(); 163 InetSocketAddress[] isaa = new InetSocketAddress[1]; 164 Throwable exc = null; 165 try { 166 begin(); 167 int n = accept(this.fd, newfd, isaa); 168 169 // spurious wakeup, is this possible? 170 if (n == IOStatus.UNAVAILABLE) { 171 synchronized (updateLock) { 172 acceptPending = true; 173 } 174 port.startPoll(fdVal, Net.POLLIN); 175 return; 176 } 177 178 } catch (Throwable x) { 179 if (x instanceof ClosedChannelException) 180 x = new AsynchronousCloseException(); 181 exc = x; 182 } finally { 183 end(); 184 } 185 186 // Connection accepted so finish it when not holding locks. 187 AsynchronousSocketChannel child = null; 188 if (exc == null) { 189 try { 190 child = finishAccept(newfd, isaa[0], acceptAcc); 191 } catch (Throwable x) { 192 if (!(x instanceof IOException) && !(x instanceof SecurityException)) 193 x = new IOException(x); 194 exc = x; 195 } 196 } 197 198 // copy field befores accept is re-renabled 199 CompletionHandler<AsynchronousSocketChannel,Object> handler = acceptHandler; 200 Object att = acceptAttachment; 201 PendingFuture<AsynchronousSocketChannel,Object> future = acceptFuture; 202 203 // re-enable accepting and invoke handler 204 enableAccept(); 205 206 if (handler == null) { 207 future.setResult(child, exc); 208 // if an async cancel has already cancelled the operation then 209 // close the new channel so as to free resources 210 if (child != null && future.isCancelled()) { 211 try { 212 child.close(); 213 } catch (IOException ignore) { } 214 } 215 } else { 216 Invoker.invoke(this, handler, att, child, exc); 217 } 218 } 219 220 /** 221 * Completes the accept by creating the AsynchronousSocketChannel for 222 * the given file descriptor and remote address. If this method completes 223 * with an IOException or SecurityException then the channel/file descriptor 224 * will be closed. 225 */ finishAccept(FileDescriptor newfd, final InetSocketAddress remote, AccessControlContext acc)226 private AsynchronousSocketChannel finishAccept(FileDescriptor newfd, 227 final InetSocketAddress remote, 228 AccessControlContext acc) 229 throws IOException, SecurityException 230 { 231 AsynchronousSocketChannel ch = null; 232 try { 233 ch = new UnixAsynchronousSocketChannelImpl(port, newfd, remote); 234 } catch (IOException x) { 235 nd.close(newfd); 236 throw x; 237 } 238 239 // permission check must always be in initiator's context 240 try { 241 if (acc != null) { 242 AccessController.doPrivileged(new PrivilegedAction<>() { 243 public Void run() { 244 SecurityManager sm = System.getSecurityManager(); 245 if (sm != null) { 246 sm.checkAccept(remote.getAddress().getHostAddress(), 247 remote.getPort()); 248 } 249 return null; 250 } 251 }, acc); 252 } else { 253 SecurityManager sm = System.getSecurityManager(); 254 if (sm != null) { 255 sm.checkAccept(remote.getAddress().getHostAddress(), 256 remote.getPort()); 257 } 258 } 259 } catch (SecurityException x) { 260 try { 261 ch.close(); 262 } catch (Throwable suppressed) { 263 x.addSuppressed(suppressed); 264 } 265 throw x; 266 } 267 return ch; 268 } 269 270 @Override implAccept(Object att, CompletionHandler<AsynchronousSocketChannel,Object> handler)271 Future<AsynchronousSocketChannel> implAccept(Object att, 272 CompletionHandler<AsynchronousSocketChannel,Object> handler) 273 { 274 // complete immediately if channel is closed 275 if (!isOpen()) { 276 Throwable e = new ClosedChannelException(); 277 if (handler == null) { 278 return CompletedFuture.withFailure(e); 279 } else { 280 Invoker.invoke(this, handler, att, null, e); 281 return null; 282 } 283 } 284 if (localAddress == null) 285 throw new NotYetBoundException(); 286 287 // cancel was invoked with pending accept so connection may have been 288 // dropped. 289 if (isAcceptKilled()) 290 throw new RuntimeException("Accept not allowed due cancellation"); 291 292 // check and set flag to prevent concurrent accepting 293 if (!accepting.compareAndSet(false, true)) 294 throw new AcceptPendingException(); 295 296 // attempt accept 297 FileDescriptor newfd = new FileDescriptor(); 298 InetSocketAddress[] isaa = new InetSocketAddress[1]; 299 Throwable exc = null; 300 try { 301 begin(); 302 303 int n = accept(this.fd, newfd, isaa); 304 if (n == IOStatus.UNAVAILABLE) { 305 306 // need calling context when there is security manager as 307 // permission check may be done in a different thread without 308 // any application call frames on the stack 309 PendingFuture<AsynchronousSocketChannel,Object> result = null; 310 synchronized (updateLock) { 311 if (handler == null) { 312 this.acceptHandler = null; 313 result = new PendingFuture<>(this); 314 this.acceptFuture = result; 315 } else { 316 this.acceptHandler = handler; 317 this.acceptAttachment = att; 318 } 319 this.acceptAcc = (System.getSecurityManager() == null) ? 320 null : AccessController.getContext(); 321 this.acceptPending = true; 322 } 323 324 // register for connections 325 port.startPoll(fdVal, Net.POLLIN); 326 return result; 327 } 328 } catch (Throwable x) { 329 // accept failed 330 if (x instanceof ClosedChannelException) 331 x = new AsynchronousCloseException(); 332 exc = x; 333 } finally { 334 end(); 335 } 336 337 AsynchronousSocketChannel child = null; 338 if (exc == null) { 339 // connection accepted immediately 340 try { 341 child = finishAccept(newfd, isaa[0], null); 342 } catch (Throwable x) { 343 exc = x; 344 } 345 } 346 347 // re-enable accepting before invoking handler 348 enableAccept(); 349 350 if (handler == null) { 351 return CompletedFuture.withResult(child, exc); 352 } else { 353 Invoker.invokeIndirectly(this, handler, att, child, exc); 354 return null; 355 } 356 } 357 358 /** 359 * Accept a connection on a socket. 360 * 361 * @implNote Wrap native call to allow instrumentation. 362 */ accept(FileDescriptor ssfd, FileDescriptor newfd, InetSocketAddress[] isaa)363 private int accept(FileDescriptor ssfd, FileDescriptor newfd, 364 InetSocketAddress[] isaa) 365 throws IOException 366 { 367 return accept0(ssfd, newfd, isaa); 368 } 369 370 // -- Native methods -- 371 initIDs()372 private static native void initIDs(); 373 374 // Accepts a new connection, setting the given file descriptor to refer to 375 // the new socket and setting isaa[0] to the socket's remote address. 376 // Returns 1 on success, or IOStatus.UNAVAILABLE. 377 // accept0(FileDescriptor ssfd, FileDescriptor newfd, InetSocketAddress[] isaa)378 private native int accept0(FileDescriptor ssfd, FileDescriptor newfd, 379 InetSocketAddress[] isaa) 380 throws IOException; 381 382 static { 383 // Android-removed: Code to load native libraries, doesn't make sense on Android. 384 // IOUtil.load(); initIDs()385 initIDs(); 386 } 387 } 388