Add contrib/lax_der_parsing.h
This shows a snippet of code to do lax DER parsing, without obeying to any particular standard.
This commit is contained in:
parent
3bb9c44719
commit
fea19e7bb7
@ -40,6 +40,7 @@ noinst_HEADERS += src/hash_impl.h
|
|||||||
noinst_HEADERS += src/field.h
|
noinst_HEADERS += src/field.h
|
||||||
noinst_HEADERS += src/field_impl.h
|
noinst_HEADERS += src/field_impl.h
|
||||||
noinst_HEADERS += src/bench.h
|
noinst_HEADERS += src/bench.h
|
||||||
|
noinst_HEADERS += contrib/lax_der_parsing.h
|
||||||
|
|
||||||
pkgconfigdir = $(libdir)/pkgconfig
|
pkgconfigdir = $(libdir)/pkgconfig
|
||||||
pkgconfig_DATA = libsecp256k1.pc
|
pkgconfig_DATA = libsecp256k1.pc
|
||||||
@ -64,7 +65,7 @@ endif
|
|||||||
if USE_TESTS
|
if USE_TESTS
|
||||||
noinst_PROGRAMS += tests
|
noinst_PROGRAMS += tests
|
||||||
tests_SOURCES = src/tests.c
|
tests_SOURCES = src/tests.c
|
||||||
tests_CPPFLAGS = -DVERIFY -I$(top_srcdir)/src $(SECP_INCLUDES) $(SECP_TEST_INCLUDES)
|
tests_CPPFLAGS = -DVERIFY -I$(top_srcdir)/src -I$(top_srcdir)/include $(SECP_INCLUDES) $(SECP_TEST_INCLUDES)
|
||||||
tests_LDADD = $(SECP_LIBS) $(SECP_TEST_LIBS)
|
tests_LDADD = $(SECP_LIBS) $(SECP_TEST_LIBS)
|
||||||
tests_LDFLAGS = -static
|
tests_LDFLAGS = -static
|
||||||
TESTS = tests
|
TESTS = tests
|
||||||
|
191
contrib/lax_der_parsing.h
Normal file
191
contrib/lax_der_parsing.h
Normal file
@ -0,0 +1,191 @@
|
|||||||
|
/**********************************************************************
|
||||||
|
* Copyright (c) 2015 Pieter Wuille *
|
||||||
|
* Distributed under the MIT software license, see the accompanying *
|
||||||
|
* file COPYING or http://www.opensource.org/licenses/mit-license.php.*
|
||||||
|
**********************************************************************/
|
||||||
|
|
||||||
|
/* This file contains a code snippet that parses DER with various errors and
|
||||||
|
* violations. This is not a part of the library itself, because the allowed
|
||||||
|
* violations are chosen arbitrarily and do not follow or establish any
|
||||||
|
* standard.
|
||||||
|
*
|
||||||
|
* In many places it matters that different implementations do not only accept
|
||||||
|
* the same set of valid signatures, but also reject the same set of signatures.
|
||||||
|
* The only means to accomplish that is by strictly obeying a standard, and not
|
||||||
|
* accepting anything else.
|
||||||
|
*
|
||||||
|
* Nonetheless, sometimes there is a need for compatibility with systems that
|
||||||
|
* use signatures which do not strictly obey DER. The snippet below shows how
|
||||||
|
* certain violations are easily supported. You may need to adapt it.
|
||||||
|
*
|
||||||
|
* Do not use this for new systems. Use well-defined DER or compact signatures
|
||||||
|
* instead if you have the choice (see secp256k1_ecdsa_signature_parse_der and
|
||||||
|
* secp256k1_ecdsa_signature_parse_compact).
|
||||||
|
*
|
||||||
|
* The supported violations are:
|
||||||
|
* - All numbers are parsed as nonnegative integers, even though X.609-0207
|
||||||
|
* section 8.3.3 specifies that integers are always encoded as two's
|
||||||
|
* complement.
|
||||||
|
* - Integers can have length 0, even though section 8.3.1 says they can't.
|
||||||
|
* - Integers with overly long padding are accepted, violation section
|
||||||
|
* 8.3.2.
|
||||||
|
* - 127-byte long length descriptors are accepted, even though section
|
||||||
|
* 8.1.3.5.c says that they are not.
|
||||||
|
* - Trailing garbage data inside or after the signature is ignored.
|
||||||
|
* - The length descriptor of the sequence is ignored.
|
||||||
|
*
|
||||||
|
* Compared to for example OpenSSL, many violations are NOT supported:
|
||||||
|
* - Using overly long tag descriptors for the sequence or integers inside,
|
||||||
|
* violating section 8.1.2.2.
|
||||||
|
* - Encoding primitive integers as constructed values, violating section
|
||||||
|
* 8.3.1.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef _SECP256K1_CONTRIB_LAX_DER_PARSING_H_
|
||||||
|
#define _SECP256K1_CONTRIB_LAX_DER_PARSING_H_
|
||||||
|
|
||||||
|
#include <string.h>
|
||||||
|
#include <secp256k1.h>
|
||||||
|
|
||||||
|
static int secp256k1_ecdsa_signature_parse_der_lax(const secp256k1_context* ctx, secp256k1_ecdsa_signature* sig, const unsigned char *input, size_t inputlen);
|
||||||
|
|
||||||
|
static int secp256k1_ecdsa_signature_parse_der_lax(const secp256k1_context* ctx, secp256k1_ecdsa_signature* sig, const unsigned char *input, size_t inputlen) {
|
||||||
|
size_t rpos, rlen, spos, slen;
|
||||||
|
size_t pos = 0;
|
||||||
|
size_t lenbyte;
|
||||||
|
unsigned char tmpsig[64] = {0};
|
||||||
|
int overflow = 0;
|
||||||
|
|
||||||
|
/* Hack to initialize sig with a correctly-parsed but invalid signature. */
|
||||||
|
secp256k1_ecdsa_signature_parse_compact(ctx, sig, tmpsig);
|
||||||
|
|
||||||
|
/* Sequence tag byte */
|
||||||
|
if (pos == inputlen || input[pos] != 0x30) {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
pos++;
|
||||||
|
|
||||||
|
/* Sequence length bytes */
|
||||||
|
if (pos == inputlen) {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
lenbyte = input[pos++];
|
||||||
|
if (lenbyte & 0x80) {
|
||||||
|
lenbyte -= 0x80;
|
||||||
|
if (pos + lenbyte > inputlen) {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
pos += lenbyte;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Integer tag byte for R */
|
||||||
|
if (pos == inputlen || input[pos] != 0x02) {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
pos++;
|
||||||
|
|
||||||
|
/* Integer length for R */
|
||||||
|
if (pos == inputlen) {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
lenbyte = input[pos++];
|
||||||
|
if (lenbyte & 0x80) {
|
||||||
|
lenbyte -= 0x80;
|
||||||
|
if (pos + lenbyte > inputlen) {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
while (lenbyte > 0 && input[pos] == 0) {
|
||||||
|
pos++;
|
||||||
|
lenbyte--;
|
||||||
|
}
|
||||||
|
if (lenbyte >= sizeof(size_t)) {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
rlen = 0;
|
||||||
|
while (lenbyte > 0) {
|
||||||
|
rlen = (rlen << 8) + input[pos];
|
||||||
|
pos++;
|
||||||
|
lenbyte--;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
rlen = lenbyte;
|
||||||
|
}
|
||||||
|
if (rlen > inputlen - pos) {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
rpos = pos;
|
||||||
|
pos += rlen;
|
||||||
|
|
||||||
|
/* Integer tag byte for S */
|
||||||
|
if (pos == inputlen || input[pos] != 0x02) {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
pos++;
|
||||||
|
|
||||||
|
/* Integer length for S */
|
||||||
|
if (pos == inputlen) {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
lenbyte = input[pos++];
|
||||||
|
if (lenbyte & 0x80) {
|
||||||
|
lenbyte -= 0x80;
|
||||||
|
if (pos + lenbyte > inputlen) {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
while (lenbyte > 0 && input[pos] == 0) {
|
||||||
|
pos++;
|
||||||
|
lenbyte--;
|
||||||
|
}
|
||||||
|
if (lenbyte >= sizeof(size_t)) {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
slen = 0;
|
||||||
|
while (lenbyte > 0) {
|
||||||
|
slen = (slen << 8) + input[pos];
|
||||||
|
pos++;
|
||||||
|
lenbyte--;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
slen = lenbyte;
|
||||||
|
}
|
||||||
|
if (slen > inputlen - pos) {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
spos = pos;
|
||||||
|
pos += slen;
|
||||||
|
|
||||||
|
/* Ignore leading zeroes in R */
|
||||||
|
while (rlen > 0 && input[rpos] == 0) {
|
||||||
|
rlen--;
|
||||||
|
rpos++;
|
||||||
|
}
|
||||||
|
/* Copy R value */
|
||||||
|
if (rlen > 32) {
|
||||||
|
overflow = 1;
|
||||||
|
} else {
|
||||||
|
memcpy(tmpsig + 32 - rlen, input + rpos, rlen);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Ignore leading zeroes in S */
|
||||||
|
while (slen > 0 && input[spos] == 0) {
|
||||||
|
slen--;
|
||||||
|
spos++;
|
||||||
|
}
|
||||||
|
/* Copy S value */
|
||||||
|
if (slen > 32) {
|
||||||
|
overflow = 1;
|
||||||
|
} else {
|
||||||
|
memcpy(tmpsig + 64 - slen, input + spos, slen);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!overflow) {
|
||||||
|
overflow = !secp256k1_ecdsa_signature_parse_compact(ctx, sig, tmpsig);
|
||||||
|
}
|
||||||
|
if (overflow) {
|
||||||
|
memset(tmpsig, 0, 64);
|
||||||
|
secp256k1_ecdsa_signature_parse_compact(ctx, sig, tmpsig);
|
||||||
|
}
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif
|
@ -1,5 +1,5 @@
|
|||||||
/**********************************************************************
|
/**********************************************************************
|
||||||
* Copyright (c) 2013, 2014 Pieter Wuille *
|
* Copyright (c) 2013-2015 Pieter Wuille *
|
||||||
* Distributed under the MIT software license, see the accompanying *
|
* Distributed under the MIT software license, see the accompanying *
|
||||||
* file COPYING or http://www.opensource.org/licenses/mit-license.php.*
|
* file COPYING or http://www.opensource.org/licenses/mit-license.php.*
|
||||||
**********************************************************************/
|
**********************************************************************/
|
||||||
|
28
src/tests.c
28
src/tests.c
@ -24,6 +24,8 @@
|
|||||||
#include "openssl/obj_mac.h"
|
#include "openssl/obj_mac.h"
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
#include "contrib/lax_der_parsing.h"
|
||||||
|
|
||||||
#if !defined(VG_CHECK)
|
#if !defined(VG_CHECK)
|
||||||
# if defined(VALGRIND)
|
# if defined(VALGRIND)
|
||||||
# include <valgrind/memcheck.h>
|
# include <valgrind/memcheck.h>
|
||||||
@ -2512,6 +2514,12 @@ int test_ecdsa_der_parse(const unsigned char *sig, size_t siglen, int certainly_
|
|||||||
size_t len_der = 2048;
|
size_t len_der = 2048;
|
||||||
int parsed_der = 0, valid_der = 0, roundtrips_der = 0;
|
int parsed_der = 0, valid_der = 0, roundtrips_der = 0;
|
||||||
|
|
||||||
|
secp256k1_ecdsa_signature sig_der_lax;
|
||||||
|
unsigned char roundtrip_der_lax[2048];
|
||||||
|
unsigned char compact_der_lax[64];
|
||||||
|
size_t len_der_lax = 2048;
|
||||||
|
int parsed_der_lax = 0, valid_der_lax = 0, roundtrips_der_lax = 0;
|
||||||
|
|
||||||
#ifdef ENABLE_OPENSSL_TESTS
|
#ifdef ENABLE_OPENSSL_TESTS
|
||||||
ECDSA_SIG *sig_openssl;
|
ECDSA_SIG *sig_openssl;
|
||||||
const unsigned char *sigptr;
|
const unsigned char *sigptr;
|
||||||
@ -2530,6 +2538,16 @@ int test_ecdsa_der_parse(const unsigned char *sig, size_t siglen, int certainly_
|
|||||||
roundtrips_der = (len_der == siglen) && memcmp(roundtrip_der, sig, siglen) == 0;
|
roundtrips_der = (len_der == siglen) && memcmp(roundtrip_der, sig, siglen) == 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
parsed_der_lax = secp256k1_ecdsa_signature_parse_der_lax(ctx, &sig_der_lax, sig, siglen);
|
||||||
|
if (parsed_der_lax) {
|
||||||
|
ret |= (!secp256k1_ecdsa_signature_serialize_compact(ctx, compact_der_lax, &sig_der_lax)) << 10;
|
||||||
|
valid_der_lax = (memcmp(compact_der_lax, zeroes, 32) != 0) && (memcmp(compact_der_lax + 32, zeroes, 32) != 0);
|
||||||
|
}
|
||||||
|
if (valid_der_lax) {
|
||||||
|
ret |= (!secp256k1_ecdsa_signature_serialize_der(ctx, roundtrip_der_lax, &len_der_lax, &sig_der_lax)) << 11;
|
||||||
|
roundtrips_der_lax = (len_der_lax == siglen) && memcmp(roundtrip_der_lax, sig, siglen) == 0;
|
||||||
|
}
|
||||||
|
|
||||||
if (certainly_der) {
|
if (certainly_der) {
|
||||||
ret |= (!parsed_der) << 2;
|
ret |= (!parsed_der) << 2;
|
||||||
}
|
}
|
||||||
@ -2540,6 +2558,16 @@ int test_ecdsa_der_parse(const unsigned char *sig, size_t siglen, int certainly_
|
|||||||
ret |= (!roundtrips_der) << 3;
|
ret |= (!roundtrips_der) << 3;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (valid_der) {
|
||||||
|
ret |= (!roundtrips_der_lax) << 12;
|
||||||
|
ret |= (len_der != len_der_lax) << 13;
|
||||||
|
ret |= (memcmp(roundtrip_der_lax, roundtrip_der, len_der) != 0) << 14;
|
||||||
|
}
|
||||||
|
ret |= (roundtrips_der != roundtrips_der_lax) << 15;
|
||||||
|
if (parsed_der) {
|
||||||
|
ret |= (!parsed_der_lax) << 16;
|
||||||
|
}
|
||||||
|
|
||||||
#ifdef ENABLE_OPENSSL_TESTS
|
#ifdef ENABLE_OPENSSL_TESTS
|
||||||
sig_openssl = ECDSA_SIG_new();
|
sig_openssl = ECDSA_SIG_new();
|
||||||
sigptr = sig;
|
sigptr = sig;
|
||||||
|
Loading…
x
Reference in New Issue
Block a user