| 
									
										
										
										
											2016-05-18 02:51:34 +08:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * Copyright 1995-2016 The OpenSSL Project Authors. All Rights Reserved. | 
					
						
							| 
									
										
										
										
											1998-12-21 18:52:47 +08:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2018-12-06 20:17:34 +08:00
										 |  |  |  * Licensed under the Apache License 2.0 (the "License").  You may not use | 
					
						
							| 
									
										
										
										
											2016-05-18 02:51:34 +08:00
										 |  |  |  * this file except in compliance with the License.  You can obtain a copy | 
					
						
							|  |  |  |  * in the file LICENSE in the source distribution or at | 
					
						
							|  |  |  |  * https://www.openssl.org/source/license.html
 | 
					
						
							| 
									
										
										
										
											1998-12-21 18:52:47 +08:00
										 |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-09-19 03:58:41 +08:00
										 |  |  | #include <limits.h>
 | 
					
						
							| 
									
										
										
										
											1998-12-21 18:52:47 +08:00
										 |  |  | #include <stdio.h>
 | 
					
						
							| 
									
										
										
										
											2015-05-14 22:56:48 +08:00
										 |  |  | #include "internal/cryptlib.h"
 | 
					
						
							| 
									
										
										
										
											1999-04-24 06:13:45 +08:00
										 |  |  | #include <openssl/asn1.h>
 | 
					
						
							| 
									
										
										
										
											2019-09-28 06:45:40 +08:00
										 |  |  | #include "asn1_local.h"
 | 
					
						
							| 
									
										
										
										
											1998-12-21 18:52:47 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-11-12 11:58:08 +08:00
										 |  |  | int ASN1_BIT_STRING_set(ASN1_BIT_STRING *x, unsigned char *d, int len) | 
					
						
							| 
									
										
										
										
											2015-01-22 11:40:55 +08:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2015-03-14 12:16:42 +08:00
										 |  |  |     return ASN1_STRING_set(x, d, len); | 
					
						
							| 
									
										
										
										
											2015-01-22 11:40:55 +08:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											1999-10-20 09:50:23 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-07-07 21:24:36 +08:00
										 |  |  | int i2c_ASN1_BIT_STRING(ASN1_BIT_STRING *a, unsigned char **pp) | 
					
						
							| 
									
										
										
										
											2015-01-22 11:40:55 +08:00
										 |  |  | { | 
					
						
							|  |  |  |     int ret, j, bits, len; | 
					
						
							|  |  |  |     unsigned char *p, *d; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (a == NULL) | 
					
						
							| 
									
										
										
										
											2017-10-09 19:05:58 +08:00
										 |  |  |         return 0; | 
					
						
							| 
									
										
										
										
											2015-01-22 11:40:55 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     len = a->length; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (len > 0) { | 
					
						
							|  |  |  |         if (a->flags & ASN1_STRING_FLAG_BITS_LEFT) { | 
					
						
							|  |  |  |             bits = (int)a->flags & 0x07; | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             for (; len > 0; len--) { | 
					
						
							|  |  |  |                 if (a->data[len - 1]) | 
					
						
							|  |  |  |                     break; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             j = a->data[len - 1]; | 
					
						
							|  |  |  |             if (j & 0x01) | 
					
						
							|  |  |  |                 bits = 0; | 
					
						
							|  |  |  |             else if (j & 0x02) | 
					
						
							|  |  |  |                 bits = 1; | 
					
						
							|  |  |  |             else if (j & 0x04) | 
					
						
							|  |  |  |                 bits = 2; | 
					
						
							|  |  |  |             else if (j & 0x08) | 
					
						
							|  |  |  |                 bits = 3; | 
					
						
							|  |  |  |             else if (j & 0x10) | 
					
						
							|  |  |  |                 bits = 4; | 
					
						
							|  |  |  |             else if (j & 0x20) | 
					
						
							|  |  |  |                 bits = 5; | 
					
						
							|  |  |  |             else if (j & 0x40) | 
					
						
							|  |  |  |                 bits = 6; | 
					
						
							|  |  |  |             else if (j & 0x80) | 
					
						
							|  |  |  |                 bits = 7; | 
					
						
							|  |  |  |             else | 
					
						
							|  |  |  |                 bits = 0;       /* should not happen */ | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } else | 
					
						
							|  |  |  |         bits = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ret = 1 + len; | 
					
						
							|  |  |  |     if (pp == NULL) | 
					
						
							| 
									
										
										
										
											2017-10-17 22:04:09 +08:00
										 |  |  |         return ret; | 
					
						
							| 
									
										
										
										
											2015-01-22 11:40:55 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     p = *pp; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     *(p++) = (unsigned char)bits; | 
					
						
							|  |  |  |     d = a->data; | 
					
						
							| 
									
										
										
										
											2016-05-27 00:40:32 +08:00
										 |  |  |     if (len > 0) { | 
					
						
							|  |  |  |         memcpy(p, d, len); | 
					
						
							|  |  |  |         p += len; | 
					
						
							| 
									
										
										
										
											2015-01-22 11:40:55 +08:00
										 |  |  |         p[-1] &= (0xff << bits); | 
					
						
							| 
									
										
										
										
											2016-05-27 00:40:32 +08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2015-01-22 11:40:55 +08:00
										 |  |  |     *pp = p; | 
					
						
							| 
									
										
										
										
											2017-10-17 22:04:09 +08:00
										 |  |  |     return ret; | 
					
						
							| 
									
										
										
										
											2015-01-22 11:40:55 +08:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											1998-12-21 18:52:47 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-03-16 07:15:26 +08:00
										 |  |  | ASN1_BIT_STRING *c2i_ASN1_BIT_STRING(ASN1_BIT_STRING **a, | 
					
						
							| 
									
										
										
										
											2015-01-22 11:40:55 +08:00
										 |  |  |                                      const unsigned char **pp, long len) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     ASN1_BIT_STRING *ret = NULL; | 
					
						
							|  |  |  |     const unsigned char *p; | 
					
						
							|  |  |  |     unsigned char *s; | 
					
						
							|  |  |  |     int i; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (len < 1) { | 
					
						
							|  |  |  |         i = ASN1_R_STRING_TOO_SHORT; | 
					
						
							|  |  |  |         goto err; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-09-19 03:58:41 +08:00
										 |  |  |     if (len > INT_MAX) { | 
					
						
							|  |  |  |         i = ASN1_R_STRING_TOO_LONG; | 
					
						
							|  |  |  |         goto err; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-22 11:40:55 +08:00
										 |  |  |     if ((a == NULL) || ((*a) == NULL)) { | 
					
						
							| 
									
										
										
										
											2015-03-14 12:16:42 +08:00
										 |  |  |         if ((ret = ASN1_BIT_STRING_new()) == NULL) | 
					
						
							| 
									
										
										
										
											2017-10-17 22:04:09 +08:00
										 |  |  |             return NULL; | 
					
						
							| 
									
										
										
										
											2015-01-22 11:40:55 +08:00
										 |  |  |     } else | 
					
						
							|  |  |  |         ret = (*a); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     p = *pp; | 
					
						
							|  |  |  |     i = *(p++); | 
					
						
							|  |  |  |     if (i > 7) { | 
					
						
							|  |  |  |         i = ASN1_R_INVALID_BIT_STRING_BITS_LEFT; | 
					
						
							|  |  |  |         goto err; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     /*
 | 
					
						
							|  |  |  |      * We do this to preserve the settings.  If we modify the settings, via | 
					
						
							|  |  |  |      * the _set_bit function, we will recalculate on output | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     ret->flags &= ~(ASN1_STRING_FLAG_BITS_LEFT | 0x07); /* clear */ | 
					
						
							|  |  |  |     ret->flags |= (ASN1_STRING_FLAG_BITS_LEFT | i); /* set */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (len-- > 1) {            /* using one because of the bits left byte */ | 
					
						
							| 
									
										
										
										
											2015-04-29 03:28:14 +08:00
										 |  |  |         s = OPENSSL_malloc((int)len); | 
					
						
							| 
									
										
										
										
											2015-01-22 11:40:55 +08:00
										 |  |  |         if (s == NULL) { | 
					
						
							|  |  |  |             i = ERR_R_MALLOC_FAILURE; | 
					
						
							|  |  |  |             goto err; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         memcpy(s, p, (int)len); | 
					
						
							|  |  |  |         s[len - 1] &= (0xff << i); | 
					
						
							|  |  |  |         p += len; | 
					
						
							|  |  |  |     } else | 
					
						
							|  |  |  |         s = NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ret->length = (int)len; | 
					
						
							| 
									
										
										
										
											2015-05-01 22:02:07 +08:00
										 |  |  |     OPENSSL_free(ret->data); | 
					
						
							| 
									
										
										
										
											2015-01-22 11:40:55 +08:00
										 |  |  |     ret->data = s; | 
					
						
							|  |  |  |     ret->type = V_ASN1_BIT_STRING; | 
					
						
							|  |  |  |     if (a != NULL) | 
					
						
							|  |  |  |         (*a) = ret; | 
					
						
							|  |  |  |     *pp = p; | 
					
						
							| 
									
										
										
										
											2017-10-17 22:04:09 +08:00
										 |  |  |     return ret; | 
					
						
							| 
									
										
										
										
											2015-01-22 11:40:55 +08:00
										 |  |  |  err: | 
					
						
							| 
									
										
										
										
											2020-11-04 19:23:19 +08:00
										 |  |  |     ERR_raise(ERR_LIB_ASN1, i); | 
					
						
							| 
									
										
										
										
											2015-03-24 19:52:24 +08:00
										 |  |  |     if ((a == NULL) || (*a != ret)) | 
					
						
							| 
									
										
										
										
											2015-03-14 12:16:42 +08:00
										 |  |  |         ASN1_BIT_STRING_free(ret); | 
					
						
							| 
									
										
										
										
											2017-10-17 22:04:09 +08:00
										 |  |  |     return NULL; | 
					
						
							| 
									
										
										
										
											2015-01-22 11:40:55 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*
 | 
					
						
							| 
									
										
										
										
											2017-10-11 05:55:09 +08:00
										 |  |  |  * These next 2 functions from Goetz Babin-Ebell. | 
					
						
							| 
									
										
										
										
											1998-12-21 18:56:39 +08:00
										 |  |  |  */ | 
					
						
							| 
									
										
										
										
											2008-11-12 11:58:08 +08:00
										 |  |  | int ASN1_BIT_STRING_set_bit(ASN1_BIT_STRING *a, int n, int value) | 
					
						
							| 
									
										
										
										
											2015-01-22 11:40:55 +08:00
										 |  |  | { | 
					
						
							|  |  |  |     int w, v, iv; | 
					
						
							|  |  |  |     unsigned char *c; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     w = n / 8; | 
					
						
							|  |  |  |     v = 1 << (7 - (n & 0x07)); | 
					
						
							|  |  |  |     iv = ~v; | 
					
						
							|  |  |  |     if (!value) | 
					
						
							|  |  |  |         v = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (a == NULL) | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     a->flags &= ~(ASN1_STRING_FLAG_BITS_LEFT | 0x07); /* clear, set on write */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if ((a->length < (w + 1)) || (a->data == NULL)) { | 
					
						
							|  |  |  |         if (!value) | 
					
						
							| 
									
										
										
										
											2017-10-17 22:04:09 +08:00
										 |  |  |             return 1;         /* Don't need to set */ | 
					
						
							| 
									
										
										
										
											2015-12-17 05:51:12 +08:00
										 |  |  |         c = OPENSSL_clear_realloc(a->data, a->length, w + 1); | 
					
						
							| 
									
										
										
										
											2015-01-22 11:40:55 +08:00
										 |  |  |         if (c == NULL) { | 
					
						
							| 
									
										
										
										
											2020-11-04 19:23:19 +08:00
										 |  |  |             ERR_raise(ERR_LIB_ASN1, ERR_R_MALLOC_FAILURE); | 
					
						
							| 
									
										
										
										
											2015-01-22 11:40:55 +08:00
										 |  |  |             return 0; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (w + 1 - a->length > 0) | 
					
						
							|  |  |  |             memset(c + a->length, 0, w + 1 - a->length); | 
					
						
							|  |  |  |         a->data = c; | 
					
						
							|  |  |  |         a->length = w + 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     a->data[w] = ((a->data[w]) & iv) | v; | 
					
						
							|  |  |  |     while ((a->length > 0) && (a->data[a->length - 1] == 0)) | 
					
						
							|  |  |  |         a->length--; | 
					
						
							| 
									
										
										
										
											2017-10-09 19:05:58 +08:00
										 |  |  |     return 1; | 
					
						
							| 
									
										
										
										
											2015-01-22 11:40:55 +08:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											1998-12-21 18:56:39 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-08-18 23:01:15 +08:00
										 |  |  | int ASN1_BIT_STRING_get_bit(const ASN1_BIT_STRING *a, int n) | 
					
						
							| 
									
										
										
										
											2015-01-22 11:40:55 +08:00
										 |  |  | { | 
					
						
							|  |  |  |     int w, v; | 
					
						
							| 
									
										
										
										
											1998-12-21 18:56:39 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-22 11:40:55 +08:00
										 |  |  |     w = n / 8; | 
					
						
							|  |  |  |     v = 1 << (7 - (n & 0x07)); | 
					
						
							|  |  |  |     if ((a == NULL) || (a->length < (w + 1)) || (a->data == NULL)) | 
					
						
							| 
									
										
										
										
											2017-10-09 19:05:58 +08:00
										 |  |  |         return 0; | 
					
						
							| 
									
										
										
										
											2015-01-22 11:40:55 +08:00
										 |  |  |     return ((a->data[w] & v) != 0); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											1998-12-21 18:56:39 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-02-13 07:11:56 +08:00
										 |  |  | /*
 | 
					
						
							| 
									
										
										
										
											2015-01-22 11:40:55 +08:00
										 |  |  |  * Checks if the given bit string contains only bits specified by | 
					
						
							| 
									
										
										
										
											2006-02-13 07:11:56 +08:00
										 |  |  |  * the flags vector. Returns 0 if there is at least one bit set in 'a' | 
					
						
							|  |  |  |  * which is not specified in 'flags', 1 otherwise. | 
					
						
							|  |  |  |  * 'len' is the length of 'flags'. | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2014-08-18 23:01:15 +08:00
										 |  |  | int ASN1_BIT_STRING_check(const ASN1_BIT_STRING *a, | 
					
						
							| 
									
										
										
										
											2015-01-22 11:40:55 +08:00
										 |  |  |                           const unsigned char *flags, int flags_len) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     int i, ok; | 
					
						
							|  |  |  |     /* Check if there is one bit set at all. */ | 
					
						
							|  |  |  |     if (!a || !a->data) | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /*
 | 
					
						
							|  |  |  |      * Check each byte of the internal representation of the bit string. | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     ok = 1; | 
					
						
							|  |  |  |     for (i = 0; i < a->length && ok; ++i) { | 
					
						
							|  |  |  |         unsigned char mask = i < flags_len ? ~flags[i] : 0xff; | 
					
						
							|  |  |  |         /* We are done if there is an unneeded bit set. */ | 
					
						
							|  |  |  |         ok = (a->data[i] & mask) == 0; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return ok; | 
					
						
							|  |  |  | } |