1// Copyright 2010, Shuo Chen. All rights reserved. 2// http://code.google.com/p/muduo/ 3// 4// Use of this source code is governed by a BSD-style license 5// that can be found in the License file. 6 7// Author: Shuo Chen (chenshuo at chenshuo dot com) 8 9#include "SocketsOps.h" 10#include "logging/Logging.h" 11 12#include <errno.h> 13#include <fcntl.h> 14#include <stdio.h> // snprintf 15#include <strings.h> // bzero 16#include <sys/socket.h> 17#include <unistd.h> 18 19using namespace muduo; 20 21namespace 22{ 23 24typedef struct sockaddr SA; 25 26const SA* sockaddr_cast(const struct sockaddr_in* addr) 27{ 28 return static_cast<const SA*>(implicit_cast<const void*>(addr)); 29} 30 31SA* sockaddr_cast(struct sockaddr_in* addr) 32{ 33 return static_cast<SA*>(implicit_cast<void*>(addr)); 34} 35 36void setNonBlockAndCloseOnExec(int sockfd) 37{ 38 // non-block 39 int flags = ::fcntl(sockfd, F_GETFL, 0); 40 flags |= O_NONBLOCK; 41 int ret = ::fcntl(sockfd, F_SETFL, flags); 42 // FIXME check 43 44 // close-on-exec 45 flags = ::fcntl(sockfd, F_GETFD, 0); 46 flags |= FD_CLOEXEC; 47 ret = ::fcntl(sockfd, F_SETFD, flags); 48 // FIXME check 49} 50 51} 52 53int sockets::createNonblockingOrDie() 54{ 55 // socket 56#if VALGRIND 57 int sockfd = ::socket(AF_INET, SOCK_STREAM, IPPROTO_TCP); 58 if (sockfd < 0) 59 { 60 LOG_SYSFATAL << "sockets::createNonblockingOrDie"; 61 } 62 63 setNonBlockAndCloseOnExec(sockfd); 64#else 65 int sockfd = ::socket(AF_INET, 66 SOCK_STREAM | SOCK_NONBLOCK | SOCK_CLOEXEC, 67 IPPROTO_TCP); 68 if (sockfd < 0) 69 { 70 LOG_SYSFATAL << "sockets::createNonblockingOrDie"; 71 } 72#endif 73 return sockfd; 74} 75 76void sockets::bindOrDie(int sockfd, const struct sockaddr_in& addr) 77{ 78 int ret = ::bind(sockfd, sockaddr_cast(&addr), sizeof addr); 79 if (ret < 0) 80 { 81 LOG_SYSFATAL << "sockets::bindOrDie"; 82 } 83} 84 85void sockets::listenOrDie(int sockfd) 86{ 87 int ret = ::listen(sockfd, SOMAXCONN); 88 if (ret < 0) 89 { 90 LOG_SYSFATAL << "sockets::listenOrDie"; 91 } 92} 93 94int sockets::accept(int sockfd, struct sockaddr_in* addr) 95{ 96 socklen_t addrlen = sizeof *addr; 97#if VALGRIND 98 int connfd = ::accept(sockfd, sockaddr_cast(addr), &addrlen); 99 setNonBlockAndCloseOnExec(connfd); 100#else 101 int connfd = ::accept4(sockfd, sockaddr_cast(addr), 102 &addrlen, SOCK_NONBLOCK | SOCK_CLOEXEC); 103#endif 104 if (connfd < 0) 105 { 106 int savedErrno = errno; 107 LOG_SYSERR << "Socket::accept"; 108 switch (savedErrno) 109 { 110 case EAGAIN: 111 case ECONNABORTED: 112 case EINTR: 113 case EPROTO: // ??? 114 case EPERM: 115 case EMFILE: // per-process lmit of open file desctiptor ??? 116 // expected errors 117 errno = savedErrno; 118 break; 119 case EBADF: 120 case EFAULT: 121 case EINVAL: 122 case ENFILE: 123 case ENOBUFS: 124 case ENOMEM: 125 case ENOTSOCK: 126 case EOPNOTSUPP: 127 // unexpected errors 128 LOG_FATAL << "unexpected error of ::accept " << savedErrno; 129 break; 130 default: 131 LOG_FATAL << "unknown error of ::accept " << savedErrno; 132 break; 133 } 134 } 135 return connfd; 136} 137 138void sockets::close(int sockfd) 139{ 140 if (::close(sockfd) < 0) 141 { 142 LOG_SYSERR << "sockets::close"; 143 } 144} 145 146void sockets::toHostPort(char* buf, size_t size, 147 const struct sockaddr_in& addr) 148{ 149 char host[INET_ADDRSTRLEN] = "INVALID"; 150 ::inet_ntop(AF_INET, &addr.sin_addr, host, sizeof host); 151 uint16_t port = sockets::networkToHost16(addr.sin_port); 152 snprintf(buf, size, "%s:%u", host, port); 153} 154 155void sockets::fromHostPort(const char* ip, uint16_t port, 156 struct sockaddr_in* addr) 157{ 158 addr->sin_family = AF_INET; 159 addr->sin_port = hostToNetwork16(port); 160 if (::inet_pton(AF_INET, ip, &addr->sin_addr) <= 0) 161 { 162 LOG_SYSERR << "sockets::fromHostPort"; 163 } 164} 165 166