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 */