1 /* 2 * Copyright (C) 2011 The Android Open Source Project 3 * 4 * Licensed under the Apache License, Version 2.0 (the "License"); 5 * you may not use this file except in compliance with the License. 6 * You may obtain a copy of the License at 7 * 8 * http://www.apache.org/licenses/LICENSE-2.0 9 * 10 * Unless required by applicable law or agreed to in writing, software 11 * distributed under the License is distributed on an "AS IS" BASIS, 12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13 * See the License for the specific language governing permissions and 14 * limitations under the License. 15 */ 16 17 package libcore.io; 18 19 import android.system.ErrnoException; 20 import android.system.StructGroupReq; 21 import android.system.StructGroupSourceReq; 22 import android.system.StructLinger; 23 import android.system.StructPollfd; 24 import android.system.StructTimeval; 25 import android.util.MutableInt; 26 import java.io.FileDescriptor; 27 import java.io.FileNotFoundException; 28 import java.io.IOException; 29 import java.net.BindException; 30 import java.net.ConnectException; 31 import java.net.DatagramPacket; 32 import java.net.Inet4Address; 33 import java.net.Inet6Address; 34 import java.net.InetAddress; 35 import java.net.InetSocketAddress; 36 import java.net.NetworkInterface; 37 import java.net.PortUnreachableException; 38 import java.net.SocketAddress; 39 import java.net.SocketException; 40 import java.net.SocketOptions; 41 import java.net.SocketTimeoutException; 42 import java.net.UnknownHostException; 43 import java.nio.ByteBuffer; 44 import java.util.Arrays; 45 import java.util.concurrent.TimeUnit; 46 47 import static android.system.OsConstants.*; 48 49 /** 50 * Implements java.io/java.net/java.nio semantics in terms of the underlying POSIX system calls. 51 */ 52 public final class IoBridge { 53 IoBridge()54 private IoBridge() { 55 } 56 available(FileDescriptor fd)57 public static int available(FileDescriptor fd) throws IOException { 58 try { 59 MutableInt available = new MutableInt(0); 60 Libcore.os.ioctlInt(fd, FIONREAD, available); 61 if (available.value < 0) { 62 // If the fd refers to a regular file, the result is the difference between 63 // the file size and the file position. This may be negative if the position 64 // is past the end of the file. If the fd refers to a special file masquerading 65 // as a regular file, the result may be negative because the special file 66 // may appear to have zero size and yet a previous read call may have 67 // read some amount of data and caused the file position to be advanced. 68 available.value = 0; 69 } 70 return available.value; 71 } catch (ErrnoException errnoException) { 72 if (errnoException.errno == ENOTTY) { 73 // The fd is unwilling to opine about its read buffer. 74 return 0; 75 } 76 throw errnoException.rethrowAsIOException(); 77 } 78 } 79 80 bind(FileDescriptor fd, InetAddress address, int port)81 public static void bind(FileDescriptor fd, InetAddress address, int port) throws SocketException { 82 if (address instanceof Inet6Address) { 83 Inet6Address inet6Address = (Inet6Address) address; 84 if (inet6Address.getScopeId() == 0 && inet6Address.isLinkLocalAddress()) { 85 // Linux won't let you bind a link-local address without a scope id. 86 // Find one. 87 NetworkInterface nif = NetworkInterface.getByInetAddress(address); 88 if (nif == null) { 89 throw new SocketException("Can't bind to a link-local address without a scope id: " + address); 90 } 91 try { 92 address = Inet6Address.getByAddress(address.getHostName(), address.getAddress(), nif.getIndex()); 93 } catch (UnknownHostException ex) { 94 throw new AssertionError(ex); // Can't happen. 95 } 96 } 97 } 98 try { 99 Libcore.os.bind(fd, address, port); 100 } catch (ErrnoException errnoException) { 101 throw new BindException(errnoException.getMessage(), errnoException); 102 } 103 } 104 105 106 /** 107 * Connects socket 'fd' to 'inetAddress' on 'port', with no timeout. The lack of a timeout 108 * means this method won't throw SocketTimeoutException. 109 */ connect(FileDescriptor fd, InetAddress inetAddress, int port)110 public static void connect(FileDescriptor fd, InetAddress inetAddress, int port) throws SocketException { 111 try { 112 IoBridge.connect(fd, inetAddress, port, 0); 113 } catch (SocketTimeoutException ex) { 114 throw new AssertionError(ex); // Can't happen for a connect without a timeout. 115 } 116 } 117 118 /** 119 * Connects socket 'fd' to 'inetAddress' on 'port', with a the given 'timeoutMs'. 120 * Use timeoutMs == 0 for a blocking connect with no timeout. 121 */ connect(FileDescriptor fd, InetAddress inetAddress, int port, int timeoutMs)122 public static void connect(FileDescriptor fd, InetAddress inetAddress, int port, int timeoutMs) throws SocketException, SocketTimeoutException { 123 try { 124 connectErrno(fd, inetAddress, port, timeoutMs); 125 } catch (ErrnoException errnoException) { 126 throw new ConnectException(connectDetail(inetAddress, port, timeoutMs, errnoException), errnoException); 127 } catch (SocketException ex) { 128 throw ex; // We don't want to doubly wrap these. 129 } catch (SocketTimeoutException ex) { 130 throw ex; // We don't want to doubly wrap these. 131 } catch (IOException ex) { 132 throw new SocketException(ex); 133 } 134 } 135 connectErrno(FileDescriptor fd, InetAddress inetAddress, int port, int timeoutMs)136 private static void connectErrno(FileDescriptor fd, InetAddress inetAddress, int port, int timeoutMs) throws ErrnoException, IOException { 137 // With no timeout, just call connect(2) directly. 138 if (timeoutMs == 0) { 139 Libcore.os.connect(fd, inetAddress, port); 140 return; 141 } 142 143 // For connect with a timeout, we: 144 // 1. set the socket to non-blocking, 145 // 2. connect(2), 146 // 3. loop using poll(2) to decide whether we're connected, whether we should keep 147 // waiting, or whether we've seen a permanent failure and should give up, 148 // 4. set the socket back to blocking. 149 150 // 1. set the socket to non-blocking. 151 IoUtils.setBlocking(fd, false); 152 153 // 2. call connect(2) non-blocking. 154 long finishTimeNanos = System.nanoTime() + TimeUnit.MILLISECONDS.toNanos(timeoutMs); 155 try { 156 Libcore.os.connect(fd, inetAddress, port); 157 IoUtils.setBlocking(fd, true); // 4. set the socket back to blocking. 158 return; // We connected immediately. 159 } catch (ErrnoException errnoException) { 160 if (errnoException.errno != EINPROGRESS) { 161 throw errnoException; 162 } 163 // EINPROGRESS means we should keep trying... 164 } 165 166 // 3. loop using poll(2). 167 int remainingTimeoutMs; 168 do { 169 remainingTimeoutMs = 170 (int) TimeUnit.NANOSECONDS.toMillis(finishTimeNanos - System.nanoTime()); 171 if (remainingTimeoutMs <= 0) { 172 throw new SocketTimeoutException(connectDetail(inetAddress, port, timeoutMs, null)); 173 } 174 } while (!IoBridge.isConnected(fd, inetAddress, port, timeoutMs, remainingTimeoutMs)); 175 IoUtils.setBlocking(fd, true); // 4. set the socket back to blocking. 176 } 177 connectDetail(InetAddress inetAddress, int port, int timeoutMs, ErrnoException cause)178 private static String connectDetail(InetAddress inetAddress, int port, int timeoutMs, ErrnoException cause) { 179 String detail = "failed to connect to " + inetAddress + " (port " + port + ")"; 180 if (timeoutMs > 0) { 181 detail += " after " + timeoutMs + "ms"; 182 } 183 if (cause != null) { 184 detail += ": " + cause.getMessage(); 185 } 186 return detail; 187 } 188 189 /** 190 * Closes the supplied file descriptor and sends a signal to any threads are currently blocking. 191 * In order for the signal to be sent the blocked threads must have registered with 192 * the AsynchronousCloseMonitor before they entered the blocking operation. 193 * 194 * <p>This method is a no-op if passed a {@code null} or already-closed file descriptor. 195 */ closeAndSignalBlockedThreads(FileDescriptor fd)196 public static void closeAndSignalBlockedThreads(FileDescriptor fd) throws IOException { 197 if (fd == null || !fd.valid()) { 198 return; 199 } 200 int intFd = fd.getInt$(); 201 fd.setInt$(-1); 202 FileDescriptor oldFd = new FileDescriptor(); 203 oldFd.setInt$(intFd); 204 AsynchronousCloseMonitor.signalBlockedThreads(oldFd); 205 try { 206 Libcore.os.close(oldFd); 207 } catch (ErrnoException errnoException) { 208 // TODO: are there any cases in which we should throw? 209 } 210 } 211 isConnected(FileDescriptor fd, InetAddress inetAddress, int port, int timeoutMs, int remainingTimeoutMs)212 public static boolean isConnected(FileDescriptor fd, InetAddress inetAddress, int port, int timeoutMs, int remainingTimeoutMs) throws IOException { 213 ErrnoException cause; 214 try { 215 StructPollfd[] pollFds = new StructPollfd[] { new StructPollfd() }; 216 pollFds[0].fd = fd; 217 pollFds[0].events = (short) POLLOUT; 218 int rc = Libcore.os.poll(pollFds, remainingTimeoutMs); 219 if (rc == 0) { 220 return false; // Timeout. 221 } 222 int connectError = Libcore.os.getsockoptInt(fd, SOL_SOCKET, SO_ERROR); 223 if (connectError == 0) { 224 return true; // Success! 225 } 226 throw new ErrnoException("isConnected", connectError); // The connect(2) failed. 227 } catch (ErrnoException errnoException) { 228 if (!fd.valid()) { 229 throw new SocketException("Socket closed"); 230 } 231 cause = errnoException; 232 } 233 String detail = connectDetail(inetAddress, port, timeoutMs, cause); 234 if (cause.errno == ETIMEDOUT) { 235 throw new SocketTimeoutException(detail, cause); 236 } 237 throw new ConnectException(detail, cause); 238 } 239 240 // Socket options used by java.net but not exposed in SocketOptions. 241 public static final int JAVA_MCAST_JOIN_GROUP = 19; 242 public static final int JAVA_MCAST_LEAVE_GROUP = 20; 243 public static final int JAVA_MCAST_JOIN_SOURCE_GROUP = 21; 244 public static final int JAVA_MCAST_LEAVE_SOURCE_GROUP = 22; 245 public static final int JAVA_MCAST_BLOCK_SOURCE = 23; 246 public static final int JAVA_MCAST_UNBLOCK_SOURCE = 24; 247 public static final int JAVA_IP_MULTICAST_TTL = 17; 248 249 /** 250 * java.net has its own socket options similar to the underlying Unix ones. We paper over the 251 * differences here. 252 */ getSocketOption(FileDescriptor fd, int option)253 public static Object getSocketOption(FileDescriptor fd, int option) throws SocketException { 254 try { 255 return getSocketOptionErrno(fd, option); 256 } catch (ErrnoException errnoException) { 257 throw errnoException.rethrowAsSocketException(); 258 } 259 } 260 getSocketOptionErrno(FileDescriptor fd, int option)261 private static Object getSocketOptionErrno(FileDescriptor fd, int option) throws ErrnoException, SocketException { 262 switch (option) { 263 case SocketOptions.IP_MULTICAST_IF: 264 // This is IPv4-only. 265 return Libcore.os.getsockoptInAddr(fd, IPPROTO_IP, IP_MULTICAST_IF); 266 case SocketOptions.IP_MULTICAST_IF2: 267 // This is IPv6-only. 268 return Libcore.os.getsockoptInt(fd, IPPROTO_IPV6, IPV6_MULTICAST_IF); 269 case SocketOptions.IP_MULTICAST_LOOP: 270 // Since setting this from java.net always sets IPv4 and IPv6 to the same value, 271 // it doesn't matter which we return. 272 return booleanFromInt(Libcore.os.getsockoptInt(fd, IPPROTO_IPV6, IPV6_MULTICAST_LOOP)); 273 case IoBridge.JAVA_IP_MULTICAST_TTL: 274 // Since setting this from java.net always sets IPv4 and IPv6 to the same value, 275 // it doesn't matter which we return. 276 return Libcore.os.getsockoptInt(fd, IPPROTO_IPV6, IPV6_MULTICAST_HOPS); 277 case SocketOptions.IP_TOS: 278 // Since setting this from java.net always sets IPv4 and IPv6 to the same value, 279 // it doesn't matter which we return. 280 return Libcore.os.getsockoptInt(fd, IPPROTO_IPV6, IPV6_TCLASS); 281 case SocketOptions.SO_BROADCAST: 282 return booleanFromInt(Libcore.os.getsockoptInt(fd, SOL_SOCKET, SO_BROADCAST)); 283 case SocketOptions.SO_KEEPALIVE: 284 return booleanFromInt(Libcore.os.getsockoptInt(fd, SOL_SOCKET, SO_KEEPALIVE)); 285 case SocketOptions.SO_LINGER: 286 StructLinger linger = Libcore.os.getsockoptLinger(fd, SOL_SOCKET, SO_LINGER); 287 if (!linger.isOn()) { 288 return false; 289 } 290 return linger.l_linger; 291 case SocketOptions.SO_OOBINLINE: 292 return booleanFromInt(Libcore.os.getsockoptInt(fd, SOL_SOCKET, SO_OOBINLINE)); 293 case SocketOptions.SO_RCVBUF: 294 return Libcore.os.getsockoptInt(fd, SOL_SOCKET, SO_RCVBUF); 295 case SocketOptions.SO_REUSEADDR: 296 return booleanFromInt(Libcore.os.getsockoptInt(fd, SOL_SOCKET, SO_REUSEADDR)); 297 case SocketOptions.SO_SNDBUF: 298 return Libcore.os.getsockoptInt(fd, SOL_SOCKET, SO_SNDBUF); 299 case SocketOptions.SO_TIMEOUT: 300 return (int) Libcore.os.getsockoptTimeval(fd, SOL_SOCKET, SO_RCVTIMEO).toMillis(); 301 case SocketOptions.TCP_NODELAY: 302 return booleanFromInt(Libcore.os.getsockoptInt(fd, IPPROTO_TCP, TCP_NODELAY)); 303 default: 304 throw new SocketException("Unknown socket option: " + option); 305 } 306 } 307 booleanFromInt(int i)308 private static boolean booleanFromInt(int i) { 309 return (i != 0); 310 } 311 booleanToInt(boolean b)312 private static int booleanToInt(boolean b) { 313 return b ? 1 : 0; 314 } 315 316 /** 317 * java.net has its own socket options similar to the underlying Unix ones. We paper over the 318 * differences here. 319 */ setSocketOption(FileDescriptor fd, int option, Object value)320 public static void setSocketOption(FileDescriptor fd, int option, Object value) throws SocketException { 321 try { 322 setSocketOptionErrno(fd, option, value); 323 } catch (ErrnoException errnoException) { 324 throw errnoException.rethrowAsSocketException(); 325 } 326 } 327 setSocketOptionErrno(FileDescriptor fd, int option, Object value)328 private static void setSocketOptionErrno(FileDescriptor fd, int option, Object value) throws ErrnoException, SocketException { 329 switch (option) { 330 case SocketOptions.IP_MULTICAST_IF: 331 throw new UnsupportedOperationException("Use IP_MULTICAST_IF2 on Android"); 332 case SocketOptions.IP_MULTICAST_IF2: 333 // Although IPv6 was cleaned up to use int, IPv4 uses an ip_mreqn containing an int. 334 Libcore.os.setsockoptIpMreqn(fd, IPPROTO_IP, IP_MULTICAST_IF, (Integer) value); 335 Libcore.os.setsockoptInt(fd, IPPROTO_IPV6, IPV6_MULTICAST_IF, (Integer) value); 336 return; 337 case SocketOptions.IP_MULTICAST_LOOP: 338 // Although IPv6 was cleaned up to use int, IPv4 multicast loopback uses a byte. 339 Libcore.os.setsockoptByte(fd, IPPROTO_IP, IP_MULTICAST_LOOP, booleanToInt((Boolean) value)); 340 Libcore.os.setsockoptInt(fd, IPPROTO_IPV6, IPV6_MULTICAST_LOOP, booleanToInt((Boolean) value)); 341 return; 342 case IoBridge.JAVA_IP_MULTICAST_TTL: 343 // Although IPv6 was cleaned up to use int, and IPv4 non-multicast TTL uses int, 344 // IPv4 multicast TTL uses a byte. 345 Libcore.os.setsockoptByte(fd, IPPROTO_IP, IP_MULTICAST_TTL, (Integer) value); 346 Libcore.os.setsockoptInt(fd, IPPROTO_IPV6, IPV6_MULTICAST_HOPS, (Integer) value); 347 return; 348 case SocketOptions.IP_TOS: 349 Libcore.os.setsockoptInt(fd, IPPROTO_IP, IP_TOS, (Integer) value); 350 Libcore.os.setsockoptInt(fd, IPPROTO_IPV6, IPV6_TCLASS, (Integer) value); 351 return; 352 case SocketOptions.SO_BROADCAST: 353 Libcore.os.setsockoptInt(fd, SOL_SOCKET, SO_BROADCAST, booleanToInt((Boolean) value)); 354 return; 355 case SocketOptions.SO_KEEPALIVE: 356 Libcore.os.setsockoptInt(fd, SOL_SOCKET, SO_KEEPALIVE, booleanToInt((Boolean) value)); 357 return; 358 case SocketOptions.SO_LINGER: 359 boolean on = false; 360 int seconds = 0; 361 if (value instanceof Integer) { 362 on = true; 363 seconds = Math.min((Integer) value, 65535); 364 } 365 StructLinger linger = new StructLinger(booleanToInt(on), seconds); 366 Libcore.os.setsockoptLinger(fd, SOL_SOCKET, SO_LINGER, linger); 367 return; 368 case SocketOptions.SO_OOBINLINE: 369 Libcore.os.setsockoptInt(fd, SOL_SOCKET, SO_OOBINLINE, booleanToInt((Boolean) value)); 370 return; 371 case SocketOptions.SO_RCVBUF: 372 Libcore.os.setsockoptInt(fd, SOL_SOCKET, SO_RCVBUF, (Integer) value); 373 return; 374 case SocketOptions.SO_REUSEADDR: 375 Libcore.os.setsockoptInt(fd, SOL_SOCKET, SO_REUSEADDR, booleanToInt((Boolean) value)); 376 return; 377 case SocketOptions.SO_SNDBUF: 378 Libcore.os.setsockoptInt(fd, SOL_SOCKET, SO_SNDBUF, (Integer) value); 379 return; 380 case SocketOptions.SO_TIMEOUT: 381 int millis = (Integer) value; 382 StructTimeval tv = StructTimeval.fromMillis(millis); 383 Libcore.os.setsockoptTimeval(fd, SOL_SOCKET, SO_RCVTIMEO, tv); 384 return; 385 case SocketOptions.TCP_NODELAY: 386 Libcore.os.setsockoptInt(fd, IPPROTO_TCP, TCP_NODELAY, booleanToInt((Boolean) value)); 387 return; 388 case IoBridge.JAVA_MCAST_JOIN_GROUP: 389 case IoBridge.JAVA_MCAST_LEAVE_GROUP: 390 { 391 StructGroupReq groupReq = (StructGroupReq) value; 392 int level = (groupReq.gr_group instanceof Inet4Address) ? IPPROTO_IP : IPPROTO_IPV6; 393 int op = (option == JAVA_MCAST_JOIN_GROUP) ? MCAST_JOIN_GROUP : MCAST_LEAVE_GROUP; 394 Libcore.os.setsockoptGroupReq(fd, level, op, groupReq); 395 return; 396 } 397 case IoBridge.JAVA_MCAST_JOIN_SOURCE_GROUP: 398 case IoBridge.JAVA_MCAST_LEAVE_SOURCE_GROUP: 399 case IoBridge.JAVA_MCAST_BLOCK_SOURCE: 400 case IoBridge.JAVA_MCAST_UNBLOCK_SOURCE: 401 { 402 StructGroupSourceReq groupSourceReq = (StructGroupSourceReq) value; 403 int level = (groupSourceReq.gsr_group instanceof Inet4Address) 404 ? IPPROTO_IP : IPPROTO_IPV6; 405 int op = getGroupSourceReqOp(option); 406 Libcore.os.setsockoptGroupSourceReq(fd, level, op, groupSourceReq); 407 return; 408 } 409 default: 410 throw new SocketException("Unknown socket option: " + option); 411 } 412 } 413 getGroupSourceReqOp(int javaValue)414 private static int getGroupSourceReqOp(int javaValue) { 415 switch (javaValue) { 416 case IoBridge.JAVA_MCAST_JOIN_SOURCE_GROUP: 417 return MCAST_JOIN_SOURCE_GROUP; 418 case IoBridge.JAVA_MCAST_LEAVE_SOURCE_GROUP: 419 return MCAST_LEAVE_SOURCE_GROUP; 420 case IoBridge.JAVA_MCAST_BLOCK_SOURCE: 421 return MCAST_BLOCK_SOURCE; 422 case IoBridge.JAVA_MCAST_UNBLOCK_SOURCE: 423 return MCAST_UNBLOCK_SOURCE; 424 default: 425 throw new AssertionError( 426 "Unknown java value for setsocketopt op lookup: " + javaValue); 427 } 428 } 429 430 /** 431 * java.io only throws FileNotFoundException when opening files, regardless of what actually 432 * went wrong. Additionally, java.io is more restrictive than POSIX when it comes to opening 433 * directories: POSIX says read-only is okay, but java.io doesn't even allow that. We also 434 * have an Android-specific hack to alter the default permissions. 435 */ open(String path, int flags)436 public static FileDescriptor open(String path, int flags) throws FileNotFoundException { 437 FileDescriptor fd = null; 438 try { 439 // On Android, we don't want default permissions to allow global access. 440 int mode = ((flags & O_ACCMODE) == O_RDONLY) ? 0 : 0600; 441 fd = Libcore.os.open(path, flags, mode); 442 // Posix open(2) fails with EISDIR only if you ask for write permission. 443 // Java disallows reading directories too. 444 if (S_ISDIR(Libcore.os.fstat(fd).st_mode)) { 445 throw new ErrnoException("open", EISDIR); 446 } 447 return fd; 448 } catch (ErrnoException errnoException) { 449 try { 450 if (fd != null) { 451 IoUtils.close(fd); 452 } 453 } catch (IOException ignored) { 454 } 455 FileNotFoundException ex = new FileNotFoundException(path + ": " + errnoException.getMessage()); 456 ex.initCause(errnoException); 457 throw ex; 458 } 459 } 460 461 /** 462 * java.io thinks that a read at EOF is an error and should return -1, contrary to traditional 463 * Unix practice where you'd read until you got 0 bytes (and any future read would return -1). 464 */ read(FileDescriptor fd, byte[] bytes, int byteOffset, int byteCount)465 public static int read(FileDescriptor fd, byte[] bytes, int byteOffset, int byteCount) throws IOException { 466 Arrays.checkOffsetAndCount(bytes.length, byteOffset, byteCount); 467 if (byteCount == 0) { 468 return 0; 469 } 470 try { 471 int readCount = Libcore.os.read(fd, bytes, byteOffset, byteCount); 472 if (readCount == 0) { 473 return -1; 474 } 475 return readCount; 476 } catch (ErrnoException errnoException) { 477 if (errnoException.errno == EAGAIN) { 478 // We return 0 rather than throw if we try to read from an empty non-blocking pipe. 479 return 0; 480 } 481 throw errnoException.rethrowAsIOException(); 482 } 483 } 484 485 /** 486 * java.io always writes every byte it's asked to, or fails with an error. (That is, unlike 487 * Unix it never just writes as many bytes as happens to be convenient.) 488 */ write(FileDescriptor fd, byte[] bytes, int byteOffset, int byteCount)489 public static void write(FileDescriptor fd, byte[] bytes, int byteOffset, int byteCount) throws IOException { 490 Arrays.checkOffsetAndCount(bytes.length, byteOffset, byteCount); 491 if (byteCount == 0) { 492 return; 493 } 494 try { 495 while (byteCount > 0) { 496 int bytesWritten = Libcore.os.write(fd, bytes, byteOffset, byteCount); 497 byteCount -= bytesWritten; 498 byteOffset += bytesWritten; 499 } 500 } catch (ErrnoException errnoException) { 501 throw errnoException.rethrowAsIOException(); 502 } 503 } 504 sendto(FileDescriptor fd, byte[] bytes, int byteOffset, int byteCount, int flags, InetAddress inetAddress, int port)505 public static int sendto(FileDescriptor fd, byte[] bytes, int byteOffset, int byteCount, int flags, InetAddress inetAddress, int port) throws IOException { 506 boolean isDatagram = (inetAddress != null); 507 if (!isDatagram && byteCount <= 0) { 508 return 0; 509 } 510 int result; 511 try { 512 result = Libcore.os.sendto(fd, bytes, byteOffset, byteCount, flags, inetAddress, port); 513 } catch (ErrnoException errnoException) { 514 result = maybeThrowAfterSendto(isDatagram, errnoException); 515 } 516 return result; 517 } 518 sendto(FileDescriptor fd, ByteBuffer buffer, int flags, InetAddress inetAddress, int port)519 public static int sendto(FileDescriptor fd, ByteBuffer buffer, int flags, InetAddress inetAddress, int port) throws IOException { 520 boolean isDatagram = (inetAddress != null); 521 if (!isDatagram && buffer.remaining() == 0) { 522 return 0; 523 } 524 int result; 525 try { 526 result = Libcore.os.sendto(fd, buffer, flags, inetAddress, port); 527 } catch (ErrnoException errnoException) { 528 result = maybeThrowAfterSendto(isDatagram, errnoException); 529 } 530 return result; 531 } 532 maybeThrowAfterSendto(boolean isDatagram, ErrnoException errnoException)533 private static int maybeThrowAfterSendto(boolean isDatagram, ErrnoException errnoException) throws SocketException { 534 if (isDatagram) { 535 if (errnoException.errno == ECONNRESET || errnoException.errno == ECONNREFUSED) { 536 return 0; 537 } 538 } else { 539 if (errnoException.errno == EAGAIN) { 540 // We were asked to write to a non-blocking socket, but were told 541 // it would block, so report "no bytes written". 542 return 0; 543 } 544 } 545 throw errnoException.rethrowAsSocketException(); 546 } 547 recvfrom(boolean isRead, FileDescriptor fd, byte[] bytes, int byteOffset, int byteCount, int flags, DatagramPacket packet, boolean isConnected)548 public static int recvfrom(boolean isRead, FileDescriptor fd, byte[] bytes, int byteOffset, int byteCount, int flags, DatagramPacket packet, boolean isConnected) throws IOException { 549 int result; 550 try { 551 InetSocketAddress srcAddress = (packet != null && !isConnected) ? new InetSocketAddress() : null; 552 result = Libcore.os.recvfrom(fd, bytes, byteOffset, byteCount, flags, srcAddress); 553 result = postRecvfrom(isRead, packet, isConnected, srcAddress, result); 554 } catch (ErrnoException errnoException) { 555 result = maybeThrowAfterRecvfrom(isRead, isConnected, errnoException); 556 } 557 return result; 558 } 559 recvfrom(boolean isRead, FileDescriptor fd, ByteBuffer buffer, int flags, DatagramPacket packet, boolean isConnected)560 public static int recvfrom(boolean isRead, FileDescriptor fd, ByteBuffer buffer, int flags, DatagramPacket packet, boolean isConnected) throws IOException { 561 int result; 562 try { 563 InetSocketAddress srcAddress = (packet != null && !isConnected) ? new InetSocketAddress() : null; 564 result = Libcore.os.recvfrom(fd, buffer, flags, srcAddress); 565 result = postRecvfrom(isRead, packet, isConnected, srcAddress, result); 566 } catch (ErrnoException errnoException) { 567 result = maybeThrowAfterRecvfrom(isRead, isConnected, errnoException); 568 } 569 return result; 570 } 571 postRecvfrom(boolean isRead, DatagramPacket packet, boolean isConnected, InetSocketAddress srcAddress, int byteCount)572 private static int postRecvfrom(boolean isRead, DatagramPacket packet, boolean isConnected, InetSocketAddress srcAddress, int byteCount) { 573 if (isRead && byteCount == 0) { 574 return -1; 575 } 576 if (packet != null) { 577 packet.setReceivedLength(byteCount); 578 if (!isConnected) { 579 packet.setAddress(srcAddress.getAddress()); 580 packet.setPort(srcAddress.getPort()); 581 } 582 } 583 return byteCount; 584 } 585 maybeThrowAfterRecvfrom(boolean isRead, boolean isConnected, ErrnoException errnoException)586 private static int maybeThrowAfterRecvfrom(boolean isRead, boolean isConnected, ErrnoException errnoException) throws SocketException, SocketTimeoutException { 587 if (isRead) { 588 if (errnoException.errno == EAGAIN) { 589 return 0; 590 } else { 591 throw errnoException.rethrowAsSocketException(); 592 } 593 } else { 594 if (isConnected && errnoException.errno == ECONNREFUSED) { 595 throw new PortUnreachableException("", errnoException); 596 } else if (errnoException.errno == EAGAIN) { 597 throw new SocketTimeoutException(errnoException); 598 } else { 599 throw errnoException.rethrowAsSocketException(); 600 } 601 } 602 } 603 socket(boolean stream)604 public static FileDescriptor socket(boolean stream) throws SocketException { 605 FileDescriptor fd; 606 try { 607 fd = Libcore.os.socket(AF_INET6, stream ? SOCK_STREAM : SOCK_DGRAM, 0); 608 609 // The RFC (http://www.ietf.org/rfc/rfc3493.txt) says that IPV6_MULTICAST_HOPS defaults 610 // to 1. The Linux kernel (at least up to 2.6.38) accidentally defaults to 64 (which 611 // would be correct for the *unicast* hop limit). 612 // See http://www.spinics.net/lists/netdev/msg129022.html, though no patch appears to 613 // have been applied as a result of that discussion. If that bug is ever fixed, we can 614 // remove this code. Until then, we manually set the hop limit on IPv6 datagram sockets. 615 // (IPv4 is already correct.) 616 if (!stream) { 617 Libcore.os.setsockoptInt(fd, IPPROTO_IPV6, IPV6_MULTICAST_HOPS, 1); 618 } 619 620 return fd; 621 } catch (ErrnoException errnoException) { 622 throw errnoException.rethrowAsSocketException(); 623 } 624 } 625 getSocketLocalAddress(FileDescriptor fd)626 public static InetAddress getSocketLocalAddress(FileDescriptor fd) throws SocketException { 627 try { 628 SocketAddress sa = Libcore.os.getsockname(fd); 629 InetSocketAddress isa = (InetSocketAddress) sa; 630 return isa.getAddress(); 631 } catch (ErrnoException errnoException) { 632 throw errnoException.rethrowAsSocketException(); 633 } 634 } 635 getSocketLocalPort(FileDescriptor fd)636 public static int getSocketLocalPort(FileDescriptor fd) throws SocketException { 637 try { 638 SocketAddress sa = Libcore.os.getsockname(fd); 639 InetSocketAddress isa = (InetSocketAddress) sa; 640 return isa.getPort(); 641 } catch (ErrnoException errnoException) { 642 throw errnoException.rethrowAsSocketException(); 643 } 644 } 645 } 646