hsocket.h 5.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187
  1. #ifndef HW_SOCKET_H_
  2. #define HW_SOCKET_H_
  3. #include "hplatform.h"
  4. #include "hdef.h"
  5. #ifdef _MSC_VER
  6. #pragma comment(lib, "ws2_32.lib")
  7. #endif
  8. BEGIN_EXTERN_C
  9. static inline int socket_errno() {
  10. #ifdef OS_WIN
  11. return WSAGetLastError();
  12. #else
  13. return errno;
  14. #endif
  15. }
  16. char* socket_strerror(int err);
  17. // socket -> setsockopt -> bind
  18. // @param type: SOCK_STREAM(tcp) SOCK_DGRAM(udp)
  19. // @return sockfd
  20. int Bind(int port, int type DEFAULT(SOCK_STREAM));
  21. // Bind -> listen
  22. // @return sockfd
  23. int Listen(int port);
  24. // @param host: domain or ip
  25. // @retval 0:succeed
  26. int Resolver(const char* host, struct sockaddr* addr);
  27. // @return sockfd
  28. // Resolver -> socket -> nonblocking -> connect
  29. int Connect(const char* host, int port, int nonblock DEFAULT(0));
  30. // Connect(host, port, 1)
  31. int ConnectNonblock(const char* host, int port);
  32. // Connect(host, port, 1) -> select -> blocking
  33. int ConnectTimeout(const char* host, int port, int ms);
  34. // @param cnt: ping count
  35. // @return: ok count
  36. // @note: printd $CC -DPRINT_DEBUG
  37. int Ping(const char* host, int cnt DEFAULT(4));
  38. #ifdef OS_WIN
  39. typedef int socklen_t;
  40. static inline int blocking(int sockfd) {
  41. unsigned long nb = 0;
  42. return ioctlsocket(sockfd, FIONBIO, &nb);
  43. }
  44. static inline int nonblocking(int sockfd) {
  45. unsigned long nb = 1;
  46. return ioctlsocket(sockfd, FIONBIO, &nb);
  47. }
  48. #define poll WSAPoll
  49. #undef EAGAIN
  50. #define EAGAIN WSAEWOULDBLOCK
  51. #undef EINPROGRESS
  52. #define EINPROGRESS WSAEINPROGRESS
  53. #undef ENOTSOCK
  54. #define ENOTSOCK WSAENOTSOCK
  55. #else
  56. #define blocking(s) fcntl(s, F_SETFL, fcntl(s, F_GETFL) & ~O_NONBLOCK)
  57. #define nonblocking(s) fcntl(s, F_SETFL, fcntl(s, F_GETFL) | O_NONBLOCK)
  58. typedef int SOCKET;
  59. #define closesocket close
  60. #endif
  61. static inline const char* sockaddr_ntop(const struct sockaddr* addr, char *ip, int len) {
  62. if (addr->sa_family == AF_INET) {
  63. struct sockaddr_in* sin = (struct sockaddr_in*)addr;
  64. return inet_ntop(AF_INET, &sin->sin_addr, ip, len);
  65. }
  66. else if (addr->sa_family == AF_INET6) {
  67. struct sockaddr_in6* sin6 = (struct sockaddr_in6*)addr;
  68. return inet_ntop(AF_INET6, &sin6->sin6_addr, ip, len);
  69. }
  70. return ip;
  71. }
  72. static inline uint16_t sockaddr_htons(const struct sockaddr* addr) {
  73. if (addr->sa_family == AF_INET) {
  74. struct sockaddr_in* sin = (struct sockaddr_in*)addr;
  75. return htons(sin->sin_port);
  76. }
  77. else if (addr->sa_family == AF_INET6) {
  78. struct sockaddr_in6* sin6 = (struct sockaddr_in6*)addr;
  79. return htons(sin6->sin6_port);
  80. }
  81. return 0;
  82. }
  83. static inline void sockaddr_printf(const struct sockaddr* addr) {
  84. char ip[INET6_ADDRSTRLEN] = {0};
  85. int port = 0;
  86. if (addr->sa_family == AF_INET) {
  87. struct sockaddr_in* sin = (struct sockaddr_in*)addr;
  88. inet_ntop(AF_INET, &sin->sin_addr, ip, sizeof(ip));
  89. port = htons(sin->sin_port);
  90. }
  91. else if (addr->sa_family == AF_INET6) {
  92. struct sockaddr_in6* sin6 = (struct sockaddr_in6*)addr;
  93. inet_ntop(AF_INET6, &sin6->sin6_addr, ip, sizeof(ip));
  94. port = htons(sin6->sin6_port);
  95. }
  96. printf("%s:%d\n", ip, port);
  97. }
  98. static inline const char* sockaddr_snprintf(const struct sockaddr* addr, char* buf, int len) {
  99. int port = 0;
  100. if (addr->sa_family == AF_INET) {
  101. struct sockaddr_in* sin = (struct sockaddr_in*)addr;
  102. inet_ntop(AF_INET, &sin->sin_addr, buf, len);
  103. port = htons(sin->sin_port);
  104. }
  105. else if (addr->sa_family == AF_INET6) {
  106. struct sockaddr_in6* sin6 = (struct sockaddr_in6*)addr;
  107. inet_ntop(AF_INET6, &sin6->sin6_addr, buf, len);
  108. port = htons(sin6->sin6_port);
  109. }
  110. char sport[16] = {0};
  111. snprintf(sport, sizeof(sport), ":%d", port);
  112. safe_strncat(buf, sport, len);
  113. return buf;
  114. }
  115. static inline int tcp_nodelay(int sockfd, int on DEFAULT(1)) {
  116. return setsockopt(sockfd, IPPROTO_TCP, TCP_NODELAY, (const char*)&on, sizeof(int));
  117. }
  118. static inline int tcp_nopush(int sockfd, int on DEFAULT(1)) {
  119. #ifdef TCP_NOPUSH
  120. return setsockopt(sockfd, IPPROTO_TCP, TCP_NOPUSH, (const char*)&on, sizeof(int));
  121. #elif defined(TCP_CORK)
  122. return setsockopt(sockfd, IPPROTO_TCP, TCP_CORK, (const char*)&on, sizeof(int));
  123. #else
  124. return -10;
  125. #endif
  126. }
  127. static inline int tcp_keepalive(int sockfd, int on DEFAULT(1), int delay DEFAULT(60)) {
  128. if (setsockopt(sockfd, SOL_SOCKET, SO_KEEPALIVE, (const char*)&on, sizeof(int)) != 0) {
  129. return socket_errno();
  130. }
  131. #ifdef TCP_KEEPALIVE
  132. return setsockopt(sockfd, IPPROTO_TCP, TCP_KEEPALIVE, (const char*)&delay, sizeof(int));
  133. #elif defined(TCP_KEEPIDLE)
  134. // TCP_KEEPIDLE => tcp_keepalive_time
  135. // TCP_KEEPCNT => tcp_keepalive_probes
  136. // TCP_KEEPINTVL => tcp_keepalive_intvl
  137. return setsockopt(sockfd, IPPROTO_TCP, TCP_KEEPIDLE, (const char*)&delay, sizeof(int));
  138. #else
  139. return 0;
  140. #endif
  141. }
  142. static inline int udp_broadcast(int sockfd, int on DEFAULT(1)) {
  143. return setsockopt(sockfd, SOL_SOCKET, SO_BROADCAST, (const char*)&on, sizeof(int));
  144. }
  145. // send timeout
  146. static inline int so_sndtimeo(int sockfd, int timeout) {
  147. #ifdef OS_WIN
  148. return setsockopt(sockfd, SOL_SOCKET, SO_SNDTIMEO, (const char*)&timeout, sizeof(int));
  149. #else
  150. struct timeval tv = {timeout/1000, (timeout%1000)*1000};
  151. return setsockopt(sockfd, SOL_SOCKET, SO_SNDTIMEO, &tv, sizeof(tv));
  152. #endif
  153. }
  154. // recv timeout
  155. static inline int so_rcvtimeo(int sockfd, int timeout) {
  156. #ifdef OS_WIN
  157. return setsockopt(sockfd, SOL_SOCKET, SO_RCVTIMEO, (const char*)&timeout, sizeof(int));
  158. #else
  159. struct timeval tv = {timeout/1000, (timeout%1000)*1000};
  160. return setsockopt(sockfd, SOL_SOCKET, SO_RCVTIMEO, &tv, sizeof(tv));
  161. #endif
  162. }
  163. END_EXTERN_C
  164. #endif // HW_SOCKET_H_