haproxy/src/quic_fctl.c
Amaury Denoyelle 25493ca036 MINOR: mux-quic: define a flow control related type
Create a new module dedicated to flow control handling. It will be used
to implement earlier flow control update on snd_buf stream callback.

For the moment, only Tx part is implemented (i.e. limit set by the peer
that haproxy must respect for sending). A type quic_fctl is defined to
count emitted data bytes. Two offsets are used : a real one and a soft
one. The difference is that soft offset can be incremented beyond limit
unless it is already in excess.

Soft offset will be used for HTX to H3 parsing. As size of generated H3
is unknown before parsing, it allows to surpass the limit one time. Real
offset will be used during STREAM frame generation : this time the limit
must not be exceeded to prevent protocol violation.
2024-01-31 16:28:54 +01:00

97 lines
2.6 KiB
C

#include <haproxy/quic_fctl.h>
#include <haproxy/api.h>
void qfctl_init(struct quic_fctl *fctl, uint64_t limit)
{
fctl->limit = limit;
fctl->off_real = 0;
fctl->off_soft = 0;
}
/* Returns true if real limit is blocked for <fctl> flow control instance.
* This happens if it is equal than current max value.
*/
int qfctl_rblocked(const struct quic_fctl *fctl)
{
/* Real limit must never be exceeded. */
BUG_ON(fctl->off_real > fctl->limit);
return fctl->off_real == fctl->limit;
}
/* Returns true if soft limit is blocked for <fctl> flow control instance.
* This happens if it is equal or greater than current max value.
*/
int qfctl_sblocked(const struct quic_fctl *fctl)
{
return fctl->off_soft >= fctl->limit;
}
/* Set a new <val> maximum value for <fctl> flow control instance. If current
* offset is already equal or more, the new value is ignored. Additionally,
* <unblocked_soft> and <unblocked_real> can be used as output parameters to
* detect if the current update result in one or both of these offsets to be
* unblocked.
*
* Returns true if max is incremented else false.
*/
int qfctl_set_max(struct quic_fctl *fctl, uint64_t val,
int *out_unblock_soft, int *out_unblock_real)
{
int unblock_soft = 0, unblock_real = 0;
int ret = 0;
if (fctl->limit < val) {
if (fctl->off_soft >= fctl->limit && fctl->off_soft < val)
unblock_soft = 1;
if (fctl->off_real == fctl->limit && fctl->off_real < val)
unblock_real = 1;
fctl->limit = val;
ret = 1;
}
if (out_unblock_soft)
*out_unblock_soft = unblock_soft;
if (out_unblock_real)
*out_unblock_real = unblock_real;
return ret;
}
/* Increment real offset of <fctl> flow control instance by <diff>. This cannot
* exceed <fctl> limit.
*
* Returns true if limit is reached after increment.
*/
int qfctl_rinc(struct quic_fctl *fctl, uint64_t diff)
{
/* Real limit must never be exceeded. */
BUG_ON(fctl->off_real + diff > fctl->limit);
fctl->off_real += diff;
return fctl->off_real == fctl->limit;
}
/* Increment soft offset of <fctl> flow control instance by <diff>. This cannot
* be done if <fctl> limit was already reached.
*
* Returns true if limit is reached after increment.
*/
int qfctl_sinc(struct quic_fctl *fctl, uint64_t diff)
{
/* Soft limit must not be incremented if already in excess. */
BUG_ON(qfctl_sblocked(fctl));
fctl->off_soft += diff;
return fctl->off_soft >= fctl->limit;
}
/* Return the remaining offset before reaching <fctl> limit. */
uint64_t qfctl_rcap(const struct quic_fctl *fctl)
{
/* Real limit must never be exceeded. */
BUG_ON(fctl->off_real > fctl->limit);
return fctl->limit - fctl->off_real;
}