tomcrypt/src/pk/asn1/der/sequence/der_encode_sequence_ex.c

346 lines
9.4 KiB
C
Raw Normal View History

2005-06-09 00:08:13 +00:00
/* LibTomCrypt, modular cryptographic library -- Tom St Denis
*
* LibTomCrypt is a library that provides various cryptographic
* algorithms in a highly modular and flexible manner.
*
* The library is free for all purposes without any express
* guarantee it works.
*
2007-07-20 17:48:02 +00:00
* Tom St Denis, tomstdenis@gmail.com, http://libtom.org
2005-06-09 00:08:13 +00:00
*/
#include "tomcrypt.h"
#include <stdarg.h>
/**
2005-11-24 11:57:58 +00:00
@file der_encode_sequence_ex.c
2005-06-09 00:08:13 +00:00
ASN.1 DER, encode a SEQUENCE, Tom St Denis
*/
#ifdef LTC_DER
/**
Encode a SEQUENCE
@param list The list of items to encode
@param inlen The number of items in the list
2014-08-22 12:41:46 +02:00
@param out [out] The destination
2005-06-09 00:08:13 +00:00
@param outlen [in/out] The size of the output
2005-11-24 11:57:58 +00:00
@param type_of LTC_ASN1_SEQUENCE or LTC_ASN1_SET/LTC_ASN1_SETOF
2005-06-09 00:08:13 +00:00
@return CRYPT_OK on success
*/
2005-11-24 11:57:58 +00:00
int der_encode_sequence_ex(ltc_asn1_list *list, unsigned long inlen,
2014-08-22 12:41:46 +02:00
unsigned char *out, unsigned long *outlen, int type_of)
2005-06-09 00:08:13 +00:00
{
int err, type;
unsigned long size, x, y, z, i;
void *data;
LTC_ARGCHK(list != NULL);
LTC_ARGCHK(out != NULL);
LTC_ARGCHK(outlen != NULL);
/* get size of output that will be required */
y = 0;
for (i = 0; i < inlen; i++) {
type = list[i].type;
size = list[i].size;
data = list[i].data;
2014-08-22 12:41:46 +02:00
if (type == LTC_ASN1_EOL) {
2005-06-09 00:08:13 +00:00
break;
}
switch (type) {
2006-05-29 23:12:56 +00:00
case LTC_ASN1_BOOLEAN:
2006-12-16 18:10:04 +00:00
if ((err = der_length_boolean(&x)) != CRYPT_OK) {
goto LBL_ERR;
}
y += x;
break;
2006-05-29 23:12:56 +00:00
2005-06-09 00:08:13 +00:00
case LTC_ASN1_INTEGER:
if ((err = der_length_integer(data, &x)) != CRYPT_OK) {
goto LBL_ERR;
}
y += x;
break;
case LTC_ASN1_SHORT_INTEGER:
if ((err = der_length_short_integer(*((unsigned long*)data), &x)) != CRYPT_OK) {
goto LBL_ERR;
}
y += x;
break;
case LTC_ASN1_BIT_STRING:
case LTC_ASN1_RAW_BIT_STRING:
2005-06-09 00:08:13 +00:00
if ((err = der_length_bit_string(size, &x)) != CRYPT_OK) {
goto LBL_ERR;
}
y += x;
break;
case LTC_ASN1_OCTET_STRING:
if ((err = der_length_octet_string(size, &x)) != CRYPT_OK) {
goto LBL_ERR;
}
y += x;
break;
case LTC_ASN1_NULL:
y += 2;
break;
case LTC_ASN1_OBJECT_IDENTIFIER:
if ((err = der_length_object_identifier(data, size, &x)) != CRYPT_OK) {
goto LBL_ERR;
}
y += x;
break;
case LTC_ASN1_IA5_STRING:
if ((err = der_length_ia5_string(data, size, &x)) != CRYPT_OK) {
goto LBL_ERR;
}
y += x;
break;
case LTC_ASN1_PRINTABLE_STRING:
if ((err = der_length_printable_string(data, size, &x)) != CRYPT_OK) {
goto LBL_ERR;
}
y += x;
break;
2006-12-16 18:10:04 +00:00
case LTC_ASN1_UTF8_STRING:
if ((err = der_length_utf8_string(data, size, &x)) != CRYPT_OK) {
goto LBL_ERR;
}
y += x;
break;
2005-06-27 11:47:35 +00:00
case LTC_ASN1_UTCTIME:
if ((err = der_length_utctime(data, &x)) != CRYPT_OK) {
goto LBL_ERR;
}
y += x;
break;
2005-11-24 11:57:58 +00:00
case LTC_ASN1_SET:
case LTC_ASN1_SETOF:
2005-06-09 00:08:13 +00:00
case LTC_ASN1_SEQUENCE:
if ((err = der_length_sequence(data, size, &x)) != CRYPT_OK) {
goto LBL_ERR;
}
y += x;
break;
2014-08-22 12:41:46 +02:00
2005-06-09 00:08:13 +00:00
default:
err = CRYPT_INVALID_ARG;
goto LBL_ERR;
}
}
/* calc header size */
z = y;
if (y < 128) {
y += 2;
} else if (y < 256) {
/* 0x30 0x81 LL */
y += 3;
} else if (y < 65536UL) {
/* 0x30 0x82 LL LL */
y += 4;
} else if (y < 16777216UL) {
/* 0x30 0x83 LL LL LL */
y += 5;
} else {
err = CRYPT_INVALID_ARG;
goto LBL_ERR;
}
/* too big ? */
if (*outlen < y) {
2006-06-18 01:37:50 +00:00
*outlen = y;
2005-06-09 00:08:13 +00:00
err = CRYPT_BUFFER_OVERFLOW;
goto LBL_ERR;
}
/* store header */
x = 0;
2005-11-24 11:57:58 +00:00
out[x++] = (type_of == LTC_ASN1_SEQUENCE) ? 0x30 : 0x31;
2014-08-22 12:41:46 +02:00
2005-06-09 00:08:13 +00:00
if (z < 128) {
2006-12-16 18:10:04 +00:00
out[x++] = (unsigned char)z;
2005-06-09 00:08:13 +00:00
} else if (z < 256) {
out[x++] = 0x81;
2006-12-16 18:10:04 +00:00
out[x++] = (unsigned char)z;
2005-06-09 00:08:13 +00:00
} else if (z < 65536UL) {
out[x++] = 0x82;
2006-12-16 18:10:04 +00:00
out[x++] = (unsigned char)((z>>8UL)&255);
out[x++] = (unsigned char)(z&255);
2005-06-09 00:08:13 +00:00
} else if (z < 16777216UL) {
out[x++] = 0x83;
2006-12-16 18:10:04 +00:00
out[x++] = (unsigned char)((z>>16UL)&255);
out[x++] = (unsigned char)((z>>8UL)&255);
out[x++] = (unsigned char)(z&255);
2005-06-09 00:08:13 +00:00
}
/* store data */
*outlen -= x;
for (i = 0; i < inlen; i++) {
type = list[i].type;
size = list[i].size;
data = list[i].data;
2014-08-22 12:41:46 +02:00
if (type == LTC_ASN1_EOL) {
2005-06-09 00:08:13 +00:00
break;
}
switch (type) {
2006-05-29 23:12:56 +00:00
case LTC_ASN1_BOOLEAN:
2006-12-16 18:10:04 +00:00
z = *outlen;
if ((err = der_encode_boolean(*((int *)data), out + x, &z)) != CRYPT_OK) {
goto LBL_ERR;
}
x += z;
*outlen -= z;
break;
2014-08-22 12:41:46 +02:00
2005-06-09 00:08:13 +00:00
case LTC_ASN1_INTEGER:
z = *outlen;
if ((err = der_encode_integer(data, out + x, &z)) != CRYPT_OK) {
goto LBL_ERR;
}
x += z;
*outlen -= z;
break;
case LTC_ASN1_SHORT_INTEGER:
z = *outlen;
if ((err = der_encode_short_integer(*((unsigned long*)data), out + x, &z)) != CRYPT_OK) {
goto LBL_ERR;
}
x += z;
*outlen -= z;
break;
case LTC_ASN1_BIT_STRING:
z = *outlen;
if ((err = der_encode_bit_string(data, size, out + x, &z)) != CRYPT_OK) {
goto LBL_ERR;
}
x += z;
*outlen -= z;
break;
case LTC_ASN1_RAW_BIT_STRING:
z = *outlen;
if ((err = der_encode_raw_bit_string(data, size, out + x, &z)) != CRYPT_OK) {
2005-06-09 00:08:13 +00:00
goto LBL_ERR;
}
x += z;
*outlen -= z;
break;
case LTC_ASN1_OCTET_STRING:
z = *outlen;
if ((err = der_encode_octet_string(data, size, out + x, &z)) != CRYPT_OK) {
goto LBL_ERR;
}
x += z;
*outlen -= z;
break;
case LTC_ASN1_NULL:
out[x++] = 0x05;
out[x++] = 0x00;
*outlen -= 2;
break;
case LTC_ASN1_OBJECT_IDENTIFIER:
z = *outlen;
if ((err = der_encode_object_identifier(data, size, out + x, &z)) != CRYPT_OK) {
goto LBL_ERR;
}
x += z;
*outlen -= z;
break;
case LTC_ASN1_IA5_STRING:
z = *outlen;
if ((err = der_encode_ia5_string(data, size, out + x, &z)) != CRYPT_OK) {
goto LBL_ERR;
}
x += z;
*outlen -= z;
break;
2014-08-22 12:41:46 +02:00
2005-06-09 00:08:13 +00:00
case LTC_ASN1_PRINTABLE_STRING:
z = *outlen;
if ((err = der_encode_printable_string(data, size, out + x, &z)) != CRYPT_OK) {
goto LBL_ERR;
}
x += z;
*outlen -= z;
break;
2006-12-16 18:10:04 +00:00
case LTC_ASN1_UTF8_STRING:
z = *outlen;
if ((err = der_encode_utf8_string(data, size, out + x, &z)) != CRYPT_OK) {
goto LBL_ERR;
}
x += z;
*outlen -= z;
break;
2005-06-27 11:47:35 +00:00
case LTC_ASN1_UTCTIME:
z = *outlen;
if ((err = der_encode_utctime(data, out + x, &z)) != CRYPT_OK) {
goto LBL_ERR;
}
x += z;
*outlen -= z;
break;
2005-11-24 11:57:58 +00:00
case LTC_ASN1_SET:
z = *outlen;
if ((err = der_encode_set(data, size, out + x, &z)) != CRYPT_OK) {
goto LBL_ERR;
}
x += z;
*outlen -= z;
break;
case LTC_ASN1_SETOF:
2005-06-09 00:08:13 +00:00
z = *outlen;
2005-11-24 11:57:58 +00:00
if ((err = der_encode_setof(data, size, out + x, &z)) != CRYPT_OK) {
2005-06-09 00:08:13 +00:00
goto LBL_ERR;
}
x += z;
*outlen -= z;
break;
2005-11-24 11:57:58 +00:00
case LTC_ASN1_SEQUENCE:
z = *outlen;
if ((err = der_encode_sequence_ex(data, size, out + x, &z, type)) != CRYPT_OK) {
goto LBL_ERR;
}
x += z;
*outlen -= z;
break;
2014-08-22 12:41:46 +02:00
2005-06-09 00:08:13 +00:00
default:
err = CRYPT_INVALID_ARG;
goto LBL_ERR;
}
}
*outlen = x;
2014-08-22 12:41:46 +02:00
err = CRYPT_OK;
2005-06-09 00:08:13 +00:00
LBL_ERR:
return err;
}
#endif