summaryrefslogtreecommitdiffstats
path: root/src
diff options
context:
space:
mode:
authorLennart Poettering <lennart@poettering.net>2023-03-23 23:16:43 +0100
committerLennart Poettering <lennart@poettering.net>2023-05-24 10:52:08 +0200
commit2fdc274c66b6735114ddc00df7f537bcc5372e41 (patch)
treeb5719536b0883f1367335a662d8510a67b3ea651 /src
parentcore: split out default network dep generation into own function (diff)
downloadsystemd-2fdc274c66b6735114ddc00df7f537bcc5372e41.tar.xz
systemd-2fdc274c66b6735114ddc00df7f537bcc5372e41.zip
sd-event: add an explicit API for leaving the ratelimit state
Sometimes, it might make sense to end the ratelimit window early.
Diffstat (limited to 'src')
-rw-r--r--src/libsystemd/libsystemd.sym1
-rw-r--r--src/libsystemd/sd-event/sd-event.c21
-rw-r--r--src/libsystemd/sd-event/test-event.c71
-rw-r--r--src/systemd/sd-event.h1
4 files changed, 94 insertions, 0 deletions
diff --git a/src/libsystemd/libsystemd.sym b/src/libsystemd/libsystemd.sym
index 8b99d0a7aa..35352dc832 100644
--- a/src/libsystemd/libsystemd.sym
+++ b/src/libsystemd/libsystemd.sym
@@ -824,4 +824,5 @@ global:
sd_event_source_set_memory_pressure_period;
sd_event_trim_memory;
sd_pid_notify_barrier;
+ sd_event_source_leave_ratelimit;
} LIBSYSTEMD_253;
diff --git a/src/libsystemd/sd-event/sd-event.c b/src/libsystemd/sd-event/sd-event.c
index f4ede985de..9c624ab56f 100644
--- a/src/libsystemd/sd-event/sd-event.c
+++ b/src/libsystemd/sd-event/sd-event.c
@@ -5193,6 +5193,27 @@ _public_ int sd_event_source_is_ratelimited(sd_event_source *s) {
return s->ratelimited;
}
+_public_ int sd_event_source_leave_ratelimit(sd_event_source *s) {
+ int r;
+
+ assert_return(s, -EINVAL);
+
+ if (!EVENT_SOURCE_CAN_RATE_LIMIT(s->type))
+ return 0;
+
+ if (!ratelimit_configured(&s->rate_limit))
+ return 0;
+
+ if (!s->ratelimited)
+ return 0;
+
+ r = event_source_leave_ratelimit(s, /* run_callback */ false);
+ if (r < 0)
+ return r;
+
+ return 1; /* tell caller that we indeed just left the ratelimit state */
+}
+
_public_ int sd_event_set_signal_exit(sd_event *e, int b) {
bool change = false;
int r;
diff --git a/src/libsystemd/sd-event/test-event.c b/src/libsystemd/sd-event/test-event.c
index 7892f2bc2e..c5497e5148 100644
--- a/src/libsystemd/sd-event/test-event.c
+++ b/src/libsystemd/sd-event/test-event.c
@@ -828,4 +828,75 @@ TEST(fork) {
assert_se(r >= 0);
}
+static int hup_callback(sd_event_source *s, int fd, uint32_t revents, void *userdata) {
+ unsigned *c = userdata;
+
+ assert_se(revents == EPOLLHUP);
+
+ (*c)++;
+ return 0;
+}
+
+TEST(leave_ratelimit) {
+ bool expect_ratelimit = false, manually_left_ratelimit = false;
+ _cleanup_(sd_event_source_unrefp) sd_event_source *s = NULL;
+ _cleanup_(sd_event_unrefp) sd_event *e = NULL;
+ _cleanup_(close_pairp) int pfd[2] = PIPE_EBADF;
+ unsigned c = 0;
+ int r;
+
+ assert_se(sd_event_default(&e) >= 0);
+
+ /* Create an event source that will continously fire by creating a pipe whose write side is closed,
+ * and which hence will only see EOF and constant EPOLLHUP */
+ assert_se(pipe2(pfd, O_CLOEXEC) >= 0);
+ assert_se(sd_event_add_io(e, &s, pfd[0], EPOLLIN, hup_callback, &c) >= 0);
+ assert_se(sd_event_source_set_io_fd_own(s, true) >= 0);
+ assert_se(sd_event_source_set_ratelimit(s, 5*USEC_PER_MINUTE, 5) >= 0);
+
+ pfd[0] = -EBADF;
+ pfd[1] = safe_close(pfd[1]); /* Trigger continous EOF */
+
+ for (;;) {
+ r = sd_event_prepare(e);
+ assert_se(r >= 0);
+
+ if (r == 0) {
+ r = sd_event_wait(e, UINT64_MAX);
+ assert_se(r > 0);
+ }
+
+ r = sd_event_dispatch(e);
+ assert_se(r > 0);
+
+ r = sd_event_source_is_ratelimited(s);
+ assert_se(r >= 0);
+
+ if (c < 5)
+ /* First four dispatches should just work */
+ assert_se(!r);
+ else if (c == 5) {
+ /* The fifth dispatch should still work, but we now expect the ratelimit to be hit subsequently */
+ if (!expect_ratelimit) {
+ assert_se(!r);
+ assert_se(sd_event_source_leave_ratelimit(s) == 0); /* this should be a NOP, and return 0 hence */
+ expect_ratelimit = true;
+ } else {
+ /* We expected the ratelimit, let's leave it manually, and verify it */
+ assert_se(r);
+ assert_se(sd_event_source_leave_ratelimit(s) > 0); /* we are ratelimited, hence should return > 0 */
+ assert_se(sd_event_source_is_ratelimited(s) == 0);
+
+ manually_left_ratelimit = true;
+ }
+
+ } else if (c == 6)
+ /* On the sixth iteration let's just exit */
+ break;
+ }
+
+ /* Verify we definitely hit the ratelimit and left it manually again */
+ assert_se(manually_left_ratelimit);
+}
+
DEFINE_TEST_MAIN(LOG_DEBUG);
diff --git a/src/systemd/sd-event.h b/src/systemd/sd-event.h
index 5ca8528895..49d6975967 100644
--- a/src/systemd/sd-event.h
+++ b/src/systemd/sd-event.h
@@ -173,6 +173,7 @@ int sd_event_source_set_ratelimit(sd_event_source *s, uint64_t interval_usec, un
int sd_event_source_get_ratelimit(sd_event_source *s, uint64_t *ret_interval_usec, unsigned *ret_burst);
int sd_event_source_is_ratelimited(sd_event_source *s);
int sd_event_source_set_ratelimit_expire_callback(sd_event_source *s, sd_event_handler_t callback);
+int sd_event_source_leave_ratelimit(sd_event_source *s);
int sd_event_trim_memory(void);