LCOV - code coverage report
Current view: top level - waltz/quic/templ - fd_quic_transport_params.c (source / functions) Hit Total Coverage
Test: cov.lcov Lines: 103 117 88.0 %
Date: 2025-01-08 12:08:44 Functions: 5 5 100.0 %

          Line data    Source code
       1             : #include "../fd_quic_common.h"
       2             : 
       3             : #include "fd_quic_transport_params.h"
       4             : #include "fd_quic_parse_util.h"
       5             : 
       6             : #include <stdio.h>
       7             : 
       8             : void
       9           3 : fd_quic_dump_transport_param_desc( FILE * out ) {
      10           3 :   fprintf( out, "Transport parameter descriptions:\n" );
      11             : 
      12           3 : #define __( NAME, ID, TYPE, DFT, DESC, ... ) \
      13          51 :   fprintf( out, #ID " " #TYPE " " #NAME "\n" DESC "\n\n" );
      14             : 
      15          51 :   FD_QUIC_TRANSPORT_PARAMS( __, _ )
      16           3 : #undef __
      17           3 : }
      18             : 
      19             : 
      20             : #define FD_QUIC_PARSE_TP_VARINT(NAME) \
      21       96270 :   do {                                                 \
      22       96270 :     if( FD_UNLIKELY( sz==0    ) ) return -2;           \
      23       96270 :     uint width = 1u << ( (unsigned)buf[0] >> 6u );     \
      24       96270 :     if( FD_UNLIKELY( sz<width ) ) return -3;           \
      25       96270 :     ulong value = (ulong)( buf[0] & 0x3f );            \
      26      294798 :     for( ulong j=1; j<width; ++j ) {                   \
      27      198528 :       value= ( value<<8u ) + (ulong)buf[j];            \
      28      198528 :     }                                                  \
      29       96270 :     params->NAME = value;                              \
      30       96270 :     params->NAME##_present = 1;                        \
      31       96270 :   } while(0)
      32             : 
      33             : #define FD_QUIC_PARSE_TP_CONN_ID(NAME)                      \
      34       18339 :   do {                                                      \
      35       18339 :     if( FD_UNLIKELY( sz>sizeof(params->NAME) ) ) return -1; \
      36       18339 :     fd_memcpy( params->NAME, buf, sz );                     \
      37       18339 :     params->NAME##_len = (uchar)sz;                         \
      38       18339 :     params->NAME##_present = 1;                             \
      39       18339 :   } while(0)
      40             : 
      41             : #define FD_QUIC_PARSE_TP_ZERO_LENGTH(NAME) \
      42       12033 :   params->NAME##_present = 1;
      43             : 
      44             : #define FD_QUIC_PARSE_TP_TOKEN(NAME)                        \
      45           0 :   do {                                                      \
      46           0 :     if( FD_UNLIKELY( sz>sizeof(params->NAME) ) ) return -1; \
      47           0 :     fd_memcpy( params->NAME, buf, sz );                     \
      48           0 :     params->NAME##_len = (uchar)sz;                         \
      49           0 :     params->NAME##_present = 1;                             \
      50           0 :   } while(0)
      51             : 
      52             : #define FD_QUIC_PARSE_TP_PREFERRED_ADDRESS(NAME)            \
      53           0 :   do {                                                      \
      54           0 :     if( FD_UNLIKELY( sz>sizeof(params->NAME) ) ) return -1; \
      55           0 :     fd_memcpy( params->NAME, buf, sz );                     \
      56           0 :     params->NAME##_len = (uchar)sz;                         \
      57           0 :     params->NAME##_present = 1;                             \
      58           0 :   } while(0)
      59             : 
      60             : 
      61             : static int
      62             : fd_quic_decode_transport_param( fd_quic_transport_params_t * params,
      63             :                                 ulong                        id,
      64             :                                 uchar const *                buf,
      65      126651 :                                 ulong                        sz ) {
      66             :   // This compiles into a jump table, which is reasonably fast
      67      126651 :   switch( id ) {
      68           0 : #define __( NAME, ID, TYPE, DFT, DESC, ... ) \
      69      126642 :   case ID: { \
      70      126642 :       FD_QUIC_PARSE_TP_##TYPE(NAME); \
      71      126642 :       return 0; \
      72      114609 :     } \
      73           0 : 
      74      126651 :   FD_QUIC_TRANSPORT_PARAMS( __, _ )
      75      126651 : #undef __
      76             : 
      77      126651 :   }
      78             : 
      79           9 :   return 0; /* ignore unknown IDs */
      80      126651 : }
      81             : 
      82             : int
      83             : fd_quic_decode_transport_params( fd_quic_transport_params_t * params,
      84             :                                  uchar const *                buf,
      85       12042 :                                  ulong                        buf_sz ) {
      86      138693 :   while( buf_sz > 0 ) {
      87             :     /* upon success, this function adjusts buf and sz by bytes consumed */
      88      126654 :     ulong param_id = fd_quic_tp_parse_varint( &buf, &buf_sz );
      89      126654 :     ulong param_sz = fd_quic_tp_parse_varint( &buf, &buf_sz );
      90      126654 :     if( FD_UNLIKELY( param_sz > buf_sz ) ) return -1; /* length OOB */
      91             : 
      92      126651 :     int param_err = fd_quic_decode_transport_param( params, param_id, buf, param_sz );
      93      126651 :     if( FD_UNLIKELY( param_err ) ) return -1; /* parse failure */
      94             : 
      95             :     /* update buf and buf_sz */
      96      126651 :     buf    += param_sz;
      97      126651 :     buf_sz -= param_sz;
      98      126651 :   }
      99             : 
     100       12039 :   return 0; /* success */
     101       12042 : }
     102             : 
     103             : #define FD_QUIC_DUMP_TP_VARINT(NAME) \
     104          33 :   fprintf( out, "%lu", (ulong)params->NAME )
     105             : #define FD_QUIC_DUMP_TP_CONN_ID(NAME) \
     106          15 :   do { \
     107          15 :     ulong sz = params->NAME##_len; \
     108          15 :     fprintf( out, "len(%d) ", (int)sz ); \
     109          39 :     for( ulong j = 0; j < sz; ++j ) { \
     110          24 :       fprintf( out, "%2.2x ", (unsigned)params->NAME[j] ); \
     111          24 :     } \
     112          15 :   } while(0)
     113             : #define FD_QUIC_DUMP_TP_ZERO_LENGTH(NAME) \
     114           3 :   fprintf( out, "%u", (unsigned)params->NAME##_present )
     115           3 : #define FD_QUIC_DUMP_TP_TOKEN(NAME) FD_QUIC_DUMP_TP_CONN_ID(NAME)
     116           3 : #define FD_QUIC_DUMP_TP_PREFERRED_ADDRESS(NAME) FD_QUIC_DUMP_TP_CONN_ID(NAME)
     117             : 
     118             : void
     119           3 : fd_quic_dump_transport_params( fd_quic_transport_params_t const * params, FILE * out ) {
     120           3 :   fprintf( out, "Transport params:\n" );
     121           3 : #define __( NAME, ID, TYPE, DFT, DESC, ... ) \
     122          51 :   fprintf( out, "  %-50s: %s ", #NAME " (" #ID ")", params->NAME##_present ? "*" : " " ); \
     123          51 :   FD_QUIC_DUMP_TP_##TYPE(NAME); \
     124          51 :   fprintf( out, "\n" );
     125          51 :   FD_QUIC_TRANSPORT_PARAMS( __, _ )
     126           3 : #undef __
     127           3 : }
     128             : 
     129             : 
     130             : #define FD_QUIC_ENCODE_TP_VARINT(NAME,ID)                              \
     131       96333 :   do {                                                                 \
     132       96333 :     ulong val_len = fd_quic_varint_min_sz( params->NAME );             \
     133       96333 :     if( val_len == FD_QUIC_ENCODE_FAIL ) return FD_QUIC_ENCODE_FAIL;   \
     134       96333 :     FD_QUIC_ENCODE_VARINT( buf, buf_sz, ID );                          \
     135       96333 :     FD_QUIC_ENCODE_VARINT( buf, buf_sz, val_len );                     \
     136       96333 :     FD_QUIC_ENCODE_VARINT(buf,buf_sz,params->NAME);                    \
     137       96333 :   } while(0)
     138             : 
     139             : #define FD_QUIC_ENCODE_TP_CONN_ID(NAME,ID)                             \
     140       18069 :   do {                                                                 \
     141       18069 :     ulong val_len = params->NAME##_len;                                \
     142       18069 :     FD_QUIC_ENCODE_VARINT( buf, buf_sz, ID );                          \
     143       18069 :     FD_QUIC_ENCODE_VARINT( buf, buf_sz, val_len );                     \
     144       18069 :     if( val_len + 1 > buf_sz ) return FD_QUIC_ENCODE_FAIL;             \
     145      162912 :     for( ulong j = 0; j < val_len; ++j ) {                             \
     146      144843 :       buf[j] = params->NAME[j];                                        \
     147      144843 :     }                                                                  \
     148       18069 :     buf += val_len; buf_sz -= val_len;                                 \
     149       18069 :   } while(0);
     150             : 
     151             : #define FD_QUIC_ENCODE_TP_ZERO_LENGTH(NAME,ID)                         \
     152       12033 :   do {                                                                 \
     153       12033 :     if( params->NAME##_present ) {                                               \
     154       12033 :       FD_QUIC_ENCODE_VARINT( buf, buf_sz, ID );                        \
     155       12033 :       FD_QUIC_ENCODE_VARINT( buf, buf_sz, 0 );                         \
     156       12033 :     }                                                                  \
     157       12033 :   } while(0)
     158             : 
     159             : #define FD_QUIC_ENCODE_TP_TOKEN(NAME,ID) \
     160           3 :   FD_QUIC_ENCODE_TP_CONN_ID(NAME,ID)
     161             : 
     162             : #define FD_QUIC_ENCODE_TP_PREFERRED_ADDRESS(NAME,ID) \
     163           3 :   FD_QUIC_ENCODE_TP_CONN_ID(NAME,ID)
     164             : 
     165             : 
     166             : /* determine footprint of each type */
     167             : 
     168             : #define FD_QUIC_FOOTPRINT_FAIL ((ulong)1 << (ulong)62)
     169             : 
     170             : /* we need the length of the ID + the length of the length of the parameter value
     171             :    plus the length of the parameter value */
     172             : #define FD_QUIC_FOOTPRINT_TP_VARINT(NAME,ID)                           \
     173             :   (                                                                    \
     174             :     fd_quic_varint_min_sz( ID ) +                                      \
     175             :     fd_quic_varint_min_sz( fd_quic_varint_min_sz( params->NAME ) ) +   \
     176             :     fd_quic_varint_min_sz( params->NAME )                              \
     177             :   )
     178             : 
     179             : /* the length of a connection id is specified by *_len */
     180             : #define FD_QUIC_FOOTPRINT_TP_CONN_ID(NAME,ID)                          \
     181             :   (                                                                    \
     182             :     fd_quic_varint_min_sz( ID ) +                                      \
     183             :     fd_quic_varint_min_sz( params->NAME##_len ) +                      \
     184             :     params->NAME##_len                                                 \
     185             :   )
     186             : 
     187             : #define FD_QUIC_FOOTPRINT_TP_ZERO_LENGTH(NAME,ID)                      \
     188             :   (                                                                    \
     189             :     fd_quic_varint_min_sz( ID ) +                                      \
     190             :     fd_quic_varint_min_sz( 0 )                                         \
     191             :   )
     192             : 
     193             : #define FD_QUIC_FOOTPRINT_TP_TOKEN(NAME,ID) \
     194             :   FD_QUIC_FOOTPRINT_TP_CONN_ID(NAME,ID)
     195             : 
     196             : #define FD_QUIC_FOOTPRINT_TP_PREFERRED_ADDRESS(NAME,ID) \
     197             :   FD_QUIC_FOOTPRINT_TP_CONN_ID(NAME,ID)
     198             : 
     199             : 
     200             : #define FD_QUIC_VALIDATE_TP_TOKEN(NAME,ID) \
     201             :   FD_QUIC_VALIDATE_TP_CONN_ID(NAME,ID)
     202             : 
     203             : #define FD_QUIC_VALIDATE_TP_PREFERRED_ADDRESS(NAME,ID) \
     204             :   FD_QUIC_VALIDATE_TP_CONN_ID(NAME,ID)
     205             : 
     206             : 
     207             : // encode transport parameters into a buffer
     208             : // returns the number of bytes written
     209             : ulong
     210             : fd_quic_encode_transport_params( uchar *                            buf,
     211             :                                  ulong                              buf_sz,
     212       12039 :                                  fd_quic_transport_params_t const * params ) {
     213       12039 :   ulong orig_buf_sz = buf_sz;
     214       12039 : #define __( NAME, ID, TYPE, DFT, DESC, ... )                           \
     215      204663 :   if( params->NAME##_present ) {                                       \
     216      126435 :     FD_QUIC_ENCODE_TP_##TYPE(NAME,ID);                                 \
     217       18069 :   }
     218      204663 :   FD_QUIC_TRANSPORT_PARAMS( __, _ )
     219       12039 : #undef __
     220             : 
     221       12039 :   return orig_buf_sz - buf_sz;
     222      192624 : }

Generated by: LCOV version 1.14