| OLD | NEW |
| 1 // Copyright (c) 2013, the Dart project authors. Please see the AUTHORS file | 1 // Copyright (c) 2013, the Dart project authors. Please see the AUTHORS file |
| 2 // for details. All rights reserved. Use of this source code is governed by a | 2 // for details. All rights reserved. Use of this source code is governed by a |
| 3 // BSD-style license that can be found in the LICENSE file. | 3 // BSD-style license that can be found in the LICENSE file. |
| 4 | 4 |
| 5 #include "platform/globals.h" | 5 #include "platform/globals.h" |
| 6 #if defined(TARGET_OS_MACOS) | 6 #if defined(TARGET_OS_MACOS) |
| 7 | 7 |
| 8 #include <errno.h> // NOLINT | 8 #include <errno.h> // NOLINT |
| 9 #include <stdio.h> // NOLINT | 9 #include <stdio.h> // NOLINT |
| 10 #include <stdlib.h> // NOLINT | 10 #include <stdlib.h> // NOLINT |
| 11 #include <string.h> // NOLINT | 11 #include <string.h> // NOLINT |
| 12 #include <sys/stat.h> // NOLINT | 12 #include <sys/stat.h> // NOLINT |
| 13 #include <unistd.h> // NOLINT | 13 #include <unistd.h> // NOLINT |
| 14 #include <netinet/tcp.h> // NOLINT | 14 #include <netinet/tcp.h> // NOLINT |
| 15 | 15 |
| 16 #include "bin/fdutils.h" | 16 #include "bin/fdutils.h" |
| 17 #include "bin/file.h" | 17 #include "bin/file.h" |
| 18 #include "bin/log.h" | 18 #include "bin/log.h" |
| 19 #include "bin/socket.h" | 19 #include "bin/socket.h" |
| 20 | 20 |
| 21 | 21 |
| 22 #define SOCKADDR_STORAGE_SET_PORT(addr, port) \ |
| 23 if (addr.ss_family == AF_INET) { \ |
| 24 reinterpret_cast<struct sockaddr_in*>(&addr)->sin_port = htons(port); \ |
| 25 } else { \ |
| 26 reinterpret_cast<struct sockaddr_in6*>(&addr)->sin6_port = htons(port); \ |
| 27 } |
| 28 |
| 29 |
| 30 #define SOCKADDR_STORAGE_GET_PORT(addr) \ |
| 31 addr.ss_family == AF_INET ? \ |
| 32 ntohs(reinterpret_cast<struct sockaddr_in*>(&addr)->sin_port) : \ |
| 33 ntohs(reinterpret_cast<struct sockaddr_in6*>(&addr)->sin6_port) |
| 34 |
| 35 |
| 36 SocketAddress::SocketAddress(struct addrinfo* addrinfo) { |
| 37 ASSERT(INET6_ADDRSTRLEN >= INET_ADDRSTRLEN); |
| 38 sockaddr_in *sockaddr = reinterpret_cast<sockaddr_in *>(addrinfo->ai_addr); |
| 39 const char* result = inet_ntop(addrinfo->ai_family, |
| 40 &sockaddr->sin_addr, |
| 41 as_string_, |
| 42 INET6_ADDRSTRLEN); |
| 43 if (result == NULL) as_string_[0] = 0; |
| 44 memmove(reinterpret_cast<void *>(&addr_), |
| 45 addrinfo->ai_addr, |
| 46 addrinfo->ai_addrlen); |
| 47 } |
| 48 |
| 49 |
| 22 bool Socket::Initialize() { | 50 bool Socket::Initialize() { |
| 23 // Nothing to do on Mac OS. | 51 // Nothing to do on Mac OS. |
| 24 return true; | 52 return true; |
| 25 } | 53 } |
| 26 | 54 |
| 27 | 55 |
| 28 intptr_t Socket::CreateConnect(const char* host, const intptr_t port) { | 56 intptr_t Socket::CreateConnect(sockaddr_storage addr, const intptr_t port) { |
| 29 intptr_t fd; | 57 intptr_t fd; |
| 30 struct hostent* server; | |
| 31 struct sockaddr_in server_address; | |
| 32 | 58 |
| 33 fd = TEMP_FAILURE_RETRY(socket(AF_INET, SOCK_STREAM, 0)); | 59 fd = TEMP_FAILURE_RETRY(socket(addr.ss_family, SOCK_STREAM, 0)); |
| 34 if (fd < 0) { | 60 if (fd < 0) { |
| 35 Log::PrintErr("Error CreateConnect: %s\n", strerror(errno)); | 61 Log::PrintErr("Error CreateConnect: %s\n", strerror(errno)); |
| 36 return -1; | 62 return -1; |
| 37 } | 63 } |
| 38 | 64 |
| 39 FDUtils::SetCloseOnExec(fd); | 65 FDUtils::SetCloseOnExec(fd); |
| 40 Socket::SetNonBlocking(fd); | 66 Socket::SetNonBlocking(fd); |
| 41 | 67 |
| 42 server = gethostbyname(host); | 68 SOCKADDR_STORAGE_SET_PORT(addr, port); |
| 43 if (server == NULL) { | |
| 44 VOID_TEMP_FAILURE_RETRY(close(fd)); | |
| 45 Log::PrintErr("Error CreateConnect: %s\n", strerror(errno)); | |
| 46 return -1; | |
| 47 } | |
| 48 | |
| 49 server_address.sin_family = AF_INET; | |
| 50 server_address.sin_port = htons(port); | |
| 51 bcopy(server->h_addr, &server_address.sin_addr.s_addr, server->h_length); | |
| 52 memset(&server_address.sin_zero, 0, sizeof(server_address.sin_zero)); | |
| 53 intptr_t result = TEMP_FAILURE_RETRY( | 69 intptr_t result = TEMP_FAILURE_RETRY( |
| 54 connect(fd, | 70 connect(fd, |
| 55 reinterpret_cast<struct sockaddr *>(&server_address), | 71 reinterpret_cast<struct sockaddr*>(&addr), |
| 56 sizeof(server_address))); | 72 SocketAddress::GetAddrLength(addr))); |
| 57 if (result == 0 || errno == EINPROGRESS) { | 73 if (result == 0 || errno == EINPROGRESS) { |
| 58 return fd; | 74 return fd; |
| 59 } | 75 } |
| 76 TEMP_FAILURE_RETRY(close(fd)); |
| 60 return -1; | 77 return -1; |
| 61 } | 78 } |
| 62 | 79 |
| 63 | 80 |
| 64 intptr_t Socket::Available(intptr_t fd) { | 81 intptr_t Socket::Available(intptr_t fd) { |
| 65 return FDUtils::AvailableBytes(fd); | 82 return FDUtils::AvailableBytes(fd); |
| 66 } | 83 } |
| 67 | 84 |
| 68 | 85 |
| 69 int Socket::Read(intptr_t fd, void* buffer, intptr_t num_bytes) { | 86 int Socket::Read(intptr_t fd, void* buffer, intptr_t num_bytes) { |
| (...skipping 17 matching lines...) Expand all Loading... |
| 87 // If the would block we need to retry and therefore return 0 as | 104 // If the would block we need to retry and therefore return 0 as |
| 88 // the number of bytes written. | 105 // the number of bytes written. |
| 89 written_bytes = 0; | 106 written_bytes = 0; |
| 90 } | 107 } |
| 91 return written_bytes; | 108 return written_bytes; |
| 92 } | 109 } |
| 93 | 110 |
| 94 | 111 |
| 95 intptr_t Socket::GetPort(intptr_t fd) { | 112 intptr_t Socket::GetPort(intptr_t fd) { |
| 96 ASSERT(fd >= 0); | 113 ASSERT(fd >= 0); |
| 97 struct sockaddr_in socket_address; | 114 struct sockaddr_storage socket_address; |
| 98 socklen_t size = sizeof(socket_address); | 115 socklen_t size = sizeof(socket_address); |
| 99 if (TEMP_FAILURE_RETRY( | 116 if (TEMP_FAILURE_RETRY( |
| 100 getsockname(fd, | 117 getsockname(fd, |
| 101 reinterpret_cast<struct sockaddr *>(&socket_address), | 118 reinterpret_cast<struct sockaddr *>(&socket_address), |
| 102 &size))) { | 119 &size))) { |
| 103 Log::PrintErr("Error getsockname: %s\n", strerror(errno)); | 120 Log::PrintErr("Error getsockname: %s\n", strerror(errno)); |
| 104 return 0; | 121 return 0; |
| 105 } | 122 } |
| 106 return ntohs(socket_address.sin_port); | 123 return SOCKADDR_STORAGE_GET_PORT(socket_address); |
| 107 } | 124 } |
| 108 | 125 |
| 109 | 126 |
| 110 bool Socket::GetRemotePeer(intptr_t fd, char *host, intptr_t *port) { | 127 bool Socket::GetRemotePeer(intptr_t fd, char *host, intptr_t *port) { |
| 111 ASSERT(fd >= 0); | 128 ASSERT(fd >= 0); |
| 112 struct sockaddr_in socket_address; | 129 struct sockaddr_storage socket_address; |
| 113 socklen_t size = sizeof(socket_address); | 130 socklen_t size = sizeof(socket_address); |
| 114 if (TEMP_FAILURE_RETRY( | 131 if (TEMP_FAILURE_RETRY( |
| 115 getpeername(fd, | 132 getpeername(fd, |
| 116 reinterpret_cast<struct sockaddr *>(&socket_address), | 133 reinterpret_cast<struct sockaddr *>(&socket_address), |
| 117 &size))) { | 134 &size))) { |
| 118 Log::PrintErr("Error getpeername: %s\n", strerror(errno)); | 135 Log::PrintErr("Error getpeername: %s\n", strerror(errno)); |
| 119 return false; | 136 return false; |
| 120 } | 137 } |
| 121 if (inet_ntop(socket_address.sin_family, | 138 const void* src; |
| 122 reinterpret_cast<const void *>(&socket_address.sin_addr), | 139 if (socket_address.ss_family == AF_INET6) { |
| 140 const sockaddr_in6* in6 = |
| 141 reinterpret_cast<const sockaddr_in6*>(&socket_address); |
| 142 src = reinterpret_cast<const void*>(&in6->sin6_addr); |
| 143 } else { |
| 144 const sockaddr_in* in = |
| 145 reinterpret_cast<const sockaddr_in*>(&socket_address); |
| 146 src = reinterpret_cast<const void*>(&in->sin_addr); |
| 147 } |
| 148 if (inet_ntop(socket_address.ss_family, |
| 149 src, |
| 123 host, | 150 host, |
| 124 INET_ADDRSTRLEN) == NULL) { | 151 INET_ADDRSTRLEN) == NULL) { |
| 125 Log::PrintErr("Error inet_ntop: %s\n", strerror(errno)); | 152 Log::PrintErr("Error inet_ntop: %s\n", strerror(errno)); |
| 126 return false; | 153 return false; |
| 127 } | 154 } |
| 128 *port = ntohs(socket_address.sin_port); | 155 *port = SOCKADDR_STORAGE_GET_PORT(socket_address); |
| 129 return true; | 156 return true; |
| 130 } | 157 } |
| 131 | 158 |
| 132 | 159 |
| 133 void Socket::GetError(intptr_t fd, OSError* os_error) { | 160 void Socket::GetError(intptr_t fd, OSError* os_error) { |
| 134 int len = sizeof(errno); | 161 int len = sizeof(errno); |
| 135 getsockopt(fd, | 162 getsockopt(fd, |
| 136 SOL_SOCKET, | 163 SOL_SOCKET, |
| 137 SO_ERROR, | 164 SO_ERROR, |
| 138 &errno, | 165 &errno, |
| (...skipping 11 matching lines...) Expand all Loading... |
| 150 if (S_ISREG(buf.st_mode)) return File::kFile; | 177 if (S_ISREG(buf.st_mode)) return File::kFile; |
| 151 return File::kOther; | 178 return File::kOther; |
| 152 } | 179 } |
| 153 | 180 |
| 154 | 181 |
| 155 intptr_t Socket::GetStdioHandle(int num) { | 182 intptr_t Socket::GetStdioHandle(int num) { |
| 156 return static_cast<intptr_t>(num); | 183 return static_cast<intptr_t>(num); |
| 157 } | 184 } |
| 158 | 185 |
| 159 | 186 |
| 160 const char* Socket::LookupIPv4Address(char* host, OSError** os_error) { | 187 SocketAddresses* Socket::LookupAddress(const char* host, |
| 161 // Perform a name lookup for an IPv4 address. | 188 int type, |
| 189 OSError** os_error) { |
| 190 // Perform a name lookup for a host name. |
| 162 struct addrinfo hints; | 191 struct addrinfo hints; |
| 163 memset(&hints, 0, sizeof(hints)); | 192 memset(&hints, 0, sizeof(hints)); |
| 164 hints.ai_family = AF_INET; | 193 hints.ai_family = SocketAddress::FromType(type); |
| 165 hints.ai_socktype = SOCK_STREAM; | 194 hints.ai_socktype = SOCK_STREAM; |
| 195 hints.ai_flags = 0; |
| 166 hints.ai_protocol = IPPROTO_TCP; | 196 hints.ai_protocol = IPPROTO_TCP; |
| 167 struct addrinfo* info = NULL; | 197 struct addrinfo* info = NULL; |
| 168 int status = getaddrinfo(host, 0, &hints, &info); | 198 int status = getaddrinfo(host, 0, &hints, &info); |
| 169 if (status != 0) { | 199 if (status != 0) { |
| 170 ASSERT(*os_error == NULL); | 200 ASSERT(*os_error == NULL); |
| 171 *os_error = new OSError(status, | 201 *os_error = new OSError(status, |
| 172 gai_strerror(status), | 202 gai_strerror(status), |
| 173 OSError::kGetAddressInfo); | 203 OSError::kGetAddressInfo); |
| 174 return NULL; | 204 return NULL; |
| 175 } | 205 } |
| 176 // Convert the address into IPv4 dotted decimal notation. | 206 intptr_t count = 0; |
| 177 char* buffer = reinterpret_cast<char*>(malloc(INET_ADDRSTRLEN)); | 207 for (struct addrinfo* c = info; c != NULL; c = c->ai_next) { |
| 178 sockaddr_in *sockaddr = reinterpret_cast<sockaddr_in *>(info->ai_addr); | 208 if (c->ai_family == AF_INET || c->ai_family == AF_INET6) count++; |
| 179 const char* result = inet_ntop(AF_INET, | |
| 180 reinterpret_cast<void *>(&sockaddr->sin_addr), | |
| 181 buffer, | |
| 182 INET_ADDRSTRLEN); | |
| 183 if (result == NULL) { | |
| 184 free(buffer); | |
| 185 return NULL; | |
| 186 } | 209 } |
| 187 ASSERT(result == buffer); | 210 SocketAddresses* addresses = new SocketAddresses(count); |
| 188 return buffer; | 211 intptr_t i = 0; |
| 212 for (struct addrinfo* c = info; c != NULL; c = c->ai_next) { |
| 213 if (c->ai_family == AF_INET || c->ai_family == AF_INET6) { |
| 214 addresses->SetAt(i, new SocketAddress(c)); |
| 215 i++; |
| 216 } |
| 217 } |
| 218 freeaddrinfo(info); |
| 219 return addresses; |
| 189 } | 220 } |
| 190 | 221 |
| 191 | 222 |
| 192 intptr_t ServerSocket::CreateBindListen(const char* host, | 223 intptr_t ServerSocket::CreateBindListen(sockaddr_storage addr, |
| 193 intptr_t port, | 224 intptr_t port, |
| 194 intptr_t backlog) { | 225 intptr_t backlog) { |
| 195 intptr_t fd; | 226 intptr_t fd; |
| 196 struct sockaddr_in server_address; | |
| 197 | 227 |
| 198 in_addr_t s_addr = inet_addr(host); | 228 fd = TEMP_FAILURE_RETRY(socket(addr.ss_family, SOCK_STREAM, 0)); |
| 199 if (s_addr == INADDR_NONE) { | |
| 200 return -5; | |
| 201 } | |
| 202 | |
| 203 fd = TEMP_FAILURE_RETRY(socket(AF_INET, SOCK_STREAM, 0)); | |
| 204 if (fd < 0) return -1; | 229 if (fd < 0) return -1; |
| 205 | 230 |
| 206 FDUtils::SetCloseOnExec(fd); | 231 FDUtils::SetCloseOnExec(fd); |
| 207 | 232 |
| 208 int optval = 1; | 233 int optval = 1; |
| 209 VOID_TEMP_FAILURE_RETRY( | 234 VOID_TEMP_FAILURE_RETRY( |
| 210 setsockopt(fd, SOL_SOCKET, SO_REUSEADDR, &optval, sizeof(optval))); | 235 setsockopt(fd, SOL_SOCKET, SO_REUSEADDR, &optval, sizeof(optval))); |
| 211 | 236 |
| 212 server_address.sin_family = AF_INET; | 237 if (addr.ss_family == AF_INET6) { |
| 213 server_address.sin_port = htons(port); | 238 optval = 0; |
| 214 server_address.sin_addr.s_addr = s_addr; | 239 VOID_TEMP_FAILURE_RETRY( |
| 215 memset(&server_address.sin_zero, 0, sizeof(server_address.sin_zero)); | 240 setsockopt(fd, IPPROTO_IPV6, IPV6_V6ONLY, &optval, sizeof(optval))); |
| 241 } |
| 216 | 242 |
| 243 SOCKADDR_STORAGE_SET_PORT(addr, port); |
| 217 if (TEMP_FAILURE_RETRY( | 244 if (TEMP_FAILURE_RETRY( |
| 218 bind(fd, | 245 bind(fd, |
| 219 reinterpret_cast<struct sockaddr *>(&server_address), | 246 reinterpret_cast<struct sockaddr*>(&addr), |
| 220 sizeof(server_address))) < 0) { | 247 SocketAddress::GetAddrLength(addr))) < 0) { |
| 221 VOID_TEMP_FAILURE_RETRY(close(fd)); | 248 VOID_TEMP_FAILURE_RETRY(close(fd)); |
| 222 return -1; | 249 return -1; |
| 223 } | 250 } |
| 224 | 251 |
| 225 if (TEMP_FAILURE_RETRY(listen(fd, backlog > 0 ? backlog : SOMAXCONN)) != 0) { | 252 if (TEMP_FAILURE_RETRY(listen(fd, backlog > 0 ? backlog : SOMAXCONN)) != 0) { |
| 226 VOID_TEMP_FAILURE_RETRY(close(fd)); | 253 VOID_TEMP_FAILURE_RETRY(close(fd)); |
| 227 return -1; | 254 return -1; |
| 228 } | 255 } |
| 229 | 256 |
| 230 Socket::SetNonBlocking(fd); | 257 Socket::SetNonBlocking(fd); |
| (...skipping 46 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 277 bool Socket::SetNoDelay(intptr_t fd, bool enabled) { | 304 bool Socket::SetNoDelay(intptr_t fd, bool enabled) { |
| 278 int on = enabled ? 1 : 0; | 305 int on = enabled ? 1 : 0; |
| 279 return TEMP_FAILURE_RETRY(setsockopt(fd, | 306 return TEMP_FAILURE_RETRY(setsockopt(fd, |
| 280 IPPROTO_TCP, | 307 IPPROTO_TCP, |
| 281 TCP_NODELAY, | 308 TCP_NODELAY, |
| 282 reinterpret_cast<char *>(&on), | 309 reinterpret_cast<char *>(&on), |
| 283 sizeof(on))) == 0; | 310 sizeof(on))) == 0; |
| 284 } | 311 } |
| 285 | 312 |
| 286 #endif // defined(TARGET_OS_MACOS) | 313 #endif // defined(TARGET_OS_MACOS) |
| OLD | NEW |