os/ossrv/ssl/libcrypto/src/crypto/asn1/asn1_par.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_par.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
 
sl@0
    62
sl@0
    63
#include <stdio.h>
sl@0
    64
#include "cryptlib.h"
sl@0
    65
#include <openssl/buffer.h>
sl@0
    66
#include <openssl/objects.h>
sl@0
    67
#include <openssl/asn1.h>
sl@0
    68
#if (defined(SYMBIAN) && (defined(__WINSCW__) || defined(__WINS__)))
sl@0
    69
#include "libcrypto_wsd.h"
sl@0
    70
#include "libcrypto_wsd_macros.h"
sl@0
    71
#endif
sl@0
    72
sl@0
    73
static int asn1_print_info(BIO *bp, int tag, int xclass,int constructed,
sl@0
    74
	int indent);
sl@0
    75
static int asn1_parse2(BIO *bp, const unsigned char **pp, long length,
sl@0
    76
	int offset, int depth, int indent, int dump);
sl@0
    77
static int asn1_print_info(BIO *bp, int tag, int xclass, int constructed,
sl@0
    78
	     int indent)
sl@0
    79
	{
sl@0
    80
	static const char fmt[]="%-18s";
sl@0
    81
	static const char fmt2[]="%2d %-15s";
sl@0
    82
#ifndef SYMBIAN	
sl@0
    83
	char str[128];
sl@0
    84
#else
sl@0
    85
  char str[50];
sl@0
    86
#endif	
sl@0
    87
	const char *p,*p2=NULL;
sl@0
    88
sl@0
    89
	if (constructed & V_ASN1_CONSTRUCTED)
sl@0
    90
		p="cons: ";
sl@0
    91
	else
sl@0
    92
		p="prim: ";
sl@0
    93
	if (BIO_write(bp,p,6) < 6) goto err;
sl@0
    94
	BIO_indent(bp,indent,128);
sl@0
    95
sl@0
    96
	p=str;
sl@0
    97
	if ((xclass & V_ASN1_PRIVATE) == V_ASN1_PRIVATE)
sl@0
    98
		BIO_snprintf(str,sizeof str,"priv [ %d ] ",tag);
sl@0
    99
	else if ((xclass & V_ASN1_CONTEXT_SPECIFIC) == V_ASN1_CONTEXT_SPECIFIC)
sl@0
   100
		BIO_snprintf(str,sizeof str,"cont [ %d ]",tag);
sl@0
   101
	else if ((xclass & V_ASN1_APPLICATION) == V_ASN1_APPLICATION)
sl@0
   102
		BIO_snprintf(str,sizeof str,"appl [ %d ]",tag);
sl@0
   103
	else if (tag > 30)
sl@0
   104
		BIO_snprintf(str,sizeof str,"<ASN1 %d>",tag);
sl@0
   105
	else p = ASN1_tag2str(tag);
sl@0
   106
sl@0
   107
	if (p2 != NULL)
sl@0
   108
		{
sl@0
   109
		if (BIO_printf(bp,fmt2,tag,p2) <= 0) goto err;
sl@0
   110
		}
sl@0
   111
	else
sl@0
   112
		{
sl@0
   113
		if (BIO_printf(bp,fmt,p) <= 0) goto err;
sl@0
   114
		}
sl@0
   115
	return(1);
sl@0
   116
err:
sl@0
   117
	return(0);
sl@0
   118
	}
sl@0
   119
sl@0
   120
EXPORT_C int ASN1_parse(BIO *bp, const unsigned char *pp, long len, int indent)
sl@0
   121
	{
sl@0
   122
	return(asn1_parse2(bp,&pp,len,0,0,indent,0));
sl@0
   123
	}
sl@0
   124
sl@0
   125
EXPORT_C int ASN1_parse_dump(BIO *bp, const unsigned char *pp, long len, int indent, int dump)
sl@0
   126
	{
sl@0
   127
	return(asn1_parse2(bp,&pp,len,0,0,indent,dump));
sl@0
   128
	}
sl@0
   129
sl@0
   130
static int asn1_parse2(BIO *bp, const unsigned char **pp, long length, int offset,
sl@0
   131
	     int depth, int indent, int dump)
sl@0
   132
	{
sl@0
   133
	const unsigned char *p,*ep,*tot,*op,*opp;
sl@0
   134
	long len;
sl@0
   135
	int tag,xclass,ret=0;
sl@0
   136
	int nl,hl,j,r;
sl@0
   137
	ASN1_OBJECT *o=NULL;
sl@0
   138
	ASN1_OCTET_STRING *os=NULL;
sl@0
   139
	/* ASN1_BMPSTRING *bmp=NULL;*/
sl@0
   140
	int dump_indent;
sl@0
   141
sl@0
   142
#if 0
sl@0
   143
	dump_indent = indent;
sl@0
   144
#else
sl@0
   145
	dump_indent = 6;	/* Because we know BIO_dump_indent() */
sl@0
   146
#endif
sl@0
   147
	p= *pp;
sl@0
   148
	tot=p+length;
sl@0
   149
	op=p-1;
sl@0
   150
	while ((p < tot) && (op < p))
sl@0
   151
		{
sl@0
   152
		op=p;
sl@0
   153
		j=ASN1_get_object(&p,&len,&tag,&xclass,length);
sl@0
   154
#ifdef LINT
sl@0
   155
		j=j;
sl@0
   156
#endif
sl@0
   157
		if (j & 0x80)
sl@0
   158
			{
sl@0
   159
			if (BIO_write(bp,"Error in encoding\n",18) <= 0)
sl@0
   160
				goto end;
sl@0
   161
			ret=0;
sl@0
   162
			goto end;
sl@0
   163
			}
sl@0
   164
		hl=(p-op);
sl@0
   165
		length-=hl;
sl@0
   166
		/* if j == 0x21 it is a constructed indefinite length object */
sl@0
   167
		if (BIO_printf(bp,"%5ld:",(long)offset+(long)(op- *pp))
sl@0
   168
			<= 0) goto end;
sl@0
   169
sl@0
   170
		if (j != (V_ASN1_CONSTRUCTED | 1))
sl@0
   171
			{
sl@0
   172
			if (BIO_printf(bp,"d=%-2d hl=%ld l=%4ld ",
sl@0
   173
				depth,(long)hl,len) <= 0)
sl@0
   174
				goto end;
sl@0
   175
			}
sl@0
   176
		else
sl@0
   177
			{
sl@0
   178
			if (BIO_printf(bp,"d=%-2d hl=%ld l=inf  ",
sl@0
   179
				depth,(long)hl) <= 0)
sl@0
   180
				goto end;
sl@0
   181
			}
sl@0
   182
		if (!asn1_print_info(bp,tag,xclass,j,(indent)?depth:0))
sl@0
   183
			goto end;
sl@0
   184
		if (j & V_ASN1_CONSTRUCTED)
sl@0
   185
			{
sl@0
   186
			ep=p+len;
sl@0
   187
			if (BIO_write(bp,"\n",1) <= 0) goto end;
sl@0
   188
			if (len > length)
sl@0
   189
				{
sl@0
   190
				BIO_printf(bp,
sl@0
   191
					"length is greater than %ld\n",length);
sl@0
   192
				ret=0;
sl@0
   193
				goto end;
sl@0
   194
				}
sl@0
   195
			if ((j == 0x21) && (len == 0))
sl@0
   196
				{
sl@0
   197
				for (;;)
sl@0
   198
					{
sl@0
   199
					r=asn1_parse2(bp,&p,(long)(tot-p),
sl@0
   200
						offset+(p - *pp),depth+1,
sl@0
   201
						indent,dump);
sl@0
   202
					if (r == 0) { ret=0; goto end; }
sl@0
   203
					if ((r == 2) || (p >= tot)) break;
sl@0
   204
					}
sl@0
   205
				}
sl@0
   206
			else
sl@0
   207
				while (p < ep)
sl@0
   208
					{
sl@0
   209
					r=asn1_parse2(bp,&p,(long)len,
sl@0
   210
						offset+(p - *pp),depth+1,
sl@0
   211
						indent,dump);
sl@0
   212
					if (r == 0) { ret=0; goto end; }
sl@0
   213
					}
sl@0
   214
			}
sl@0
   215
		else if (xclass != 0)
sl@0
   216
			{
sl@0
   217
			p+=len;
sl@0
   218
			if (BIO_write(bp,"\n",1) <= 0) goto end;
sl@0
   219
			}
sl@0
   220
		else
sl@0
   221
			{
sl@0
   222
			nl=0;
sl@0
   223
			if (	(tag == V_ASN1_PRINTABLESTRING) ||
sl@0
   224
				(tag == V_ASN1_T61STRING) ||
sl@0
   225
				(tag == V_ASN1_IA5STRING) ||
sl@0
   226
				(tag == V_ASN1_VISIBLESTRING) ||
sl@0
   227
				(tag == V_ASN1_UTCTIME) ||
sl@0
   228
				(tag == V_ASN1_GENERALIZEDTIME))
sl@0
   229
				{
sl@0
   230
				if (BIO_write(bp,":",1) <= 0) goto end;
sl@0
   231
				if ((len > 0) &&
sl@0
   232
					BIO_write(bp,(const char *)p,(int)len)
sl@0
   233
					!= (int)len)
sl@0
   234
					goto end;
sl@0
   235
				}
sl@0
   236
			else if (tag == V_ASN1_OBJECT)
sl@0
   237
				{
sl@0
   238
				opp=op;
sl@0
   239
				if (d2i_ASN1_OBJECT(&o,&opp,len+hl) != NULL)
sl@0
   240
					{
sl@0
   241
					if (BIO_write(bp,":",1) <= 0) goto end;
sl@0
   242
					i2a_ASN1_OBJECT(bp,o);
sl@0
   243
					}
sl@0
   244
				else
sl@0
   245
					{
sl@0
   246
					if (BIO_write(bp,":BAD OBJECT",11) <= 0)
sl@0
   247
						goto end;
sl@0
   248
					}
sl@0
   249
				}
sl@0
   250
			else if (tag == V_ASN1_BOOLEAN)
sl@0
   251
				{
sl@0
   252
				int ii;
sl@0
   253
sl@0
   254
				opp=op;
sl@0
   255
				ii=d2i_ASN1_BOOLEAN(NULL,&opp,len+hl);
sl@0
   256
				if (ii < 0)
sl@0
   257
					{
sl@0
   258
					if (BIO_write(bp,"Bad boolean\n",12))
sl@0
   259
						goto end;
sl@0
   260
					}
sl@0
   261
				BIO_printf(bp,":%d",ii);
sl@0
   262
				}
sl@0
   263
			else if (tag == V_ASN1_BMPSTRING)
sl@0
   264
				{
sl@0
   265
				/* do the BMP thang */
sl@0
   266
				}
sl@0
   267
			else if (tag == V_ASN1_OCTET_STRING)
sl@0
   268
				{
sl@0
   269
				int i,printable=1;
sl@0
   270
sl@0
   271
				opp=op;
sl@0
   272
				os=d2i_ASN1_OCTET_STRING(NULL,&opp,len+hl);
sl@0
   273
				if (os != NULL && os->length > 0)
sl@0
   274
					{
sl@0
   275
					opp = os->data;
sl@0
   276
					/* testing whether the octet string is
sl@0
   277
					 * printable */
sl@0
   278
					for (i=0; i<os->length; i++)
sl@0
   279
						{
sl@0
   280
						if ((	(opp[i] < ' ') &&
sl@0
   281
							(opp[i] != '\n') &&
sl@0
   282
							(opp[i] != '\r') &&
sl@0
   283
							(opp[i] != '\t')) ||
sl@0
   284
							(opp[i] > '~'))
sl@0
   285
							{
sl@0
   286
							printable=0;
sl@0
   287
							break;
sl@0
   288
							}
sl@0
   289
						}
sl@0
   290
					if (printable)
sl@0
   291
					/* printable string */
sl@0
   292
						{
sl@0
   293
						if (BIO_write(bp,":",1) <= 0)
sl@0
   294
							goto end;
sl@0
   295
						if (BIO_write(bp,(const char *)opp,
sl@0
   296
							os->length) <= 0)
sl@0
   297
							goto end;
sl@0
   298
						}
sl@0
   299
					else if (!dump)
sl@0
   300
					/* not printable => print octet string
sl@0
   301
					 * as hex dump */
sl@0
   302
						{
sl@0
   303
						if (BIO_write(bp,"[HEX DUMP]:",11) <= 0)
sl@0
   304
							goto end;
sl@0
   305
						for (i=0; i<os->length; i++)
sl@0
   306
							{
sl@0
   307
							if (BIO_printf(bp,"%02X"
sl@0
   308
								, opp[i]) <= 0)
sl@0
   309
								goto end;
sl@0
   310
							}
sl@0
   311
						}
sl@0
   312
					else
sl@0
   313
					/* print the normal dump */
sl@0
   314
						{
sl@0
   315
						if (!nl) 
sl@0
   316
							{
sl@0
   317
							if (BIO_write(bp,"\n",1) <= 0)
sl@0
   318
								goto end;
sl@0
   319
							}
sl@0
   320
						if (BIO_dump_indent(bp,
sl@0
   321
							(const char *)opp,
sl@0
   322
							((dump == -1 || dump > 
sl@0
   323
							os->length)?os->length:dump),
sl@0
   324
							dump_indent) <= 0)
sl@0
   325
							goto end;
sl@0
   326
						nl=1;
sl@0
   327
						}
sl@0
   328
					}
sl@0
   329
				if (os != NULL)
sl@0
   330
					{
sl@0
   331
					M_ASN1_OCTET_STRING_free(os);
sl@0
   332
					os=NULL;
sl@0
   333
					}
sl@0
   334
				}
sl@0
   335
			else if (tag == V_ASN1_INTEGER)
sl@0
   336
				{
sl@0
   337
				ASN1_INTEGER *bs;
sl@0
   338
				int i;
sl@0
   339
sl@0
   340
				opp=op;
sl@0
   341
				bs=d2i_ASN1_INTEGER(NULL,&opp,len+hl);
sl@0
   342
				if (bs != NULL)
sl@0
   343
					{
sl@0
   344
					if (BIO_write(bp,":",1) <= 0) goto end;
sl@0
   345
					if (bs->type == V_ASN1_NEG_INTEGER)
sl@0
   346
						if (BIO_write(bp,"-",1) <= 0)
sl@0
   347
							goto end;
sl@0
   348
					for (i=0; i<bs->length; i++)
sl@0
   349
						{
sl@0
   350
						if (BIO_printf(bp,"%02X",
sl@0
   351
							bs->data[i]) <= 0)
sl@0
   352
							goto end;
sl@0
   353
						}
sl@0
   354
					if (bs->length == 0)
sl@0
   355
						{
sl@0
   356
						if (BIO_write(bp,"00",2) <= 0)
sl@0
   357
							goto end;
sl@0
   358
						}
sl@0
   359
					}
sl@0
   360
				else
sl@0
   361
					{
sl@0
   362
					if (BIO_write(bp,"BAD INTEGER",11) <= 0)
sl@0
   363
						goto end;
sl@0
   364
					}
sl@0
   365
				M_ASN1_INTEGER_free(bs);
sl@0
   366
				}
sl@0
   367
			else if (tag == V_ASN1_ENUMERATED)
sl@0
   368
				{
sl@0
   369
				ASN1_ENUMERATED *bs;
sl@0
   370
				int i;
sl@0
   371
sl@0
   372
				opp=op;
sl@0
   373
				bs=d2i_ASN1_ENUMERATED(NULL,&opp,len+hl);
sl@0
   374
				if (bs != NULL)
sl@0
   375
					{
sl@0
   376
					if (BIO_write(bp,":",1) <= 0) goto end;
sl@0
   377
					if (bs->type == V_ASN1_NEG_ENUMERATED)
sl@0
   378
						if (BIO_write(bp,"-",1) <= 0)
sl@0
   379
							goto end;
sl@0
   380
					for (i=0; i<bs->length; i++)
sl@0
   381
						{
sl@0
   382
						if (BIO_printf(bp,"%02X",
sl@0
   383
							bs->data[i]) <= 0)
sl@0
   384
							goto end;
sl@0
   385
						}
sl@0
   386
					if (bs->length == 0)
sl@0
   387
						{
sl@0
   388
						if (BIO_write(bp,"00",2) <= 0)
sl@0
   389
							goto end;
sl@0
   390
						}
sl@0
   391
					}
sl@0
   392
				else
sl@0
   393
					{
sl@0
   394
					if (BIO_write(bp,"BAD ENUMERATED",11) <= 0)
sl@0
   395
						goto end;
sl@0
   396
					}
sl@0
   397
				M_ASN1_ENUMERATED_free(bs);
sl@0
   398
				}
sl@0
   399
			else if (len > 0 && dump)
sl@0
   400
				{
sl@0
   401
				if (!nl) 
sl@0
   402
					{
sl@0
   403
					if (BIO_write(bp,"\n",1) <= 0)
sl@0
   404
						goto end;
sl@0
   405
					}
sl@0
   406
				if (BIO_dump_indent(bp,(const char *)p,
sl@0
   407
					((dump == -1 || dump > len)?len:dump),
sl@0
   408
					dump_indent) <= 0)
sl@0
   409
					goto end;
sl@0
   410
				nl=1;
sl@0
   411
				}
sl@0
   412
sl@0
   413
			if (!nl) 
sl@0
   414
				{
sl@0
   415
				if (BIO_write(bp,"\n",1) <= 0) goto end;
sl@0
   416
				}
sl@0
   417
			p+=len;
sl@0
   418
			if ((tag == V_ASN1_EOC) && (xclass == 0))
sl@0
   419
				{
sl@0
   420
				ret=2; /* End of sequence */
sl@0
   421
				goto end;
sl@0
   422
				}
sl@0
   423
			}
sl@0
   424
		length-=len;
sl@0
   425
		}
sl@0
   426
	ret=1;
sl@0
   427
end:
sl@0
   428
	if (o != NULL) ASN1_OBJECT_free(o);
sl@0
   429
	if (os != NULL) M_ASN1_OCTET_STRING_free(os);
sl@0
   430
	*pp=p;
sl@0
   431
	return(ret);
sl@0
   432
	}
sl@0
   433
sl@0
   434
EXPORT_C const char *ASN1_tag2str(int tag)
sl@0
   435
{
sl@0
   436
#ifndef EMULATOR
sl@0
   437
	static const char *tag2str[] = {
sl@0
   438
	 "EOC", "BOOLEAN", "INTEGER", "BIT STRING", "OCTET STRING", /* 0-4 */
sl@0
   439
	 "NULL", "OBJECT", "OBJECT DESCRIPTOR", "EXTERNAL", "REAL", /* 5-9 */
sl@0
   440
	 "ENUMERATED", "<ASN1 11>", "UTF8STRING", "<ASN1 13>", 	    /* 10-13 */
sl@0
   441
	"<ASN1 14>", "<ASN1 15>", "SEQUENCE", "SET", 		    /* 15-17 */
sl@0
   442
	"NUMERICSTRING", "PRINTABLESTRING", "T61STRING",	    /* 18-20 */
sl@0
   443
	"VIDEOTEXSTRING", "IA5STRING", "UTCTIME","GENERALIZEDTIME", /* 21-24 */
sl@0
   444
	"GRAPHICSTRING", "VISIBLESTRING", "GENERALSTRING",	    /* 25-27 */
sl@0
   445
	"UNIVERSALSTRING", "<ASN1 29>", "BMPSTRING"		    /* 28-30 */
sl@0
   446
	};
sl@0
   447
#else
sl@0
   448
	static const char *const tag2str[] = {
sl@0
   449
	 "EOC", "BOOLEAN", "INTEGER", "BIT STRING", "OCTET STRING", /* 0-4 */
sl@0
   450
	 "NULL", "OBJECT", "OBJECT DESCRIPTOR", "EXTERNAL", "REAL", /* 5-9 */
sl@0
   451
	 "ENUMERATED", "<ASN1 11>", "UTF8STRING", "<ASN1 13>", 	    /* 10-13 */
sl@0
   452
	"<ASN1 14>", "<ASN1 15>", "SEQUENCE", "SET", 		    /* 15-17 */
sl@0
   453
	"NUMERICSTRING", "PRINTABLESTRING", "T61STRING",	    /* 18-20 */
sl@0
   454
	"VIDEOTEXSTRING", "IA5STRING", "UTCTIME","GENERALIZEDTIME", /* 21-24 */
sl@0
   455
	"GRAPHICSTRING", "VISIBLESTRING", "GENERALSTRING",	    /* 25-27 */
sl@0
   456
	"UNIVERSALSTRING", "<ASN1 29>", "BMPSTRING"		    /* 28-30 */
sl@0
   457
	};
sl@0
   458
	
sl@0
   459
	
sl@0
   460
#endif
sl@0
   461
	if((tag == V_ASN1_NEG_INTEGER) || (tag == V_ASN1_NEG_ENUMERATED))
sl@0
   462
							tag &= ~0x100;
sl@0
   463
sl@0
   464
	if(tag < 0 || tag > 30) return "(unknown)";
sl@0
   465
	return tag2str[tag];
sl@0
   466
}
sl@0
   467