os/ossrv/ssl/libcrypto/src/crypto/asn1/asn1_lib.c
author sl
Tue, 10 Jun 2014 14:32:02 +0200
changeset 1 260cb5ec6c19
permissions -rw-r--r--
Update contrib.
sl@0
     1
/* crypto/asn1/asn1_lib.c */
sl@0
     2
/* Copyright (C) 1995-1998 Eric Young (eay@cryptsoft.com)
sl@0
     3
 * All rights reserved.
sl@0
     4
 *
sl@0
     5
 * This package is an SSL implementation written
sl@0
     6
 * by Eric Young (eay@cryptsoft.com).
sl@0
     7
 * The implementation was written so as to conform with Netscapes SSL.
sl@0
     8
 * 
sl@0
     9
 * This library is free for commercial and non-commercial use as long as
sl@0
    10
 * the following conditions are aheared to.  The following conditions
sl@0
    11
 * apply to all code found in this distribution, be it the RC4, RSA,
sl@0
    12
 * lhash, DES, etc., code; not just the SSL code.  The SSL documentation
sl@0
    13
 * included with this distribution is covered by the same copyright terms
sl@0
    14
 * except that the holder is Tim Hudson (tjh@cryptsoft.com).
sl@0
    15
 * 
sl@0
    16
 * Copyright remains Eric Young's, and as such any Copyright notices in
sl@0
    17
 * the code are not to be removed.
sl@0
    18
 * If this package is used in a product, Eric Young should be given attribution
sl@0
    19
 * as the author of the parts of the library used.
sl@0
    20
 * This can be in the form of a textual message at program startup or
sl@0
    21
 * in documentation (online or textual) provided with the package.
sl@0
    22
 * 
sl@0
    23
 * Redistribution and use in source and binary forms, with or without
sl@0
    24
 * modification, are permitted provided that the following conditions
sl@0
    25
 * are met:
sl@0
    26
 * 1. Redistributions of source code must retain the copyright
sl@0
    27
 *    notice, this list of conditions and the following disclaimer.
sl@0
    28
 * 2. Redistributions in binary form must reproduce the above copyright
sl@0
    29
 *    notice, this list of conditions and the following disclaimer in the
sl@0
    30
 *    documentation and/or other materials provided with the distribution.
sl@0
    31
 * 3. All advertising materials mentioning features or use of this software
sl@0
    32
 *    must display the following acknowledgement:
sl@0
    33
 *    "This product includes cryptographic software written by
sl@0
    34
 *     Eric Young (eay@cryptsoft.com)"
sl@0
    35
 *    The word 'cryptographic' can be left out if the rouines from the library
sl@0
    36
 *    being used are not cryptographic related :-).
sl@0
    37
 * 4. If you include any Windows specific code (or a derivative thereof) from 
sl@0
    38
 *    the apps directory (application code) you must include an acknowledgement:
sl@0
    39
 *    "This product includes software written by Tim Hudson (tjh@cryptsoft.com)"
sl@0
    40
 * 
sl@0
    41
 * THIS SOFTWARE IS PROVIDED BY ERIC YOUNG ``AS IS'' AND
sl@0
    42
 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
sl@0
    43
 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
sl@0
    44
 * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
sl@0
    45
 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
sl@0
    46
 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
sl@0
    47
 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
sl@0
    48
 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
sl@0
    49
 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
sl@0
    50
 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
sl@0
    51
 * SUCH DAMAGE.
sl@0
    52
 * 
sl@0
    53
 * The licence and distribution terms for any publically available version or
sl@0
    54
 * derivative of this code cannot be changed.  i.e. this code cannot simply be
sl@0
    55
 * copied and put under another distribution licence
sl@0
    56
 * [including the GNU Public Licence.]
sl@0
    57
 */
sl@0
    58
sl@0
    59
#include <stdio.h>
sl@0
    60
#include <limits.h>
sl@0
    61
#include "cryptlib.h"
sl@0
    62
#include <openssl/asn1.h>
sl@0
    63
#include <openssl/asn1_mac.h>
sl@0
    64
sl@0
    65
static int asn1_get_length(const unsigned char **pp,int *inf,long *rl,int max);
sl@0
    66
static void asn1_put_length(unsigned char **pp, int length);
sl@0
    67
const char ASN1_version[]="ASN.1" OPENSSL_VERSION_PTEXT;
sl@0
    68
sl@0
    69
static int _asn1_check_infinite_end(const unsigned char **p, long len)
sl@0
    70
	{
sl@0
    71
	/* If there is 0 or 1 byte left, the length check should pick
sl@0
    72
	 * things up */
sl@0
    73
	if (len <= 0)
sl@0
    74
		return(1);
sl@0
    75
	else if ((len >= 2) && ((*p)[0] == 0) && ((*p)[1] == 0))
sl@0
    76
		{
sl@0
    77
		(*p)+=2;
sl@0
    78
		return(1);
sl@0
    79
		}
sl@0
    80
	return(0);
sl@0
    81
	}
sl@0
    82
sl@0
    83
EXPORT_C int ASN1_check_infinite_end(unsigned char **p, long len)
sl@0
    84
	{
sl@0
    85
	return _asn1_check_infinite_end((const unsigned char **)p, len);
sl@0
    86
	}
sl@0
    87
sl@0
    88
EXPORT_C int ASN1_const_check_infinite_end(const unsigned char **p, long len)
sl@0
    89
	{
sl@0
    90
	return _asn1_check_infinite_end(p, len);
sl@0
    91
	}
sl@0
    92
sl@0
    93
sl@0
    94
EXPORT_C int ASN1_get_object(const unsigned char **pp, long *plength, int *ptag,
sl@0
    95
	int *pclass, long omax)
sl@0
    96
	{
sl@0
    97
	int i,ret;
sl@0
    98
	long l;
sl@0
    99
	const unsigned char *p= *pp;
sl@0
   100
	int tag,xclass,inf;
sl@0
   101
	long max=omax;
sl@0
   102
sl@0
   103
	if (!max) goto err;
sl@0
   104
	ret=(*p&V_ASN1_CONSTRUCTED);
sl@0
   105
	xclass=(*p&V_ASN1_PRIVATE);
sl@0
   106
	i= *p&V_ASN1_PRIMITIVE_TAG;
sl@0
   107
	if (i == V_ASN1_PRIMITIVE_TAG)
sl@0
   108
		{		/* high-tag */
sl@0
   109
		p++;
sl@0
   110
		if (--max == 0) goto err;
sl@0
   111
		l=0;
sl@0
   112
		while (*p&0x80)
sl@0
   113
			{
sl@0
   114
			l<<=7L;
sl@0
   115
			l|= *(p++)&0x7f;
sl@0
   116
			if (--max == 0) goto err;
sl@0
   117
			if (l > (INT_MAX >> 7L)) goto err;
sl@0
   118
			}
sl@0
   119
		l<<=7L;
sl@0
   120
		l|= *(p++)&0x7f;
sl@0
   121
		tag=(int)l;
sl@0
   122
		if (--max == 0) goto err;
sl@0
   123
		}
sl@0
   124
	else
sl@0
   125
		{ 
sl@0
   126
		tag=i;
sl@0
   127
		p++;
sl@0
   128
		if (--max == 0) goto err;
sl@0
   129
		}
sl@0
   130
	*ptag=tag;
sl@0
   131
	*pclass=xclass;
sl@0
   132
	if (!asn1_get_length(&p,&inf,plength,(int)max)) goto err;
sl@0
   133
sl@0
   134
#if 0
sl@0
   135
	fprintf(stderr,"p=%d + *plength=%ld > omax=%ld + *pp=%d  (%d > %d)\n", 
sl@0
   136
		(int)p,*plength,omax,(int)*pp,(int)(p+ *plength),
sl@0
   137
		(int)(omax+ *pp));
sl@0
   138
sl@0
   139
#endif
sl@0
   140
	if (*plength > (omax - (p - *pp)))
sl@0
   141
		{
sl@0
   142
		ASN1err(ASN1_F_ASN1_GET_OBJECT,ASN1_R_TOO_LONG);
sl@0
   143
		/* Set this so that even if things are not long enough
sl@0
   144
		 * the values are set correctly */
sl@0
   145
		ret|=0x80;
sl@0
   146
		}
sl@0
   147
	*pp=p;
sl@0
   148
	return(ret|inf);
sl@0
   149
err:
sl@0
   150
	ASN1err(ASN1_F_ASN1_GET_OBJECT,ASN1_R_HEADER_TOO_LONG);
sl@0
   151
	return(0x80);
sl@0
   152
	}
sl@0
   153
sl@0
   154
static int asn1_get_length(const unsigned char **pp, int *inf, long *rl, int max)
sl@0
   155
	{
sl@0
   156
	const unsigned char *p= *pp;
sl@0
   157
	unsigned long ret=0;
sl@0
   158
	unsigned int i;
sl@0
   159
sl@0
   160
	if (max-- < 1) return(0);
sl@0
   161
	if (*p == 0x80)
sl@0
   162
		{
sl@0
   163
		*inf=1;
sl@0
   164
		ret=0;
sl@0
   165
		p++;
sl@0
   166
		}
sl@0
   167
	else
sl@0
   168
		{
sl@0
   169
		*inf=0;
sl@0
   170
		i= *p&0x7f;
sl@0
   171
		if (*(p++) & 0x80)
sl@0
   172
			{
sl@0
   173
			if (i > sizeof(long))
sl@0
   174
				return 0;
sl@0
   175
			if (max-- == 0) return(0);
sl@0
   176
			while (i-- > 0)
sl@0
   177
				{
sl@0
   178
				ret<<=8L;
sl@0
   179
				ret|= *(p++);
sl@0
   180
				if (max-- == 0) return(0);
sl@0
   181
				}
sl@0
   182
			}
sl@0
   183
		else
sl@0
   184
			ret=i;
sl@0
   185
		}
sl@0
   186
	if (ret > LONG_MAX)
sl@0
   187
		return 0;
sl@0
   188
	*pp=p;
sl@0
   189
	*rl=(long)ret;
sl@0
   190
	return(1);
sl@0
   191
	}
sl@0
   192
sl@0
   193
/* class 0 is constructed
sl@0
   194
 * constructed == 2 for indefinite length constructed */
sl@0
   195
EXPORT_C void ASN1_put_object(unsigned char **pp, int constructed, int length, int tag,
sl@0
   196
	     int xclass)
sl@0
   197
	{
sl@0
   198
	unsigned char *p= *pp;
sl@0
   199
	int i, ttag;
sl@0
   200
sl@0
   201
	i=(constructed)?V_ASN1_CONSTRUCTED:0;
sl@0
   202
	i|=(xclass&V_ASN1_PRIVATE);
sl@0
   203
	if (tag < 31)
sl@0
   204
		*(p++)=i|(tag&V_ASN1_PRIMITIVE_TAG);
sl@0
   205
	else
sl@0
   206
		{
sl@0
   207
		*(p++)=i|V_ASN1_PRIMITIVE_TAG;
sl@0
   208
		for(i = 0, ttag = tag; ttag > 0; i++) ttag >>=7;
sl@0
   209
		ttag = i;
sl@0
   210
		while(i-- > 0)
sl@0
   211
			{
sl@0
   212
			p[i] = tag & 0x7f;
sl@0
   213
			if(i != (ttag - 1)) p[i] |= 0x80;
sl@0
   214
			tag >>= 7;
sl@0
   215
			}
sl@0
   216
		p += ttag;
sl@0
   217
		}
sl@0
   218
	if (constructed == 2)
sl@0
   219
		*(p++)=0x80;
sl@0
   220
	else
sl@0
   221
		asn1_put_length(&p,length);
sl@0
   222
	*pp=p;
sl@0
   223
	}
sl@0
   224
sl@0
   225
EXPORT_C int ASN1_put_eoc(unsigned char **pp)
sl@0
   226
	{
sl@0
   227
	unsigned char *p = *pp;
sl@0
   228
	*p++ = 0;
sl@0
   229
	*p++ = 0;
sl@0
   230
	*pp = p;
sl@0
   231
	return 2;
sl@0
   232
	}
sl@0
   233
sl@0
   234
static void asn1_put_length(unsigned char **pp, int length)
sl@0
   235
	{
sl@0
   236
	unsigned char *p= *pp;
sl@0
   237
	int i,l;
sl@0
   238
	if (length <= 127)
sl@0
   239
		*(p++)=(unsigned char)length;
sl@0
   240
	else
sl@0
   241
		{
sl@0
   242
		l=length;
sl@0
   243
		for (i=0; l > 0; i++)
sl@0
   244
			l>>=8;
sl@0
   245
		*(p++)=i|0x80;
sl@0
   246
		l=i;
sl@0
   247
		while (i-- > 0)
sl@0
   248
			{
sl@0
   249
			p[i]=length&0xff;
sl@0
   250
			length>>=8;
sl@0
   251
			}
sl@0
   252
		p+=l;
sl@0
   253
		}
sl@0
   254
	*pp=p;
sl@0
   255
	}
sl@0
   256
sl@0
   257
EXPORT_C int ASN1_object_size(int constructed, int length, int tag)
sl@0
   258
	{
sl@0
   259
	int ret;
sl@0
   260
sl@0
   261
	ret=length;
sl@0
   262
	ret++;
sl@0
   263
	if (tag >= 31)
sl@0
   264
		{
sl@0
   265
		while (tag > 0)
sl@0
   266
			{
sl@0
   267
			tag>>=7;
sl@0
   268
			ret++;
sl@0
   269
			}
sl@0
   270
		}
sl@0
   271
	if (constructed == 2)
sl@0
   272
		return ret + 3;
sl@0
   273
	ret++;
sl@0
   274
	if (length > 127)
sl@0
   275
		{
sl@0
   276
		while (length > 0)
sl@0
   277
			{
sl@0
   278
			length>>=8;
sl@0
   279
			ret++;
sl@0
   280
			}
sl@0
   281
		}
sl@0
   282
	return(ret);
sl@0
   283
	}
sl@0
   284
sl@0
   285
static int _asn1_Finish(ASN1_const_CTX *c)
sl@0
   286
	{
sl@0
   287
	if ((c->inf == (1|V_ASN1_CONSTRUCTED)) && (!c->eos))
sl@0
   288
		{
sl@0
   289
		if (!ASN1_const_check_infinite_end(&c->p,c->slen))
sl@0
   290
			{
sl@0
   291
			c->error=ERR_R_MISSING_ASN1_EOS;
sl@0
   292
			return(0);
sl@0
   293
			}
sl@0
   294
		}
sl@0
   295
	if (	((c->slen != 0) && !(c->inf & 1)) ||
sl@0
   296
		((c->slen < 0) && (c->inf & 1)))
sl@0
   297
		{
sl@0
   298
		c->error=ERR_R_ASN1_LENGTH_MISMATCH;
sl@0
   299
		return(0);
sl@0
   300
		}
sl@0
   301
	return(1);
sl@0
   302
	}
sl@0
   303
sl@0
   304
EXPORT_C int asn1_Finish(ASN1_CTX *c)
sl@0
   305
	{
sl@0
   306
	return _asn1_Finish((ASN1_const_CTX *)c);
sl@0
   307
	}
sl@0
   308
sl@0
   309
EXPORT_C int asn1_const_Finish(ASN1_const_CTX *c)
sl@0
   310
	{
sl@0
   311
	return _asn1_Finish(c);
sl@0
   312
	}
sl@0
   313
sl@0
   314
EXPORT_C int asn1_GetSequence(ASN1_const_CTX *c, long *length)
sl@0
   315
	{
sl@0
   316
	const unsigned char *q;
sl@0
   317
sl@0
   318
	q=c->p;
sl@0
   319
	c->inf=ASN1_get_object(&(c->p),&(c->slen),&(c->tag),&(c->xclass),
sl@0
   320
		*length);
sl@0
   321
	if (c->inf & 0x80)
sl@0
   322
		{
sl@0
   323
		c->error=ERR_R_BAD_GET_ASN1_OBJECT_CALL;
sl@0
   324
		return(0);
sl@0
   325
		}
sl@0
   326
	if (c->tag != V_ASN1_SEQUENCE)
sl@0
   327
		{
sl@0
   328
		c->error=ERR_R_EXPECTING_AN_ASN1_SEQUENCE;
sl@0
   329
		return(0);
sl@0
   330
		}
sl@0
   331
	(*length)-=(c->p-q);
sl@0
   332
	if (c->max && (*length < 0))
sl@0
   333
		{
sl@0
   334
		c->error=ERR_R_ASN1_LENGTH_MISMATCH;
sl@0
   335
		return(0);
sl@0
   336
		}
sl@0
   337
	if (c->inf == (1|V_ASN1_CONSTRUCTED))
sl@0
   338
		c->slen= *length+ *(c->pp)-c->p;
sl@0
   339
	c->eos=0;
sl@0
   340
	return(1);
sl@0
   341
	}
sl@0
   342
sl@0
   343
EXPORT_C ASN1_STRING *ASN1_STRING_dup(ASN1_STRING *str)
sl@0
   344
	{
sl@0
   345
	ASN1_STRING *ret;
sl@0
   346
sl@0
   347
	if (str == NULL) return(NULL);
sl@0
   348
	if ((ret=ASN1_STRING_type_new(str->type)) == NULL)
sl@0
   349
		return(NULL);
sl@0
   350
	if (!ASN1_STRING_set(ret,str->data,str->length))
sl@0
   351
		{
sl@0
   352
		ASN1_STRING_free(ret);
sl@0
   353
		return(NULL);
sl@0
   354
		}
sl@0
   355
	ret->flags = str->flags;
sl@0
   356
	return(ret);
sl@0
   357
	}
sl@0
   358
sl@0
   359
EXPORT_C int ASN1_STRING_set(ASN1_STRING *str, const void *_data, int len)
sl@0
   360
	{
sl@0
   361
	unsigned char *c;
sl@0
   362
	const char *data=_data;
sl@0
   363
sl@0
   364
	if (len < 0)
sl@0
   365
		{
sl@0
   366
		if (data == NULL)
sl@0
   367
			return(0);
sl@0
   368
		else
sl@0
   369
			len=strlen(data);
sl@0
   370
		}
sl@0
   371
	if ((str->length < len) || (str->data == NULL))
sl@0
   372
		{
sl@0
   373
		c=str->data;
sl@0
   374
		if (c == NULL)
sl@0
   375
			str->data=OPENSSL_malloc(len+1);
sl@0
   376
		else
sl@0
   377
			str->data=OPENSSL_realloc(c,len+1);
sl@0
   378
sl@0
   379
		if (str->data == NULL)
sl@0
   380
			{
sl@0
   381
			ASN1err(ASN1_F_ASN1_STRING_SET,ERR_R_MALLOC_FAILURE);
sl@0
   382
			str->data=c;
sl@0
   383
			return(0);
sl@0
   384
			}
sl@0
   385
		}
sl@0
   386
	str->length=len;
sl@0
   387
	if (data != NULL)
sl@0
   388
		{
sl@0
   389
		memcpy(str->data,data,len);
sl@0
   390
		/* an allowance for strings :-) */
sl@0
   391
		str->data[len]='\0';
sl@0
   392
		}
sl@0
   393
	return(1);
sl@0
   394
	}
sl@0
   395
sl@0
   396
EXPORT_C ASN1_STRING *ASN1_STRING_new(void)
sl@0
   397
	{
sl@0
   398
	return(ASN1_STRING_type_new(V_ASN1_OCTET_STRING));
sl@0
   399
	}
sl@0
   400
sl@0
   401
sl@0
   402
EXPORT_C ASN1_STRING *ASN1_STRING_type_new(int type)
sl@0
   403
	{
sl@0
   404
	ASN1_STRING *ret;
sl@0
   405
sl@0
   406
	ret=(ASN1_STRING *)OPENSSL_malloc(sizeof(ASN1_STRING));
sl@0
   407
	if (ret == NULL)
sl@0
   408
		{
sl@0
   409
		ASN1err(ASN1_F_ASN1_STRING_TYPE_NEW,ERR_R_MALLOC_FAILURE);
sl@0
   410
		return(NULL);
sl@0
   411
		}
sl@0
   412
	ret->length=0;
sl@0
   413
	ret->type=type;
sl@0
   414
	ret->data=NULL;
sl@0
   415
	ret->flags=0;
sl@0
   416
	return(ret);
sl@0
   417
	}
sl@0
   418
sl@0
   419
EXPORT_C void ASN1_STRING_free(ASN1_STRING *a)
sl@0
   420
	{
sl@0
   421
	if (a == NULL) return;
sl@0
   422
	if (a->data != NULL) OPENSSL_free(a->data);
sl@0
   423
	OPENSSL_free(a);
sl@0
   424
	}
sl@0
   425
sl@0
   426
EXPORT_C int ASN1_STRING_cmp(ASN1_STRING *a, ASN1_STRING *b)
sl@0
   427
	{
sl@0
   428
	int i;
sl@0
   429
sl@0
   430
	i=(a->length-b->length);
sl@0
   431
	if (i == 0)
sl@0
   432
		{
sl@0
   433
		i=memcmp(a->data,b->data,a->length);
sl@0
   434
		if (i == 0)
sl@0
   435
			return(a->type-b->type);
sl@0
   436
		else
sl@0
   437
			return(i);
sl@0
   438
		}
sl@0
   439
	else
sl@0
   440
		return(i);
sl@0
   441
	}
sl@0
   442
sl@0
   443
EXPORT_C void asn1_add_error(const unsigned char *address, int offset)
sl@0
   444
	{
sl@0
   445
	char buf1[DECIMAL_SIZE(address)+1],buf2[DECIMAL_SIZE(offset)+1];
sl@0
   446
sl@0
   447
	BIO_snprintf(buf1,sizeof buf1,"%lu",(unsigned long)address);
sl@0
   448
	BIO_snprintf(buf2,sizeof buf2,"%d",offset);
sl@0
   449
	ERR_add_error_data(4,"address=",buf1," offset=",buf2);
sl@0
   450
	}
sl@0
   451
sl@0
   452
EXPORT_C int ASN1_STRING_length(ASN1_STRING *x)
sl@0
   453
{ return M_ASN1_STRING_length(x); }
sl@0
   454
sl@0
   455
EXPORT_C void ASN1_STRING_length_set(ASN1_STRING *x, int len)
sl@0
   456
{ M_ASN1_STRING_length_set(x, len); return; }
sl@0
   457
sl@0
   458
EXPORT_C int ASN1_STRING_type(ASN1_STRING *x)
sl@0
   459
{ return M_ASN1_STRING_type(x); }
sl@0
   460
sl@0
   461
EXPORT_C unsigned char * ASN1_STRING_data(ASN1_STRING *x)
sl@0
   462
{ return M_ASN1_STRING_data(x); }