mirror of
git://git.gnupg.org/gnupg.git
synced 2025-01-03 12:11:33 +01:00
sm: Improve the octet string cramming for pkcs#12
* sm/minip12.c (need_octet_string_cramming): New. (tlv_expect_object, tlv_expect_octet_string): Run the test before cramming. * sm/minip12.c (ENABLE_DER_STRUCT_DUMPING): New but undefined macro for debug purposes. (bag_decrypted_data_p, bag_data_p): Use macro to allow dumping. -- This bug was exhibited by importing a gpgsm exported EC certificate. We use an extra test instead of retrying to allow retruning an error from malloc failure. And well, for easier reading of the code. GnuPG-bug-id: 6536 (cherry picked from commit c1f78634ec3927ddcfdc4687bc6e408c658a0ece)
This commit is contained in:
parent
a0ac529d08
commit
bb157044a0
79
sm/minip12.c
79
sm/minip12.c
@ -50,6 +50,8 @@
|
|||||||
#define DIM(v) (sizeof(v)/sizeof((v)[0]))
|
#define DIM(v) (sizeof(v)/sizeof((v)[0]))
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
/* Enable the next macro to dump stuff for debugging. */
|
||||||
|
#undef ENABLE_DER_STRUCT_DUMPING
|
||||||
|
|
||||||
|
|
||||||
static unsigned char const oid_data[9] = {
|
static unsigned char const oid_data[9] = {
|
||||||
@ -111,6 +113,8 @@ static unsigned char const data_mactemplate[51] = {
|
|||||||
#define DATA_MACTEMPLATE_MAC_OFF 17
|
#define DATA_MACTEMPLATE_MAC_OFF 17
|
||||||
#define DATA_MACTEMPLATE_SALT_OFF 39
|
#define DATA_MACTEMPLATE_SALT_OFF 39
|
||||||
|
|
||||||
|
/* Note that the BMP String in this template reads:
|
||||||
|
* "GnuPG exported certificate ffffffff" */
|
||||||
static unsigned char const data_attrtemplate[106] = {
|
static unsigned char const data_attrtemplate[106] = {
|
||||||
0x31, 0x7c, 0x30, 0x55, 0x06, 0x09, 0x2a, 0x86,
|
0x31, 0x7c, 0x30, 0x55, 0x06, 0x09, 0x2a, 0x86,
|
||||||
0x48, 0x86, 0xf7, 0x0d, 0x01, 0x09, 0x14, 0x31,
|
0x48, 0x86, 0xf7, 0x0d, 0x01, 0x09, 0x14, 0x31,
|
||||||
@ -210,6 +214,8 @@ static int opt_verbose;
|
|||||||
|
|
||||||
static unsigned char *cram_octet_string (const unsigned char *input,
|
static unsigned char *cram_octet_string (const unsigned char *input,
|
||||||
size_t length, size_t *r_newlength);
|
size_t length, size_t *r_newlength);
|
||||||
|
static int need_octet_string_cramming (const unsigned char *input,
|
||||||
|
size_t length);
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
@ -560,7 +566,7 @@ tlv_expect_sequence (struct tlv_ctx_s *tlv)
|
|||||||
return _tlv_push (tlv);
|
return _tlv_push (tlv);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Variant of tlv_expect_sequence to be used for the ouyter sequence
|
/* Variant of tlv_expect_sequence to be used for the outer sequence
|
||||||
* of an object which might have padding after the ASN.1 data. */
|
* of an object which might have padding after the ASN.1 data. */
|
||||||
static gpg_error_t
|
static gpg_error_t
|
||||||
tlv_expect_top_sequence (struct tlv_ctx_s *tlv)
|
tlv_expect_top_sequence (struct tlv_ctx_s *tlv)
|
||||||
@ -618,7 +624,8 @@ tlv_expect_object (struct tlv_ctx_s *tlv, int class, int tag,
|
|||||||
if (!tlv->ti.length)
|
if (!tlv->ti.length)
|
||||||
return (tlv->lasterr = gpg_error (GPG_ERR_TOO_SHORT));
|
return (tlv->lasterr = gpg_error (GPG_ERR_TOO_SHORT));
|
||||||
|
|
||||||
if (class == CLASS_CONTEXT && tag == 0 && tlv->ti.is_constructed)
|
if (class == CLASS_CONTEXT && tag == 0 && tlv->ti.is_constructed
|
||||||
|
&& need_octet_string_cramming (p, tlv->ti.length))
|
||||||
{
|
{
|
||||||
char *newbuffer;
|
char *newbuffer;
|
||||||
|
|
||||||
@ -665,7 +672,8 @@ tlv_expect_octet_string (struct tlv_ctx_s *tlv, int encapsulates,
|
|||||||
if (!(n=tlv->ti.length))
|
if (!(n=tlv->ti.length))
|
||||||
return (tlv->lasterr = gpg_error (GPG_ERR_TOO_SHORT));
|
return (tlv->lasterr = gpg_error (GPG_ERR_TOO_SHORT));
|
||||||
|
|
||||||
if (encapsulates && tlv->ti.is_constructed)
|
if (encapsulates && tlv->ti.is_constructed
|
||||||
|
&& need_octet_string_cramming (p, n))
|
||||||
{
|
{
|
||||||
char *newbuffer;
|
char *newbuffer;
|
||||||
|
|
||||||
@ -859,6 +867,39 @@ cram_octet_string (const unsigned char *input, size_t length,
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/* Return true if (INPUT,LENGTH) is a structure which should be passed
|
||||||
|
* to cram_octet_string. This is basically the same loop as in
|
||||||
|
* cram_octet_string but without any actual copying. */
|
||||||
|
static int
|
||||||
|
need_octet_string_cramming (const unsigned char *input, size_t length)
|
||||||
|
{
|
||||||
|
const unsigned char *s = input;
|
||||||
|
size_t n = length;
|
||||||
|
struct tag_info ti;
|
||||||
|
|
||||||
|
if (!length)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
while (n)
|
||||||
|
{
|
||||||
|
if (parse_tag (&s, &n, &ti))
|
||||||
|
return 0;
|
||||||
|
if (ti.class == CLASS_UNIVERSAL && ti.tag == TAG_OCTET_STRING
|
||||||
|
&& !ti.ndef && !ti.is_constructed)
|
||||||
|
{
|
||||||
|
s += ti.length;
|
||||||
|
n -= ti.length;
|
||||||
|
}
|
||||||
|
else if (ti.class == CLASS_UNIVERSAL && !ti.tag && !ti.is_constructed)
|
||||||
|
break; /* Ready */
|
||||||
|
else
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
static int
|
static int
|
||||||
string_to_key (int id, char *salt, size_t saltlen, int iter, const char *pw,
|
string_to_key (int id, char *salt, size_t saltlen, int iter, const char *pw,
|
||||||
int req_keylen, unsigned char *keybuf)
|
int req_keylen, unsigned char *keybuf)
|
||||||
@ -1173,13 +1214,15 @@ bag_decrypted_data_p (const void *plaintext, size_t length)
|
|||||||
const unsigned char *p = plaintext;
|
const unsigned char *p = plaintext;
|
||||||
size_t n = length;
|
size_t n = length;
|
||||||
|
|
||||||
/* { */
|
#ifdef ENABLE_DER_STRUCT_DUMPING
|
||||||
/* # warning debug code is enabled */
|
{
|
||||||
/* FILE *fp = fopen ("tmp-minip12-plain-data.der", "wb"); */
|
# warning debug code is enabled
|
||||||
/* if (!fp || fwrite (p, n, 1, fp) != 1) */
|
FILE *fp = fopen ("tmp-minip12-plain-data.der", "wb");
|
||||||
/* exit (2); */
|
if (!fp || fwrite (p, n, 1, fp) != 1)
|
||||||
/* fclose (fp); */
|
exit (2);
|
||||||
/* } */
|
fclose (fp);
|
||||||
|
}
|
||||||
|
#endif /*ENABLE_DER_STRUCT_DUMPING*/
|
||||||
|
|
||||||
if (parse_tag (&p, &n, &ti))
|
if (parse_tag (&p, &n, &ti))
|
||||||
return 0;
|
return 0;
|
||||||
@ -1696,13 +1739,15 @@ bag_data_p (const void *plaintext, size_t length)
|
|||||||
const unsigned char *p = plaintext;
|
const unsigned char *p = plaintext;
|
||||||
size_t n = length;
|
size_t n = length;
|
||||||
|
|
||||||
/* { */
|
#ifdef ENABLE_DER_STRUCT_DUMPING
|
||||||
/* # warning debug code is enabled */
|
{
|
||||||
/* FILE *fp = fopen ("tmp-minip12-plain-key.der", "wb"); */
|
# warning debug code is enabled
|
||||||
/* if (!fp || fwrite (p, n, 1, fp) != 1) */
|
FILE *fp = fopen ("tmp-minip12-plain-key.der", "wb");
|
||||||
/* exit (2); */
|
if (!fp || fwrite (p, n, 1, fp) != 1)
|
||||||
/* fclose (fp); */
|
exit (2);
|
||||||
/* } */
|
fclose (fp);
|
||||||
|
}
|
||||||
|
#endif /*ENABLE_DER_STRUCT_DUMPING*/
|
||||||
|
|
||||||
if (parse_tag (&p, &n, &ti) || ti.class || ti.tag != TAG_SEQUENCE)
|
if (parse_tag (&p, &n, &ti) || ti.class || ti.tag != TAG_SEQUENCE)
|
||||||
return 0;
|
return 0;
|
||||||
|
@ -51,6 +51,7 @@ EXTRA_DIST = runtest inittests $(testscripts) ChangeLog-2011 \
|
|||||||
samplekeys/opensc-test.p12 \
|
samplekeys/opensc-test.p12 \
|
||||||
samplekeys/t5793-openssl.pfx \
|
samplekeys/t5793-openssl.pfx \
|
||||||
samplekeys/t5793-test.pfx \
|
samplekeys/t5793-test.pfx \
|
||||||
|
samplekeys/edward.tester@demo.gnupg.com.p12 \
|
||||||
run-tests.scm
|
run-tests.scm
|
||||||
|
|
||||||
# We used to run $(testscripts) here but these asschk scripts are not
|
# We used to run $(testscripts) here but these asschk scripts are not
|
||||||
|
@ -1,4 +1,6 @@
|
|||||||
# Description-p12 - Machine readable description of our P12 test vectors
|
# Description-p12 - Machine readable description of our P12 test vectors
|
||||||
|
# The Cert line gives the SHA1 fingerprint of the certificate
|
||||||
|
# The Key line gives a hash of the key parameters as returned by minip12.c
|
||||||
|
|
||||||
Name: ov-user.p12
|
Name: ov-user.p12
|
||||||
Desc: Private test key from www.openvalidation.org
|
Desc: Private test key from www.openvalidation.org
|
||||||
@ -30,3 +32,11 @@ Desc: QuaVadis format of t5793-openssl
|
|||||||
Pass: test
|
Pass: test
|
||||||
Cert: 80348a438e4b803b99e708da0b7fdd0659dedd15
|
Cert: 80348a438e4b803b99e708da0b7fdd0659dedd15
|
||||||
Key: c271e44ab4fb19ca1aae71102ea4d7292ccc981d
|
Key: c271e44ab4fb19ca1aae71102ea4d7292ccc981d
|
||||||
|
|
||||||
|
Name: edward.tester@demo.gnupg.com.p12
|
||||||
|
Desc: GnuPG exported Brainpool certificate
|
||||||
|
Pass: abc,123456
|
||||||
|
Cert: ff810b9281a43c394aa138e9c7fd4c0193216fa6
|
||||||
|
Key: 94c6d0b067370a8f2a09ae43cfe8d700bbd61e75
|
||||||
|
|
||||||
|
# eof #
|
||||||
|
BIN
tests/samplekeys/edward.tester@demo.gnupg.com.p12
Normal file
BIN
tests/samplekeys/edward.tester@demo.gnupg.com.p12
Normal file
Binary file not shown.
Loading…
x
Reference in New Issue
Block a user