summaryrefslogtreecommitdiffstats
path: root/src/lib
diff options
context:
space:
mode:
Diffstat (limited to 'src/lib')
-rw-r--r--src/lib/krb5/asn.1/asn1_decode.c26
-rw-r--r--src/lib/krb5/asn.1/asn1_decode.h3
-rw-r--r--src/lib/krb5/asn.1/asn1_k_decode.c11
3 files changed, 35 insertions, 5 deletions
diff --git a/src/lib/krb5/asn.1/asn1_decode.c b/src/lib/krb5/asn.1/asn1_decode.c
index 53e204549..1ded579b8 100644
--- a/src/lib/krb5/asn.1/asn1_decode.c
+++ b/src/lib/krb5/asn.1/asn1_decode.c
@@ -197,6 +197,32 @@ asn1_decode_generalstring(asn1buf *buf, unsigned int *retlen, char **val)
cleanup();
}
+asn1_error_code
+asn1_decode_bitstring(asn1buf *buf, unsigned int *retlen, char **val)
+{
+ setup();
+ asn1_octet unused;
+
+ tag(ASN1_BITSTRING);
+
+ /* Get the number of unused bits in the last byte (0-7). */
+ retval = asn1buf_remove_octet(buf, &unused);
+ if (retval)
+ return retval;
+ if (unused > 7)
+ return ASN1_BAD_FORMAT;
+
+ retval = asn1buf_remove_charstring(buf, length - 1, val);
+ if (retval)
+ return retval;
+
+ /* Mask out unused bits (unnecessary for correct DER, but be safe). */
+ if (length > 1)
+ (*val)[length - 2] &= (0xff << unused);
+
+ *retlen = length - 1;
+ return 0;
+}
asn1_error_code
asn1_decode_null(asn1buf *buf)
diff --git a/src/lib/krb5/asn.1/asn1_decode.h b/src/lib/krb5/asn.1/asn1_decode.h
index 0e14491de..7573b4fb7 100644
--- a/src/lib/krb5/asn.1/asn1_decode.h
+++ b/src/lib/krb5/asn.1/asn1_decode.h
@@ -45,6 +45,7 @@
* asn1_decode_octetstring
* asn1_decode_charstring
* asn1_decode_generalstring
+ * asn1_decode_bitstring
* asn1_decode_null
* asn1_decode_printablestring
* asn1_decode_ia5string
@@ -72,6 +73,8 @@ asn1_error_code asn1_decode_octetstring(asn1buf *buf, unsigned int *retlen,
asn1_octet **val);
asn1_error_code asn1_decode_generalstring(asn1buf *buf, unsigned int *retlen,
char **val);
+asn1_error_code asn1_decode_bitstring(asn1buf *buf, unsigned int *retlen,
+ char **val);
asn1_error_code asn1_decode_charstring(asn1buf *buf, unsigned int *retlen,
char **val);
/*
diff --git a/src/lib/krb5/asn.1/asn1_k_decode.c b/src/lib/krb5/asn.1/asn1_k_decode.c
index b2471004a..e0c827c4f 100644
--- a/src/lib/krb5/asn.1/asn1_k_decode.c
+++ b/src/lib/krb5/asn.1/asn1_k_decode.c
@@ -1515,11 +1515,12 @@ asn1_decode_kdc_dh_key_info(asn1buf *buf, krb5_kdc_dh_key_info *val)
setup();
val->subjectPublicKey.data = NULL;
{ begin_structure();
- retval = asn1buf_remove_charstring(&subbuf, taglen,
- &val->subjectPublicKey.data);
- if (retval) clean_return(retval);
- val->subjectPublicKey.length = taglen;
- next_tag();
+ /* Special handling for [0] IMPLICIT BIT STRING */
+ error_if_bad_tag(0);
+ if (asn1class != CONTEXT_SPECIFIC || construction != CONSTRUCTED)
+ clean_return(ASN1_BAD_ID);
+ get_lenfield_body(val->subjectPublicKey.length,
+ val->subjectPublicKey.data, asn1_decode_bitstring);
get_field(val->nonce, 1, asn1_decode_int32);
opt_field(val->dhKeyExpiration, 2, asn1_decode_kerberos_time, 0);
end_structure();