xs_socket.h 2.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129
  1. /* copyright (c) 2022 - 2023 grunfink / 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, int port);
  6. FILE *xs_socket_accept(int rs);
  7. xs_str *xs_socket_peername(int s);
  8. #ifdef XS_IMPLEMENTATION
  9. #include <sys/socket.h>
  10. #include <netdb.h>
  11. #include <netinet/in.h>
  12. #include <arpa/inet.h>
  13. int xs_socket_timeout(int s, double rto, double sto)
  14. /* sets the socket timeout in seconds */
  15. {
  16. struct timeval tv;
  17. int ret = 0;
  18. if (rto > 0.0) {
  19. tv.tv_sec = (int)rto;
  20. tv.tv_usec = (int)((rto - (double)(int)rto) * 1000000.0);
  21. ret = setsockopt(s, SOL_SOCKET, SO_RCVTIMEO, (char *)&tv, sizeof(tv));
  22. }
  23. if (sto > 0.0) {
  24. tv.tv_sec = (int)sto;
  25. tv.tv_usec = (int)((sto - (double)(int)sto) * 1000000.0);
  26. ret = setsockopt(s, SOL_SOCKET, SO_SNDTIMEO, (char *)&tv, sizeof(tv));
  27. }
  28. return ret;
  29. }
  30. int xs_socket_server(const char *addr, int port)
  31. /* opens a server socket */
  32. {
  33. int rs = -1;
  34. struct sockaddr_in host;
  35. memset(&host, '\0', sizeof(host));
  36. if (addr != NULL) {
  37. struct hostent *he;
  38. if ((he = gethostbyname(addr)) != NULL)
  39. memcpy(&host.sin_addr, he->h_addr_list[0], he->h_length);
  40. else
  41. goto end;
  42. }
  43. host.sin_family = AF_INET;
  44. host.sin_port = htons(port);
  45. if ((rs = socket(AF_INET, SOCK_STREAM, 0)) != -1) {
  46. /* reuse addr */
  47. int i = 1;
  48. setsockopt(rs, SOL_SOCKET, SO_REUSEADDR, (char *)&i, sizeof(i));
  49. if (bind(rs, (struct sockaddr *)&host, sizeof(host)) == -1) {
  50. close(rs);
  51. rs = -1;
  52. }
  53. else
  54. listen(rs, SOMAXCONN);
  55. }
  56. end:
  57. return rs;
  58. }
  59. FILE *xs_socket_accept(int rs)
  60. /* accepts an incoming connection */
  61. {
  62. int cs = -1;
  63. struct sockaddr_storage addr;
  64. socklen_t l = sizeof(addr);
  65. cs = accept(rs, (struct sockaddr *)&addr, &l);
  66. return cs == -1 ? NULL : fdopen(cs, "r+");
  67. }
  68. xs_str *xs_socket_peername(int s)
  69. /* returns the remote address as a string */
  70. {
  71. xs_str *ip = NULL;
  72. struct sockaddr_storage addr;
  73. socklen_t slen = sizeof(addr);
  74. if (getpeername(s, (struct sockaddr *)&addr, &slen) != -1) {
  75. char buf[1024];
  76. const char *p = NULL;
  77. if (addr.ss_family == AF_INET) {
  78. struct sockaddr_in *sa = (struct sockaddr_in *)&addr;
  79. p = inet_ntop(AF_INET, &sa->sin_addr, buf, sizeof(buf));
  80. }
  81. else
  82. if (addr.ss_family == AF_INET6) {
  83. struct sockaddr_in6 *sa = (struct sockaddr_in6 *)&addr;
  84. p = inet_ntop(AF_INET6, &sa->sin6_addr, buf, sizeof(buf));
  85. }
  86. if (p != NULL)
  87. ip = xs_str_new(p);
  88. }
  89. return ip;
  90. }
  91. #endif /* XS_IMPLEMENTATION */
  92. #endif /* _XS_SOCKET_H */