提交 be0d8517 编写于 作者: A Adam Langley 提交者: Emilia Kasper

psk_client_callback, 128-byte id bug.

Fix a bug in handling of 128 byte long PSK identity in
psk_client_callback.

OpenSSL supports PSK identities of up to (and including) 128 bytes in
length. PSK identity is obtained via the psk_client_callback,
implementors of which are expected to provide a NULL-terminated
identity. However, the callback is invoked with only 128 bytes of
storage thus making it impossible to return a 128 byte long identity and
the required additional NULL byte.

This CL fixes the issue by passing in a 129 byte long buffer into the
psk_client_callback. As a safety precaution, this CL also zeroes out the
buffer before passing it into the callback, uses strnlen for obtaining
the length of the identity returned by the callback, and aborts the
handshake if the identity (without the NULL terminator) is longer than
128 bytes.

(Original patch amended to achieve strnlen in a different way.)
Reviewed-by: NRich Salz <rsalz@openssl.org>
上级 360928b7
...@@ -3119,7 +3119,11 @@ int ssl3_send_client_key_exchange(SSL *s) ...@@ -3119,7 +3119,11 @@ int ssl3_send_client_key_exchange(SSL *s)
#ifndef OPENSSL_NO_PSK #ifndef OPENSSL_NO_PSK
else if (alg_k & SSL_kPSK) else if (alg_k & SSL_kPSK)
{ {
char identity[PSK_MAX_IDENTITY_LEN]; /* The callback needs PSK_MAX_IDENTITY_LEN + 1 bytes
* to return a \0-terminated identity. The last byte
* is for us for simulating strnlen. */
char identity[PSK_MAX_IDENTITY_LEN + 2];
size_t identity_len;
unsigned char *t = NULL; unsigned char *t = NULL;
unsigned char psk_or_pre_ms[PSK_MAX_PSK_LEN*2+4]; unsigned char psk_or_pre_ms[PSK_MAX_PSK_LEN*2+4];
unsigned int pre_ms_len = 0, psk_len = 0; unsigned int pre_ms_len = 0, psk_len = 0;
...@@ -3133,8 +3137,9 @@ int ssl3_send_client_key_exchange(SSL *s) ...@@ -3133,8 +3137,9 @@ int ssl3_send_client_key_exchange(SSL *s)
goto err; goto err;
} }
memset(identity, 0, sizeof(identity));
psk_len = s->psk_client_callback(s, s->ctx->psk_identity_hint, psk_len = s->psk_client_callback(s, s->ctx->psk_identity_hint,
identity, PSK_MAX_IDENTITY_LEN, identity, sizeof(identity) - 1,
psk_or_pre_ms, sizeof(psk_or_pre_ms)); psk_or_pre_ms, sizeof(psk_or_pre_ms));
if (psk_len > PSK_MAX_PSK_LEN) if (psk_len > PSK_MAX_PSK_LEN)
{ {
...@@ -3148,7 +3153,14 @@ int ssl3_send_client_key_exchange(SSL *s) ...@@ -3148,7 +3153,14 @@ int ssl3_send_client_key_exchange(SSL *s)
SSL_R_PSK_IDENTITY_NOT_FOUND); SSL_R_PSK_IDENTITY_NOT_FOUND);
goto psk_err; goto psk_err;
} }
identity[PSK_MAX_IDENTITY_LEN + 1] = '\0';
identity_len = strlen(identity);
if (identity_len > PSK_MAX_IDENTITY_LEN)
{
SSLerr(SSL_F_SSL3_SEND_CLIENT_KEY_EXCHANGE,
ERR_R_INTERNAL_ERROR);
goto psk_err;
}
/* create PSK pre_master_secret */ /* create PSK pre_master_secret */
pre_ms_len = 2+psk_len+2+psk_len; pre_ms_len = 2+psk_len+2+psk_len;
t = psk_or_pre_ms; t = psk_or_pre_ms;
...@@ -3182,14 +3194,13 @@ int ssl3_send_client_key_exchange(SSL *s) ...@@ -3182,14 +3194,13 @@ int ssl3_send_client_key_exchange(SSL *s)
s->session->master_key_length = s->session->master_key_length =
s->method->ssl3_enc->generate_master_secret(s, s->method->ssl3_enc->generate_master_secret(s,
s->session->master_key, s->session->master_key,
psk_or_pre_ms, pre_ms_len); psk_or_pre_ms, pre_ms_len);
n = strlen(identity); s2n(identity_len, p);
s2n(n, p); memcpy(p, identity, identity_len);
memcpy(p, identity, n); n = 2 + identity_len;
n+=2;
psk_err = 0; psk_err = 0;
psk_err: psk_err:
OPENSSL_cleanse(identity, PSK_MAX_IDENTITY_LEN); OPENSSL_cleanse(identity, sizeof(identity));
OPENSSL_cleanse(psk_or_pre_ms, sizeof(psk_or_pre_ms)); OPENSSL_cleanse(psk_or_pre_ms, sizeof(psk_or_pre_ms));
if (psk_err != 0) if (psk_err != 0)
{ {
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册