polling.c 3.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180
  1. /* This file is part of libepistle.
  2. *
  3. * libepistle is free software: you can redistribute it and/or modify
  4. * it under the terms of the GNU Lesser General Public License as published by
  5. * the Free Software Foundation, either version 3 of the License, or
  6. * (at your option) any later version.
  7. *
  8. * libepistle is distributed in the hope that it will be useful,
  9. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  10. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  11. * GNU Lesser General Public License for more details.
  12. *
  13. * You should have received a copy of the GNU Lesser General Public License
  14. * along with libepistle. If not, see <http://www.gnu.org/licenses/>.
  15. */
  16. #include <sys/epoll.h>
  17. #include <sys/socket.h>
  18. #include <unistd.h>
  19. #include <errno.h>
  20. #include "polling.h"
  21. struct pm_extra {
  22. Pm_react react;
  23. Server_check s_check;
  24. void *extra;
  25. };
  26. int
  27. poll_add(int epoll_fd, int fd, Err *err)
  28. {
  29. struct epoll_event event = { 0 };
  30. event.events = EPOLLIN;
  31. event.data.fd = fd;
  32. if (epoll_ctl(epoll_fd, EPOLL_CTL_ADD, fd, &event) < 0) {
  33. err_std(err);
  34. return -1;
  35. }
  36. return 0;
  37. }
  38. int
  39. poll_remove(int epoll_fd, int fd, Err *err)
  40. {
  41. if (epoll_ctl(epoll_fd, EPOLL_CTL_DEL, fd, NULL) < 0) {
  42. err_std(err);
  43. return -1;
  44. }
  45. return 0;
  46. }
  47. int
  48. poll_out(int epoll_fd, int fd, Err *err)
  49. {
  50. struct epoll_event event = { 0 };
  51. event.events = EPOLLIN | EPOLLOUT;
  52. event.data.fd = fd;
  53. if (epoll_ctl(epoll_fd, EPOLL_CTL_MOD, fd, &event) < 0) {
  54. err_std(err);
  55. return -1;
  56. }
  57. return 0;
  58. }
  59. int
  60. poll_out_off(int epoll_fd, int fd, Err *err)
  61. {
  62. struct epoll_event event = { 0 };
  63. event.events = EPOLLIN;
  64. event.data.fd = fd;
  65. if (epoll_ctl(epoll_fd, EPOLL_CTL_MOD, fd, &event) < 0) {
  66. err_std(err);
  67. return -1;
  68. }
  69. return 0;
  70. }
  71. int
  72. polling(int *epoll_fd, Polling_react react, int fds[], Err *err, void *extra)
  73. {
  74. struct epoll_event events[16];
  75. if (*epoll_fd < 0 && (*epoll_fd = epoll_create1(0)) < 0) {
  76. err_std(err);
  77. return -1;
  78. }
  79. if (fds)
  80. for (int i = 0; fds[i] >= 0; ++i)
  81. if (poll_add(*epoll_fd, fds[i], err) < 0)
  82. return -1;
  83. while (1) {
  84. int react_val = POLLING_OKAY;
  85. int nfds = epoll_wait(*epoll_fd, events,
  86. sizeof(events) / sizeof(events[0]), -1);
  87. if (nfds < 0) {
  88. /* laptop closing can cause this! */
  89. if (errno == EINTR)
  90. continue;
  91. err_std(err);
  92. return -1;
  93. }
  94. for (int i = 0; i < nfds && react_val == POLLING_OKAY; ++i)
  95. react_val = react(*epoll_fd, events[i].data.fd,
  96. events[i].events, err, extra);
  97. if (react_val == POLLING_OKAY)
  98. continue;
  99. if (react_val > 0) /* POLLING END */
  100. break;
  101. /* POLLING FATAL */
  102. return -1;
  103. }
  104. return 0;
  105. }
  106. static int
  107. pm_react(int epoll_fd, int fd, uint32_t events, Err *err, void *extra)
  108. {
  109. int pm_event = 0;
  110. struct pm_extra *pm_extra = extra;
  111. int server_fd = -1;
  112. char check;
  113. if (pm_extra->s_check &&
  114. pm_extra->s_check(fd, pm_extra->extra)) {
  115. int client = accept(fd, NULL, NULL);
  116. /* We don't do anything, it's the clients problem */
  117. if (client < 0)
  118. return POLLING_OKAY;
  119. server_fd = fd;
  120. fd = client;
  121. pm_event = PM_ADD;
  122. } else if (events & EPOLLIN) {
  123. pm_event = recv(fd, &check, 1, MSG_PEEK | MSG_DONTWAIT) > 0 ?
  124. PM_INPUT : PM_REMOVE;
  125. } else if (events & EPOLLOUT) {
  126. pm_event = PM_OUTPUT;
  127. }
  128. if (!pm_event)
  129. return POLLING_OKAY;
  130. return pm_extra->react(epoll_fd, fd, server_fd, pm_event,
  131. err, pm_extra->extra);
  132. }
  133. int
  134. polling_mode(Pm_react react, int *epoll_fd, int fds[], Server_check s_check,
  135. Err *err, void *extra)
  136. {
  137. struct pm_extra pm_extra = {
  138. .react = react,
  139. .s_check = s_check,
  140. .extra = extra
  141. };
  142. return polling(epoll_fd, pm_react, fds, err, &pm_extra);
  143. }