summaryrefslogtreecommitdiff
path: root/lib
diff options
context:
space:
mode:
Diffstat (limited to 'lib')
-rw-r--r--lib/libssl/ssl_tlsext.c136
1 files changed, 86 insertions, 50 deletions
diff --git a/lib/libssl/ssl_tlsext.c b/lib/libssl/ssl_tlsext.c
index c74772f683e..3502e5a7214 100644
--- a/lib/libssl/ssl_tlsext.c
+++ b/lib/libssl/ssl_tlsext.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: ssl_tlsext.c,v 1.40 2019/01/31 08:11:55 tb Exp $ */
+/* $OpenBSD: ssl_tlsext.c,v 1.41 2019/02/03 14:09:58 jsing Exp $ */
/*
* Copyright (c) 2016, 2017, 2019 Joel Sing <jsing@openbsd.org>
* Copyright (c) 2017 Doug Hogan <doug@openbsd.org>
@@ -25,24 +25,6 @@
#include "ssl_sigalgs.h"
#include "ssl_tlsext.h"
-
-static int
-tlsext_u16_prefixed_builder(CBB *parent, uint8_t *bytes, size_t len)
-{
- CBB child;
-
- if (!CBB_add_u16_length_prefixed(parent, &child))
- return 0;
-
- if (!CBB_add_bytes(&child, bytes, len))
- return 0;
-
- if (!CBB_flush(parent))
- return 0;
-
- return 1;
-}
-
/*
* Supported Application-Layer Protocol Negotiation - RFC 7301
*/
@@ -58,10 +40,19 @@ tlsext_alpn_client_needs(SSL *s)
int
tlsext_alpn_client_build(SSL *s, CBB *cbb)
{
+ CBB protolist;
+
+ if (!CBB_add_u16_length_prefixed(cbb, &protolist))
+ return 0;
- return (tlsext_u16_prefixed_builder(cbb,
- s->internal->alpn_client_proto_list,
- s->internal->alpn_client_proto_list_len));
+ if (!CBB_add_bytes(&protolist, s->internal->alpn_client_proto_list,
+ s->internal->alpn_client_proto_list_len))
+ return 0;
+
+ if (!CBB_flush(cbb))
+ return 0;
+
+ return 1;
}
int
@@ -1233,10 +1224,16 @@ tlsext_keyshare_client_needs(SSL *s)
}
int
-tlsext_keyshare_x25519_generate(SSL *s, CBB *keyshare)
+tlsext_keyshare_client_build(SSL *s, CBB *cbb)
{
uint8_t *public_key = NULL, *private_key = NULL;
- CBB key_exchange;
+ CBB client_shares, key_exchange;
+
+ /* Generate and provide key shares. */
+ if (!CBB_add_u16_length_prefixed(cbb, &client_shares))
+ return 0;
+
+ /* XXX - other groups. */
/* Generate X25519 key pair. */
if ((public_key = malloc(X25519_KEY_LENGTH)) == NULL)
@@ -1246,13 +1243,16 @@ tlsext_keyshare_x25519_generate(SSL *s, CBB *keyshare)
X25519_keypair(public_key, private_key);
/* Add the group and serialize the public key. */
- if (!CBB_add_u16(keyshare, tls1_ec_nid2curve_id(NID_X25519)))
+ if (!CBB_add_u16(&client_shares, tls1_ec_nid2curve_id(NID_X25519)))
goto err;
- if (!CBB_add_u16_length_prefixed(keyshare, &key_exchange))
+ if (!CBB_add_u16_length_prefixed(&client_shares, &key_exchange))
goto err;
if (!CBB_add_bytes(&key_exchange, public_key, X25519_KEY_LENGTH))
goto err;
+ if (!CBB_flush(cbb))
+ goto err;
+
S3I(s)->hs_tls13.x25519_public = public_key;
S3I(s)->hs_tls13.x25519_private = private_key;
@@ -1266,22 +1266,6 @@ tlsext_keyshare_x25519_generate(SSL *s, CBB *keyshare)
}
int
-tlsext_keyshare_client_build(SSL *s, CBB *cbb)
-{
- CBB client_shares;
-
- /* Generate and provide key shares. */
- if (!CBB_add_u16_length_prefixed(cbb, &client_shares))
- return 0;
-
- if (!tlsext_keyshare_x25519_generate(s, &client_shares) ||
- !CBB_flush(cbb))
- return 0;
-
- return (1);
-}
-
-int
tlsext_keyshare_server_parse(SSL *s, CBS *cbs, int *alert)
{
CBS client_shares;
@@ -1343,15 +1327,43 @@ tlsext_keyshare_server_needs(SSL *s)
int
tlsext_keyshare_server_build(SSL *s, CBB *cbb)
{
+ uint8_t *public_key = NULL, *private_key = NULL;
+ CBB key_exchange;
+
+ /* XXX deduplicate with client code */
+
/* X25519 */
if (S3I(s)->hs_tls13.x25519_peer_public == NULL)
return 0;
- if (!tlsext_keyshare_x25519_generate(s, cbb) ||
- !CBB_flush(cbb))
- return 0;
+ /* Generate X25519 key pair. */
+ if ((public_key = malloc(X25519_KEY_LENGTH)) == NULL)
+ goto err;
+ if ((private_key = malloc(X25519_KEY_LENGTH)) == NULL)
+ goto err;
+ X25519_keypair(public_key, private_key);
+
+ /* Add the group and serialize the public key. */
+ if (!CBB_add_u16(cbb, tls1_ec_nid2curve_id(NID_X25519)))
+ goto err;
+ if (!CBB_add_u16_length_prefixed(cbb, &key_exchange))
+ goto err;
+ if (!CBB_add_bytes(&key_exchange, public_key, X25519_KEY_LENGTH))
+ goto err;
+
+ if (!CBB_flush(cbb))
+ goto err;
+
+ S3I(s)->hs_tls13.x25519_public = public_key;
+ S3I(s)->hs_tls13.x25519_private = private_key;
return 1;
+
+ err:
+ freezero(public_key, X25519_KEY_LENGTH);
+ freezero(private_key, X25519_KEY_LENGTH);
+
+ return 0;
}
int
@@ -1528,8 +1540,19 @@ tlsext_cookie_client_needs(SSL *s)
int
tlsext_cookie_client_build(SSL *s, CBB *cbb)
{
- return (tlsext_u16_prefixed_builder(cbb,
- S3I(s)->hs_tls13.cookie, S3I(s)->hs_tls13.cookie_len));
+ CBB cookie;
+
+ if (!CBB_add_u16_length_prefixed(cbb, &cookie))
+ return 0;
+
+ if (!CBB_add_bytes(&cookie, S3I(s)->hs_tls13.cookie,
+ S3I(s)->hs_tls13.cookie_len))
+ return 0;
+
+ if (!CBB_flush(cbb))
+ return 0;
+
+ return 1;
}
int
@@ -1581,8 +1604,21 @@ tlsext_cookie_server_needs(SSL *s)
int
tlsext_cookie_server_build(SSL *s, CBB *cbb)
{
- return (tlsext_u16_prefixed_builder(cbb,
- S3I(s)->hs_tls13.cookie, S3I(s)->hs_tls13.cookie_len));
+ CBB cookie;
+
+ /* XXX deduplicate with client code */
+
+ if (!CBB_add_u16_length_prefixed(cbb, &cookie))
+ return 0;
+
+ if (!CBB_add_bytes(&cookie, S3I(s)->hs_tls13.cookie,
+ S3I(s)->hs_tls13.cookie_len))
+ return 0;
+
+ if (!CBB_flush(cbb))
+ return 0;
+
+ return 1;
}
int
@@ -1840,7 +1876,7 @@ tlsext_funcs(struct tls_extension *tlsext, int is_server)
if (is_server)
return &tlsext->server;
- return &tlsext->client;
+ return &tlsext->client;
}
static int