LCOV - code coverage report
Current view: top level - waltz/tls - fd_tls_estate.h (source / functions) Hit Total Coverage
Test: cov.lcov Lines: 17 17 100.0 %
Date: 2025-01-08 12:08:44 Functions: 6 160 3.8 %

          Line data    Source code
       1             : #ifndef HEADER_fd_src_waltz_tls_fd_tls_estate_h
       2             : #define HEADER_fd_src_waltz_tls_fd_tls_estate_h
       3             : 
       4             : #include "../fd_waltz_base.h"
       5             : #include "../../ballet/sha256/fd_sha256.h"
       6             : 
       7             : /* Base ***************************************************************/
       8             : 
       9             : /* fd_tls_estate_base_t is the shared header of the
      10             :    fd_tls_estate_{srv,cli} objects. */
      11             : 
      12             : struct fd_tls_estate_base {
      13             :   uchar  state;
      14             :   uchar  server : 1;  /* 1 if server, 0 if client */
      15             :   ushort reason;      /* FD_TLS_REASON_{...} */
      16             : 
      17             :   /* Sadly required for SSLKEYLOGFILE */
      18             :   uchar client_random[ 32 ];
      19             : };
      20             : 
      21             : typedef struct fd_tls_estate_base fd_tls_estate_base_t;
      22             : 
      23             : /* The transcript is a running hash over all handshake messages.  The
      24             :    hash state depends on the current handshake progression.  The hash
      25             :    order is as follows:
      26             : 
      27             :      client   ClientHello           always
      28             :      server   ServerHello           always
      29             :      server   EncryptedExtensions   always
      30             :      server   CertificateRequest    optional
      31             :      server   Certificate           always
      32             :      server   CertificateVerify     always
      33             :      server   Finished              always
      34             :      client   Certificate           optional
      35             :      client   CertificateVerify     optional
      36             :      client   Finished              always */
      37             : 
      38             : struct fd_tls_transcript {
      39             :   uchar buf[ 64 ];  /* Pending SHA block */
      40             :   uint  sha[ 8 ];   /* Current internal SHA state */
      41             :   uint  len;        /* Number of bytes so far compressed into SHA state
      42             :                        plus number of bytes in pending in buf */
      43             : };
      44             : 
      45             : typedef struct fd_tls_transcript fd_tls_transcript_t;
      46             : 
      47             : /* Note:  An experimental memory optimization that is not implemented
      48             :    here is alignment of SHA state.  It might be possible to craft a
      49             :    transcript preimage that is aligned to SHA block size.  This allows
      50             :    omitting the SHA block buffer, saving 64 bytes per transcript (and
      51             :    thus per in-flight handshake). */
      52             : 
      53             : FD_PROTOTYPES_BEGIN
      54             : 
      55             : static inline void
      56             : fd_tls_transcript_store( fd_tls_transcript_t * script,
      57        6021 :                          fd_sha256_t const *   sha ) {
      58        6021 :   memcpy( script->buf, sha->buf,   64UL );
      59        6021 :   memcpy( script->sha, sha->state, 32UL );
      60        6021 :   script->len = (uint)( sha->bit_cnt / 8UL );
      61        6021 : }
      62             : 
      63             : static inline void
      64             : fd_tls_transcript_load( fd_tls_transcript_t const * script,
      65        6021 :                         fd_sha256_t *               sha ) {
      66        6021 :   memcpy( sha->buf,   script->buf, 64UL );
      67        6021 :   memcpy( sha->state, script->sha, 32UL );
      68        6021 :   sha->bit_cnt = (ulong)( script->len ) * 8UL;
      69        6021 :   sha->buf_used = (uint)( script->len % 64U );
      70        6021 : }
      71             : 
      72             : FD_PROTOTYPES_END
      73             : 
      74             : /* Server *************************************************************/
      75             : 
      76             : /* fd_tls_estate_srv contains compressed TLS server handshake state
      77             :    while waiting for the client to send a message.  estate is optimized
      78             :    for small memory use for incoming conns (128 byte per handshake).
      79             : 
      80             :    *** Security ********************************************************
      81             : 
      82             :    fd_tls servers must handle handshake requests from untrusted clients,
      83             :    and thus require hardening.  A typical lifetime of an estate object
      84             :    is in the order of ~3 seconds (conn timeout).  This requires some
      85             :    care to avoid memory exhaustion attacks.
      86             : 
      87             :    For example, a default OpenSSL app uses a handshake state size of
      88             :    about 10 KB (via multiple allocations on global heap).  Assuming 3
      89             :    second conn timeout, and flood rate of 1 million ClientHello msg/s,
      90             :    an attacker could indefinitely occupy ~30 GB of raw heap allocations!
      91             :    More concerning -- Global heap pressure will cause latent allocation
      92             :    failures in unrelated code, which might escalate to OOM kills.
      93             : 
      94             :    fd_tls_estate would only require 0.38 GB of memory for the same spam
      95             :    rate.  The memory of estate_srv objects is provided by the transport
      96             :    layer (e.g. QUIC or TLS record layer).  Each estate_srv object has
      97             :    static footprint.  This allows dense packing in custom memory arenas.
      98             : 
      99             :    This allows for robust handling of packet floods:  Even in the event
     100             :    the estate_srv arena is exhausted, the impact is limited to temporary
     101             :    inability to accept new connections.  Processing of established conns
     102             :    and unrelated code is unaffected.
     103             : 
     104             :    *** State Machine ***************************************************
     105             : 
     106             :    Each estate_srv object corresponds to an instance of a server-side
     107             :    TLS handshake state machine.  It is first instantiated when the
     108             :    client sent its ClientHello, the first message of a TLS handshake.
     109             : 
     110             :    Currently, only one external state exists:
     111             : 
     112             :      FD_TLS_HS_WAIT_FINISHED:  Processed ClientHello.
     113             : 
     114             :        At this point, the server has responded with all messages up to
     115             :        server Finished and is waiting for the client to respond with
     116             :        client Finished (and optionally, a certificate).
     117             : 
     118             :    State data contains:
     119             : 
     120             :    - The transcript hash state, which commits both sides to the entire
     121             :      sequence of handshake messages (such that they cannot be tampered
     122             :      with).
     123             : 
     124             :    - The client handshake secret, which is used to derive the "client
     125             :      Finished" verify data. */
     126             : 
     127             : struct fd_tls_estate_srv {
     128             :   /* TLS base (hs) handles are deliberately placed at the start.
     129             :    Allows for type punning between fd_quic_tls_hs_t and
     130             :    fd_tls_estate_{srv,cli}_t.  DO NOT MOVE.
     131             :    Type of handshake object depends on is_server. */
     132             :   fd_tls_estate_base_t base;
     133             : 
     134             :   uchar  server_cert_rpk : 1;  /* 0: X.509  1: raw public key */
     135             :   uchar  client_cert     : 1;  /* 0: no client auth  1: client cert */
     136             :   uchar  client_cert_rpk : 1;  /* 0: X.509  1: raw public key */
     137             :   uchar  hello_retry     : 1;
     138             : 
     139             :   fd_tls_transcript_t transcript;
     140             :   uchar               client_hs_secret[32];
     141             :   uchar               client_pubkey[32];
     142             : };
     143             : 
     144             : typedef struct fd_tls_estate_srv fd_tls_estate_srv_t;
     145             : 
     146             : 
     147             : /* Note:  When requesting a cert, the server should consider stopping
     148             :    after CertificateVerify to save compute resources (at the expense of
     149             :    more memory required to save state carried over from
     150             :    CertificateRequest...server Finished). */
     151             : 
     152             : FD_PROTOTYPES_BEGIN
     153             : 
     154             : /* fd_tls_estate_srv_new initializes an estate object for an incoming
     155             :    conn.  mem points to a memory region suitable for storing an
     156             :    fd_tls_estate_srv_t.  Returns cast of mem, which will be initialized
     157             :    to state FD_TLS_HS_START. */
     158             : 
     159             : fd_tls_estate_srv_t *
     160             : fd_tls_estate_srv_new( void * mem );
     161             : 
     162             : /* fd_tls_estate_srv_delete is currently a no-op. */
     163             : 
     164             : static inline void *
     165        6285 : fd_tls_estate_srv_delete( fd_tls_estate_srv_t * estate ) {
     166        6285 :   return (void *)estate;
     167        6285 : }
     168             : 
     169             : FD_PROTOTYPES_END
     170             : 
     171             : 
     172             : /* Client *************************************************************/
     173             : 
     174             : /* fd_tls_estate_cli contains TLS client handshake state while waiting
     175             :    for the server to send a message.
     176             : 
     177             :    TLS client side handshake state is considerably more complex than
     178             :    the server-side state.  Thus, estate_cli memory requirements are
     179             :    larger than for estate_srv.
     180             : 
     181             :    However, clients are not vulnerable to handshake packet floods:
     182             :    - The packet count per handshake is limited
     183             :    - Peers cannot initiate connections to clients (clients will simply
     184             :      drop unsolicited packets)
     185             : 
     186             :    Thus, estate_cli is not optimized for memory use. */
     187             : 
     188             : struct fd_tls_estate_cli {
     189             :   /* TLS handshake handles are deliberately placed at the start.
     190             :    Allows for type punning between fd_quic_tls_hs_t and
     191             :    fd_tls_estate_{srv,cli}_t.  DO NOT MOVE.
     192             :    Type of handshake object depends on is_server. */
     193             :   fd_tls_estate_base_t base;
     194             : 
     195             :   uchar server_pubkey   [ 32 ];
     196             :   uchar server_hs_secret[ 32 ];
     197             :   uchar client_hs_secret[ 32 ];
     198             :   uchar master_secret   [ 32 ];
     199             : 
     200             :   uchar client_cert        : 1;  /* 0=anon  1=client auth */
     201             :   uchar server_cert_rpk    : 1;
     202             :   uchar client_cert_nox509 : 1;
     203             :   uchar client_cert_rpk    : 1;
     204             :   uchar server_pubkey_pin  : 1;  /* if 1, require cert to match server_pubkey */
     205             : 
     206             :   fd_sha256_t transcript;
     207             : };
     208             : 
     209             : typedef struct fd_tls_estate_cli fd_tls_estate_cli_t;
     210             : 
     211             : FD_PROTOTYPES_BEGIN
     212             : 
     213             : fd_tls_estate_cli_t *
     214             : fd_tls_estate_cli_new( void * mem );
     215             : 
     216             : static inline void *
     217        6027 : fd_tls_estate_cli_delete( fd_tls_estate_cli_t * estate ) {
     218        6027 :   return (void *)estate;
     219        6027 : }
     220             : 
     221             : FD_PROTOTYPES_END
     222             : 
     223             : 
     224             : /* Common *************************************************************/
     225             : 
     226             : union fd_tls_estate {
     227             :   fd_tls_estate_base_t base;
     228             :   fd_tls_estate_srv_t  srv;
     229             :   fd_tls_estate_cli_t  cli;
     230             : };
     231             : 
     232             : typedef union fd_tls_estate fd_tls_estate_t;
     233             : 
     234             : #endif /* HEADER_fd_src_waltz_tls_fd_tls_estate_h */

Generated by: LCOV version 1.14