| /* |
| * Copyright 2009, 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 <stdlib.h> |
| #include <errno.h> |
| #include <unistd.h> |
| #include <fcntl.h> |
| #include <sys/socket.h> |
| #include <sys/poll.h> |
| |
| #include "cutils/abort_socket.h" |
| |
| struct asocket *asocket_init(int fd) { |
| int abort_fd[2]; |
| int flags; |
| struct asocket *s; |
| |
| /* set primary socket to non-blocking */ |
| flags = fcntl(fd, F_GETFL); |
| if (flags == -1) |
| return NULL; |
| if (fcntl(fd, F_SETFL, flags | O_NONBLOCK)) |
| return NULL; |
| |
| /* create pipe with non-blocking write, so that asocket_close() cannot |
| block */ |
| if (pipe(abort_fd)) |
| return NULL; |
| flags = fcntl(abort_fd[1], F_GETFL); |
| if (flags == -1) |
| return NULL; |
| if (fcntl(abort_fd[1], F_SETFL, flags | O_NONBLOCK)) |
| return NULL; |
| |
| s = malloc(sizeof(struct asocket)); |
| if (!s) |
| return NULL; |
| |
| s->fd = fd; |
| s->abort_fd[0] = abort_fd[0]; |
| s->abort_fd[1] = abort_fd[1]; |
| |
| return s; |
| } |
| |
| int asocket_connect(struct asocket *s, const struct sockaddr *addr, |
| socklen_t addrlen, int timeout) { |
| |
| int ret; |
| |
| do { |
| ret = connect(s->fd, addr, addrlen); |
| } while (ret && errno == EINTR); |
| |
| if (ret && errno == EINPROGRESS) { |
| /* ready to poll() */ |
| socklen_t retlen; |
| struct pollfd pfd[2]; |
| |
| pfd[0].fd = s->fd; |
| pfd[0].events = POLLOUT; |
| pfd[0].revents = 0; |
| pfd[1].fd = s->abort_fd[0]; |
| pfd[1].events = POLLIN; |
| pfd[1].revents = 0; |
| |
| do { |
| ret = poll(pfd, 2, timeout); |
| } while (ret < 0 && errno == EINTR); |
| |
| if (ret < 0) |
| return -1; |
| else if (ret == 0) { |
| /* timeout */ |
| errno = ETIMEDOUT; |
| return -1; |
| } |
| |
| if (pfd[1].revents) { |
| /* abort due to asocket_abort() */ |
| errno = ECANCELED; |
| return -1; |
| } |
| |
| if (pfd[0].revents) { |
| if (pfd[0].revents & POLLOUT) { |
| /* connect call complete, read return code */ |
| retlen = sizeof(ret); |
| if (getsockopt(s->fd, SOL_SOCKET, SO_ERROR, &ret, &retlen)) |
| return -1; |
| /* got connect() return code */ |
| if (ret) { |
| errno = ret; |
| } |
| } else { |
| /* some error event on this fd */ |
| errno = ECONNABORTED; |
| return -1; |
| } |
| } |
| } |
| |
| return ret; |
| } |
| |
| int asocket_accept(struct asocket *s, struct sockaddr *addr, |
| socklen_t *addrlen, int timeout) { |
| |
| int ret; |
| struct pollfd pfd[2]; |
| |
| pfd[0].fd = s->fd; |
| pfd[0].events = POLLIN; |
| pfd[0].revents = 0; |
| pfd[1].fd = s->abort_fd[0]; |
| pfd[1].events = POLLIN; |
| pfd[1].revents = 0; |
| |
| do { |
| ret = poll(pfd, 2, timeout); |
| } while (ret < 0 && errno == EINTR); |
| |
| if (ret < 0) |
| return -1; |
| else if (ret == 0) { |
| /* timeout */ |
| errno = ETIMEDOUT; |
| return -1; |
| } |
| |
| if (pfd[1].revents) { |
| /* abort due to asocket_abort() */ |
| errno = ECANCELED; |
| return -1; |
| } |
| |
| if (pfd[0].revents) { |
| if (pfd[0].revents & POLLIN) { |
| /* ready to accept() without blocking */ |
| do { |
| ret = accept(s->fd, addr, addrlen); |
| } while (ret < 0 && errno == EINTR); |
| } else { |
| /* some error event on this fd */ |
| errno = ECONNABORTED; |
| return -1; |
| } |
| } |
| |
| return ret; |
| } |
| |
| int asocket_read(struct asocket *s, void *buf, size_t count, int timeout) { |
| int ret; |
| struct pollfd pfd[2]; |
| |
| pfd[0].fd = s->fd; |
| pfd[0].events = POLLIN; |
| pfd[0].revents = 0; |
| pfd[1].fd = s->abort_fd[0]; |
| pfd[1].events = POLLIN; |
| pfd[1].revents = 0; |
| |
| do { |
| ret = poll(pfd, 2, timeout); |
| } while (ret < 0 && errno == EINTR); |
| |
| if (ret < 0) |
| return -1; |
| else if (ret == 0) { |
| /* timeout */ |
| errno = ETIMEDOUT; |
| return -1; |
| } |
| |
| if (pfd[1].revents) { |
| /* abort due to asocket_abort() */ |
| errno = ECANCELED; |
| return -1; |
| } |
| |
| if (pfd[0].revents) { |
| if (pfd[0].revents & POLLIN) { |
| /* ready to read() without blocking */ |
| do { |
| ret = read(s->fd, buf, count); |
| } while (ret < 0 && errno == EINTR); |
| } else { |
| /* some error event on this fd */ |
| errno = ECONNABORTED; |
| return -1; |
| } |
| } |
| |
| return ret; |
| } |
| |
| int asocket_write(struct asocket *s, const void *buf, size_t count, |
| int timeout) { |
| int ret; |
| struct pollfd pfd[2]; |
| |
| pfd[0].fd = s->fd; |
| pfd[0].events = POLLOUT; |
| pfd[0].revents = 0; |
| pfd[1].fd = s->abort_fd[0]; |
| pfd[1].events = POLLIN; |
| pfd[1].revents = 0; |
| |
| do { |
| ret = poll(pfd, 2, timeout); |
| } while (ret < 0 && errno == EINTR); |
| |
| if (ret < 0) |
| return -1; |
| else if (ret == 0) { |
| /* timeout */ |
| errno = ETIMEDOUT; |
| return -1; |
| } |
| |
| if (pfd[1].revents) { |
| /* abort due to asocket_abort() */ |
| errno = ECANCELED; |
| return -1; |
| } |
| |
| if (pfd[0].revents) { |
| if (pfd[0].revents & POLLOUT) { |
| /* ready to write() without blocking */ |
| do { |
| ret = write(s->fd, buf, count); |
| } while (ret < 0 && errno == EINTR); |
| } else { |
| /* some error event on this fd */ |
| errno = ECONNABORTED; |
| return -1; |
| } |
| } |
| |
| return ret; |
| } |
| |
| void asocket_abort(struct asocket *s) { |
| int ret; |
| char buf = 0; |
| |
| /* Prevent further use of fd, without yet releasing the fd */ |
| shutdown(s->fd, SHUT_RDWR); |
| |
| /* wake up calls blocked at poll() */ |
| do { |
| ret = write(s->abort_fd[1], &buf, 1); |
| } while (ret < 0 && errno == EINTR); |
| } |
| |
| void asocket_destroy(struct asocket *s) { |
| struct asocket s_copy = *s; |
| |
| /* Clients should *not* be using these fd's after calling |
| asocket_destroy(), but in case they do, set to -1 so they cannot use a |
| stale fd */ |
| s->fd = -1; |
| s->abort_fd[0] = -1; |
| s->abort_fd[1] = -1; |
| |
| /* Call asocket_abort() in case there are still threads blocked on this |
| socket. Clients should not rely on this behavior - it is racy because we |
| are about to close() these sockets - clients should instead make sure |
| all threads are done with the socket before calling asocket_destory(). |
| */ |
| asocket_abort(&s_copy); |
| |
| /* enough safety checks, close and release memory */ |
| close(s_copy.abort_fd[1]); |
| close(s_copy.abort_fd[0]); |
| close(s_copy.fd); |
| |
| free(s); |
| } |