]> git.ipfire.org Git - thirdparty/haproxy.git/commitdiff
MINOR: quic: create accept queue for QUIC connections
authorAmaury Denoyelle <adenoyelle@haproxy.com>
Wed, 19 Jan 2022 14:46:11 +0000 (15:46 +0100)
committerAmaury Denoyelle <adenoyelle@haproxy.com>
Wed, 26 Jan 2022 15:13:51 +0000 (16:13 +0100)
Create a new type quic_accept_queue to handle QUIC connections accept.
A queue will be allocated for each thread. It contains a list of
listeners which contains at least one quic_conn ready to be accepted and
the tasklet to run listener_accept for these listeners.

include/haproxy/quic_sock-t.h [new file with mode: 0644]
include/haproxy/quic_sock.h
src/proto_quic.c
src/quic_sock.c

diff --git a/include/haproxy/quic_sock-t.h b/include/haproxy/quic_sock-t.h
new file mode 100644 (file)
index 0000000..5a591cc
--- /dev/null
@@ -0,0 +1,12 @@
+#ifndef _HAPROXY_QUIC_SOCK_T_H
+#define _HAPROXY_QUIC_SOCK_T_H
+#ifdef USE_QUIC
+
+/* QUIC connection accept queue. One per thread. */
+struct quic_accept_queue {
+       struct mt_list listeners; /* QUIC listeners with at least one connection ready to be accepted on this queue */
+       struct tasklet *tasklet;  /* task responsible to call listener_accept */
+};
+
+#endif /* USE_QUIC */
+#endif /* _HAPROXY_QUIC_SOCK_T_H */
index 652bc414b07f1eb2c53eae440ea1d2b9ea32d879..2a2818bc514b2740ddbd3bc876e72a01e36b60c8 100644 (file)
@@ -32,6 +32,7 @@
 #include <haproxy/api.h>
 #include <haproxy/connection-t.h>
 #include <haproxy/listener-t.h>
+#include <haproxy/quic_sock-t.h>
 
 int quic_session_accept(struct connection *cli_conn);
 int quic_sock_accepting_conn(const struct receiver *rx);
index 6fdb3f779cb1c0f0eeed0cec2fbafb716c8a7781..dc3d26219f644ed07826f08d3e6d8bf853e25e98 100644 (file)
@@ -525,6 +525,7 @@ static void quic_add_listener(struct protocol *proto, struct listener *listener)
        listener->rx.cids = EB_ROOT_UNIQUE;
        listener->rx.flags |= RX_F_LOCAL_ACCEPT;
        HA_RWLOCK_INIT(&listener->rx.cids_lock);
+
        default_add_listener(proto, listener);
 }
 
index bb593738e52450a4bd4381be4bbbb29b83b472aa..866eaa28784ca2067360db786d72d1ba7639fa76 100644 (file)
@@ -222,3 +222,67 @@ void quic_sock_fd_iocb(int fd)
  out:
        MT_LIST_APPEND(&l->rx.rxbuf_list, &rxbuf->mt_list);
 }
+
+
+/*********************** QUIC accept queue management ***********************/
+/* per-thread accept queues */
+struct quic_accept_queue *quic_accept_queues;
+
+/* Tasklet handler to accept QUIC connections. Call listener_accept on every
+ * listener instances registered in the accept queue.
+ */
+static struct task *quic_accept_run(struct task *t, void *ctx, unsigned int i)
+{
+       struct li_per_thread *lthr;
+       struct mt_list *elt1, elt2;
+       struct quic_accept_queue *queue = &quic_accept_queues[tid];
+
+       mt_list_for_each_entry_safe(lthr, &queue->listeners, quic_accept.list, elt1, elt2) {
+               listener_accept(lthr->li);
+               MT_LIST_DELETE_SAFE(elt1);
+       }
+
+       return NULL;
+}
+
+static int quic_alloc_accept_queues(void)
+{
+       int i;
+
+       quic_accept_queues = calloc(global.nbthread, sizeof(struct quic_accept_queue));
+       if (!quic_accept_queues) {
+               ha_alert("Failed to allocate the quic accept queues.\n");
+               return 0;
+       }
+
+       for (i = 0; i < global.nbthread; ++i) {
+               struct tasklet *task;
+               if (!(task = tasklet_new())) {
+                       ha_alert("Failed to allocate the quic accept queue on thread %d.\n", i);
+                       return 0;
+               }
+
+               tasklet_set_tid(task, i);
+               task->process = quic_accept_run;
+               quic_accept_queues[i].tasklet = task;
+
+               MT_LIST_INIT(&quic_accept_queues[i].listeners);
+       }
+
+       return 1;
+}
+REGISTER_POST_CHECK(quic_alloc_accept_queues);
+
+static int quic_deallocate_accept_queues(void)
+{
+       int i;
+
+       if (quic_accept_queues) {
+               for (i = 0; i < global.nbthread; ++i)
+                       tasklet_free(quic_accept_queues[i].tasklet);
+               free(quic_accept_queues);
+       }
+
+       return 1;
+}
+REGISTER_POST_DEINIT(quic_deallocate_accept_queues);