123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583 |
- /*
- * Copyright 2022-2024 The OpenSSL Project Authors. All Rights Reserved.
- *
- * Licensed under the Apache License 2.0 (the "License"). You may not use
- * this file except in compliance with the License. You can obtain a copy
- * in the file LICENSE in the source distribution or at
- * https://www.openssl.org/source/license.html
- */
- #include "internal/quic_tserver.h"
- #include "internal/quic_channel.h"
- #include "internal/quic_statm.h"
- #include "internal/quic_port.h"
- #include "internal/quic_engine.h"
- #include "internal/common.h"
- #include "internal/time.h"
- #include "quic_local.h"
- /*
- * QUIC Test Server Module
- * =======================
- */
- struct quic_tserver_st {
- QUIC_TSERVER_ARGS args;
- /* Dummy SSL object for this QUIC connection for use by msg_callback */
- SSL *ssl;
- /*
- * The QUIC engine, port and channel providing the core QUIC connection
- * implementation.
- */
- QUIC_ENGINE *engine;
- QUIC_PORT *port;
- QUIC_CHANNEL *ch;
- /* The mutex we give to the QUIC channel. */
- CRYPTO_MUTEX *mutex;
- /* SSL_CTX for creating the underlying TLS connection */
- SSL_CTX *ctx;
- /* SSL for the underlying TLS connection */
- SSL *tls;
- /* The current peer L4 address. AF_UNSPEC if we do not have a peer yet. */
- BIO_ADDR cur_peer_addr;
- /* Are we connected to a peer? */
- unsigned int connected : 1;
- };
- static int alpn_select_cb(SSL *ssl, const unsigned char **out,
- unsigned char *outlen, const unsigned char *in,
- unsigned int inlen, void *arg)
- {
- QUIC_TSERVER *srv = arg;
- static const unsigned char alpndeflt[] = {
- 8, 'o', 's', 's', 'l', 't', 'e', 's', 't'
- };
- const unsigned char *alpn;
- size_t alpnlen;
- if (srv->args.alpn == NULL) {
- alpn = alpndeflt;
- alpnlen = sizeof(alpn);
- } else {
- alpn = srv->args.alpn;
- alpnlen = srv->args.alpnlen;
- }
- if (SSL_select_next_proto((unsigned char **)out, outlen, alpn, alpnlen,
- in, inlen) != OPENSSL_NPN_NEGOTIATED)
- return SSL_TLSEXT_ERR_ALERT_FATAL;
- return SSL_TLSEXT_ERR_OK;
- }
- QUIC_TSERVER *ossl_quic_tserver_new(const QUIC_TSERVER_ARGS *args,
- const char *certfile, const char *keyfile)
- {
- QUIC_TSERVER *srv = NULL;
- QUIC_ENGINE_ARGS engine_args = {0};
- QUIC_PORT_ARGS port_args = {0};
- QUIC_CONNECTION *qc = NULL;
- if (args->net_rbio == NULL || args->net_wbio == NULL)
- goto err;
- if ((srv = OPENSSL_zalloc(sizeof(*srv))) == NULL)
- goto err;
- srv->args = *args;
- #if defined(OPENSSL_THREADS)
- if ((srv->mutex = ossl_crypto_mutex_new()) == NULL)
- goto err;
- #endif
- if (args->ctx != NULL)
- srv->ctx = args->ctx;
- else
- srv->ctx = SSL_CTX_new_ex(srv->args.libctx, srv->args.propq,
- TLS_method());
- if (srv->ctx == NULL)
- goto err;
- if (certfile != NULL
- && SSL_CTX_use_certificate_file(srv->ctx, certfile, SSL_FILETYPE_PEM) <= 0)
- goto err;
- if (keyfile != NULL
- && SSL_CTX_use_PrivateKey_file(srv->ctx, keyfile, SSL_FILETYPE_PEM) <= 0)
- goto err;
- SSL_CTX_set_alpn_select_cb(srv->ctx, alpn_select_cb, srv);
- srv->tls = SSL_new(srv->ctx);
- if (srv->tls == NULL)
- goto err;
- engine_args.libctx = srv->args.libctx;
- engine_args.propq = srv->args.propq;
- engine_args.mutex = srv->mutex;
- engine_args.now_cb = srv->args.now_cb;
- engine_args.now_cb_arg = srv->args.now_cb_arg;
- if ((srv->engine = ossl_quic_engine_new(&engine_args)) == NULL)
- goto err;
- port_args.channel_ctx = srv->ctx;
- port_args.is_multi_conn = 1;
- if ((srv->port = ossl_quic_engine_create_port(srv->engine, &port_args)) == NULL)
- goto err;
- if ((srv->ch = ossl_quic_port_create_incoming(srv->port, srv->tls)) == NULL)
- goto err;
- if (!ossl_quic_port_set_net_rbio(srv->port, srv->args.net_rbio)
- || !ossl_quic_port_set_net_wbio(srv->port, srv->args.net_wbio))
- goto err;
- qc = OPENSSL_zalloc(sizeof(*qc));
- if (qc == NULL)
- goto err;
- srv->ssl = (SSL *)qc;
- qc->ch = srv->ch;
- srv->ssl->type = SSL_TYPE_QUIC_CONNECTION;
- return srv;
- err:
- if (srv != NULL) {
- if (args->ctx == NULL)
- SSL_CTX_free(srv->ctx);
- SSL_free(srv->tls);
- ossl_quic_channel_free(srv->ch);
- ossl_quic_port_free(srv->port);
- ossl_quic_engine_free(srv->engine);
- #if defined(OPENSSL_THREADS)
- ossl_crypto_mutex_free(&srv->mutex);
- #endif
- OPENSSL_free(qc);
- }
- OPENSSL_free(srv);
- return NULL;
- }
- void ossl_quic_tserver_free(QUIC_TSERVER *srv)
- {
- if (srv == NULL)
- return;
- SSL_free(srv->tls);
- ossl_quic_channel_free(srv->ch);
- ossl_quic_port_free(srv->port);
- ossl_quic_engine_free(srv->engine);
- BIO_free_all(srv->args.net_rbio);
- BIO_free_all(srv->args.net_wbio);
- OPENSSL_free(srv->ssl);
- SSL_CTX_free(srv->ctx);
- #if defined(OPENSSL_THREADS)
- ossl_crypto_mutex_free(&srv->mutex);
- #endif
- OPENSSL_free(srv);
- }
- /* Set mutator callbacks for test framework support */
- int ossl_quic_tserver_set_plain_packet_mutator(QUIC_TSERVER *srv,
- ossl_mutate_packet_cb mutatecb,
- ossl_finish_mutate_cb finishmutatecb,
- void *mutatearg)
- {
- return ossl_quic_channel_set_mutator(srv->ch, mutatecb, finishmutatecb,
- mutatearg);
- }
- int ossl_quic_tserver_set_handshake_mutator(QUIC_TSERVER *srv,
- ossl_statem_mutate_handshake_cb mutate_handshake_cb,
- ossl_statem_finish_mutate_handshake_cb finish_mutate_handshake_cb,
- void *mutatearg)
- {
- return ossl_statem_set_mutator(ossl_quic_channel_get0_ssl(srv->ch),
- mutate_handshake_cb,
- finish_mutate_handshake_cb,
- mutatearg);
- }
- int ossl_quic_tserver_tick(QUIC_TSERVER *srv)
- {
- ossl_quic_reactor_tick(ossl_quic_channel_get_reactor(srv->ch), 0);
- if (ossl_quic_channel_is_active(srv->ch))
- srv->connected = 1;
- return 1;
- }
- int ossl_quic_tserver_is_connected(QUIC_TSERVER *srv)
- {
- return ossl_quic_channel_is_active(srv->ch);
- }
- /* Returns 1 if the server is in any terminating or terminated state */
- int ossl_quic_tserver_is_term_any(const QUIC_TSERVER *srv)
- {
- return ossl_quic_channel_is_term_any(srv->ch);
- }
- const QUIC_TERMINATE_CAUSE *
- ossl_quic_tserver_get_terminate_cause(const QUIC_TSERVER *srv)
- {
- return ossl_quic_channel_get_terminate_cause(srv->ch);
- }
- /* Returns 1 if the server is in a terminated state */
- int ossl_quic_tserver_is_terminated(const QUIC_TSERVER *srv)
- {
- return ossl_quic_channel_is_terminated(srv->ch);
- }
- int ossl_quic_tserver_is_handshake_confirmed(const QUIC_TSERVER *srv)
- {
- return ossl_quic_channel_is_handshake_confirmed(srv->ch);
- }
- int ossl_quic_tserver_read(QUIC_TSERVER *srv,
- uint64_t stream_id,
- unsigned char *buf,
- size_t buf_len,
- size_t *bytes_read)
- {
- int is_fin = 0;
- QUIC_STREAM *qs;
- qs = ossl_quic_stream_map_get_by_id(ossl_quic_channel_get_qsm(srv->ch),
- stream_id);
- if (qs == NULL) {
- int is_client_init
- = ((stream_id & QUIC_STREAM_INITIATOR_MASK)
- == QUIC_STREAM_INITIATOR_CLIENT);
- /*
- * A client-initiated stream might spontaneously come into existence, so
- * allow trying to read on a client-initiated stream before it exists,
- * assuming the connection is still active.
- * Otherwise, fail.
- */
- if (!is_client_init || !ossl_quic_channel_is_active(srv->ch))
- return 0;
- *bytes_read = 0;
- return 1;
- }
- if (qs->recv_state == QUIC_RSTREAM_STATE_DATA_READ
- || !ossl_quic_stream_has_recv_buffer(qs))
- return 0;
- if (!ossl_quic_rstream_read(qs->rstream, buf, buf_len,
- bytes_read, &is_fin))
- return 0;
- if (*bytes_read > 0) {
- /*
- * We have read at least one byte from the stream. Inform stream-level
- * RXFC of the retirement of controlled bytes. Update the active stream
- * status (the RXFC may now want to emit a frame granting more credit to
- * the peer).
- */
- OSSL_RTT_INFO rtt_info;
- ossl_statm_get_rtt_info(ossl_quic_channel_get_statm(srv->ch), &rtt_info);
- if (!ossl_quic_rxfc_on_retire(&qs->rxfc, *bytes_read,
- rtt_info.smoothed_rtt))
- return 0;
- }
- if (is_fin)
- ossl_quic_stream_map_notify_totally_read(ossl_quic_channel_get_qsm(srv->ch),
- qs);
- if (*bytes_read > 0)
- ossl_quic_stream_map_update_state(ossl_quic_channel_get_qsm(srv->ch), qs);
- return 1;
- }
- int ossl_quic_tserver_has_read_ended(QUIC_TSERVER *srv, uint64_t stream_id)
- {
- QUIC_STREAM *qs;
- unsigned char buf[1];
- size_t bytes_read = 0;
- int is_fin = 0;
- qs = ossl_quic_stream_map_get_by_id(ossl_quic_channel_get_qsm(srv->ch),
- stream_id);
- if (qs == NULL)
- return 0;
- if (qs->recv_state == QUIC_RSTREAM_STATE_DATA_READ)
- return 1;
- if (!ossl_quic_stream_has_recv_buffer(qs))
- return 0;
- /*
- * If we do not have the DATA_READ, it is possible we should still return 1
- * if there is a lone FIN (but no more data) remaining to be retired from
- * the RSTREAM, for example because ossl_quic_tserver_read() has not been
- * called since the FIN was received.
- */
- if (!ossl_quic_rstream_peek(qs->rstream, buf, sizeof(buf),
- &bytes_read, &is_fin))
- return 0;
- if (is_fin && bytes_read == 0) {
- /* If we have a FIN awaiting retirement and no data before it... */
- /* Let RSTREAM know we've consumed this FIN. */
- if (!ossl_quic_rstream_read(qs->rstream, buf, sizeof(buf),
- &bytes_read, &is_fin))
- return 0;
- assert(is_fin && bytes_read == 0);
- assert(qs->recv_state == QUIC_RSTREAM_STATE_DATA_RECVD);
- ossl_quic_stream_map_notify_totally_read(ossl_quic_channel_get_qsm(srv->ch),
- qs);
- ossl_quic_stream_map_update_state(ossl_quic_channel_get_qsm(srv->ch), qs);
- return 1;
- }
- return 0;
- }
- int ossl_quic_tserver_write(QUIC_TSERVER *srv,
- uint64_t stream_id,
- const unsigned char *buf,
- size_t buf_len,
- size_t *bytes_written)
- {
- QUIC_STREAM *qs;
- if (!ossl_quic_channel_is_active(srv->ch))
- return 0;
- qs = ossl_quic_stream_map_get_by_id(ossl_quic_channel_get_qsm(srv->ch),
- stream_id);
- if (qs == NULL || !ossl_quic_stream_has_send_buffer(qs))
- return 0;
- if (!ossl_quic_sstream_append(qs->sstream,
- buf, buf_len, bytes_written))
- return 0;
- if (*bytes_written > 0)
- /*
- * We have appended at least one byte to the stream. Potentially mark
- * the stream as active, depending on FC.
- */
- ossl_quic_stream_map_update_state(ossl_quic_channel_get_qsm(srv->ch), qs);
- /* Try and send. */
- ossl_quic_tserver_tick(srv);
- return 1;
- }
- int ossl_quic_tserver_conclude(QUIC_TSERVER *srv, uint64_t stream_id)
- {
- QUIC_STREAM *qs;
- if (!ossl_quic_channel_is_active(srv->ch))
- return 0;
- qs = ossl_quic_stream_map_get_by_id(ossl_quic_channel_get_qsm(srv->ch),
- stream_id);
- if (qs == NULL || !ossl_quic_stream_has_send_buffer(qs))
- return 0;
- if (!ossl_quic_sstream_get_final_size(qs->sstream, NULL)) {
- ossl_quic_sstream_fin(qs->sstream);
- ossl_quic_stream_map_update_state(ossl_quic_channel_get_qsm(srv->ch), qs);
- }
- ossl_quic_tserver_tick(srv);
- return 1;
- }
- int ossl_quic_tserver_stream_new(QUIC_TSERVER *srv,
- int is_uni,
- uint64_t *stream_id)
- {
- QUIC_STREAM *qs;
- if (!ossl_quic_channel_is_active(srv->ch))
- return 0;
- if ((qs = ossl_quic_channel_new_stream_local(srv->ch, is_uni)) == NULL)
- return 0;
- *stream_id = qs->id;
- return 1;
- }
- BIO *ossl_quic_tserver_get0_rbio(QUIC_TSERVER *srv)
- {
- return srv->args.net_rbio;
- }
- SSL_CTX *ossl_quic_tserver_get0_ssl_ctx(QUIC_TSERVER *srv)
- {
- return srv->ctx;
- }
- int ossl_quic_tserver_stream_has_peer_stop_sending(QUIC_TSERVER *srv,
- uint64_t stream_id,
- uint64_t *app_error_code)
- {
- QUIC_STREAM *qs;
- qs = ossl_quic_stream_map_get_by_id(ossl_quic_channel_get_qsm(srv->ch),
- stream_id);
- if (qs == NULL)
- return 0;
- if (qs->peer_stop_sending && app_error_code != NULL)
- *app_error_code = qs->peer_stop_sending_aec;
- return qs->peer_stop_sending;
- }
- int ossl_quic_tserver_stream_has_peer_reset_stream(QUIC_TSERVER *srv,
- uint64_t stream_id,
- uint64_t *app_error_code)
- {
- QUIC_STREAM *qs;
- qs = ossl_quic_stream_map_get_by_id(ossl_quic_channel_get_qsm(srv->ch),
- stream_id);
- if (qs == NULL)
- return 0;
- if (ossl_quic_stream_recv_is_reset(qs) && app_error_code != NULL)
- *app_error_code = qs->peer_reset_stream_aec;
- return ossl_quic_stream_recv_is_reset(qs);
- }
- int ossl_quic_tserver_set_new_local_cid(QUIC_TSERVER *srv,
- const QUIC_CONN_ID *conn_id)
- {
- /* Replace existing local connection ID in the QUIC_CHANNEL */
- return ossl_quic_channel_replace_local_cid(srv->ch, conn_id);
- }
- uint64_t ossl_quic_tserver_pop_incoming_stream(QUIC_TSERVER *srv)
- {
- QUIC_STREAM_MAP *qsm = ossl_quic_channel_get_qsm(srv->ch);
- QUIC_STREAM *qs = ossl_quic_stream_map_peek_accept_queue(qsm);
- if (qs == NULL)
- return UINT64_MAX;
- ossl_quic_stream_map_remove_from_accept_queue(qsm, qs, ossl_time_zero());
- return qs->id;
- }
- int ossl_quic_tserver_is_stream_totally_acked(QUIC_TSERVER *srv,
- uint64_t stream_id)
- {
- QUIC_STREAM *qs;
- qs = ossl_quic_stream_map_get_by_id(ossl_quic_channel_get_qsm(srv->ch),
- stream_id);
- if (qs == NULL)
- return 1;
- return ossl_quic_sstream_is_totally_acked(qs->sstream);
- }
- int ossl_quic_tserver_get_net_read_desired(QUIC_TSERVER *srv)
- {
- return ossl_quic_reactor_net_read_desired(
- ossl_quic_channel_get_reactor(srv->ch));
- }
- int ossl_quic_tserver_get_net_write_desired(QUIC_TSERVER *srv)
- {
- return ossl_quic_reactor_net_write_desired(
- ossl_quic_channel_get_reactor(srv->ch));
- }
- OSSL_TIME ossl_quic_tserver_get_deadline(QUIC_TSERVER *srv)
- {
- return ossl_quic_reactor_get_tick_deadline(
- ossl_quic_channel_get_reactor(srv->ch));
- }
- int ossl_quic_tserver_shutdown(QUIC_TSERVER *srv, uint64_t app_error_code)
- {
- ossl_quic_channel_local_close(srv->ch, app_error_code, NULL);
- /* TODO(QUIC SERVER): !SSL_SHUTDOWN_FLAG_NO_STREAM_FLUSH */
- if (ossl_quic_channel_is_terminated(srv->ch))
- return 1;
- ossl_quic_reactor_tick(ossl_quic_channel_get_reactor(srv->ch), 0);
- return ossl_quic_channel_is_terminated(srv->ch);
- }
- int ossl_quic_tserver_ping(QUIC_TSERVER *srv)
- {
- if (ossl_quic_channel_is_terminated(srv->ch))
- return 0;
- if (!ossl_quic_channel_ping(srv->ch))
- return 0;
- ossl_quic_reactor_tick(ossl_quic_channel_get_reactor(srv->ch), 0);
- return 1;
- }
- QUIC_CHANNEL *ossl_quic_tserver_get_channel(QUIC_TSERVER *srv)
- {
- return srv->ch;
- }
- void ossl_quic_tserver_set_msg_callback(QUIC_TSERVER *srv,
- void (*f)(int write_p, int version,
- int content_type,
- const void *buf, size_t len,
- SSL *ssl, void *arg),
- void *arg)
- {
- ossl_quic_channel_set_msg_callback(srv->ch, f, srv->ssl);
- ossl_quic_channel_set_msg_callback_arg(srv->ch, arg);
- SSL_set_msg_callback(srv->tls, f);
- SSL_set_msg_callback_arg(srv->tls, arg);
- }
- int ossl_quic_tserver_new_ticket(QUIC_TSERVER *srv)
- {
- return SSL_new_session_ticket(srv->tls);
- }
- int ossl_quic_tserver_set_max_early_data(QUIC_TSERVER *srv,
- uint32_t max_early_data)
- {
- return SSL_set_max_early_data(srv->tls, max_early_data);
- }
- void ossl_quic_tserver_set_psk_find_session_cb(QUIC_TSERVER *srv,
- SSL_psk_find_session_cb_func cb)
- {
- SSL_set_psk_find_session_callback(srv->tls, cb);
- }
|