libsdevent.h 4.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177
  1. #ifndef HIREDIS_LIBSDEVENT_H
  2. #define HIREDIS_LIBSDEVENT_H
  3. #include <systemd/sd-event.h>
  4. #include "../hiredis.h"
  5. #include "../async.h"
  6. #define REDIS_LIBSDEVENT_DELETED 0x01
  7. #define REDIS_LIBSDEVENT_ENTERED 0x02
  8. typedef struct redisLibsdeventEvents {
  9. redisAsyncContext *context;
  10. struct sd_event *event;
  11. struct sd_event_source *fdSource;
  12. struct sd_event_source *timerSource;
  13. int fd;
  14. short flags;
  15. short state;
  16. } redisLibsdeventEvents;
  17. static void redisLibsdeventDestroy(redisLibsdeventEvents *e) {
  18. if (e->fdSource) {
  19. e->fdSource = sd_event_source_disable_unref(e->fdSource);
  20. }
  21. if (e->timerSource) {
  22. e->timerSource = sd_event_source_disable_unref(e->timerSource);
  23. }
  24. sd_event_unref(e->event);
  25. hi_free(e);
  26. }
  27. static int redisLibsdeventTimeoutHandler(sd_event_source *s, uint64_t usec, void *userdata) {
  28. ((void)s);
  29. ((void)usec);
  30. redisLibsdeventEvents *e = (redisLibsdeventEvents*)userdata;
  31. redisAsyncHandleTimeout(e->context);
  32. return 0;
  33. }
  34. static int redisLibsdeventHandler(sd_event_source *s, int fd, uint32_t event, void *userdata) {
  35. ((void)s);
  36. ((void)fd);
  37. redisLibsdeventEvents *e = (redisLibsdeventEvents*)userdata;
  38. e->state |= REDIS_LIBSDEVENT_ENTERED;
  39. #define CHECK_DELETED() if (e->state & REDIS_LIBSDEVENT_DELETED) {\
  40. redisLibsdeventDestroy(e);\
  41. return 0; \
  42. }
  43. if ((event & EPOLLIN) && e->context && (e->state & REDIS_LIBSDEVENT_DELETED) == 0) {
  44. redisAsyncHandleRead(e->context);
  45. CHECK_DELETED();
  46. }
  47. if ((event & EPOLLOUT) && e->context && (e->state & REDIS_LIBSDEVENT_DELETED) == 0) {
  48. redisAsyncHandleWrite(e->context);
  49. CHECK_DELETED();
  50. }
  51. e->state &= ~REDIS_LIBSDEVENT_ENTERED;
  52. #undef CHECK_DELETED
  53. return 0;
  54. }
  55. static void redisLibsdeventAddRead(void *userdata) {
  56. redisLibsdeventEvents *e = (redisLibsdeventEvents*)userdata;
  57. if (e->flags & EPOLLIN) {
  58. return;
  59. }
  60. e->flags |= EPOLLIN;
  61. if (e->flags & EPOLLOUT) {
  62. sd_event_source_set_io_events(e->fdSource, e->flags);
  63. } else {
  64. sd_event_add_io(e->event, &e->fdSource, e->fd, e->flags, redisLibsdeventHandler, e);
  65. }
  66. }
  67. static void redisLibsdeventDelRead(void *userdata) {
  68. redisLibsdeventEvents *e = (redisLibsdeventEvents*)userdata;
  69. e->flags &= ~EPOLLIN;
  70. if (e->flags) {
  71. sd_event_source_set_io_events(e->fdSource, e->flags);
  72. } else {
  73. e->fdSource = sd_event_source_disable_unref(e->fdSource);
  74. }
  75. }
  76. static void redisLibsdeventAddWrite(void *userdata) {
  77. redisLibsdeventEvents *e = (redisLibsdeventEvents*)userdata;
  78. if (e->flags & EPOLLOUT) {
  79. return;
  80. }
  81. e->flags |= EPOLLOUT;
  82. if (e->flags & EPOLLIN) {
  83. sd_event_source_set_io_events(e->fdSource, e->flags);
  84. } else {
  85. sd_event_add_io(e->event, &e->fdSource, e->fd, e->flags, redisLibsdeventHandler, e);
  86. }
  87. }
  88. static void redisLibsdeventDelWrite(void *userdata) {
  89. redisLibsdeventEvents *e = (redisLibsdeventEvents*)userdata;
  90. e->flags &= ~EPOLLOUT;
  91. if (e->flags) {
  92. sd_event_source_set_io_events(e->fdSource, e->flags);
  93. } else {
  94. e->fdSource = sd_event_source_disable_unref(e->fdSource);
  95. }
  96. }
  97. static void redisLibsdeventCleanup(void *userdata) {
  98. redisLibsdeventEvents *e = (redisLibsdeventEvents*)userdata;
  99. if (!e) {
  100. return;
  101. }
  102. if (e->state & REDIS_LIBSDEVENT_ENTERED) {
  103. e->state |= REDIS_LIBSDEVENT_DELETED;
  104. } else {
  105. redisLibsdeventDestroy(e);
  106. }
  107. }
  108. static void redisLibsdeventSetTimeout(void *userdata, struct timeval tv) {
  109. redisLibsdeventEvents *e = (redisLibsdeventEvents *)userdata;
  110. uint64_t usec = tv.tv_sec * 1000000 + tv.tv_usec;
  111. if (!e->timerSource) {
  112. sd_event_add_time_relative(e->event, &e->timerSource, CLOCK_MONOTONIC, usec, 1, redisLibsdeventTimeoutHandler, e);
  113. } else {
  114. sd_event_source_set_time_relative(e->timerSource, usec);
  115. }
  116. }
  117. static int redisLibsdeventAttach(redisAsyncContext *ac, struct sd_event *event) {
  118. redisContext *c = &(ac->c);
  119. redisLibsdeventEvents *e;
  120. /* Nothing should be attached when something is already attached */
  121. if (ac->ev.data != NULL)
  122. return REDIS_ERR;
  123. /* Create container for context and r/w events */
  124. e = (redisLibsdeventEvents*)hi_calloc(1, sizeof(*e));
  125. if (e == NULL)
  126. return REDIS_ERR;
  127. /* Initialize and increase event refcount */
  128. e->context = ac;
  129. e->event = event;
  130. e->fd = c->fd;
  131. sd_event_ref(event);
  132. /* Register functions to start/stop listening for events */
  133. ac->ev.addRead = redisLibsdeventAddRead;
  134. ac->ev.delRead = redisLibsdeventDelRead;
  135. ac->ev.addWrite = redisLibsdeventAddWrite;
  136. ac->ev.delWrite = redisLibsdeventDelWrite;
  137. ac->ev.cleanup = redisLibsdeventCleanup;
  138. ac->ev.scheduleTimer = redisLibsdeventSetTimeout;
  139. ac->ev.data = e;
  140. return REDIS_OK;
  141. }
  142. #endif