123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536 |
- #include "internal/quic_demux.h"
- #include "internal/quic_wire_pkt.h"
- #include "internal/common.h"
- #include <openssl/lhash.h>
- #define DEMUX_MAX_MSGS_PER_CALL 32
- void ossl_quic_urxe_remove(QUIC_URXE_LIST *l, QUIC_URXE *e)
- {
- /* Must be in list currently. */
- assert((e->prev != NULL || l->head == e)
- && (e->next != NULL || l->tail == e));
- if (e->prev != NULL)
- e->prev->next = e->next;
- if (e->next != NULL)
- e->next->prev = e->prev;
- if (e == l->head)
- l->head = e->next;
- if (e == l->tail)
- l->tail = e->prev;
- e->next = e->prev = NULL;
- }
- void ossl_quic_urxe_insert_head(QUIC_URXE_LIST *l, QUIC_URXE *e)
- {
- /* Must not be in list. */
- assert(e->prev == NULL && e->next == NULL);
- if (l->head == NULL) {
- l->head = l->tail = e;
- e->next = e->prev = NULL;
- return;
- }
- l->head->prev = e;
- e->next = l->head;
- e->prev = NULL;
- l->head = e;
- }
- void ossl_quic_urxe_insert_tail(QUIC_URXE_LIST *l, QUIC_URXE *e)
- {
- /* Must not be in list. */
- assert(e->prev == NULL && e->next == NULL);
- if (l->tail == NULL) {
- l->head = l->tail = e;
- e->next = e->prev = NULL;
- return;
- }
- l->tail->next = e;
- e->prev = l->tail;
- e->next = NULL;
- l->tail = e;
- }
- /* Structure used to track a given connection ID. */
- typedef struct quic_demux_conn_st QUIC_DEMUX_CONN;
- struct quic_demux_conn_st {
- QUIC_DEMUX_CONN *next; /* used when unregistering only */
- QUIC_CONN_ID dst_conn_id;
- ossl_quic_demux_cb_fn *cb;
- void *cb_arg;
- };
- DEFINE_LHASH_OF_EX(QUIC_DEMUX_CONN);
- static unsigned long demux_conn_hash(const QUIC_DEMUX_CONN *conn)
- {
- size_t i;
- unsigned long v = 0;
- assert(conn->dst_conn_id.id_len <= QUIC_MAX_CONN_ID_LEN);
- for (i = 0; i < conn->dst_conn_id.id_len; ++i)
- v ^= ((unsigned long)conn->dst_conn_id.id[i])
- << ((i * 8) % (sizeof(unsigned long) * 8));
- return v;
- }
- static int demux_conn_cmp(const QUIC_DEMUX_CONN *a, const QUIC_DEMUX_CONN *b)
- {
- return !ossl_quic_conn_id_eq(&a->dst_conn_id, &b->dst_conn_id);
- }
- struct quic_demux_st {
- /* The underlying transport BIO with datagram semantics. */
- BIO *net_bio;
- /*
- * QUIC short packets do not contain the length of the connection ID field,
- * therefore it must be known contextually. The demuxer requires connection
- * IDs of the same length to be used for all incoming packets.
- */
- size_t short_conn_id_len;
- /* Default URXE buffer size in bytes. */
- size_t default_urxe_alloc_len;
- /* Time retrieval callback. */
- OSSL_TIME (*now)(void *arg);
- void *now_arg;
- /* Hashtable mapping connection IDs to QUIC_DEMUX_CONN structures. */
- LHASH_OF(QUIC_DEMUX_CONN) *conns_by_id;
- /*
- * List of URXEs which are not currently in use (i.e., not filled with
- * unconsumed data). These are moved to the pending list as they are filled.
- */
- QUIC_URXE_LIST urx_free;
- size_t num_urx_free;
- /*
- * List of URXEs which are filled with received encrypted data. These are
- * removed from this list as we invoke the callbacks for each of them. They
- * are then not on any list managed by us; we forget about them until our
- * user calls ossl_quic_demux_release_urxe to return the URXE to us, at
- * which point we add it to the free list.
- */
- QUIC_URXE_LIST urx_pending;
- /* Whether to use local address support. */
- char use_local_addr;
- };
- QUIC_DEMUX *ossl_quic_demux_new(BIO *net_bio,
- size_t short_conn_id_len,
- size_t default_urxe_alloc_len,
- OSSL_TIME (*now)(void *arg),
- void *now_arg)
- {
- QUIC_DEMUX *demux;
- demux = OPENSSL_zalloc(sizeof(QUIC_DEMUX));
- if (demux == NULL)
- return NULL;
- demux->net_bio = net_bio;
- demux->short_conn_id_len = short_conn_id_len;
- demux->default_urxe_alloc_len = default_urxe_alloc_len;
- demux->now = now;
- demux->now_arg = now_arg;
- demux->conns_by_id
- = lh_QUIC_DEMUX_CONN_new(demux_conn_hash, demux_conn_cmp);
- if (demux->conns_by_id == NULL) {
- OPENSSL_free(demux);
- return NULL;
- }
- if (net_bio != NULL
- && BIO_dgram_get_local_addr_cap(net_bio)
- && BIO_dgram_set_local_addr_enable(net_bio, 1))
- demux->use_local_addr = 1;
- return demux;
- }
- static void demux_free_conn_it(QUIC_DEMUX_CONN *conn, void *arg)
- {
- OPENSSL_free(conn);
- }
- static void demux_free_urxl(QUIC_URXE_LIST *l)
- {
- QUIC_URXE *e, *enext;
- for (e = l->head; e != NULL; e = enext) {
- enext = e->next;
- OPENSSL_free(e);
- }
- l->head = l->tail = NULL;
- }
- void ossl_quic_demux_free(QUIC_DEMUX *demux)
- {
- if (demux == NULL)
- return;
- /* Free all connection structures. */
- lh_QUIC_DEMUX_CONN_doall_arg(demux->conns_by_id, demux_free_conn_it, NULL);
- lh_QUIC_DEMUX_CONN_free(demux->conns_by_id);
- /* Free all URXEs we are holding. */
- demux_free_urxl(&demux->urx_free);
- demux_free_urxl(&demux->urx_pending);
- OPENSSL_free(demux);
- }
- static QUIC_DEMUX_CONN *demux_get_by_conn_id(QUIC_DEMUX *demux,
- const QUIC_CONN_ID *dst_conn_id)
- {
- QUIC_DEMUX_CONN key;
- if (dst_conn_id->id_len > QUIC_MAX_CONN_ID_LEN)
- return NULL;
- key.dst_conn_id = *dst_conn_id;
- return lh_QUIC_DEMUX_CONN_retrieve(demux->conns_by_id, &key);
- }
- int ossl_quic_demux_register(QUIC_DEMUX *demux,
- const QUIC_CONN_ID *dst_conn_id,
- ossl_quic_demux_cb_fn *cb, void *cb_arg)
- {
- QUIC_DEMUX_CONN *conn;
- if (dst_conn_id == NULL
- || dst_conn_id->id_len > QUIC_MAX_CONN_ID_LEN
- || cb == NULL)
- return 0;
- /* Ensure not already registered. */
- if (demux_get_by_conn_id(demux, dst_conn_id) != NULL)
- /* Handler already registered with this connection ID. */
- return 0;
- conn = OPENSSL_zalloc(sizeof(QUIC_DEMUX_CONN));
- if (conn == NULL)
- return 0;
- conn->dst_conn_id = *dst_conn_id;
- conn->cb = cb;
- conn->cb_arg = cb_arg;
- lh_QUIC_DEMUX_CONN_insert(demux->conns_by_id, conn);
- return 1;
- }
- static void demux_unregister(QUIC_DEMUX *demux,
- QUIC_DEMUX_CONN *conn)
- {
- lh_QUIC_DEMUX_CONN_delete(demux->conns_by_id, conn);
- OPENSSL_free(conn);
- }
- int ossl_quic_demux_unregister(QUIC_DEMUX *demux,
- const QUIC_CONN_ID *dst_conn_id)
- {
- QUIC_DEMUX_CONN *conn;
- if (dst_conn_id == NULL
- || dst_conn_id->id_len > QUIC_MAX_CONN_ID_LEN)
- return 0;
- conn = demux_get_by_conn_id(demux, dst_conn_id);
- if (conn == NULL)
- return 0;
- demux_unregister(demux, conn);
- return 1;
- }
- struct unreg_arg {
- ossl_quic_demux_cb_fn *cb;
- void *cb_arg;
- QUIC_DEMUX_CONN *head;
- };
- static void demux_unregister_by_cb(QUIC_DEMUX_CONN *conn, void *arg_)
- {
- struct unreg_arg *arg = arg_;
- if (conn->cb == arg->cb && conn->cb_arg == arg->cb_arg) {
- conn->next = arg->head;
- arg->head = conn;
- }
- }
- void ossl_quic_demux_unregister_by_cb(QUIC_DEMUX *demux,
- ossl_quic_demux_cb_fn *cb,
- void *cb_arg)
- {
- QUIC_DEMUX_CONN *conn, *cnext;
- struct unreg_arg arg = {0};
- arg.cb = cb;
- arg.cb_arg = cb_arg;
- lh_QUIC_DEMUX_CONN_doall_arg(demux->conns_by_id,
- demux_unregister_by_cb, &arg);
- for (conn = arg.head; conn != NULL; conn = cnext) {
- cnext = conn->next;
- demux_unregister(demux, conn);
- }
- }
- static QUIC_URXE *demux_alloc_urxe(size_t alloc_len)
- {
- QUIC_URXE *e;
- if (alloc_len >= SIZE_MAX - sizeof(QUIC_URXE))
- return NULL;
- e = OPENSSL_malloc(sizeof(QUIC_URXE) + alloc_len);
- if (e == NULL)
- return NULL;
- e->prev = e->next = NULL;
- e->alloc_len = alloc_len;
- e->data_len = 0;
- return e;
- }
- static int demux_ensure_free_urxe(QUIC_DEMUX *demux, size_t min_num_free)
- {
- QUIC_URXE *e;
- while (demux->num_urx_free < min_num_free) {
- e = demux_alloc_urxe(demux->default_urxe_alloc_len);
- if (e == NULL)
- return 0;
- ossl_quic_urxe_insert_tail(&demux->urx_free, e);
- ++demux->num_urx_free;
- }
- return 1;
- }
- /*
- * Receive datagrams from network, placing them into URXEs.
- *
- * Returns 1 on success or 0 on failure.
- *
- * Precondition: at least one URXE is free
- * Precondition: there are no pending URXEs
- */
- static int demux_recv(QUIC_DEMUX *demux)
- {
- BIO_MSG msg[DEMUX_MAX_MSGS_PER_CALL];
- size_t rd, i;
- QUIC_URXE *urxe = demux->urx_free.head, *unext;
- OSSL_TIME now;
- /* This should never be called when we have any pending URXE. */
- assert(demux->urx_pending.head == NULL);
- if (demux->net_bio == NULL)
- return 0;
- /*
- * Opportunistically receive as many messages as possible in a single
- * syscall, determined by how many free URXEs are available.
- */
- for (i = 0; i < (ossl_ssize_t)OSSL_NELEM(msg); ++i, urxe = urxe->next) {
- if (urxe == NULL) {
- /* We need at least one URXE to receive into. */
- if (!ossl_assert(i > 0))
- return 0;
- break;
- }
- /* Ensure we zero any fields added to BIO_MSG at a later date. */
- memset(&msg[i], 0, sizeof(BIO_MSG));
- msg[i].data = ossl_quic_urxe_data(urxe);
- msg[i].data_len = urxe->alloc_len;
- msg[i].peer = &urxe->peer;
- if (demux->use_local_addr)
- msg[i].local = &urxe->local;
- else
- BIO_ADDR_clear(&urxe->local);
- }
- if (!BIO_recvmmsg(demux->net_bio, msg, sizeof(BIO_MSG), i, 0, &rd))
- return 0;
- now = demux->now != NULL ? demux->now(demux->now_arg) : ossl_time_zero();
- urxe = demux->urx_free.head;
- for (i = 0; i < rd; ++i, urxe = unext) {
- unext = urxe->next;
- /* Set URXE with actual length of received datagram. */
- urxe->data_len = msg[i].data_len;
- /* Time we received datagram. */
- urxe->time = now;
- /* Move from free list to pending list. */
- ossl_quic_urxe_remove(&demux->urx_free, urxe);
- --demux->num_urx_free;
- ossl_quic_urxe_insert_tail(&demux->urx_pending, urxe);
- }
- return 1;
- }
- /* Extract destination connection ID from the first packet in a datagram. */
- static int demux_identify_conn_id(QUIC_DEMUX *demux,
- QUIC_URXE *e,
- QUIC_CONN_ID *dst_conn_id)
- {
- return ossl_quic_wire_get_pkt_hdr_dst_conn_id(ossl_quic_urxe_data(e),
- e->data_len,
- demux->short_conn_id_len,
- dst_conn_id);
- }
- /* Identify the connection structure corresponding to a given URXE. */
- static QUIC_DEMUX_CONN *demux_identify_conn(QUIC_DEMUX *demux, QUIC_URXE *e)
- {
- QUIC_CONN_ID dst_conn_id;
- if (!demux_identify_conn_id(demux, e, &dst_conn_id))
- /*
- * Datagram is so badly malformed we can't get the DCID from the first
- * packet in it, so just give up.
- */
- return NULL;
- return demux_get_by_conn_id(demux, &dst_conn_id);
- }
- /* Process a single pending URXE. */
- static int demux_process_pending_urxe(QUIC_DEMUX *demux, QUIC_URXE *e)
- {
- QUIC_DEMUX_CONN *conn;
- /* The next URXE we process should be at the head of the pending list. */
- if (!ossl_assert(e == demux->urx_pending.head))
- return 0;
- conn = demux_identify_conn(demux, e);
- if (conn == NULL) {
- /*
- * We could not identify a connection. We will never be able to process
- * this datagram, so get rid of it.
- */
- ossl_quic_urxe_remove(&demux->urx_pending, e);
- ossl_quic_urxe_insert_tail(&demux->urx_free, e);
- ++demux->num_urx_free;
- return 1; /* keep processing pending URXEs */
- }
- /*
- * Remove from list and invoke callback. The URXE now belongs to the
- * callback. (QUIC_DEMUX_CONN never has non-NULL cb.)
- */
- ossl_quic_urxe_remove(&demux->urx_pending, e);
- conn->cb(e, conn->cb_arg);
- return 1;
- }
- /* Process pending URXEs to generate callbacks. */
- static int demux_process_pending_urxl(QUIC_DEMUX *demux)
- {
- QUIC_URXE *e;
- while ((e = demux->urx_pending.head) != NULL)
- if (!demux_process_pending_urxe(demux, e))
- return 0;
- return 1;
- }
- /*
- * Drain the pending URXE list, processing any pending URXEs by making their
- * callbacks. If no URXEs are pending, a network read is attempted first.
- */
- int ossl_quic_demux_pump(QUIC_DEMUX *demux)
- {
- int ret;
- if (demux->urx_pending.head == NULL) {
- ret = demux_ensure_free_urxe(demux, DEMUX_MAX_MSGS_PER_CALL);
- if (ret != 1)
- return 0;
- ret = demux_recv(demux);
- if (ret != 1)
- return 0;
- /*
- * If demux_recv returned successfully, we should always have something.
- */
- assert(demux->urx_pending.head != NULL);
- }
- return demux_process_pending_urxl(demux);
- }
- /* Artificially inject a packet into the demuxer for testing purposes. */
- int ossl_quic_demux_inject(QUIC_DEMUX *demux,
- const unsigned char *buf,
- size_t buf_len,
- const BIO_ADDR *peer,
- const BIO_ADDR *local)
- {
- int ret;
- QUIC_URXE *urxe;
- ret = demux_ensure_free_urxe(demux, 1);
- if (ret != 1)
- return 0;
- urxe = demux->urx_free.head;
- if (buf_len > urxe->alloc_len)
- return 0;
- memcpy(ossl_quic_urxe_data(urxe), buf, buf_len);
- urxe->data_len = buf_len;
- if (peer != NULL)
- urxe->peer = *peer;
- else
- BIO_ADDR_clear(&urxe->local);
- if (local != NULL)
- urxe->local = *local;
- else
- BIO_ADDR_clear(&urxe->local);
- /* Move from free list to pending list. */
- ossl_quic_urxe_remove(&demux->urx_free, urxe);
- --demux->num_urx_free;
- ossl_quic_urxe_insert_tail(&demux->urx_pending, urxe);
- return demux_process_pending_urxl(demux);
- }
- /* Called by our user to return a URXE to the free list. */
- void ossl_quic_demux_release_urxe(QUIC_DEMUX *demux,
- QUIC_URXE *e)
- {
- assert(e->prev == NULL && e->next == NULL);
- ossl_quic_urxe_insert_tail(&demux->urx_free, e);
- ++demux->num_urx_free;
- }
|