Coverage Report

Created: 2026-05-30 06:06

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/curl/lib/vtls/vtls_scache.c
Line
Count
Source
1
/***************************************************************************
2
 *                                  _   _ ____  _
3
 *  Project                     ___| | | |  _ \| |
4
 *                             / __| | | | |_) | |
5
 *                            | (__| |_| |  _ <| |___
6
 *                             \___|\___/|_| \_\_____|
7
 *
8
 * Copyright (C) Daniel Stenberg, <daniel@haxx.se>, et al.
9
 *
10
 * This software is licensed as described in the file COPYING, which
11
 * you should have received as part of this distribution. The terms
12
 * are also available at https://curl.se/docs/copyright.html.
13
 *
14
 * You may opt to use, copy, modify, merge, publish, distribute and/or sell
15
 * copies of the Software, and permit persons to whom the Software is
16
 * furnished to do so, under the terms of the COPYING file.
17
 *
18
 * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
19
 * KIND, either express or implied.
20
 *
21
 * SPDX-License-Identifier: curl
22
 *
23
 ***************************************************************************/
24
#include "curl_setup.h"
25
26
#ifdef USE_SSL
27
28
#ifdef HAVE_SYS_TYPES_H
29
#include <sys/types.h>
30
#endif
31
32
#include "urldata.h"
33
#include "cfilters.h"
34
35
#include "vtls/vtls.h" /* generic SSL protos etc */
36
#include "vtls/vtls_int.h"
37
#include "vtls/vtls_scache.h"
38
#include "vtls/vtls_spack.h"
39
40
#include "strcase.h"
41
#include "url.h"
42
#include "llist.h"
43
#include "curl_share.h"
44
#include "curl_trc.h"
45
#include "curl_sha256.h"
46
#include "rand.h"
47
48
49
/* a peer+tls-config we cache sessions for */
50
struct Curl_ssl_scache_peer {
51
  char *ssl_peer_key;      /* id for peer + relevant TLS configuration */
52
  char *clientcert;
53
  char *key_passwd;
54
  char *srp_username;
55
  char *srp_password;
56
  struct Curl_llist sessions;
57
  void *sobj;              /* object instance or NULL */
58
  Curl_ssl_scache_obj_dtor *sobj_free; /* free `sobj` callback */
59
  unsigned char key_salt[CURL_SHA256_DIGEST_LENGTH]; /* for entry export */
60
  unsigned char key_hmac[CURL_SHA256_DIGEST_LENGTH]; /* for entry export */
61
  size_t max_sessions;
62
  long age;                /* a number, the higher the more recent */
63
  BIT(hmac_set);           /* if key_salt and key_hmac are present */
64
  BIT(exportable);         /* sessions for this peer can be exported */
65
};
66
67
0
#define CURL_SCACHE_MAGIC 0x000e1551
68
69
0
#define GOOD_SCACHE(x) ((x) && (x)->magic == CURL_SCACHE_MAGIC)
70
71
static CURLcode cf_ssl_peer_key_add_path(struct dynbuf *buf,
72
                                         const char *name,
73
                                         const char *path,
74
                                         bool *is_local)
75
0
{
76
0
  if(path && path[0]) {
77
    /* We try to add absolute paths, so that the session key can stay valid
78
     * when used in another process with different CWD. When a path does not
79
     * exist, this does not work. Then, we add the path as is. */
80
#ifdef _WIN32
81
    char abspath[_MAX_PATH];
82
    if(_fullpath(abspath, path, _MAX_PATH))
83
      return curlx_dyn_addf(buf, ":%s-%s", name, abspath);
84
    *is_local = TRUE;
85
#elif defined(HAVE_REALPATH)
86
0
    if(path[0] != '/') {
87
0
      char *abspath = realpath(path, NULL);
88
0
      if(abspath) {
89
0
        CURLcode result = curlx_dyn_addf(buf, ":%s-%s", name, abspath);
90
        /* !checksrc! disable BANNEDFUNC 1 */
91
0
        free(abspath); /* allocated by libc, free without memdebug */
92
0
        return result;
93
0
      }
94
0
      *is_local = TRUE;
95
0
    }
96
0
#endif
97
0
    return curlx_dyn_addf(buf, ":%s-%s", name, path);
98
0
  }
99
0
  return CURLE_OK;
100
0
}
101
102
static CURLcode cf_ssl_peer_key_add_hash(struct dynbuf *buf,
103
                                         const char *name,
104
                                         struct curl_blob *blob)
105
0
{
106
0
  CURLcode result = CURLE_OK;
107
0
  if(blob && blob->len) {
108
0
    unsigned char hash[CURL_SHA256_DIGEST_LENGTH];
109
0
    size_t i;
110
111
0
    result = curlx_dyn_addf(buf, ":%s-", name);
112
0
    if(result)
113
0
      goto out;
114
0
    result = Curl_sha256it(hash, blob->data, blob->len);
115
0
    if(result)
116
0
      goto out;
117
0
    for(i = 0; i < CURL_SHA256_DIGEST_LENGTH; ++i) {
118
0
      result = curlx_dyn_addf(buf, "%02x", hash[i]);
119
0
      if(result)
120
0
        goto out;
121
0
    }
122
0
  }
123
0
out:
124
0
  return result;
125
0
}
126
127
static CURLcode cf_ssl_peer_key_add_mtls(struct dynbuf *buf,
128
                                         struct ssl_primary_config *ssl,
129
                                         bool *is_local)
130
0
{
131
0
  CURLcode result = CURLE_OK;
132
0
  if(ssl->clientcert && ssl->clientcert[0]) {
133
0
    result = cf_ssl_peer_key_add_path(buf, "CCERT", ssl->clientcert, is_local);
134
0
    if(result)
135
0
      goto out;
136
0
  }
137
0
  if(ssl->key && ssl->key[0]) {
138
0
    result = cf_ssl_peer_key_add_path(buf, "KEY", ssl->key, is_local);
139
0
    if(result)
140
0
      goto out;
141
0
  }
142
0
  if(ssl->key_blob) {
143
0
    result = cf_ssl_peer_key_add_hash(buf, "KEYBlob", ssl->key_blob);
144
0
    if(result)
145
0
      goto out;
146
0
  }
147
0
  if(ssl->cert_type && ssl->cert_type[0]) {
148
0
    size_t i;
149
0
    result = curlx_dyn_add(buf, ":CT-");
150
0
    for(i = 0; !result && ssl->cert_type[i]; i++) {
151
0
      char c = Curl_raw_toupper(ssl->cert_type[i]);
152
0
      result = curlx_dyn_addn(buf, &c, 1);
153
0
    }
154
0
    if(result)
155
0
      goto out;
156
0
  }
157
0
  if(ssl->key_type && ssl->key_type[0]) {
158
0
    size_t i;
159
0
    result = curlx_dyn_add(buf, ":KT-");
160
0
    for(i = 0; !result && ssl->key_type[i]; i++) {
161
0
      char c = Curl_raw_toupper(ssl->key_type[i]);
162
0
      result = curlx_dyn_addn(buf, &c, 1);
163
0
    }
164
0
  }
165
0
out:
166
0
  return result;
167
0
}
168
169
0
#define CURL_SSLS_LOCAL_SUFFIX     ":L"
170
0
#define CURL_SSLS_GLOBAL_SUFFIX    ":G"
171
172
static bool cf_ssl_peer_key_is_global(const char *peer_key)
173
0
{
174
0
  size_t len = peer_key ? strlen(peer_key) : 0;
175
0
  return (len > 2) &&
176
0
         (peer_key[len - 1] == 'G') &&
177
0
         (peer_key[len - 2] == ':');
178
0
}
179
180
CURLcode Curl_ssl_peer_key_build(struct ssl_primary_config *ssl,
181
                                 const struct ssl_peer *peer,
182
                                 const struct Curl_peer *via_peer,
183
                                 const char *tls_id,
184
                                 char **ppeer_key)
185
0
{
186
0
  struct dynbuf buf;
187
0
  size_t key_len;
188
0
  bool is_local = FALSE;
189
0
  CURLcode result;
190
191
0
  *ppeer_key = NULL;
192
0
  curlx_dyn_init(&buf, 10 * 1024);
193
194
0
  result = curlx_dyn_addf(&buf, "%s:%d",
195
0
                          peer->dest->hostname, peer->dest->port);
196
0
  if(result)
197
0
    goto out;
198
199
0
  switch(peer->transport) {
200
0
  case TRNSPRT_TCP:
201
0
    break;
202
0
  case TRNSPRT_UDP:
203
0
    result = curlx_dyn_add(&buf, ":UDP");
204
0
    break;
205
0
  case TRNSPRT_QUIC:
206
0
    result = curlx_dyn_add(&buf, ":QUIC");
207
0
    break;
208
0
  case TRNSPRT_UNIX:
209
0
    result = curlx_dyn_add(&buf, ":UNIX");
210
0
    break;
211
0
  default:
212
0
    result = curlx_dyn_addf(&buf, ":TRNSPRT-%d", peer->transport);
213
0
    break;
214
0
  }
215
0
  if(result)
216
0
    goto out;
217
218
0
  if(!ssl->verifypeer) {
219
0
    result = curlx_dyn_add(&buf, ":NO-VRFY-PEER");
220
0
    if(result)
221
0
      goto out;
222
0
  }
223
0
  if(!ssl->verifyhost) {
224
0
    result = curlx_dyn_add(&buf, ":NO-VRFY-HOST");
225
0
    if(result)
226
0
      goto out;
227
0
  }
228
0
  if(ssl->verifystatus) {
229
0
    result = curlx_dyn_add(&buf, ":VRFY-STATUS");
230
0
    if(result)
231
0
      goto out;
232
0
  }
233
0
  if(!ssl->verifypeer || !ssl->verifyhost) {
234
0
    if(via_peer) {
235
0
      result = curlx_dyn_addf(&buf, ":CHOST-%s:CPORT-%u",
236
0
                              via_peer->hostname,
237
0
                              via_peer->port);
238
0
      if(result)
239
0
        goto out;
240
0
    }
241
0
  }
242
243
0
  if(ssl->version || ssl->version_max) {
244
0
    result = curlx_dyn_addf(&buf, ":TLSVER-%d-%u", ssl->version,
245
0
                            (ssl->version_max >> 16));
246
0
    if(result)
247
0
      goto out;
248
0
  }
249
0
  if(ssl->ssl_options) {
250
0
    result = curlx_dyn_addf(&buf, ":TLSOPT-%x", ssl->ssl_options);
251
0
    if(result)
252
0
      goto out;
253
0
  }
254
0
  if(ssl->cipher_list) {
255
0
    result = curlx_dyn_addf(&buf, ":CIPHER-%s", ssl->cipher_list);
256
0
    if(result)
257
0
      goto out;
258
0
  }
259
0
  if(ssl->cipher_list13) {
260
0
    result = curlx_dyn_addf(&buf, ":CIPHER13-%s", ssl->cipher_list13);
261
0
    if(result)
262
0
      goto out;
263
0
  }
264
0
  if(ssl->curves) {
265
0
    result = curlx_dyn_addf(&buf, ":CURVES-%s", ssl->curves);
266
0
    if(result)
267
0
      goto out;
268
0
  }
269
0
  if(ssl->signature_algorithms) {
270
0
    result = curlx_dyn_addf(&buf, ":SIGALGS-%s",
271
0
                            ssl->signature_algorithms);
272
0
    if(result)
273
0
      goto out;
274
0
  }
275
0
  if(ssl->verifypeer) {
276
0
    result = cf_ssl_peer_key_add_path(&buf, "CA", ssl->CAfile, &is_local);
277
0
    if(result)
278
0
      goto out;
279
0
    result = cf_ssl_peer_key_add_path(&buf, "CApath", ssl->CApath, &is_local);
280
0
    if(result)
281
0
      goto out;
282
0
    result = cf_ssl_peer_key_add_path(&buf, "CRL", ssl->CRLfile, &is_local);
283
0
    if(result)
284
0
      goto out;
285
0
    result = cf_ssl_peer_key_add_path(&buf, "Issuer", ssl->issuercert,
286
0
                                      &is_local);
287
0
    if(result)
288
0
      goto out;
289
0
    if(ssl->ca_info_blob) {
290
0
      result = cf_ssl_peer_key_add_hash(&buf, "CAInfoBlob", ssl->ca_info_blob);
291
0
      if(result)
292
0
        goto out;
293
0
    }
294
0
    if(ssl->issuercert_blob) {
295
0
      result = cf_ssl_peer_key_add_hash(&buf, "IssuerBlob",
296
0
                                        ssl->issuercert_blob);
297
0
      if(result)
298
0
        goto out;
299
0
    }
300
0
  }
301
0
  if(ssl->cert_blob) {
302
0
    result = cf_ssl_peer_key_add_hash(&buf, "CertBlob", ssl->cert_blob);
303
0
    if(result)
304
0
      goto out;
305
0
  }
306
0
  if(ssl->pinned_key && ssl->pinned_key[0]) {
307
0
    result = curlx_dyn_addf(&buf, ":Pinned-%s", ssl->pinned_key);
308
0
    if(result)
309
0
      goto out;
310
0
  }
311
312
0
  result = cf_ssl_peer_key_add_mtls(&buf, ssl, &is_local);
313
0
  if(result)
314
0
    goto out;
315
0
#ifdef USE_TLS_SRP
316
0
  if(ssl->username || ssl->password) {
317
0
    result = curlx_dyn_add(&buf, ":SRP-AUTH");
318
0
    if(result)
319
0
      goto out;
320
0
  }
321
0
#endif
322
323
0
  if(!tls_id || !tls_id[0]) {
324
0
    result = CURLE_FAILED_INIT;
325
0
    goto out;
326
0
  }
327
0
  result = curlx_dyn_addf(&buf, ":IMPL-%s", tls_id);
328
0
  if(result)
329
0
    goto out;
330
331
0
  result = curlx_dyn_addf(&buf, is_local ?
332
0
                          CURL_SSLS_LOCAL_SUFFIX : CURL_SSLS_GLOBAL_SUFFIX);
333
0
  if(result)
334
0
    goto out;
335
336
0
  *ppeer_key = curlx_dyn_take(&buf, &key_len);
337
  /* we added printable char, and dynbuf always null-terminates, no need
338
   * to track length */
339
340
0
out:
341
0
  curlx_dyn_free(&buf);
342
0
  return result;
343
0
}
344
345
CURLcode Curl_ssl_peer_key_make(struct Curl_cfilter *cf,
346
                                const struct ssl_peer *peer,
347
                                const char *tls_id,
348
                                char **ppeer_key)
349
0
{
350
0
  struct ssl_primary_config *ssl = Curl_ssl_cf_get_primary_config(cf);
351
0
  return Curl_ssl_peer_key_build(ssl, peer, cf->conn->via_peer, tls_id,
352
0
                                 ppeer_key);
353
0
}
354
355
struct Curl_ssl_scache {
356
  unsigned int magic;
357
  struct Curl_ssl_scache_peer *peers;
358
  size_t peer_count;
359
  int default_lifetime_secs;
360
  long age;
361
  BIT(is_locked);
362
};
363
364
static struct Curl_ssl_scache *cf_ssl_scache_get(struct Curl_easy *data)
365
0
{
366
0
  struct Curl_ssl_scache *scache = NULL;
367
  /* If a share is present, its ssl_scache has preference over the multi */
368
0
  if(data->share && data->share->ssl_scache)
369
0
    scache = data->share->ssl_scache;
370
0
  else if(data->multi && data->multi->ssl_scache)
371
0
    scache = data->multi->ssl_scache;
372
0
  if(scache && !GOOD_SCACHE(scache)) {
373
0
    failf(data, "transfer would use an invalid scache at %p, denied",
374
0
          (void *)scache);
375
0
    DEBUGASSERT(0);
376
0
    return NULL;
377
0
  }
378
0
  return scache;
379
0
}
380
381
static void cf_ssl_scache_session_ldestroy(void *udata, void *obj)
382
0
{
383
0
  struct Curl_ssl_session *s = obj;
384
0
  (void)udata;
385
0
  curlx_free(CURL_UNCONST(s->sdata));
386
0
  curlx_free(CURL_UNCONST(s->quic_tp));
387
0
  curlx_free((void *)s->alpn);
388
0
  curlx_free(s);
389
0
}
390
391
CURLcode Curl_ssl_session_create(void *sdata, size_t sdata_len,
392
                                 int ietf_tls_id, const char *alpn,
393
                                 curl_off_t valid_until, size_t earlydata_max,
394
                                 struct Curl_ssl_session **psession)
395
0
{
396
0
  return Curl_ssl_session_create2(sdata, sdata_len, ietf_tls_id, alpn,
397
0
                                  valid_until, earlydata_max,
398
0
                                  NULL, 0, psession);
399
0
}
400
401
CURLcode Curl_ssl_session_create2(void *sdata, size_t sdata_len,
402
                                  int ietf_tls_id, const char *alpn,
403
                                  curl_off_t valid_until, size_t earlydata_max,
404
                                  unsigned char *quic_tp, size_t quic_tp_len,
405
                                  struct Curl_ssl_session **psession)
406
0
{
407
0
  struct Curl_ssl_session *s;
408
409
0
  if(!sdata || !sdata_len) {
410
0
    curlx_free(sdata);
411
0
    return CURLE_BAD_FUNCTION_ARGUMENT;
412
0
  }
413
414
0
  *psession = NULL;
415
0
  s = curlx_calloc(1, sizeof(*s));
416
0
  if(!s) {
417
0
    curlx_free(sdata);
418
0
    curlx_free(quic_tp);
419
0
    return CURLE_OUT_OF_MEMORY;
420
0
  }
421
422
0
  s->ietf_tls_id = ietf_tls_id;
423
0
  s->valid_until = valid_until;
424
0
  s->earlydata_max = earlydata_max;
425
0
  s->sdata = sdata;
426
0
  s->sdata_len = sdata_len;
427
0
  s->quic_tp = quic_tp;
428
0
  s->quic_tp_len = quic_tp_len;
429
0
  if(alpn) {
430
0
    s->alpn = curlx_strdup(alpn);
431
0
    if(!s->alpn) {
432
0
      cf_ssl_scache_session_ldestroy(NULL, s);
433
0
      return CURLE_OUT_OF_MEMORY;
434
0
    }
435
0
  }
436
0
  *psession = s;
437
0
  return CURLE_OK;
438
0
}
439
440
void Curl_ssl_session_destroy(struct Curl_ssl_session *s)
441
0
{
442
0
  if(s) {
443
    /* if in the list, the list destructor takes care of it */
444
0
    if(Curl_node_llist(&s->list))
445
0
      Curl_node_remove(&s->list);
446
0
    else {
447
0
      cf_ssl_scache_session_ldestroy(NULL, s);
448
0
    }
449
0
  }
450
0
}
451
452
static void cf_ssl_scache_clear_peer(struct Curl_ssl_scache_peer *peer)
453
0
{
454
0
  Curl_llist_destroy(&peer->sessions, NULL);
455
0
  if(peer->sobj) {
456
0
    DEBUGASSERT(peer->sobj_free);
457
0
    if(peer->sobj_free)
458
0
      peer->sobj_free(peer->sobj);
459
0
    peer->sobj = NULL;
460
0
  }
461
0
  peer->sobj_free = NULL;
462
0
  curlx_safefree(peer->clientcert);
463
0
  curlx_safefree(peer->key_passwd);
464
0
#ifdef USE_TLS_SRP
465
0
  curlx_safefree(peer->srp_username);
466
0
  curlx_safefree(peer->srp_password);
467
0
#endif
468
0
  curlx_safefree(peer->ssl_peer_key);
469
0
  peer->age = 0;
470
0
  peer->hmac_set = FALSE;
471
0
}
472
473
static void cf_ssl_scache_peer_set_obj(struct Curl_ssl_scache_peer *peer,
474
                                       void *sobj,
475
                                       Curl_ssl_scache_obj_dtor *sobj_free)
476
0
{
477
0
  DEBUGASSERT(peer);
478
0
  if(peer->sobj_free) {
479
0
    peer->sobj_free(peer->sobj);
480
0
  }
481
0
  peer->sobj = sobj;
482
0
  peer->sobj_free = sobj_free;
483
0
}
484
485
static void cf_ssl_cache_peer_update(struct Curl_ssl_scache_peer *peer)
486
0
{
487
  /* The sessions of this peer are exportable if
488
   * - it has no confidential information
489
   * - its peer key is not yet known, because sessions were
490
   *   imported using only the salt+hmac
491
   * - the peer key is global, e.g. carrying no relative paths */
492
0
  peer->exportable = (!peer->clientcert && !peer->key_passwd &&
493
0
                      !peer->srp_username && !peer->srp_password &&
494
0
                      (!peer->ssl_peer_key ||
495
0
                       cf_ssl_peer_key_is_global(peer->ssl_peer_key)));
496
0
}
497
498
static CURLcode cf_ssl_scache_peer_init(struct Curl_ssl_scache_peer *peer,
499
                                        const char *ssl_peer_key,
500
                                        const char *clientcert,
501
                                        const char *key_passwd,
502
                                        const char *srp_username,
503
                                        const char *srp_password,
504
                                        const unsigned char *salt,
505
                                        const unsigned char *hmac)
506
0
{
507
0
  CURLcode result = CURLE_OUT_OF_MEMORY;
508
509
0
  DEBUGASSERT(!peer->ssl_peer_key);
510
0
  if(ssl_peer_key) {
511
0
    peer->ssl_peer_key = curlx_strdup(ssl_peer_key);
512
0
    if(!peer->ssl_peer_key)
513
0
      goto out;
514
0
    peer->hmac_set = FALSE;
515
0
  }
516
0
  else if(salt && hmac) {
517
0
    memcpy(peer->key_salt, salt, sizeof(peer->key_salt));
518
0
    memcpy(peer->key_hmac, hmac, sizeof(peer->key_hmac));
519
0
    peer->hmac_set = TRUE;
520
0
  }
521
0
  else {
522
0
    result = CURLE_BAD_FUNCTION_ARGUMENT;
523
0
    goto out;
524
0
  }
525
0
  if(clientcert) {
526
0
    peer->clientcert = curlx_strdup(clientcert);
527
0
    if(!peer->clientcert)
528
0
      goto out;
529
0
  }
530
0
  if(key_passwd) {
531
0
    peer->key_passwd = curlx_strdup(key_passwd);
532
0
    if(!peer->key_passwd)
533
0
      goto out;
534
0
  }
535
0
  if(srp_username) {
536
0
    peer->srp_username = curlx_strdup(srp_username);
537
0
    if(!peer->srp_username)
538
0
      goto out;
539
0
  }
540
0
  if(srp_password) {
541
0
    peer->srp_password = curlx_strdup(srp_password);
542
0
    if(!peer->srp_password)
543
0
      goto out;
544
0
  }
545
546
0
  cf_ssl_cache_peer_update(peer);
547
0
  result = CURLE_OK;
548
0
out:
549
0
  if(result)
550
0
    cf_ssl_scache_clear_peer(peer);
551
0
  return result;
552
0
}
553
554
static void cf_scache_session_remove(struct Curl_ssl_scache_peer *peer,
555
                                     struct Curl_ssl_session *s)
556
0
{
557
0
  (void)peer;
558
0
  DEBUGASSERT(Curl_node_llist(&s->list) == &peer->sessions);
559
0
  Curl_ssl_session_destroy(s);
560
0
}
561
562
static bool cf_scache_session_expired(struct Curl_ssl_session *s,
563
                                      curl_off_t now)
564
0
{
565
0
  return (s->valid_until > 0) && (s->valid_until < now);
566
0
}
567
568
static void cf_scache_peer_remove_expired(struct Curl_ssl_scache_peer *peer,
569
                                          curl_off_t now)
570
0
{
571
0
  struct Curl_llist_node *n = Curl_llist_head(&peer->sessions);
572
0
  while(n) {
573
0
    struct Curl_ssl_session *s = Curl_node_elem(n);
574
0
    n = Curl_node_next(n);
575
0
    if(cf_scache_session_expired(s, now))
576
0
      cf_scache_session_remove(peer, s);
577
0
  }
578
0
}
579
580
static void cf_scache_peer_remove_non13(struct Curl_ssl_scache_peer *peer)
581
0
{
582
0
  struct Curl_llist_node *n = Curl_llist_head(&peer->sessions);
583
0
  while(n) {
584
0
    struct Curl_ssl_session *s = Curl_node_elem(n);
585
0
    n = Curl_node_next(n);
586
0
    if(s->ietf_tls_id != CURL_IETF_PROTO_TLS1_3)
587
0
      cf_scache_session_remove(peer, s);
588
0
  }
589
0
}
590
591
CURLcode Curl_ssl_scache_create(size_t max_peers,
592
                                size_t max_sessions_per_peer,
593
                                struct Curl_ssl_scache **pscache)
594
0
{
595
0
  struct Curl_ssl_scache *scache;
596
0
  struct Curl_ssl_scache_peer *peers;
597
0
  size_t i;
598
599
0
  *pscache = NULL;
600
0
  peers = curlx_calloc(max_peers, sizeof(*peers));
601
0
  if(!peers)
602
0
    return CURLE_OUT_OF_MEMORY;
603
604
0
  scache = curlx_calloc(1, sizeof(*scache));
605
0
  if(!scache) {
606
0
    curlx_free(peers);
607
0
    return CURLE_OUT_OF_MEMORY;
608
0
  }
609
610
0
  scache->magic = CURL_SCACHE_MAGIC;
611
0
  scache->default_lifetime_secs = (24 * 60 * 60); /* 1 day */
612
0
  scache->peer_count = max_peers;
613
0
  scache->peers = peers;
614
0
  scache->age = 1;
615
0
  for(i = 0; i < scache->peer_count; ++i) {
616
0
    scache->peers[i].max_sessions = max_sessions_per_peer;
617
0
    Curl_llist_init(&scache->peers[i].sessions,
618
0
                    cf_ssl_scache_session_ldestroy);
619
0
  }
620
621
0
  *pscache = scache;
622
0
  return CURLE_OK;
623
0
}
624
625
void Curl_ssl_scache_destroy(struct Curl_ssl_scache *scache)
626
0
{
627
0
  if(GOOD_SCACHE(scache)) {
628
0
    size_t i;
629
0
    scache->magic = 0;
630
0
    for(i = 0; i < scache->peer_count; ++i) {
631
0
      cf_ssl_scache_clear_peer(&scache->peers[i]);
632
0
    }
633
0
    curlx_free(scache->peers);
634
0
    curlx_free(scache);
635
0
  }
636
0
}
637
638
bool Curl_ssl_scache_use(struct Curl_cfilter *cf, struct Curl_easy *data)
639
0
{
640
0
  if(cf_ssl_scache_get(data)) {
641
0
    struct ssl_config_data *ssl_config = Curl_ssl_cf_get_config(cf, data);
642
0
    return ssl_config ? ssl_config->primary.cache_session : FALSE;
643
0
  }
644
0
  return FALSE;
645
0
}
646
647
/* Lock shared SSL session data */
648
void Curl_ssl_scache_lock(struct Curl_easy *data)
649
0
{
650
0
  struct Curl_ssl_scache *scache = cf_ssl_scache_get(data);
651
0
  if(scache) {
652
0
    if(CURL_SHARE_ssl_scache(data))
653
0
      Curl_share_lock(data, CURL_LOCK_DATA_SSL_SESSION,
654
0
                      CURL_LOCK_ACCESS_SINGLE);
655
0
    DEBUGASSERT(!scache->is_locked);
656
0
    scache->is_locked = TRUE;
657
0
  }
658
0
}
659
660
/* Unlock shared SSL session data */
661
void Curl_ssl_scache_unlock(struct Curl_easy *data)
662
0
{
663
0
  struct Curl_ssl_scache *scache = cf_ssl_scache_get(data);
664
0
  if(scache) {
665
0
    DEBUGASSERT(scache->is_locked);
666
0
    scache->is_locked = FALSE;
667
0
    if(CURL_SHARE_ssl_scache(data))
668
0
      Curl_share_unlock(data, CURL_LOCK_DATA_SSL_SESSION);
669
0
  }
670
0
}
671
672
static bool cf_ssl_scache_match_auth(struct Curl_ssl_scache_peer *peer,
673
                                     struct ssl_primary_config *conn_config)
674
0
{
675
0
  if(!conn_config) {
676
0
    if(peer->clientcert || peer->key_passwd)
677
0
      return FALSE;
678
0
#ifdef USE_TLS_SRP
679
0
    if(peer->srp_username || peer->srp_password)
680
0
      return FALSE;
681
0
#endif
682
0
    return TRUE;
683
0
  }
684
0
  else if(!Curl_safecmp(peer->clientcert, conn_config->clientcert))
685
0
    return FALSE;
686
0
  if(Curl_timestrcmp(peer->key_passwd, conn_config->key_passwd))
687
0
    return FALSE;
688
0
#ifdef USE_TLS_SRP
689
0
  if(Curl_timestrcmp(peer->srp_username, conn_config->username) ||
690
0
     Curl_timestrcmp(peer->srp_password, conn_config->password))
691
0
    return FALSE;
692
0
#endif
693
0
  return TRUE;
694
0
}
695
696
static CURLcode cf_ssl_find_peer_by_key(struct Curl_easy *data,
697
                                        struct Curl_ssl_scache *scache,
698
                                        const char *ssl_peer_key,
699
                                        struct ssl_primary_config *conn_config,
700
                                        struct Curl_ssl_scache_peer **ppeer)
701
0
{
702
0
  size_t i, peer_key_len = 0;
703
0
  CURLcode result = CURLE_OK;
704
705
0
  *ppeer = NULL;
706
0
  if(!GOOD_SCACHE(scache)) {
707
0
    return CURLE_BAD_FUNCTION_ARGUMENT;
708
0
  }
709
710
0
  CURL_TRC_SSLS(data, "find peer slot for %s among %zu slots",
711
0
                ssl_peer_key, scache->peer_count);
712
713
  /* check for entries with known peer_key */
714
0
  for(i = 0; scache && i < scache->peer_count; i++) {
715
0
    if(scache->peers[i].ssl_peer_key &&
716
0
       curl_strequal(ssl_peer_key, scache->peers[i].ssl_peer_key) &&
717
0
       cf_ssl_scache_match_auth(&scache->peers[i], conn_config)) {
718
      /* yes, we have a cached session for this! */
719
0
      *ppeer = &scache->peers[i];
720
0
      goto out;
721
0
    }
722
0
  }
723
  /* check for entries with HMAC set but no known peer_key */
724
0
  for(i = 0; scache && i < scache->peer_count; i++) {
725
0
    if(!scache->peers[i].ssl_peer_key &&
726
0
       scache->peers[i].hmac_set &&
727
0
       cf_ssl_scache_match_auth(&scache->peers[i], conn_config)) {
728
      /* possible entry with unknown peer_key, check hmac */
729
0
      unsigned char my_hmac[CURL_SHA256_DIGEST_LENGTH];
730
0
      if(!peer_key_len) /* we are lazy */
731
0
        peer_key_len = strlen(ssl_peer_key);
732
0
      result = Curl_hmacit(&Curl_HMAC_SHA256,
733
0
                           scache->peers[i].key_salt,
734
0
                           sizeof(scache->peers[i].key_salt),
735
0
                           (const unsigned char *)ssl_peer_key,
736
0
                           peer_key_len,
737
0
                           my_hmac);
738
0
      if(result)
739
0
        goto out;
740
0
      if(!memcmp(scache->peers[i].key_hmac, my_hmac, sizeof(my_hmac))) {
741
        /* remember peer_key for future lookups */
742
0
        CURL_TRC_SSLS(data, "peer entry %zu key recovered: %s",
743
0
                      i, ssl_peer_key);
744
0
        scache->peers[i].ssl_peer_key = curlx_strdup(ssl_peer_key);
745
0
        if(!scache->peers[i].ssl_peer_key) {
746
0
          result = CURLE_OUT_OF_MEMORY;
747
0
          goto out;
748
0
        }
749
0
        cf_ssl_cache_peer_update(&scache->peers[i]);
750
0
        *ppeer = &scache->peers[i];
751
0
        goto out;
752
0
      }
753
0
    }
754
0
  }
755
0
  CURL_TRC_SSLS(data, "peer not found for %s", ssl_peer_key);
756
0
out:
757
0
  return result;
758
0
}
759
760
static struct Curl_ssl_scache_peer *cf_ssl_get_free_peer(
761
  struct Curl_ssl_scache *scache)
762
0
{
763
0
  struct Curl_ssl_scache_peer *peer = NULL;
764
0
  size_t i;
765
766
  /* find empty or oldest peer */
767
0
  for(i = 0; i < scache->peer_count; ++i) {
768
    /* free peer entry? */
769
0
    if(!scache->peers[i].ssl_peer_key && !scache->peers[i].hmac_set) {
770
0
      peer = &scache->peers[i];
771
0
      break;
772
0
    }
773
    /* peer without sessions and obj */
774
0
    if(!scache->peers[i].sobj &&
775
0
       !Curl_llist_count(&scache->peers[i].sessions)) {
776
0
      peer = &scache->peers[i];
777
0
      break;
778
0
    }
779
    /* remember "oldest" peer */
780
0
    if(!peer || (scache->peers[i].age < peer->age)) {
781
0
      peer = &scache->peers[i];
782
0
    }
783
0
  }
784
0
  DEBUGASSERT(peer);
785
0
  if(peer)
786
0
    cf_ssl_scache_clear_peer(peer);
787
0
  return peer;
788
0
}
789
790
static CURLcode cf_ssl_add_peer(struct Curl_easy *data,
791
                                struct Curl_ssl_scache *scache,
792
                                const char *ssl_peer_key,
793
                                struct ssl_primary_config *conn_config,
794
                                struct Curl_ssl_scache_peer **ppeer)
795
0
{
796
0
  struct Curl_ssl_scache_peer *peer = NULL;
797
0
  CURLcode result = CURLE_OK;
798
799
0
  *ppeer = NULL;
800
0
  if(ssl_peer_key) {
801
0
    result = cf_ssl_find_peer_by_key(data, scache, ssl_peer_key, conn_config,
802
0
                                     &peer);
803
0
    if(result || !scache->peer_count)
804
0
      return result;
805
0
  }
806
807
0
  if(peer) {
808
0
    *ppeer = peer;
809
0
    return CURLE_OK;
810
0
  }
811
812
0
  peer = cf_ssl_get_free_peer(scache);
813
0
  if(peer) {
814
0
    char buffer[64];
815
0
    const char *ccert = conn_config ? conn_config->clientcert : NULL;
816
0
    const char *kpasswd = conn_config ? conn_config->key_passwd : NULL;
817
0
    const char *username = NULL, *password = NULL;
818
0
#ifdef USE_TLS_SRP
819
0
    username = conn_config ? conn_config->username : NULL;
820
0
    password = conn_config ? conn_config->password : NULL;
821
0
#endif
822
0
    if(!ccert && conn_config && conn_config->cert_blob) {
823
      /* when using a client cert blob, create a name for it */
824
0
      curl_msnprintf(buffer, sizeof(buffer),
825
0
                     "cert-%p", conn_config->cert_blob->data);
826
0
      ccert = buffer; /* data is strduped by cf_ssl_scache_peer_init */
827
0
    }
828
0
    result = cf_ssl_scache_peer_init(peer, ssl_peer_key, ccert, kpasswd,
829
0
                                     username, password, NULL, NULL);
830
0
    if(result)
831
0
      goto out;
832
    /* all ready */
833
0
    *ppeer = peer;
834
0
    result = CURLE_OK;
835
0
  }
836
837
0
out:
838
0
  if(result) {
839
0
    cf_ssl_scache_clear_peer(peer);
840
0
  }
841
0
  return result;
842
0
}
843
844
static void cf_scache_peer_add_session(struct Curl_ssl_scache_peer *peer,
845
                                       struct Curl_ssl_session *s,
846
                                       curl_off_t now)
847
0
{
848
  /* A session not from TLSv1.3 replaces all other. */
849
0
  if(s->ietf_tls_id != CURL_IETF_PROTO_TLS1_3) {
850
0
    Curl_llist_destroy(&peer->sessions, NULL);
851
0
    Curl_llist_append(&peer->sessions, s, &s->list);
852
0
  }
853
0
  else {
854
    /* Expire existing, append, trim from head to obey max_sessions */
855
0
    cf_scache_peer_remove_expired(peer, now);
856
0
    cf_scache_peer_remove_non13(peer);
857
0
    Curl_llist_append(&peer->sessions, s, &s->list);
858
0
    while(Curl_llist_count(&peer->sessions) > peer->max_sessions) {
859
0
      Curl_node_remove(Curl_llist_head(&peer->sessions));
860
0
    }
861
0
  }
862
0
}
863
864
static CURLcode cf_scache_add_session(struct Curl_cfilter *cf,
865
                                      struct Curl_easy *data,
866
                                      struct Curl_ssl_scache *scache,
867
                                      const char *ssl_peer_key,
868
                                      struct Curl_ssl_session *s)
869
0
{
870
0
  struct Curl_ssl_scache_peer *peer = NULL;
871
0
  struct ssl_primary_config *conn_config = Curl_ssl_cf_get_primary_config(cf);
872
0
  CURLcode result = CURLE_OUT_OF_MEMORY;
873
0
  curl_off_t now = (curl_off_t)time(NULL);
874
0
  curl_off_t max_lifetime;
875
876
0
  if(!scache || !scache->peer_count) {
877
0
    Curl_ssl_session_destroy(s);
878
0
    return CURLE_OK;
879
0
  }
880
881
0
  if(s->valid_until <= 0)
882
0
    s->valid_until = now + scache->default_lifetime_secs;
883
884
0
  max_lifetime = (s->ietf_tls_id == CURL_IETF_PROTO_TLS1_3) ?
885
0
                 CURL_SCACHE_MAX_13_LIFETIME_SEC :
886
0
                 CURL_SCACHE_MAX_12_LIFETIME_SEC;
887
0
  if(s->valid_until > (now + max_lifetime))
888
0
    s->valid_until = now + max_lifetime;
889
890
0
  if(cf_scache_session_expired(s, now)) {
891
0
    CURL_TRC_SSLS(data, "add, session already expired");
892
0
    Curl_ssl_session_destroy(s);
893
0
    return CURLE_OK;
894
0
  }
895
896
0
  result = cf_ssl_add_peer(data, scache, ssl_peer_key, conn_config, &peer);
897
0
  if(result || !peer) {
898
0
    CURL_TRC_SSLS(data, "unable to add scache peer: %d", result);
899
0
    Curl_ssl_session_destroy(s);
900
0
    goto out;
901
0
  }
902
903
0
  cf_scache_peer_add_session(peer, s, now);
904
905
0
out:
906
0
  if(result) {
907
0
    failf(data, "[SCACHE] failed to add session for %s, error=%d",
908
0
          ssl_peer_key, result);
909
0
  }
910
0
  else
911
0
    CURL_TRC_SSLS(data, "added session for %s [proto=0x%x, "
912
0
                  "valid_secs=%" FMT_OFF_T ", alpn=%s, earlydata=%zu, "
913
0
                  "quic_tp=%s], peer has %zu sessions now",
914
0
                  ssl_peer_key, s->ietf_tls_id, s->valid_until - now,
915
0
                  s->alpn, s->earlydata_max, s->quic_tp ? "yes" : "no",
916
0
                  peer ? Curl_llist_count(&peer->sessions) : 0);
917
0
  return result;
918
0
}
919
920
CURLcode Curl_ssl_scache_put(struct Curl_cfilter *cf,
921
                             struct Curl_easy *data,
922
                             const char *ssl_peer_key,
923
                             struct Curl_ssl_session *s)
924
0
{
925
0
  struct Curl_ssl_scache *scache = cf_ssl_scache_get(data);
926
0
  struct ssl_config_data *ssl_config = Curl_ssl_cf_get_config(cf, data);
927
0
  CURLcode result;
928
0
  DEBUGASSERT(ssl_config);
929
930
0
  if(!scache || !ssl_config->primary.cache_session) {
931
0
    Curl_ssl_session_destroy(s);
932
0
    return CURLE_OK;
933
0
  }
934
935
0
  Curl_ssl_scache_lock(data);
936
0
  result = cf_scache_add_session(cf, data, scache, ssl_peer_key, s);
937
0
  Curl_ssl_scache_unlock(data);
938
0
  return result;
939
0
}
940
941
void Curl_ssl_scache_return(struct Curl_cfilter *cf,
942
                            struct Curl_easy *data,
943
                            const char *ssl_peer_key,
944
                            struct Curl_ssl_session *s)
945
0
{
946
  /* See RFC 8446 C.4:
947
   * "Clients SHOULD NOT reuse a ticket for multiple connections." */
948
0
  if(s && s->ietf_tls_id < 0x304)
949
0
    (void)Curl_ssl_scache_put(cf, data, ssl_peer_key, s);
950
0
  else
951
0
    Curl_ssl_session_destroy(s);
952
0
}
953
954
CURLcode Curl_ssl_scache_take(struct Curl_cfilter *cf,
955
                              struct Curl_easy *data,
956
                              const char *ssl_peer_key,
957
                              struct Curl_ssl_session **ps)
958
0
{
959
0
  struct Curl_ssl_scache *scache = cf_ssl_scache_get(data);
960
0
  struct ssl_primary_config *conn_config = Curl_ssl_cf_get_primary_config(cf);
961
0
  struct Curl_ssl_scache_peer *peer = NULL;
962
0
  struct Curl_llist_node *n;
963
0
  struct Curl_ssl_session *s = NULL;
964
0
  CURLcode result;
965
966
0
  *ps = NULL;
967
0
  if(!scache)
968
0
    return CURLE_OK;
969
970
0
  Curl_ssl_scache_lock(data);
971
0
  result = cf_ssl_find_peer_by_key(data, scache, ssl_peer_key, conn_config,
972
0
                                   &peer);
973
0
  if(!result && peer) {
974
0
    cf_scache_peer_remove_expired(peer, (curl_off_t)time(NULL));
975
0
    n = Curl_llist_head(&peer->sessions);
976
0
    if(n) {
977
0
      s = Curl_node_take_elem(n);
978
0
      (scache->age)++;            /* increase general age */
979
0
      peer->age = scache->age; /* set this as used in this age */
980
0
    }
981
0
  }
982
0
  if(s) {
983
0
    *ps = s;
984
0
    CURL_TRC_SSLS(data, "took session for %s [proto=0x%x, "
985
0
                  "alpn=%s, earlydata=%zu, quic_tp=%s], %zu sessions remain",
986
0
                  ssl_peer_key, s->ietf_tls_id, s->alpn,
987
0
                  s->earlydata_max, s->quic_tp ? "yes" : "no",
988
0
                  Curl_llist_count(&peer->sessions));
989
0
  }
990
0
  else {
991
0
    CURL_TRC_SSLS(data, "no cached session for %s", ssl_peer_key);
992
0
  }
993
0
  Curl_ssl_scache_unlock(data);
994
0
  return result;
995
0
}
996
997
CURLcode Curl_ssl_scache_add_obj(struct Curl_cfilter *cf,
998
                                 struct Curl_easy *data,
999
                                 const char *ssl_peer_key,
1000
                                 void *sobj,
1001
                                 Curl_ssl_scache_obj_dtor *sobj_dtor_cb)
1002
0
{
1003
0
  struct Curl_ssl_scache *scache = cf_ssl_scache_get(data);
1004
0
  struct ssl_primary_config *conn_config = Curl_ssl_cf_get_primary_config(cf);
1005
0
  struct Curl_ssl_scache_peer *peer = NULL;
1006
0
  CURLcode result;
1007
1008
0
  DEBUGASSERT(sobj);
1009
0
  DEBUGASSERT(sobj_dtor_cb);
1010
1011
0
  if(!scache) {
1012
0
    result = CURLE_BAD_FUNCTION_ARGUMENT;
1013
0
    goto out;
1014
0
  }
1015
1016
0
  result = cf_ssl_add_peer(data, scache, ssl_peer_key, conn_config, &peer);
1017
0
  if(result || !peer) {
1018
0
    CURL_TRC_SSLS(data, "unable to add scache peer: %d", result);
1019
0
    goto out;
1020
0
  }
1021
1022
0
  cf_ssl_scache_peer_set_obj(peer, sobj, sobj_dtor_cb);
1023
0
  sobj = NULL;  /* peer took ownership */
1024
1025
0
out:
1026
0
  if(sobj && sobj_dtor_cb)
1027
0
    sobj_dtor_cb(sobj);
1028
0
  return result;
1029
0
}
1030
1031
void *Curl_ssl_scache_get_obj(struct Curl_cfilter *cf,
1032
                              struct Curl_easy *data,
1033
                              const char *ssl_peer_key)
1034
0
{
1035
0
  struct Curl_ssl_scache *scache = cf_ssl_scache_get(data);
1036
0
  struct ssl_primary_config *conn_config = Curl_ssl_cf_get_primary_config(cf);
1037
0
  struct Curl_ssl_scache_peer *peer = NULL;
1038
0
  CURLcode result;
1039
0
  void *sobj;
1040
1041
0
  if(!scache)
1042
0
    return NULL;
1043
1044
0
  result = cf_ssl_find_peer_by_key(data, scache, ssl_peer_key, conn_config,
1045
0
                                   &peer);
1046
0
  if(result)
1047
0
    return NULL;
1048
1049
0
  sobj = peer ? peer->sobj : NULL;
1050
1051
0
  CURL_TRC_SSLS(data, "%s cached session for '%s'",
1052
0
                sobj ? "Found" : "No", ssl_peer_key);
1053
0
  return sobj;
1054
0
}
1055
1056
void Curl_ssl_scache_remove_all(struct Curl_cfilter *cf,
1057
                                struct Curl_easy *data,
1058
                                const char *ssl_peer_key)
1059
0
{
1060
0
  struct Curl_ssl_scache *scache = cf_ssl_scache_get(data);
1061
0
  struct ssl_primary_config *conn_config = Curl_ssl_cf_get_primary_config(cf);
1062
0
  struct Curl_ssl_scache_peer *peer = NULL;
1063
0
  CURLcode result;
1064
1065
0
  (void)cf;
1066
0
  if(!scache)
1067
0
    return;
1068
1069
0
  Curl_ssl_scache_lock(data);
1070
0
  result = cf_ssl_find_peer_by_key(data, scache, ssl_peer_key, conn_config,
1071
0
                                   &peer);
1072
0
  if(!result && peer)
1073
0
    cf_ssl_scache_clear_peer(peer);
1074
0
  Curl_ssl_scache_unlock(data);
1075
0
}
1076
1077
#ifdef USE_SSLS_EXPORT
1078
1079
#define CURL_SSL_TICKET_MAX   (16 * 1024)
1080
1081
bool Curl_ssl_scache_is_locked(struct Curl_easy *data)
1082
{
1083
  struct Curl_ssl_scache *scache = cf_ssl_scache_get(data);
1084
  return scache && scache->is_locked;
1085
}
1086
1087
static CURLcode cf_ssl_scache_peer_set_hmac(struct Curl_ssl_scache_peer *peer)
1088
{
1089
  CURLcode result;
1090
1091
  DEBUGASSERT(peer);
1092
  if(!peer->ssl_peer_key)
1093
    return CURLE_BAD_FUNCTION_ARGUMENT;
1094
1095
  result = Curl_rand(NULL, peer->key_salt, sizeof(peer->key_salt));
1096
  if(result)
1097
    return result;
1098
1099
  result = Curl_hmacit(&Curl_HMAC_SHA256,
1100
                       peer->key_salt, sizeof(peer->key_salt),
1101
                       (const unsigned char *)peer->ssl_peer_key,
1102
                       strlen(peer->ssl_peer_key),
1103
                       peer->key_hmac);
1104
  if(!result)
1105
    peer->hmac_set = TRUE;
1106
  return result;
1107
}
1108
1109
static CURLcode cf_ssl_find_peer_by_hmac(struct Curl_ssl_scache *scache,
1110
                                         const unsigned char *salt,
1111
                                         const unsigned char *hmac,
1112
                                         struct Curl_ssl_scache_peer **ppeer)
1113
{
1114
  size_t i;
1115
  CURLcode result = CURLE_OK;
1116
1117
  *ppeer = NULL;
1118
  if(!GOOD_SCACHE(scache))
1119
    return CURLE_BAD_FUNCTION_ARGUMENT;
1120
1121
  /* look for an entry that matches salt+hmac exactly or has a known
1122
   * ssl_peer_key which salt+hmac's to the same. */
1123
  for(i = 0; scache && i < scache->peer_count; i++) {
1124
    struct Curl_ssl_scache_peer *peer = &scache->peers[i];
1125
    if(!cf_ssl_scache_match_auth(peer, NULL))
1126
      continue;
1127
    if(scache->peers[i].hmac_set &&
1128
       !memcmp(peer->key_salt, salt, sizeof(peer->key_salt)) &&
1129
       !memcmp(peer->key_hmac, hmac, sizeof(peer->key_hmac))) {
1130
      /* found exact match, return */
1131
      *ppeer = peer;
1132
      goto out;
1133
    }
1134
    else if(peer->ssl_peer_key) {
1135
      unsigned char my_hmac[CURL_SHA256_DIGEST_LENGTH];
1136
      /* compute hmac for the passed salt */
1137
      result = Curl_hmacit(&Curl_HMAC_SHA256,
1138
                           salt, sizeof(peer->key_salt),
1139
                           (const unsigned char *)peer->ssl_peer_key,
1140
                           strlen(peer->ssl_peer_key),
1141
                           my_hmac);
1142
      if(result)
1143
        goto out;
1144
      if(!memcmp(my_hmac, hmac, sizeof(my_hmac))) {
1145
        /* cryptohash match, take over salt+hmac if no set and return */
1146
        if(!peer->hmac_set) {
1147
          memcpy(peer->key_salt, salt, sizeof(peer->key_salt));
1148
          memcpy(peer->key_hmac, hmac, sizeof(peer->key_hmac));
1149
          peer->hmac_set = TRUE;
1150
        }
1151
        *ppeer = peer;
1152
        goto out;
1153
      }
1154
    }
1155
  }
1156
out:
1157
  return result;
1158
}
1159
1160
CURLcode Curl_ssl_session_import(struct Curl_easy *data,
1161
                                 const char *ssl_peer_key,
1162
                                 const unsigned char *shmac, size_t shmac_len,
1163
                                 const void *sdata, size_t sdata_len)
1164
{
1165
  struct Curl_ssl_scache *scache = cf_ssl_scache_get(data);
1166
  struct Curl_ssl_scache_peer *peer = NULL;
1167
  struct Curl_ssl_session *s = NULL;
1168
  CURLcode result;
1169
1170
  if(!scache) {
1171
    result = CURLE_BAD_FUNCTION_ARGUMENT;
1172
    goto out;
1173
  }
1174
  if(!ssl_peer_key && (!shmac || !shmac_len)) {
1175
    result = CURLE_BAD_FUNCTION_ARGUMENT;
1176
    goto out;
1177
  }
1178
1179
  result = Curl_ssl_session_unpack(data, sdata, sdata_len, &s);
1180
  if(result)
1181
    goto out;
1182
1183
  Curl_ssl_scache_lock(data);
1184
1185
  if(ssl_peer_key) {
1186
    result = cf_ssl_add_peer(data, scache, ssl_peer_key, NULL, &peer);
1187
    if(result)
1188
      goto out;
1189
  }
1190
  else if(shmac_len != (sizeof(peer->key_salt) + sizeof(peer->key_hmac))) {
1191
    /* Either salt+hmac was garbled by caller or is from a curl version
1192
     * that does things differently */
1193
    result = CURLE_BAD_FUNCTION_ARGUMENT;
1194
    goto out;
1195
  }
1196
  else {
1197
    const unsigned char *salt = shmac;
1198
    const unsigned char *hmac = shmac + sizeof(peer->key_salt);
1199
1200
    result = cf_ssl_find_peer_by_hmac(scache, salt, hmac, &peer);
1201
    if(result)
1202
      goto out;
1203
    if(!peer) {
1204
      peer = cf_ssl_get_free_peer(scache);
1205
      if(peer) {
1206
        result = cf_ssl_scache_peer_init(peer, ssl_peer_key, NULL, NULL,
1207
                                         NULL, NULL, salt, hmac);
1208
        if(result)
1209
          goto out;
1210
      }
1211
    }
1212
  }
1213
1214
  if(peer) {
1215
    cf_scache_peer_add_session(peer, s, time(NULL));
1216
    s = NULL; /* peer is now owner */
1217
    CURL_TRC_SSLS(data, "successfully imported ticket for peer %s, now "
1218
                  "with %zu tickets",
1219
                  peer->ssl_peer_key ? peer->ssl_peer_key : "without key",
1220
                  Curl_llist_count(&peer->sessions));
1221
  }
1222
1223
out:
1224
  if(scache && scache->is_locked)
1225
    Curl_ssl_scache_unlock(data);
1226
  Curl_ssl_session_destroy(s);
1227
  return result;
1228
}
1229
1230
CURLcode Curl_ssl_session_export(struct Curl_easy *data,
1231
                                 curl_ssls_export_cb *export_fn,
1232
                                 void *userptr)
1233
{
1234
  struct Curl_ssl_scache *scache = cf_ssl_scache_get(data);
1235
  struct Curl_ssl_scache_peer *peer;
1236
  struct dynbuf sbuf, hbuf;
1237
  struct Curl_llist_node *n;
1238
  size_t i;
1239
  curl_off_t now = time(NULL);
1240
  CURLcode result = CURLE_OK;
1241
#ifdef CURLVERBOSE
1242
  size_t npeers = 0, ntickets = 0;
1243
#endif
1244
1245
  if(!export_fn)
1246
    return CURLE_BAD_FUNCTION_ARGUMENT;
1247
1248
  Curl_ssl_scache_lock(data);
1249
1250
  curlx_dyn_init(&hbuf, (CURL_SHA256_DIGEST_LENGTH * 2) + 1);
1251
  curlx_dyn_init(&sbuf, CURL_SSL_TICKET_MAX);
1252
1253
  for(i = 0; scache && i < scache->peer_count; i++) {
1254
    peer = &scache->peers[i];
1255
    if(!peer->ssl_peer_key && !peer->hmac_set)
1256
      continue; /* skip free entry */
1257
    if(!peer->exportable)
1258
      continue;
1259
1260
    curlx_dyn_reset(&hbuf);
1261
    cf_scache_peer_remove_expired(peer, now);
1262
    n = Curl_llist_head(&peer->sessions);
1263
    if(n)
1264
      VERBOSE(++npeers);
1265
    while(n) {
1266
      struct Curl_ssl_session *s = Curl_node_elem(n);
1267
      if(!peer->hmac_set) {
1268
        result = cf_ssl_scache_peer_set_hmac(peer);
1269
        if(result)
1270
          goto out;
1271
      }
1272
      if(!curlx_dyn_len(&hbuf)) {
1273
        result = curlx_dyn_addn(&hbuf, peer->key_salt, sizeof(peer->key_salt));
1274
        if(result)
1275
          goto out;
1276
        result = curlx_dyn_addn(&hbuf, peer->key_hmac, sizeof(peer->key_hmac));
1277
        if(result)
1278
          goto out;
1279
      }
1280
      curlx_dyn_reset(&sbuf);
1281
      result = Curl_ssl_session_pack(data, s, &sbuf);
1282
      if(result)
1283
        goto out;
1284
1285
      result = export_fn(data, userptr, peer->ssl_peer_key,
1286
                         curlx_dyn_uptr(&hbuf), curlx_dyn_len(&hbuf),
1287
                         curlx_dyn_uptr(&sbuf), curlx_dyn_len(&sbuf),
1288
                         s->valid_until, s->ietf_tls_id,
1289
                         s->alpn, s->earlydata_max);
1290
      if(result)
1291
        goto out;
1292
      VERBOSE(++ntickets);
1293
      n = Curl_node_next(n);
1294
    }
1295
  }
1296
  result = CURLE_OK;
1297
  CURL_TRC_SSLS(data, "exported %zu session tickets for %zu peers",
1298
                ntickets, npeers);
1299
1300
out:
1301
  Curl_ssl_scache_unlock(data);
1302
  curlx_dyn_free(&hbuf);
1303
  curlx_dyn_free(&sbuf);
1304
  return result;
1305
}
1306
1307
#endif /* USE_SSLS_EXPORT */
1308
1309
#endif /* USE_SSL */