/phxsqlproxy/routineutil.cpp
https://github.com/tencent-wechat/phxsql · C++ · 157 lines · 121 code · 24 blank · 12 comment · 37 complexity · 03d4d64f2ec1af1b55ce9e5a87c6aeea MD5 · raw file
- /*
- Tencent is pleased to support the open source community by making PhxSQL available.
- Copyright (C) 2016 THL A29 Limited, a Tencent company. All rights reserved.
- Licensed under the GNU General Public 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
-
- https://opensource.org/licenses/GPL-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 <assert.h>
- #include <unistd.h>
- #include <string.h>
- #include "routineutil.h"
- #include "co_routine.h"
- #include "phxsqlproxyutil.h"
- namespace phxsqlproxy {
- int RoutineConnectWithTimeout(int fd, const struct sockaddr *address, socklen_t address_len,
- int timeout_ms) {
- assert(IsNonBlock(fd));
- //1.sys call
- int ret = connect(fd, address, address_len);
- //2.wait
- int pollret = 0;
- struct pollfd pf = { 0 };
- for (int i = 0; i < 3; i++) //25s * 3 = 75s
- {
- memset(&pf, 0, sizeof(pf));
- pf.fd = fd;
- pf.events = (POLLOUT | POLLERR | POLLHUP);
- pollret = poll(&pf, 1, timeout_ms);
- if (1 == pollret) {
- break;
- }
- }
- if (pf.revents & POLLOUT) //connect succ
- {
- errno = 0;
- return 0;
- }
- //3.set errno
- int err = 0;
- socklen_t errlen = sizeof(err);
- getsockopt(fd, SOL_SOCKET, SO_ERROR, &err, &errlen);
- if (err) {
- errno = err;
- } else {
- errno = ETIMEDOUT;
- }
- return ret;
- }
- int RoutineReadWithTimeout(int source_fd, char * buf, int buf_size, int timeout_ms) {
- assert(IsNonBlock(source_fd));
- struct pollfd pf[1];
- int nfds = 0;
- memset(pf, 0, sizeof(pf));
- pf[0].fd = source_fd;
- pf[0].events = (POLLIN | POLLERR | POLLHUP);
- nfds++;
- int return_fd_count = co_poll(co_get_epoll_ct(), pf, nfds, timeout_ms);
- if (return_fd_count < 0) {
- return return_fd_count;
- }
- if (pf[0].revents & POLLIN) {
- return read(source_fd, buf, buf_size);
- } else if (pf[0].revents & POLLHUP) {
- return return_fd_count;
- } else if (pf[0].revents & POLLERR) {
- return return_fd_count;
- }
- return return_fd_count;
- }
- int RoutineWriteWithTimeout(int dest_fd, const char * buf, int write_size, int timeout_ms) {
- assert(IsNonBlock(dest_fd));
- int written_once = 0;
- written_once = write(dest_fd, buf, write_size);
- if (written_once > 0) {
- return written_once;
- }
- if (written_once <= 0) {
- if (errno == EAGAIN || errno == EINTR) {
- written_once = 0;
- } else {
- return -__LINE__;
- }
- }
- struct pollfd pf[1];
- int nfds = 0;
- memset(pf, 0, sizeof(pf));
- pf[0].fd = dest_fd;
- pf[0].events = (POLLOUT | POLLERR | POLLHUP);
- nfds++;
- int return_fd_count = co_poll(co_get_epoll_ct(), pf, nfds, timeout_ms);
- if (return_fd_count < 0) {
- return return_fd_count;
- }
- if (pf[0].revents & POLLOUT) {
- written_once = write(dest_fd, buf, write_size);
- if (written_once <= 0) {
- if (errno == EAGAIN || errno == EINTR) {
- written_once = 0;
- } else {
- return written_once;
- }
- }
- } else if (pf[0].revents & POLLHUP) {
- return return_fd_count;
- } else if (pf[0].revents & POLLERR) {
- return return_fd_count;
- }
- return written_once;
- }
- int RoutinePeekWithTimeout(int source_fd, char * buf, int buf_size, int timeout_ms) {
- assert(IsNonBlock(source_fd));
- struct pollfd pf[1];
- int nfds = 0;
- memset(pf, 0, sizeof(pf));
- pf[0].fd = source_fd;
- pf[0].events = (POLLIN | POLLERR | POLLHUP);
- nfds++;
- int return_fd_count = co_poll(co_get_epoll_ct(), pf, nfds, timeout_ms);
- if (return_fd_count < 0) {
- return return_fd_count;
- }
- if (pf[0].revents & POLLIN) {
- return recv(source_fd, buf, buf_size, MSG_PEEK);
- } else if (pf[0].revents & POLLHUP) {
- return return_fd_count;
- } else if (pf[0].revents & POLLERR) {
- return return_fd_count;
- }
- return return_fd_count;
- }
- }