os/ossrv/ssl/libcrypto/src/crypto/asn1/a_object.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/a_object.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
 © Portions copyright (c) 2006 Nokia Corporation.  All rights reserved.
sl@0
    60
 */
sl@0
    61
#include <stdio.h>
sl@0
    62
#include <limits.h>
sl@0
    63
#include "cryptlib.h"
sl@0
    64
#include <openssl/buffer.h>
sl@0
    65
#include <openssl/asn1.h>
sl@0
    66
#include <openssl/objects.h>
sl@0
    67
#include <openssl/bn.h>
sl@0
    68
sl@0
    69
sl@0
    70
EXPORT_C int i2d_ASN1_OBJECT(ASN1_OBJECT *a, unsigned char **pp)
sl@0
    71
	{
sl@0
    72
	unsigned char *p;
sl@0
    73
	int objsize;
sl@0
    74
sl@0
    75
	if ((a == NULL) || (a->data == NULL)) return(0);
sl@0
    76
sl@0
    77
	objsize = ASN1_object_size(0,a->length,V_ASN1_OBJECT);
sl@0
    78
	if (pp == NULL) return objsize;
sl@0
    79
sl@0
    80
	p= *pp;
sl@0
    81
	ASN1_put_object(&p,0,a->length,V_ASN1_OBJECT,V_ASN1_UNIVERSAL);
sl@0
    82
	memcpy(p,a->data,a->length);
sl@0
    83
	p+=a->length;
sl@0
    84
sl@0
    85
	*pp=p;
sl@0
    86
	return(objsize);
sl@0
    87
	}
sl@0
    88
sl@0
    89
EXPORT_C int a2d_ASN1_OBJECT(unsigned char *out, int olen, const char *buf, int num)
sl@0
    90
	{
sl@0
    91
	int i,first,len=0,c, use_bn;
sl@0
    92
	char ftmp[24], *tmp = ftmp;
sl@0
    93
	int tmpsize = sizeof ftmp;
sl@0
    94
	const char *p;
sl@0
    95
	unsigned long l;
sl@0
    96
	BIGNUM *bl = NULL;
sl@0
    97
sl@0
    98
	if (num == 0)
sl@0
    99
		return(0);
sl@0
   100
	else if (num == -1)
sl@0
   101
		num=strlen(buf);
sl@0
   102
sl@0
   103
	p=buf;
sl@0
   104
	c= *(p++);
sl@0
   105
	num--;
sl@0
   106
	if ((c >= '0') && (c <= '2'))
sl@0
   107
		{
sl@0
   108
		first= c-'0';
sl@0
   109
		}
sl@0
   110
	else
sl@0
   111
		{
sl@0
   112
		ASN1err(ASN1_F_A2D_ASN1_OBJECT,ASN1_R_FIRST_NUM_TOO_LARGE);
sl@0
   113
		goto err;
sl@0
   114
		}
sl@0
   115
sl@0
   116
	if (num <= 0)
sl@0
   117
		{
sl@0
   118
		ASN1err(ASN1_F_A2D_ASN1_OBJECT,ASN1_R_MISSING_SECOND_NUMBER);
sl@0
   119
		goto err;
sl@0
   120
		}
sl@0
   121
	c= *(p++);
sl@0
   122
	num--;
sl@0
   123
	for (;;)
sl@0
   124
		{
sl@0
   125
		if (num <= 0) break;
sl@0
   126
		if ((c != '.') && (c != ' '))
sl@0
   127
			{
sl@0
   128
			ASN1err(ASN1_F_A2D_ASN1_OBJECT,ASN1_R_INVALID_SEPARATOR);
sl@0
   129
			goto err;
sl@0
   130
			}
sl@0
   131
		l=0;
sl@0
   132
		use_bn = 0;
sl@0
   133
		for (;;)
sl@0
   134
			{
sl@0
   135
			if (num <= 0) break;
sl@0
   136
			num--;
sl@0
   137
			c= *(p++);
sl@0
   138
			if ((c == ' ') || (c == '.'))
sl@0
   139
				break;
sl@0
   140
			if ((c < '0') || (c > '9'))
sl@0
   141
				{
sl@0
   142
				ASN1err(ASN1_F_A2D_ASN1_OBJECT,ASN1_R_INVALID_DIGIT);
sl@0
   143
				goto err;
sl@0
   144
				}
sl@0
   145
			if (!use_bn && l > (ULONG_MAX / 10L))
sl@0
   146
				{
sl@0
   147
				use_bn = 1;
sl@0
   148
				if (!bl)
sl@0
   149
					bl = BN_new();
sl@0
   150
				if (!bl || !BN_set_word(bl, l))
sl@0
   151
					goto err;
sl@0
   152
				}
sl@0
   153
			if (use_bn)
sl@0
   154
				{
sl@0
   155
				if (!BN_mul_word(bl, 10L)
sl@0
   156
					|| !BN_add_word(bl, c-'0'))
sl@0
   157
					goto err;
sl@0
   158
				}
sl@0
   159
			else
sl@0
   160
				l=l*10L+(long)(c-'0');
sl@0
   161
			}
sl@0
   162
		if (len == 0)
sl@0
   163
			{
sl@0
   164
			if ((first < 2) && (l >= 40))
sl@0
   165
				{
sl@0
   166
				ASN1err(ASN1_F_A2D_ASN1_OBJECT,ASN1_R_SECOND_NUMBER_TOO_LARGE);
sl@0
   167
				goto err;
sl@0
   168
				}
sl@0
   169
			if (use_bn)
sl@0
   170
				{
sl@0
   171
				if (!BN_add_word(bl, first * 40))
sl@0
   172
					goto err;
sl@0
   173
				}
sl@0
   174
			else
sl@0
   175
				l+=(long)first*40;
sl@0
   176
			}
sl@0
   177
		i=0;
sl@0
   178
		if (use_bn)
sl@0
   179
			{
sl@0
   180
			int blsize;
sl@0
   181
			blsize = BN_num_bits(bl);
sl@0
   182
			blsize = (blsize + 6)/7;
sl@0
   183
			if (blsize > tmpsize)
sl@0
   184
				{
sl@0
   185
				if (tmp != ftmp)
sl@0
   186
					OPENSSL_free(tmp);
sl@0
   187
				tmpsize = blsize + 32;
sl@0
   188
				tmp = OPENSSL_malloc(tmpsize);
sl@0
   189
				if (!tmp)
sl@0
   190
					goto err;
sl@0
   191
				}
sl@0
   192
			while(blsize--)
sl@0
   193
				tmp[i++] = (unsigned char)BN_div_word(bl, 0x80L);
sl@0
   194
			}
sl@0
   195
		else
sl@0
   196
			{
sl@0
   197
					
sl@0
   198
			for (;;)
sl@0
   199
				{
sl@0
   200
				tmp[i++]=(unsigned char)l&0x7f;
sl@0
   201
				l>>=7L;
sl@0
   202
				if (l == 0L) break;
sl@0
   203
				}
sl@0
   204
sl@0
   205
			}
sl@0
   206
		if (out != NULL)
sl@0
   207
			{
sl@0
   208
			if (len+i > olen)
sl@0
   209
				{
sl@0
   210
				ASN1err(ASN1_F_A2D_ASN1_OBJECT,ASN1_R_BUFFER_TOO_SMALL);
sl@0
   211
				goto err;
sl@0
   212
				}
sl@0
   213
			while (--i > 0)
sl@0
   214
				out[len++]=tmp[i]|0x80;
sl@0
   215
			out[len++]=tmp[0];
sl@0
   216
			}
sl@0
   217
		else
sl@0
   218
			len+=i;
sl@0
   219
		}
sl@0
   220
	if (tmp != ftmp)
sl@0
   221
		OPENSSL_free(tmp);
sl@0
   222
	if (bl)
sl@0
   223
		BN_free(bl);
sl@0
   224
	return(len);
sl@0
   225
err:
sl@0
   226
	if (tmp != ftmp)
sl@0
   227
		OPENSSL_free(tmp);
sl@0
   228
	if (bl)
sl@0
   229
		BN_free(bl);
sl@0
   230
	return(0);
sl@0
   231
sl@0
   232
	}
sl@0
   233
sl@0
   234
EXPORT_C int i2t_ASN1_OBJECT(char *buf, int buf_len, ASN1_OBJECT *a)
sl@0
   235
{
sl@0
   236
	return OBJ_obj2txt(buf, buf_len, a, 0);
sl@0
   237
}
sl@0
   238
sl@0
   239
EXPORT_C int i2a_ASN1_OBJECT(BIO *bp, ASN1_OBJECT *a)
sl@0
   240
	{
sl@0
   241
#ifndef SYMBIAN			
sl@0
   242
	char buf[80];
sl@0
   243
#else
sl@0
   244
  char buf[40];
sl@0
   245
#endif	
sl@0
   246
  char *p = buf;
sl@0
   247
	int i;
sl@0
   248
sl@0
   249
	if ((a == NULL) || (a->data == NULL))
sl@0
   250
		return(BIO_write(bp,"NULL",4));
sl@0
   251
	i=i2t_ASN1_OBJECT(buf,sizeof buf,a);
sl@0
   252
	if (i > (int)(sizeof(buf) - 1))
sl@0
   253
		{
sl@0
   254
		p = OPENSSL_malloc(i + 1);
sl@0
   255
		if (!p)
sl@0
   256
			return -1;
sl@0
   257
		i2t_ASN1_OBJECT(p,i + 1,a);
sl@0
   258
		}
sl@0
   259
	if (i <= 0)
sl@0
   260
		return BIO_write(bp, "<INVALID>", 9);
sl@0
   261
	BIO_write(bp,p,i);
sl@0
   262
	if (p != buf)
sl@0
   263
		OPENSSL_free(p);
sl@0
   264
	return(i);
sl@0
   265
	}
sl@0
   266
sl@0
   267
EXPORT_C ASN1_OBJECT *d2i_ASN1_OBJECT(ASN1_OBJECT **a, const unsigned char **pp,
sl@0
   268
	     long length)
sl@0
   269
{
sl@0
   270
	const unsigned char *p;
sl@0
   271
	long len;
sl@0
   272
	int tag,xclass;
sl@0
   273
	int inf,i;
sl@0
   274
	ASN1_OBJECT *ret = NULL;
sl@0
   275
	p= *pp;
sl@0
   276
	inf=ASN1_get_object(&p,&len,&tag,&xclass,length);
sl@0
   277
	if (inf & 0x80)
sl@0
   278
		{
sl@0
   279
		i=ASN1_R_BAD_OBJECT_HEADER;
sl@0
   280
		goto err;
sl@0
   281
		}
sl@0
   282
sl@0
   283
	if (tag != V_ASN1_OBJECT)
sl@0
   284
		{
sl@0
   285
		i=ASN1_R_EXPECTING_AN_OBJECT;
sl@0
   286
		goto err;
sl@0
   287
		}
sl@0
   288
	ret = c2i_ASN1_OBJECT(a, &p, len);
sl@0
   289
	if(ret) *pp = p;
sl@0
   290
	return ret;
sl@0
   291
err:
sl@0
   292
	ASN1err(ASN1_F_D2I_ASN1_OBJECT,i);
sl@0
   293
	if ((ret != NULL) && ((a == NULL) || (*a != ret)))
sl@0
   294
		ASN1_OBJECT_free(ret);
sl@0
   295
	return(NULL);
sl@0
   296
}
sl@0
   297
EXPORT_C ASN1_OBJECT *c2i_ASN1_OBJECT(ASN1_OBJECT **a, const unsigned char **pp,
sl@0
   298
	     long len)
sl@0
   299
	{
sl@0
   300
	ASN1_OBJECT *ret=NULL;
sl@0
   301
	const unsigned char *p;
sl@0
   302
	int i;
sl@0
   303
sl@0
   304
	/* only the ASN1_OBJECTs from the 'table' will have values
sl@0
   305
	 * for ->sn or ->ln */
sl@0
   306
	if ((a == NULL) || ((*a) == NULL) ||
sl@0
   307
		!((*a)->flags & ASN1_OBJECT_FLAG_DYNAMIC))
sl@0
   308
		{
sl@0
   309
		if ((ret=ASN1_OBJECT_new()) == NULL) return(NULL);
sl@0
   310
		}
sl@0
   311
	else	ret=(*a);
sl@0
   312
sl@0
   313
	p= *pp;
sl@0
   314
	if ((ret->data == NULL) || (ret->length < len))
sl@0
   315
		{
sl@0
   316
		if (ret->data != NULL) OPENSSL_free(ret->data);
sl@0
   317
		ret->data=(unsigned char *)OPENSSL_malloc(len ? (int)len : 1);
sl@0
   318
		ret->flags|=ASN1_OBJECT_FLAG_DYNAMIC_DATA;
sl@0
   319
		if (ret->data == NULL)
sl@0
   320
			{ i=ERR_R_MALLOC_FAILURE; goto err; }
sl@0
   321
		}
sl@0
   322
	memcpy(ret->data,p,(int)len);
sl@0
   323
	ret->length=(int)len;
sl@0
   324
	ret->sn=NULL;
sl@0
   325
	ret->ln=NULL;
sl@0
   326
	/* ret->flags=ASN1_OBJECT_FLAG_DYNAMIC; we know it is dynamic */
sl@0
   327
	p+=len;
sl@0
   328
sl@0
   329
	if (a != NULL) (*a)=ret;
sl@0
   330
	*pp=p;
sl@0
   331
	return(ret);
sl@0
   332
err:
sl@0
   333
	ASN1err(ASN1_F_C2I_ASN1_OBJECT,i);
sl@0
   334
	if ((ret != NULL) && ((a == NULL) || (*a != ret)))
sl@0
   335
		ASN1_OBJECT_free(ret);
sl@0
   336
	return(NULL);
sl@0
   337
	}
sl@0
   338
sl@0
   339
EXPORT_C ASN1_OBJECT *ASN1_OBJECT_new(void)
sl@0
   340
	{
sl@0
   341
	ASN1_OBJECT *ret;
sl@0
   342
sl@0
   343
	ret=(ASN1_OBJECT *)OPENSSL_malloc(sizeof(ASN1_OBJECT));
sl@0
   344
	if (ret == NULL)
sl@0
   345
		{
sl@0
   346
		ASN1err(ASN1_F_ASN1_OBJECT_NEW,ERR_R_MALLOC_FAILURE);
sl@0
   347
		return(NULL);
sl@0
   348
		}
sl@0
   349
	ret->length=0;
sl@0
   350
	ret->data=NULL;
sl@0
   351
	ret->nid=0;
sl@0
   352
	ret->sn=NULL;
sl@0
   353
	ret->ln=NULL;
sl@0
   354
	ret->flags=ASN1_OBJECT_FLAG_DYNAMIC;
sl@0
   355
	return(ret);
sl@0
   356
	}
sl@0
   357
sl@0
   358
EXPORT_C void ASN1_OBJECT_free(ASN1_OBJECT *a)
sl@0
   359
	{
sl@0
   360
	if (a == NULL) return;
sl@0
   361
	if (a->flags & ASN1_OBJECT_FLAG_DYNAMIC_STRINGS)
sl@0
   362
		{
sl@0
   363
#ifndef CONST_STRICT /* disable purely for compile-time strict const checking. Doing this on a "real" compile will cause memory leaks */
sl@0
   364
		if (a->sn != NULL) OPENSSL_free((void *)a->sn);
sl@0
   365
		if (a->ln != NULL) OPENSSL_free((void *)a->ln);
sl@0
   366
#endif
sl@0
   367
		a->sn=a->ln=NULL;
sl@0
   368
		}
sl@0
   369
	if (a->flags & ASN1_OBJECT_FLAG_DYNAMIC_DATA)
sl@0
   370
		{
sl@0
   371
		if (a->data != NULL) OPENSSL_free(a->data);
sl@0
   372
		a->data=NULL;
sl@0
   373
		a->length=0;
sl@0
   374
		}
sl@0
   375
	if (a->flags & ASN1_OBJECT_FLAG_DYNAMIC)
sl@0
   376
		OPENSSL_free(a);
sl@0
   377
	}
sl@0
   378
sl@0
   379
EXPORT_C ASN1_OBJECT *ASN1_OBJECT_create(int nid, unsigned char *data, int len,
sl@0
   380
	     const char *sn, const char *ln)
sl@0
   381
	{
sl@0
   382
	ASN1_OBJECT o;
sl@0
   383
sl@0
   384
	o.sn=sn;
sl@0
   385
	o.ln=ln;
sl@0
   386
	o.data=data;
sl@0
   387
	o.nid=nid;
sl@0
   388
	o.length=len;
sl@0
   389
	o.flags=ASN1_OBJECT_FLAG_DYNAMIC|ASN1_OBJECT_FLAG_DYNAMIC_STRINGS|
sl@0
   390
		ASN1_OBJECT_FLAG_DYNAMIC_DATA;
sl@0
   391
	return(OBJ_dup(&o));
sl@0
   392
	}
sl@0
   393
sl@0
   394
IMPLEMENT_STACK_OF(ASN1_OBJECT)
sl@0
   395
IMPLEMENT_ASN1_SET_OF(ASN1_OBJECT)