BUG/MINOR: quic: do not send CONNECTION_CLOSE_APP in initial/handshake
As specified by RFC 9000, it is forbidden to send a CONNECTION_CLOSE of type 0x1d (CONNECTION_CLOSE_APP) in an Initial or Handshake packet. It must be converted to type 0x1c (CONNECTION_CLOSE) with APPLICATION_ERROR code. CONNECTION_CLOSE_APP are generated by QUIC MUX interaction. Thus, special care must be taken when dealing with a 0-RTT packet, as this is the only case where the MUX can be instantiated and quic-conn still on the Initial or Handshake encryption level. To enforce RFC 9000, xprt build packet function is now responsible to translate a CONNECTION_CLOSE_APP if still on Initial/Handshake encryption. This process is done in a dedicated function named qc_build_cc_frm(). Without this patch, BUG_ON() statement in qc_build_frm() will be triggered when building a CONNECTION_CLOSE_APP frame on Initial or Handshake level. This is because QUIC_FT_CONNECTION_CLOSE_APP frame builder mask does not allow these encryption levels, as opposed to QUIC_FT_CONNECTION_CLOSE builder. This crash was reproduced by modifying the H3 layer to force emission of a CONNECTION_CLOSE_APP on first frame of a 0-RTT session. Note however that CONNECTION_CLOSE emission during Handshake is a complicated process for the server. For the moment, this is still incomplete on haproxy side. RFC 9000 requires to emit it multiple times in several packets under different encryption levels, depending on what we know about the client encryption context. This patch should be backported up to 2.6. (cherry picked from commit 0933c7b3c8d4c50303ef03ee7858b47ccac46477) Signed-off-by: Christopher Faulet <cfaulet@haproxy.com>
This commit is contained in:
parent
bf7f10efbd
commit
a495c12282
@ -5949,6 +5949,49 @@ static inline int qc_build_frms(struct list *outlist, struct list *inlist,
|
||||
return ret;
|
||||
}
|
||||
|
||||
/* Generate a CONNECTION_CLOSE frame for <qc> on <qel> encryption level. <out>
|
||||
* is used as return parameter and should be zero'ed by the caller.
|
||||
*/
|
||||
static void qc_build_cc_frm(struct quic_conn *qc, struct quic_enc_level *qel,
|
||||
struct quic_frame *out)
|
||||
{
|
||||
/* TODO improve CONNECTION_CLOSE on Initial/Handshake encryption levels
|
||||
*
|
||||
* A CONNECTION_CLOSE frame should be sent in several packets with
|
||||
* different encryption levels depending on the client context. This is
|
||||
* to ensure that the client can decrypt it. See RFC 9000 10.2.3 for
|
||||
* more details on how to implement it.
|
||||
*/
|
||||
|
||||
if (qc->flags & QUIC_FL_CONN_APP_ALERT) {
|
||||
if (unlikely(qel == &qc->els[QUIC_TLS_ENC_LEVEL_INITIAL] ||
|
||||
qel == &qc->els[QUIC_TLS_ENC_LEVEL_HANDSHAKE])) {
|
||||
/* RFC 9000 10.2.3. Immediate Close during the Handshake
|
||||
*
|
||||
* Sending a CONNECTION_CLOSE of type 0x1d in an Initial or Handshake
|
||||
* packet could expose application state or be used to alter application
|
||||
* state. A CONNECTION_CLOSE of type 0x1d MUST be replaced by a
|
||||
* CONNECTION_CLOSE of type 0x1c when sending the frame in Initial or
|
||||
* Handshake packets. Otherwise, information about the application
|
||||
* state might be revealed. Endpoints MUST clear the value of the
|
||||
* Reason Phrase field and SHOULD use the APPLICATION_ERROR code when
|
||||
* converting to a CONNECTION_CLOSE of type 0x1c.
|
||||
*/
|
||||
out->type = QUIC_FT_CONNECTION_CLOSE;
|
||||
out->connection_close.error_code = QC_ERR_APPLICATION_ERROR;
|
||||
out->connection_close.reason_phrase_len = 0;
|
||||
}
|
||||
else {
|
||||
out->type = QUIC_FT_CONNECTION_CLOSE_APP;
|
||||
out->connection_close.error_code = qc->err_code;
|
||||
}
|
||||
}
|
||||
else {
|
||||
out->type = QUIC_FT_CONNECTION_CLOSE;
|
||||
out->connection_close.error_code = qc->err_code;
|
||||
}
|
||||
}
|
||||
|
||||
/* This function builds a clear packet from <pkt> information (its type)
|
||||
* into a buffer with <pos> as position pointer and <qel> as QUIC TLS encryption
|
||||
* level for <conn> QUIC connection and <qel> as QUIC TLS encryption level,
|
||||
@ -6079,10 +6122,7 @@ static int qc_do_build_pkt(unsigned char *pos, const unsigned char *end,
|
||||
len += QUIC_TLS_TAG_LEN;
|
||||
/* CONNECTION_CLOSE frame */
|
||||
if (cc) {
|
||||
cc_frm.type = qc->flags & QUIC_FL_CONN_APP_ALERT ?
|
||||
QUIC_FT_CONNECTION_CLOSE_APP : QUIC_FT_CONNECTION_CLOSE;
|
||||
|
||||
cc_frm.connection_close.error_code = qc->err_code;
|
||||
qc_build_cc_frm(qc, qel, &cc_frm);
|
||||
len += qc_frm_len(&cc_frm);
|
||||
}
|
||||
add_ping_frm = 0;
|
||||
|
Loading…
x
Reference in New Issue
Block a user