xs_socket.h 4.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200
  1. /* copyright (c) 2022 - 2023 grunfink et al. / MIT license */
  2. #ifndef _XS_SOCKET_H
  3. #define _XS_SOCKET_H
  4. int xs_socket_timeout(int s, double rto, double sto);
  5. int xs_socket_server(const char *addr, const char *serv);
  6. FILE *xs_socket_accept(int rs);
  7. xs_str *xs_socket_peername(int s);
  8. int xs_socket_connect(const char *addr, const char *serv);
  9. #ifdef XS_IMPLEMENTATION
  10. #include <sys/socket.h>
  11. #include <netdb.h>
  12. #include <netinet/in.h>
  13. #include <arpa/inet.h>
  14. int xs_socket_timeout(int s, double rto, double sto)
  15. /* sets the socket timeout in seconds */
  16. {
  17. struct timeval tv;
  18. int ret = 0;
  19. if (rto > 0.0) {
  20. tv.tv_sec = (int)rto;
  21. tv.tv_usec = (int)((rto - (double)(int)rto) * 1000000.0);
  22. ret = setsockopt(s, SOL_SOCKET, SO_RCVTIMEO, (char *)&tv, sizeof(tv));
  23. }
  24. if (sto > 0.0) {
  25. tv.tv_sec = (int)sto;
  26. tv.tv_usec = (int)((sto - (double)(int)sto) * 1000000.0);
  27. ret = setsockopt(s, SOL_SOCKET, SO_SNDTIMEO, (char *)&tv, sizeof(tv));
  28. }
  29. return ret;
  30. }
  31. int xs_socket_server(const char *addr, const char *serv)
  32. /* opens a server socket by service name (or port as string) */
  33. {
  34. int rs = -1;
  35. struct sockaddr_in host;
  36. memset(&host, '\0', sizeof(host));
  37. if (addr != NULL) {
  38. struct hostent *he;
  39. if ((he = gethostbyname(addr)) != NULL)
  40. memcpy(&host.sin_addr, he->h_addr_list[0], he->h_length);
  41. else
  42. goto end;
  43. }
  44. struct servent *se;
  45. if ((se = getservbyname(serv, "tcp")) != NULL)
  46. host.sin_port = se->s_port;
  47. else
  48. host.sin_port = htons(atoi(serv));
  49. host.sin_family = AF_INET;
  50. if ((rs = socket(AF_INET, SOCK_STREAM, 0)) != -1) {
  51. /* reuse addr */
  52. int i = 1;
  53. setsockopt(rs, SOL_SOCKET, SO_REUSEADDR, (char *)&i, sizeof(i));
  54. if (bind(rs, (struct sockaddr *)&host, sizeof(host)) == -1) {
  55. close(rs);
  56. rs = -1;
  57. }
  58. else
  59. listen(rs, SOMAXCONN);
  60. }
  61. end:
  62. return rs;
  63. }
  64. FILE *xs_socket_accept(int rs)
  65. /* accepts an incoming connection */
  66. {
  67. int cs = -1;
  68. struct sockaddr_storage addr;
  69. socklen_t l = sizeof(addr);
  70. cs = accept(rs, (struct sockaddr *)&addr, &l);
  71. return cs == -1 ? NULL : fdopen(cs, "r+");
  72. }
  73. xs_str *xs_socket_peername(int s)
  74. /* returns the remote address as a string */
  75. {
  76. xs_str *ip = NULL;
  77. struct sockaddr_storage addr;
  78. socklen_t slen = sizeof(addr);
  79. if (getpeername(s, (struct sockaddr *)&addr, &slen) != -1) {
  80. char buf[1024];
  81. const char *p = NULL;
  82. if (addr.ss_family == AF_INET) {
  83. struct sockaddr_in *sa = (struct sockaddr_in *)&addr;
  84. p = inet_ntop(AF_INET, &sa->sin_addr, buf, sizeof(buf));
  85. }
  86. else
  87. if (addr.ss_family == AF_INET6) {
  88. struct sockaddr_in6 *sa = (struct sockaddr_in6 *)&addr;
  89. p = inet_ntop(AF_INET6, &sa->sin6_addr, buf, sizeof(buf));
  90. }
  91. if (p != NULL)
  92. ip = xs_str_new(p);
  93. }
  94. return ip;
  95. }
  96. int xs_socket_connect(const char *addr, const char *serv)
  97. /* creates a client connection socket */
  98. {
  99. int d = -1;
  100. #ifndef WITHOUT_GETADDRINFO
  101. struct addrinfo *res;
  102. struct addrinfo hints;
  103. memset(&hints, '\0', sizeof(hints));
  104. hints.ai_socktype = SOCK_STREAM;
  105. hints.ai_flags = AI_ADDRCONFIG;
  106. if (getaddrinfo(addr, serv, &hints, &res) == 0) {
  107. struct addrinfo *r;
  108. for (r = res; r != NULL; r = r->ai_next) {
  109. d = socket(r->ai_family, r->ai_socktype, r->ai_protocol);
  110. if (d != -1) {
  111. if (connect(d, r->ai_addr, r->ai_addrlen) == 0)
  112. break;
  113. close(d);
  114. d = -1;
  115. }
  116. }
  117. freeaddrinfo(res);
  118. }
  119. #else /* WITHOUT_GETADDRINFO */
  120. /* traditional socket interface */
  121. struct hostent *he;
  122. if ((he = gethostbyname(addr)) != NULL) {
  123. struct sockaddr_in host;
  124. memset(&host, '\0', sizeof(host));
  125. memcpy(&host.sin_addr, he->h_addr_list[0], he->h_length);
  126. host.sin_family = he->h_addrtype;
  127. struct servent *se;
  128. if ((se = getservbyname(serv, "tcp")) != NULL)
  129. host.sin_port = se->s_port;
  130. else
  131. host.sin_port = htons(atoi(serv));
  132. if ((d = socket(AF_INET, SOCK_STREAM, 0)) != -1) {
  133. if (connect(d, (struct sockaddr *)&host, sizeof(host)) == -1)
  134. d = -1;
  135. }
  136. }
  137. #endif /* WITHOUT_GETADDRINFO */
  138. return d;
  139. }
  140. #endif /* XS_IMPLEMENTATION */
  141. #endif /* _XS_SOCKET_H */