|  | /* | 
|  | ** Copyright 2006, The Android Open Source Project | 
|  | ** | 
|  | ** Licensed under the Apache License, Version 2.0 (the "License"); | 
|  | ** you may not use this file except in compliance with the License. | 
|  | ** You may obtain a copy of the License at | 
|  | ** | 
|  | **     http://www.apache.org/licenses/LICENSE-2.0 | 
|  | ** | 
|  | ** Unless required by applicable law or agreed to in writing, software | 
|  | ** distributed under the License is distributed on an "AS IS" BASIS, | 
|  | ** WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | 
|  | ** See the License for the specific language governing permissions and | 
|  | ** limitations under the License. | 
|  | */ | 
|  |  | 
|  | #include <errno.h> | 
|  | #include <fcntl.h> | 
|  | #include <stddef.h> | 
|  | #include <stdlib.h> | 
|  | #include <string.h> | 
|  | #include <unistd.h> | 
|  |  | 
|  | #include <sys/socket.h> | 
|  | #include <sys/select.h> | 
|  | #include <sys/types.h> | 
|  | #include <netinet/in.h> | 
|  | #include <netdb.h> | 
|  |  | 
|  | #include <cutils/sockets.h> | 
|  |  | 
|  | /* Connect to port on the IP interface. type is | 
|  | * SOCK_STREAM or SOCK_DGRAM. | 
|  | * return is a file descriptor or -1 on error | 
|  | */ | 
|  | int socket_network_client(const char *host, int port, int type) | 
|  | { | 
|  | return socket_network_client_timeout(host, port, type, 0); | 
|  | } | 
|  |  | 
|  | /* Connect to port on the IP interface. type is SOCK_STREAM or SOCK_DGRAM. | 
|  | * timeout in seconds return is a file descriptor or -1 on error | 
|  | */ | 
|  | int socket_network_client_timeout(const char *host, int port, int type, int timeout) | 
|  | { | 
|  | struct hostent *hp; | 
|  | struct sockaddr_in addr; | 
|  | socklen_t alen; | 
|  | int s; | 
|  | int flags = 0, error = 0, ret = 0; | 
|  | fd_set rset, wset; | 
|  | socklen_t len = sizeof(error); | 
|  | struct timeval ts; | 
|  |  | 
|  | ts.tv_sec = timeout; | 
|  | ts.tv_usec = 0; | 
|  |  | 
|  | hp = gethostbyname(host); | 
|  | if (hp == 0) return -1; | 
|  |  | 
|  | memset(&addr, 0, sizeof(addr)); | 
|  | addr.sin_family = hp->h_addrtype; | 
|  | addr.sin_port = htons(port); | 
|  | memcpy(&addr.sin_addr, hp->h_addr, hp->h_length); | 
|  |  | 
|  | s = socket(hp->h_addrtype, type, 0); | 
|  | if (s < 0) return -1; | 
|  |  | 
|  | if ((flags = fcntl(s, F_GETFL, 0)) < 0) { | 
|  | close(s); | 
|  | return -1; | 
|  | } | 
|  |  | 
|  | if (fcntl(s, F_SETFL, flags | O_NONBLOCK) < 0) { | 
|  | close(s); | 
|  | return -1; | 
|  | } | 
|  |  | 
|  | if ((ret = connect(s, (struct sockaddr *) &addr, sizeof(addr))) < 0) { | 
|  | if (errno != EINPROGRESS) { | 
|  | close(s); | 
|  | return -1; | 
|  | } | 
|  | } | 
|  |  | 
|  | if (ret == 0) | 
|  | goto done; | 
|  |  | 
|  | FD_ZERO(&rset); | 
|  | FD_SET(s, &rset); | 
|  | wset = rset; | 
|  |  | 
|  | if ((ret = select(s + 1, &rset, &wset, NULL, (timeout) ? &ts : NULL)) < 0) { | 
|  | close(s); | 
|  | return -1; | 
|  | } | 
|  | if (ret == 0) {   // we had a timeout | 
|  | errno = ETIMEDOUT; | 
|  | close(s); | 
|  | return -1; | 
|  | } | 
|  |  | 
|  | if (FD_ISSET(s, &rset) || FD_ISSET(s, &wset)) { | 
|  | if (getsockopt(s, SOL_SOCKET, SO_ERROR, &error, &len) < 0) { | 
|  | close(s); | 
|  | return -1; | 
|  | } | 
|  | } else { | 
|  | close(s); | 
|  | return -1; | 
|  | } | 
|  |  | 
|  | if (error) {  // check if we had a socket error | 
|  | errno = error; | 
|  | close(s); | 
|  | return -1; | 
|  | } | 
|  |  | 
|  | done: | 
|  | if (fcntl(s, F_SETFL, flags) < 0) { | 
|  | close(s); | 
|  | return -1; | 
|  | } | 
|  |  | 
|  | return s; | 
|  | } |