os/ossrv/ssl/libcrypto/src/crypto/asn1/a_mbstr.c
author sl
Tue, 10 Jun 2014 14:32:02 +0200
changeset 1 260cb5ec6c19
permissions -rw-r--r--
Update contrib.
sl@0
     1
/* a_mbstr.c */
sl@0
     2
/* Written by Dr Stephen N Henson (shenson@bigfoot.com) for the OpenSSL
sl@0
     3
 * project 1999.
sl@0
     4
 */
sl@0
     5
/* ====================================================================
sl@0
     6
 * Copyright (c) 1999 The OpenSSL Project.  All rights reserved.
sl@0
     7
 *
sl@0
     8
 * Redistribution and use in source and binary forms, with or without
sl@0
     9
 * modification, are permitted provided that the following conditions
sl@0
    10
 * are met:
sl@0
    11
 *
sl@0
    12
 * 1. Redistributions of source code must retain the above copyright
sl@0
    13
 *    notice, this list of conditions and the following disclaimer. 
sl@0
    14
 *
sl@0
    15
 * 2. Redistributions in binary form must reproduce the above copyright
sl@0
    16
 *    notice, this list of conditions and the following disclaimer in
sl@0
    17
 *    the documentation and/or other materials provided with the
sl@0
    18
 *    distribution.
sl@0
    19
 *
sl@0
    20
 * 3. All advertising materials mentioning features or use of this
sl@0
    21
 *    software must display the following acknowledgment:
sl@0
    22
 *    "This product includes software developed by the OpenSSL Project
sl@0
    23
 *    for use in the OpenSSL Toolkit. (http://www.OpenSSL.org/)"
sl@0
    24
 *
sl@0
    25
 * 4. The names "OpenSSL Toolkit" and "OpenSSL Project" must not be used to
sl@0
    26
 *    endorse or promote products derived from this software without
sl@0
    27
 *    prior written permission. For written permission, please contact
sl@0
    28
 *    licensing@OpenSSL.org.
sl@0
    29
 *
sl@0
    30
 * 5. Products derived from this software may not be called "OpenSSL"
sl@0
    31
 *    nor may "OpenSSL" appear in their names without prior written
sl@0
    32
 *    permission of the OpenSSL Project.
sl@0
    33
 *
sl@0
    34
 * 6. Redistributions of any form whatsoever must retain the following
sl@0
    35
 *    acknowledgment:
sl@0
    36
 *    "This product includes software developed by the OpenSSL Project
sl@0
    37
 *    for use in the OpenSSL Toolkit (http://www.OpenSSL.org/)"
sl@0
    38
 *
sl@0
    39
 * THIS SOFTWARE IS PROVIDED BY THE OpenSSL PROJECT ``AS IS'' AND ANY
sl@0
    40
 * EXPRESSED OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
sl@0
    41
 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
sl@0
    42
 * PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL THE OpenSSL PROJECT OR
sl@0
    43
 * ITS CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
sl@0
    44
 * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
sl@0
    45
 * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
sl@0
    46
 * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
sl@0
    47
 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
sl@0
    48
 * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
sl@0
    49
 * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED
sl@0
    50
 * OF THE POSSIBILITY OF SUCH DAMAGE.
sl@0
    51
 * ====================================================================
sl@0
    52
 *
sl@0
    53
 * This product includes cryptographic software written by Eric Young
sl@0
    54
 * (eay@cryptsoft.com).  This product includes software written by Tim
sl@0
    55
 * Hudson (tjh@cryptsoft.com).
sl@0
    56
 *
sl@0
    57
 */
sl@0
    58
sl@0
    59
#include <stdio.h>
sl@0
    60
#include <ctype.h>
sl@0
    61
#include "cryptlib.h"
sl@0
    62
#include <openssl/asn1.h>
sl@0
    63
sl@0
    64
static int traverse_string(const unsigned char *p, int len, int inform,
sl@0
    65
		 int (*rfunc)(unsigned long value, void *in), void *arg);
sl@0
    66
static int in_utf8(unsigned long value, void *arg);
sl@0
    67
static int out_utf8(unsigned long value, void *arg);
sl@0
    68
static int type_str(unsigned long value, void *arg);
sl@0
    69
static int cpy_asc(unsigned long value, void *arg);
sl@0
    70
static int cpy_bmp(unsigned long value, void *arg);
sl@0
    71
static int cpy_univ(unsigned long value, void *arg);
sl@0
    72
static int cpy_utf8(unsigned long value, void *arg);
sl@0
    73
static int is_printable(unsigned long value);
sl@0
    74
sl@0
    75
/* These functions take a string in UTF8, ASCII or multibyte form and
sl@0
    76
 * a mask of permissible ASN1 string types. It then works out the minimal
sl@0
    77
 * type (using the order Printable < IA5 < T61 < BMP < Universal < UTF8)
sl@0
    78
 * and creates a string of the correct type with the supplied data.
sl@0
    79
 * Yes this is horrible: it has to be :-(
sl@0
    80
 * The 'ncopy' form checks minimum and maximum size limits too.
sl@0
    81
 */
sl@0
    82
sl@0
    83
EXPORT_C int ASN1_mbstring_copy(ASN1_STRING **out, const unsigned char *in, int len,
sl@0
    84
					int inform, unsigned long mask)
sl@0
    85
{
sl@0
    86
	return ASN1_mbstring_ncopy(out, in, len, inform, mask, 0, 0);
sl@0
    87
}
sl@0
    88
sl@0
    89
EXPORT_C int ASN1_mbstring_ncopy(ASN1_STRING **out, const unsigned char *in, int len,
sl@0
    90
					int inform, unsigned long mask, 
sl@0
    91
					long minsize, long maxsize)
sl@0
    92
{
sl@0
    93
	int str_type;
sl@0
    94
	int ret;
sl@0
    95
	char free_out;
sl@0
    96
	int outform, outlen;
sl@0
    97
	ASN1_STRING *dest;
sl@0
    98
	unsigned char *p;
sl@0
    99
	int nchar;
sl@0
   100
	char strbuf[32];
sl@0
   101
	int (*cpyfunc)(unsigned long,void *) = NULL;
sl@0
   102
	if(len == -1) len = strlen((const char *)in);
sl@0
   103
	if(!mask) mask = DIRSTRING_TYPE;
sl@0
   104
sl@0
   105
	/* First do a string check and work out the number of characters */
sl@0
   106
	switch(inform) {
sl@0
   107
sl@0
   108
		case MBSTRING_BMP:
sl@0
   109
		if(len & 1) {
sl@0
   110
			ASN1err(ASN1_F_ASN1_MBSTRING_NCOPY,
sl@0
   111
					 ASN1_R_INVALID_BMPSTRING_LENGTH);
sl@0
   112
			return -1;
sl@0
   113
		}
sl@0
   114
		nchar = len >> 1;
sl@0
   115
		break;
sl@0
   116
sl@0
   117
		case MBSTRING_UNIV:
sl@0
   118
		if(len & 3) {
sl@0
   119
			ASN1err(ASN1_F_ASN1_MBSTRING_NCOPY,
sl@0
   120
					 ASN1_R_INVALID_UNIVERSALSTRING_LENGTH);
sl@0
   121
			return -1;
sl@0
   122
		}
sl@0
   123
		nchar = len >> 2;
sl@0
   124
		break;
sl@0
   125
sl@0
   126
		case MBSTRING_UTF8:
sl@0
   127
		nchar = 0;
sl@0
   128
		/* This counts the characters and does utf8 syntax checking */
sl@0
   129
		ret = traverse_string(in, len, MBSTRING_UTF8, in_utf8, &nchar);
sl@0
   130
		if(ret < 0) {
sl@0
   131
			ASN1err(ASN1_F_ASN1_MBSTRING_NCOPY,
sl@0
   132
						 ASN1_R_INVALID_UTF8STRING);
sl@0
   133
			return -1;
sl@0
   134
		}
sl@0
   135
		break;
sl@0
   136
sl@0
   137
		case MBSTRING_ASC:
sl@0
   138
		nchar = len;
sl@0
   139
		break;
sl@0
   140
sl@0
   141
		default:
sl@0
   142
		ASN1err(ASN1_F_ASN1_MBSTRING_NCOPY, ASN1_R_UNKNOWN_FORMAT);
sl@0
   143
		return -1;
sl@0
   144
	}
sl@0
   145
sl@0
   146
	if((minsize > 0) && (nchar < minsize)) {
sl@0
   147
		ASN1err(ASN1_F_ASN1_MBSTRING_NCOPY, ASN1_R_STRING_TOO_SHORT);
sl@0
   148
		BIO_snprintf(strbuf, sizeof strbuf, "%ld", minsize);
sl@0
   149
		ERR_add_error_data(2, "minsize=", strbuf);
sl@0
   150
		return -1;
sl@0
   151
	}
sl@0
   152
sl@0
   153
	if((maxsize > 0) && (nchar > maxsize)) {
sl@0
   154
		ASN1err(ASN1_F_ASN1_MBSTRING_NCOPY, ASN1_R_STRING_TOO_LONG);
sl@0
   155
		BIO_snprintf(strbuf, sizeof strbuf, "%ld", maxsize);
sl@0
   156
		ERR_add_error_data(2, "maxsize=", strbuf);
sl@0
   157
		return -1;
sl@0
   158
	}
sl@0
   159
sl@0
   160
	/* Now work out minimal type (if any) */
sl@0
   161
	if(traverse_string(in, len, inform, type_str, &mask) < 0) {
sl@0
   162
		ASN1err(ASN1_F_ASN1_MBSTRING_NCOPY, ASN1_R_ILLEGAL_CHARACTERS);
sl@0
   163
		return -1;
sl@0
   164
	}
sl@0
   165
sl@0
   166
sl@0
   167
	/* Now work out output format and string type */
sl@0
   168
	outform = MBSTRING_ASC;
sl@0
   169
	if(mask & B_ASN1_PRINTABLESTRING) str_type = V_ASN1_PRINTABLESTRING;
sl@0
   170
	else if(mask & B_ASN1_IA5STRING) str_type = V_ASN1_IA5STRING;
sl@0
   171
	else if(mask & B_ASN1_T61STRING) str_type = V_ASN1_T61STRING;
sl@0
   172
	else if(mask & B_ASN1_BMPSTRING) {
sl@0
   173
		str_type = V_ASN1_BMPSTRING;
sl@0
   174
		outform = MBSTRING_BMP;
sl@0
   175
	} else if(mask & B_ASN1_UNIVERSALSTRING) {
sl@0
   176
		str_type = V_ASN1_UNIVERSALSTRING;
sl@0
   177
		outform = MBSTRING_UNIV;
sl@0
   178
	} else {
sl@0
   179
		str_type = V_ASN1_UTF8STRING;
sl@0
   180
		outform = MBSTRING_UTF8;
sl@0
   181
	}
sl@0
   182
	if(!out) return str_type;
sl@0
   183
	if(*out) {
sl@0
   184
		free_out = 0;
sl@0
   185
		dest = *out;
sl@0
   186
		if(dest->data) {
sl@0
   187
			dest->length = 0;
sl@0
   188
			OPENSSL_free(dest->data);
sl@0
   189
			dest->data = NULL;
sl@0
   190
		}
sl@0
   191
		dest->type = str_type;
sl@0
   192
	} else {
sl@0
   193
		free_out = 1;
sl@0
   194
		dest = ASN1_STRING_type_new(str_type);
sl@0
   195
		if(!dest) {
sl@0
   196
			ASN1err(ASN1_F_ASN1_MBSTRING_NCOPY,
sl@0
   197
							ERR_R_MALLOC_FAILURE);
sl@0
   198
			return -1;
sl@0
   199
		}
sl@0
   200
		*out = dest;
sl@0
   201
	}
sl@0
   202
	/* If both the same type just copy across */
sl@0
   203
	if(inform == outform) {
sl@0
   204
		if(!ASN1_STRING_set(dest, in, len)) {
sl@0
   205
			ASN1err(ASN1_F_ASN1_MBSTRING_NCOPY,ERR_R_MALLOC_FAILURE);
sl@0
   206
			return -1;
sl@0
   207
		}
sl@0
   208
		return str_type;
sl@0
   209
	} 
sl@0
   210
sl@0
   211
	/* Work out how much space the destination will need */
sl@0
   212
	switch(outform) {
sl@0
   213
		case MBSTRING_ASC:
sl@0
   214
		outlen = nchar;
sl@0
   215
		cpyfunc = cpy_asc;
sl@0
   216
		break;
sl@0
   217
sl@0
   218
		case MBSTRING_BMP:
sl@0
   219
		outlen = nchar << 1;
sl@0
   220
		cpyfunc = cpy_bmp;
sl@0
   221
		break;
sl@0
   222
sl@0
   223
		case MBSTRING_UNIV:
sl@0
   224
		outlen = nchar << 2;
sl@0
   225
		cpyfunc = cpy_univ;
sl@0
   226
		break;
sl@0
   227
sl@0
   228
		case MBSTRING_UTF8:
sl@0
   229
		outlen = 0;
sl@0
   230
		traverse_string(in, len, inform, out_utf8, &outlen);
sl@0
   231
		cpyfunc = cpy_utf8;
sl@0
   232
		break;
sl@0
   233
	}
sl@0
   234
	if(!(p = OPENSSL_malloc(outlen + 1))) {
sl@0
   235
		if(free_out) ASN1_STRING_free(dest);
sl@0
   236
		ASN1err(ASN1_F_ASN1_MBSTRING_NCOPY,ERR_R_MALLOC_FAILURE);
sl@0
   237
		return -1;
sl@0
   238
	}
sl@0
   239
	dest->length = outlen;
sl@0
   240
	dest->data = p;
sl@0
   241
	p[outlen] = 0;
sl@0
   242
	traverse_string(in, len, inform, cpyfunc, &p);
sl@0
   243
	return str_type;	
sl@0
   244
}
sl@0
   245
sl@0
   246
/* This function traverses a string and passes the value of each character
sl@0
   247
 * to an optional function along with a void * argument.
sl@0
   248
 */
sl@0
   249
sl@0
   250
static int traverse_string(const unsigned char *p, int len, int inform,
sl@0
   251
		 int (*rfunc)(unsigned long value, void *in), void *arg)
sl@0
   252
{
sl@0
   253
	unsigned long value;
sl@0
   254
	int ret;
sl@0
   255
	while(len) {
sl@0
   256
		if(inform == MBSTRING_ASC) {
sl@0
   257
			value = *p++;
sl@0
   258
			len--;
sl@0
   259
		} else if(inform == MBSTRING_BMP) {
sl@0
   260
			value = *p++ << 8;
sl@0
   261
			value |= *p++;
sl@0
   262
			len -= 2;
sl@0
   263
		} else if(inform == MBSTRING_UNIV) {
sl@0
   264
			value = ((unsigned long)*p++) << 24;
sl@0
   265
			value |= ((unsigned long)*p++) << 16;
sl@0
   266
			value |= *p++ << 8;
sl@0
   267
			value |= *p++;
sl@0
   268
			len -= 4;
sl@0
   269
		} else {
sl@0
   270
			ret = UTF8_getc(p, len, &value);
sl@0
   271
			if(ret < 0) return -1;
sl@0
   272
			len -= ret;
sl@0
   273
			p += ret;
sl@0
   274
		}
sl@0
   275
		if(rfunc) {
sl@0
   276
			ret = rfunc(value, arg);
sl@0
   277
			if(ret <= 0) return ret;
sl@0
   278
		}
sl@0
   279
	}
sl@0
   280
	return 1;
sl@0
   281
}
sl@0
   282
sl@0
   283
/* Various utility functions for traverse_string */
sl@0
   284
sl@0
   285
/* Just count number of characters */
sl@0
   286
sl@0
   287
static int in_utf8(unsigned long value, void *arg)
sl@0
   288
{
sl@0
   289
	int *nchar;
sl@0
   290
	nchar = arg;
sl@0
   291
	(*nchar)++;
sl@0
   292
	return 1;
sl@0
   293
}
sl@0
   294
sl@0
   295
/* Determine size of output as a UTF8 String */
sl@0
   296
sl@0
   297
static int out_utf8(unsigned long value, void *arg)
sl@0
   298
{
sl@0
   299
	int *outlen;
sl@0
   300
	outlen = arg;
sl@0
   301
	*outlen += UTF8_putc(NULL, -1, value);
sl@0
   302
	return 1;
sl@0
   303
}
sl@0
   304
sl@0
   305
/* Determine the "type" of a string: check each character against a
sl@0
   306
 * supplied "mask".
sl@0
   307
 */
sl@0
   308
sl@0
   309
static int type_str(unsigned long value, void *arg)
sl@0
   310
{
sl@0
   311
	unsigned long types;
sl@0
   312
	types = *((unsigned long *)arg);
sl@0
   313
	if((types & B_ASN1_PRINTABLESTRING) && !is_printable(value))
sl@0
   314
					types &= ~B_ASN1_PRINTABLESTRING;
sl@0
   315
	if((types & B_ASN1_IA5STRING) && (value > 127))
sl@0
   316
					types &= ~B_ASN1_IA5STRING;
sl@0
   317
	if((types & B_ASN1_T61STRING) && (value > 0xff))
sl@0
   318
					types &= ~B_ASN1_T61STRING;
sl@0
   319
	if((types & B_ASN1_BMPSTRING) && (value > 0xffff))
sl@0
   320
					types &= ~B_ASN1_BMPSTRING;
sl@0
   321
	if(!types) return -1;
sl@0
   322
	*((unsigned long *)arg) = types;
sl@0
   323
	return 1;
sl@0
   324
}
sl@0
   325
sl@0
   326
/* Copy one byte per character ASCII like strings */
sl@0
   327
sl@0
   328
static int cpy_asc(unsigned long value, void *arg)
sl@0
   329
{
sl@0
   330
	unsigned char **p, *q;
sl@0
   331
	p = arg;
sl@0
   332
	q = *p;
sl@0
   333
	*q = (unsigned char) value;
sl@0
   334
	(*p)++;
sl@0
   335
	return 1;
sl@0
   336
}
sl@0
   337
sl@0
   338
/* Copy two byte per character BMPStrings */
sl@0
   339
sl@0
   340
static int cpy_bmp(unsigned long value, void *arg)
sl@0
   341
{
sl@0
   342
	unsigned char **p, *q;
sl@0
   343
	p = arg;
sl@0
   344
	q = *p;
sl@0
   345
	*q++ = (unsigned char) ((value >> 8) & 0xff);
sl@0
   346
	*q = (unsigned char) (value & 0xff);
sl@0
   347
	*p += 2;
sl@0
   348
	return 1;
sl@0
   349
}
sl@0
   350
sl@0
   351
/* Copy four byte per character UniversalStrings */
sl@0
   352
sl@0
   353
static int cpy_univ(unsigned long value, void *arg)
sl@0
   354
{
sl@0
   355
	unsigned char **p, *q;
sl@0
   356
	p = arg;
sl@0
   357
	q = *p;
sl@0
   358
	*q++ = (unsigned char) ((value >> 24) & 0xff);
sl@0
   359
	*q++ = (unsigned char) ((value >> 16) & 0xff);
sl@0
   360
	*q++ = (unsigned char) ((value >> 8) & 0xff);
sl@0
   361
	*q = (unsigned char) (value & 0xff);
sl@0
   362
	*p += 4;
sl@0
   363
	return 1;
sl@0
   364
}
sl@0
   365
sl@0
   366
/* Copy to a UTF8String */
sl@0
   367
sl@0
   368
static int cpy_utf8(unsigned long value, void *arg)
sl@0
   369
{
sl@0
   370
	unsigned char **p;
sl@0
   371
	int ret;
sl@0
   372
	p = arg;
sl@0
   373
	/* We already know there is enough room so pass 0xff as the length */
sl@0
   374
	ret = UTF8_putc(*p, 0xff, value);
sl@0
   375
	*p += ret;
sl@0
   376
	return 1;
sl@0
   377
}
sl@0
   378
sl@0
   379
/* Return 1 if the character is permitted in a PrintableString */
sl@0
   380
static int is_printable(unsigned long value)
sl@0
   381
{
sl@0
   382
	int ch;
sl@0
   383
	if(value > 0x7f) return 0;
sl@0
   384
	ch = (int) value;
sl@0
   385
	/* Note: we can't use 'isalnum' because certain accented 
sl@0
   386
	 * characters may count as alphanumeric in some environments.
sl@0
   387
	 */
sl@0
   388
#ifndef CHARSET_EBCDIC
sl@0
   389
	if((ch >= 'a') && (ch <= 'z')) return 1;
sl@0
   390
	if((ch >= 'A') && (ch <= 'Z')) return 1;
sl@0
   391
	if((ch >= '0') && (ch <= '9')) return 1;
sl@0
   392
	if ((ch == ' ') || strchr("'()+,-./:=?", ch)) return 1;
sl@0
   393
#else /*CHARSET_EBCDIC*/
sl@0
   394
	if((ch >= os_toascii['a']) && (ch <= os_toascii['z'])) return 1;
sl@0
   395
	if((ch >= os_toascii['A']) && (ch <= os_toascii['Z'])) return 1;
sl@0
   396
	if((ch >= os_toascii['0']) && (ch <= os_toascii['9'])) return 1;
sl@0
   397
	if ((ch == os_toascii[' ']) || strchr("'()+,-./:=?", os_toebcdic[ch])) return 1;
sl@0
   398
#endif /*CHARSET_EBCDIC*/
sl@0
   399
	return 0;
sl@0
   400
}