For ASN.1 tags with indefinite length, curl's own parser for TLS backends that do not support certificate inspection calls itself recursively. A malicious server certificate can then lead to high recursion level exhausting the stack space. This PR limits the recursion level to 16 which should be safe on all architectures. Added unit test 1657 to verify behaviour. Fixes #16135 Reported-by: z2_ Closes #16137
135 lines
3.4 KiB
C
135 lines
3.4 KiB
C
/***************************************************************************
|
|
* _ _ ____ _
|
|
* Project ___| | | | _ \| |
|
|
* / __| | | | |_) | |
|
|
* | (__| |_| | _ <| |___
|
|
* \___|\___/|_| \_\_____|
|
|
*
|
|
* Copyright (C) Daniel Stenberg, <daniel@haxx.se>, et al.
|
|
*
|
|
* This software is licensed as described in the file COPYING, which
|
|
* you should have received as part of this distribution. The terms
|
|
* are also available at https://curl.se/docs/copyright.html.
|
|
*
|
|
* You may opt to use, copy, modify, merge, publish, distribute and/or sell
|
|
* copies of the Software, and permit persons to whom the Software is
|
|
* furnished to do so, under the terms of the COPYING file.
|
|
*
|
|
* This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
|
|
* KIND, either express or implied.
|
|
*
|
|
* SPDX-License-Identifier: curl
|
|
*
|
|
***************************************************************************/
|
|
#include "curlcheck.h"
|
|
|
|
#include "vtls/x509asn1.h"
|
|
|
|
static CURLcode unit_setup(void)
|
|
{
|
|
return CURLE_OK;
|
|
}
|
|
|
|
static void unit_stop(void)
|
|
{
|
|
|
|
}
|
|
|
|
#if defined(USE_GNUTLS) || defined(USE_SCHANNEL) || defined(USE_SECTRANSP) || \
|
|
defined(USE_MBEDTLS)
|
|
|
|
#ifndef ARRAYSIZE
|
|
#define ARRAYSIZE(A) (sizeof(A)/sizeof((A)[0]))
|
|
#endif
|
|
|
|
struct test1657_spec {
|
|
CURLcode (*setbuf)(struct test1657_spec *spec, struct dynbuf *buf);
|
|
size_t n;
|
|
CURLcode exp_result;
|
|
};
|
|
|
|
static CURLcode make1657_nested(struct test1657_spec *spec, struct dynbuf *buf)
|
|
{
|
|
CURLcode r;
|
|
size_t i;
|
|
unsigned char open_undef[] = { 0x32, 0x80 };
|
|
unsigned char close_undef[] = { 0x00, 0x00 };
|
|
|
|
for(i = 0; i < spec->n; ++i) {
|
|
r = Curl_dyn_addn(buf, open_undef, sizeof(open_undef));
|
|
if(r)
|
|
return r;
|
|
}
|
|
for(i = 0; i < spec->n; ++i) {
|
|
r = Curl_dyn_addn(buf, close_undef, sizeof(close_undef));
|
|
if(r)
|
|
return r;
|
|
}
|
|
return CURLE_OK;
|
|
}
|
|
|
|
static struct test1657_spec test1657_specs[] = {
|
|
{ make1657_nested, 3, CURLE_OK },
|
|
{ make1657_nested, 16, CURLE_OK },
|
|
{ make1657_nested, 17, CURLE_BAD_FUNCTION_ARGUMENT },
|
|
{ make1657_nested, 1024, CURLE_BAD_FUNCTION_ARGUMENT },
|
|
};
|
|
|
|
static bool do_test1657(struct test1657_spec *spec, size_t i,
|
|
struct dynbuf *buf)
|
|
{
|
|
CURLcode result;
|
|
struct Curl_asn1Element elem;
|
|
const char *in;
|
|
|
|
memset(&elem, 0, sizeof(elem));
|
|
Curl_dyn_reset(buf);
|
|
result = spec->setbuf(spec, buf);
|
|
if(result) {
|
|
fprintf(stderr, "test %zu: error setting buf %d\n", i, result);
|
|
return FALSE;
|
|
}
|
|
in = Curl_dyn_ptr(buf);
|
|
result = Curl_x509_getASN1Element(&elem, in, in + Curl_dyn_len(buf));
|
|
if(result != spec->exp_result) {
|
|
fprintf(stderr, "test %zu: expect result %d, got %d\n",
|
|
i, spec->exp_result, result);
|
|
return FALSE;
|
|
}
|
|
return TRUE;
|
|
}
|
|
|
|
UNITTEST_START
|
|
{
|
|
size_t i;
|
|
bool all_ok = TRUE;
|
|
struct dynbuf dbuf;
|
|
|
|
Curl_dyn_init(&dbuf, 32*1024);
|
|
|
|
if(curl_global_init(CURL_GLOBAL_ALL) != CURLE_OK) {
|
|
fprintf(stderr, "curl_global_init() failed\n");
|
|
return TEST_ERR_MAJOR_BAD;
|
|
}
|
|
|
|
for(i = 0; i < ARRAYSIZE(test1657_specs); ++i) {
|
|
if(!do_test1657(&test1657_specs[i], i, &dbuf))
|
|
all_ok = FALSE;
|
|
}
|
|
fail_unless(all_ok, "some tests of Curl_x509_getASN1Element() fails");
|
|
|
|
Curl_dyn_free(&dbuf);
|
|
curl_global_cleanup();
|
|
}
|
|
UNITTEST_STOP
|
|
|
|
#else
|
|
|
|
UNITTEST_START
|
|
{
|
|
puts("not tested since Curl_x509_getASN1Element() is not built-in");
|
|
}
|
|
UNITTEST_STOP
|
|
|
|
#endif
|