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_LINUX) | 6 #if defined(TARGET_OS_LINUX) |
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 SocketAddress::SocketAddress(struct addrinfo* addrinfo) { | |
23 char* buffer = new char[INET6_ADDRSTRLEN]; | |
24 sockaddr_in *sockaddr = reinterpret_cast<sockaddr_in *>(addrinfo->ai_addr); | |
25 const char* as_string = inet_ntop(addrinfo->ai_family, | |
26 &sockaddr->sin_addr, | |
27 buffer, | |
28 INET6_ADDRSTRLEN); | |
29 if (as_string == NULL) delete[] buffer; | |
30 memmove(reinterpret_cast<void *>(&addr_), | |
31 addrinfo->ai_addr, | |
32 addrinfo->ai_addrlen); | |
33 as_string_ = as_string; | |
34 } | |
35 | |
36 | |
22 bool Socket::Initialize() { | 37 bool Socket::Initialize() { |
23 // Nothing to do on Linux. | 38 // Nothing to do on Linux. |
24 return true; | 39 return true; |
25 } | 40 } |
26 | 41 |
27 | 42 |
28 intptr_t Socket::CreateConnect(const char* host, const intptr_t port) { | 43 intptr_t Socket::CreateConnect(sockaddr_storage addr, const intptr_t port) { |
29 intptr_t fd; | 44 intptr_t fd; |
30 struct hostent server; | |
31 struct sockaddr_in server_address; | |
32 | 45 |
33 fd = TEMP_FAILURE_RETRY(socket(AF_INET, SOCK_STREAM, 0)); | 46 fd = TEMP_FAILURE_RETRY(socket(addr.ss_family, SOCK_STREAM, 0)); |
34 if (fd < 0) { | 47 if (fd < 0) { |
35 Log::PrintErr("Error CreateConnect: %s\n", strerror(errno)); | 48 Log::PrintErr("Error CreateConnect: %s\n", strerror(errno)); |
36 return -1; | 49 return -1; |
37 } | 50 } |
38 | 51 |
39 FDUtils::SetCloseOnExec(fd); | 52 FDUtils::SetCloseOnExec(fd); |
40 Socket::SetNonBlocking(fd); | 53 Socket::SetNonBlocking(fd); |
41 | 54 |
42 static const size_t kTempBufSize = 1024; | 55 reinterpret_cast<struct sockaddr_in*>(&addr)->sin_port = htons(port); |
Søren Gjesse
2013/04/18 12:59:24
We better switch on family here
if (addr.ss_famil
Anders Johnsen
2013/04/19 09:45:36
Done.
| |
43 char temp_buf[kTempBufSize]; | |
44 struct hostent *unused; | |
45 int err; | |
46 if (gethostbyname_r( | |
47 host, &server, temp_buf, kTempBufSize, &unused, &err) != 0) { | |
48 TEMP_FAILURE_RETRY(close(fd)); | |
49 Log::PrintErr("Error CreateConnect: %s\n", strerror(errno)); | |
50 return -1; | |
51 } | |
52 | |
53 server_address.sin_family = AF_INET; | |
54 server_address.sin_port = htons(port); | |
55 bcopy(server.h_addr, &server_address.sin_addr.s_addr, server.h_length); | |
56 memset(&server_address.sin_zero, 0, sizeof(server_address.sin_zero)); | |
57 intptr_t result = TEMP_FAILURE_RETRY( | 56 intptr_t result = TEMP_FAILURE_RETRY( |
58 connect(fd, | 57 connect(fd, |
59 reinterpret_cast<struct sockaddr *>(&server_address), | 58 reinterpret_cast<struct sockaddr*>(&addr), |
60 sizeof(server_address))); | 59 SOCKADDR_STORAGE_LENGTH(addr))); |
61 if (result == 0 || errno == EINPROGRESS) { | 60 if (result == 0 || errno == EINPROGRESS) { |
62 return fd; | 61 return fd; |
63 } | 62 } |
63 TEMP_FAILURE_RETRY(close(fd)); | |
64 return -1; | 64 return -1; |
65 } | 65 } |
66 | 66 |
67 | 67 |
68 intptr_t Socket::Available(intptr_t fd) { | 68 intptr_t Socket::Available(intptr_t fd) { |
69 return FDUtils::AvailableBytes(fd); | 69 return FDUtils::AvailableBytes(fd); |
70 } | 70 } |
71 | 71 |
72 | 72 |
73 int Socket::Read(intptr_t fd, void* buffer, intptr_t num_bytes) { | 73 int Socket::Read(intptr_t fd, void* buffer, intptr_t num_bytes) { |
(...skipping 79 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
153 if (S_ISFIFO(buf.st_mode)) return File::kPipe; | 153 if (S_ISFIFO(buf.st_mode)) return File::kPipe; |
154 if (S_ISREG(buf.st_mode)) return File::kFile; | 154 if (S_ISREG(buf.st_mode)) return File::kFile; |
155 return File::kOther; | 155 return File::kOther; |
156 } | 156 } |
157 | 157 |
158 | 158 |
159 intptr_t Socket::GetStdioHandle(int num) { | 159 intptr_t Socket::GetStdioHandle(int num) { |
160 return static_cast<intptr_t>(num); | 160 return static_cast<intptr_t>(num); |
161 } | 161 } |
162 | 162 |
163 | 163 SocketAddresses* Socket::LookupAddress(char* host, OSError** os_error) { |
164 const char* Socket::LookupIPv4Address(char* host, OSError** os_error) { | |
165 // Perform a name lookup for an IPv4 address. | 164 // Perform a name lookup for an IPv4 address. |
166 struct addrinfo hints; | 165 struct addrinfo hints; |
167 memset(&hints, 0, sizeof(hints)); | 166 memset(&hints, 0, sizeof(hints)); |
168 hints.ai_family = AF_INET; | 167 hints.ai_family = AF_UNSPEC; |
169 hints.ai_socktype = SOCK_STREAM; | 168 hints.ai_socktype = SOCK_STREAM; |
170 hints.ai_protocol = IPPROTO_TCP; | 169 hints.ai_flags = 0; |
170 hints.ai_protocol = 0; | |
Søren Gjesse
2013/04/18 12:59:24
Don't you still want only IPPROTO_TCP?
Anders Johnsen
2013/04/19 09:45:36
Done.
| |
171 struct addrinfo* info = NULL; | 171 struct addrinfo* info = NULL; |
172 int status = getaddrinfo(host, 0, &hints, &info); | 172 int status = getaddrinfo(host, 0, &hints, &info); |
173 if (status != 0) { | 173 if (status != 0) { |
174 ASSERT(*os_error == NULL); | 174 ASSERT(*os_error == NULL); |
175 *os_error = new OSError(status, | 175 *os_error = new OSError(status, |
176 gai_strerror(status), | 176 gai_strerror(status), |
177 OSError::kGetAddressInfo); | 177 OSError::kGetAddressInfo); |
178 return NULL; | 178 return NULL; |
179 } | 179 } |
180 // Convert the address into IPv4 dotted decimal notation. | 180 intptr_t count = 0; |
181 char* buffer = reinterpret_cast<char*>(malloc(INET_ADDRSTRLEN)); | 181 for (struct addrinfo* c = info; c != NULL; c = c->ai_next) { |
182 sockaddr_in *sockaddr = reinterpret_cast<sockaddr_in *>(info->ai_addr); | 182 if (c->ai_family == AF_INET || c->ai_family == AF_INET6) count++; |
183 const char* result = inet_ntop(AF_INET, | |
184 reinterpret_cast<void *>(&sockaddr->sin_addr), | |
185 buffer, | |
186 INET_ADDRSTRLEN); | |
187 if (result == NULL) { | |
188 free(buffer); | |
189 return NULL; | |
190 } | 183 } |
191 ASSERT(result == buffer); | 184 SocketAddresses* addresses = new SocketAddresses(count); |
192 return buffer; | 185 intptr_t i = 0; |
186 for (struct addrinfo* c = info; c != NULL; c = c->ai_next) { | |
187 if (c->ai_family == AF_INET || c->ai_family == AF_INET6) { | |
188 addresses->SetAt(i, new SocketAddress(c)); | |
189 i++; | |
190 } | |
191 } | |
192 freeaddrinfo(info); | |
193 return addresses; | |
193 } | 194 } |
194 | 195 |
195 | 196 |
196 intptr_t ServerSocket::CreateBindListen(const char* host, | 197 intptr_t ServerSocket::CreateBindListen(sockaddr_storage addr, |
197 intptr_t port, | 198 intptr_t port, |
198 intptr_t backlog) { | 199 intptr_t backlog) { |
199 intptr_t fd; | 200 intptr_t fd; |
200 struct sockaddr_in server_address; | |
201 | 201 |
202 in_addr_t s_addr = inet_addr(host); | 202 fd = TEMP_FAILURE_RETRY(socket(addr.ss_family, SOCK_STREAM, 0)); |
203 if (s_addr == INADDR_NONE) { | |
204 return -5; | |
205 } | |
206 | |
207 fd = TEMP_FAILURE_RETRY(socket(AF_INET, SOCK_STREAM, 0)); | |
208 if (fd < 0) return -1; | 203 if (fd < 0) return -1; |
209 | 204 |
210 FDUtils::SetCloseOnExec(fd); | 205 FDUtils::SetCloseOnExec(fd); |
211 | 206 |
212 int optval = 1; | 207 int optval = 1; |
213 TEMP_FAILURE_RETRY( | 208 TEMP_FAILURE_RETRY( |
214 setsockopt(fd, SOL_SOCKET, SO_REUSEADDR, &optval, sizeof(optval))); | 209 setsockopt(fd, SOL_SOCKET, SO_REUSEADDR, &optval, sizeof(optval))); |
215 | 210 |
216 server_address.sin_family = AF_INET; | 211 if (addr.ss_family == AF_INET6) { |
217 server_address.sin_port = htons(port); | 212 optval = 0; |
218 server_address.sin_addr.s_addr = s_addr; | 213 TEMP_FAILURE_RETRY( |
219 memset(&server_address.sin_zero, 0, sizeof(server_address.sin_zero)); | 214 setsockopt(fd, IPPROTO_IPV6, IPV6_V6ONLY, &optval, sizeof(optval))); |
215 } | |
220 | 216 |
217 reinterpret_cast<struct sockaddr_in*>(&addr)->sin_port = htons(port); | |
221 if (TEMP_FAILURE_RETRY( | 218 if (TEMP_FAILURE_RETRY( |
222 bind(fd, | 219 bind(fd, |
223 reinterpret_cast<struct sockaddr *>(&server_address), | 220 reinterpret_cast<struct sockaddr*>(&addr), |
224 sizeof(server_address))) < 0) { | 221 SOCKADDR_STORAGE_LENGTH(addr))) < 0) { |
225 TEMP_FAILURE_RETRY(close(fd)); | 222 TEMP_FAILURE_RETRY(close(fd)); |
226 return -1; | 223 return -1; |
227 } | 224 } |
228 | 225 |
229 if (TEMP_FAILURE_RETRY(listen(fd, backlog > 0 ? backlog : SOMAXCONN)) != 0) { | 226 if (TEMP_FAILURE_RETRY(listen(fd, backlog > 0 ? backlog : SOMAXCONN)) != 0) { |
230 TEMP_FAILURE_RETRY(close(fd)); | 227 TEMP_FAILURE_RETRY(close(fd)); |
231 return -1; | 228 return -1; |
232 } | 229 } |
233 | 230 |
234 Socket::SetNonBlocking(fd); | 231 Socket::SetNonBlocking(fd); |
(...skipping 56 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
291 bool Socket::SetNoDelay(intptr_t fd, bool enabled) { | 288 bool Socket::SetNoDelay(intptr_t fd, bool enabled) { |
292 int on = enabled ? 1 : 0; | 289 int on = enabled ? 1 : 0; |
293 return TEMP_FAILURE_RETRY(setsockopt(fd, | 290 return TEMP_FAILURE_RETRY(setsockopt(fd, |
294 SOL_TCP, | 291 SOL_TCP, |
295 TCP_NODELAY, | 292 TCP_NODELAY, |
296 reinterpret_cast<char *>(&on), | 293 reinterpret_cast<char *>(&on), |
297 sizeof(on))) == 0; | 294 sizeof(on))) == 0; |
298 } | 295 } |
299 | 296 |
300 #endif // defined(TARGET_OS_LINUX) | 297 #endif // defined(TARGET_OS_LINUX) |
OLD | NEW |