2021-01-27 11:06:13 -08:00
// SPDX-License-Identifier: GPL-2.0-only
/*
* Simple encoder primitives for ASN .1 BER / DER / CER
*
* Copyright ( C ) 2019 James . Bottomley @ HansenPartnership . com
*/
# include <linux/asn1_encoder.h>
# include <linux/bug.h>
# include <linux/string.h>
# include <linux/module.h>
/**
* asn1_encode_integer ( ) - encode positive integer to ASN .1
* @ data : pointer to the pointer to the data
* @ end_data : end of data pointer , points one beyond last usable byte in @ data
* @ integer : integer to be encoded
*
* This is a simplified encoder : it only currently does
* positive integers , but it should be simple enough to add the
* negative case if a use comes along .
*/
unsigned char *
asn1_encode_integer ( unsigned char * data , const unsigned char * end_data ,
s64 integer )
{
int data_len = end_data - data ;
unsigned char * d = & data [ 2 ] ;
bool found = false ;
int i ;
if ( WARN ( integer < 0 ,
" BUG: integer encode only supports positive integers " ) )
return ERR_PTR ( - EINVAL ) ;
if ( IS_ERR ( data ) )
return data ;
/* need at least 3 bytes for tag, length and integer encoding */
if ( data_len < 3 )
return ERR_PTR ( - EINVAL ) ;
/* remaining length where at d (the start of the integer encoding) */
data_len - = 2 ;
data [ 0 ] = _tag ( UNIV , PRIM , INT ) ;
if ( integer = = 0 ) {
* d + + = 0 ;
goto out ;
}
for ( i = sizeof ( integer ) ; i > 0 ; i - - ) {
int byte = integer > > ( 8 * ( i - 1 ) ) ;
if ( ! found & & byte = = 0 )
continue ;
/*
* for a positive number the first byte must have bit
* 7 clear in two ' s complement ( otherwise it ' s a
* negative number ) so prepend a leading zero if
* that ' s not the case
*/
if ( ! found & & ( byte & 0x80 ) ) {
/*
* no check needed here , we already know we
* have len > = 1
*/
* d + + = 0 ;
data_len - - ;
}
found = true ;
if ( data_len = = 0 )
return ERR_PTR ( - EINVAL ) ;
* d + + = byte ;
data_len - - ;
}
out :
data [ 1 ] = d - data - 2 ;
return d ;
}
EXPORT_SYMBOL_GPL ( asn1_encode_integer ) ;
/* calculate the base 128 digit values setting the top bit of the first octet */
static int asn1_encode_oid_digit ( unsigned char * * _data , int * data_len , u32 oid )
{
unsigned char * data = * _data ;
int start = 7 + 7 + 7 + 7 ;
int ret = 0 ;
if ( * data_len < 1 )
return - EINVAL ;
/* quick case */
if ( oid = = 0 ) {
* data + + = 0x80 ;
( * data_len ) - - ;
goto out ;
}
while ( oid > > start = = 0 )
start - = 7 ;
while ( start > 0 & & * data_len > 0 ) {
u8 byte ;
byte = oid > > start ;
oid = oid - ( byte < < start ) ;
start - = 7 ;
byte | = 0x80 ;
* data + + = byte ;
( * data_len ) - - ;
}
if ( * data_len > 0 ) {
* data + + = oid ;
( * data_len ) - - ;
} else {
ret = - EINVAL ;
}
out :
* _data = data ;
return ret ;
}
/**
* asn1_encode_oid ( ) - encode an oid to ASN .1
* @ data : position to begin encoding at
* @ end_data : end of data pointer , points one beyond last usable byte in @ data
* @ oid : array of oids
* @ oid_len : length of oid array
*
* this encodes an OID up to ASN .1 when presented as an array of OID values
*/
unsigned char *
asn1_encode_oid ( unsigned char * data , const unsigned char * end_data ,
u32 oid [ ] , int oid_len )
{
int data_len = end_data - data ;
unsigned char * d = data + 2 ;
int i , ret ;
if ( WARN ( oid_len < 2 , " OID must have at least two elements " ) )
return ERR_PTR ( - EINVAL ) ;
if ( WARN ( oid_len > 32 , " OID is too large " ) )
return ERR_PTR ( - EINVAL ) ;
if ( IS_ERR ( data ) )
return data ;
/* need at least 3 bytes for tag, length and OID encoding */
if ( data_len < 3 )
return ERR_PTR ( - EINVAL ) ;
data [ 0 ] = _tag ( UNIV , PRIM , OID ) ;
* d + + = oid [ 0 ] * 40 + oid [ 1 ] ;
data_len - = 3 ;
ret = 0 ;
for ( i = 2 ; i < oid_len ; i + + ) {
ret = asn1_encode_oid_digit ( & d , & data_len , oid [ i ] ) ;
if ( ret < 0 )
return ERR_PTR ( ret ) ;
}
data [ 1 ] = d - data - 2 ;
return d ;
}
EXPORT_SYMBOL_GPL ( asn1_encode_oid ) ;
/**
* asn1_encode_length ( ) - encode a length to follow an ASN .1 tag
* @ data : pointer to encode at
2021-07-07 18:07:31 -07:00
* @ data_len : pointer to remaining length ( adjusted by routine )
2021-01-27 11:06:13 -08:00
* @ len : length to encode
*
* This routine can encode lengths up to 65535 using the ASN .1 rules .
* It will accept a negative length and place a zero length tag
* instead ( to keep the ASN .1 valid ) . This convention allows other
* encoder primitives to accept negative lengths as singalling the
* sequence will be re - encoded when the length is known .
*/
static int asn1_encode_length ( unsigned char * * data , int * data_len , int len )
{
if ( * data_len < 1 )
return - EINVAL ;
if ( len < 0 ) {
* ( ( * data ) + + ) = 0 ;
( * data_len ) - - ;
return 0 ;
}
if ( len < = 0x7f ) {
* ( ( * data ) + + ) = len ;
( * data_len ) - - ;
return 0 ;
}
if ( * data_len < 2 )
return - EINVAL ;
if ( len < = 0xff ) {
* ( ( * data ) + + ) = 0x81 ;
* ( ( * data ) + + ) = len & 0xff ;
* data_len - = 2 ;
return 0 ;
}
if ( * data_len < 3 )
return - EINVAL ;
if ( len < = 0xffff ) {
* ( ( * data ) + + ) = 0x82 ;
* ( ( * data ) + + ) = ( len > > 8 ) & 0xff ;
* ( ( * data ) + + ) = len & 0xff ;
* data_len - = 3 ;
return 0 ;
}
if ( WARN ( len > 0xffffff , " ASN.1 length can't be > 0xffffff " ) )
return - EINVAL ;
if ( * data_len < 4 )
return - EINVAL ;
* ( ( * data ) + + ) = 0x83 ;
* ( ( * data ) + + ) = ( len > > 16 ) & 0xff ;
* ( ( * data ) + + ) = ( len > > 8 ) & 0xff ;
* ( ( * data ) + + ) = len & 0xff ;
* data_len - = 4 ;
return 0 ;
}
/**
* asn1_encode_tag ( ) - add a tag for optional or explicit value
* @ data : pointer to place tag at
* @ end_data : end of data pointer , points one beyond last usable byte in @ data
* @ tag : tag to be placed
* @ string : the data to be tagged
* @ len : the length of the data to be tagged
*
* Note this currently only handles short form tags < 31.
*
* Standard usage is to pass in a @ tag , @ string and @ length and the
* @ string will be ASN .1 encoded with @ tag and placed into @ data . If
* the encoding would put data past @ end_data then an error is
* returned , otherwise a pointer to a position one beyond the encoding
* is returned .
*
* To encode in place pass a NULL @ string and - 1 for @ len and the
* maximum allowable beginning and end of the data ; all this will do
* is add the current maximum length and update the data pointer to
* the place where the tag contents should be placed is returned . The
* data should be copied in by the calling routine which should then
* repeat the prior statement but now with the known length . In order
* to avoid having to keep both before and after pointers , the repeat
* expects to be called with @ data pointing to where the first encode
* returned it and still NULL for @ string but the real length in @ len .
*/
unsigned char *
asn1_encode_tag ( unsigned char * data , const unsigned char * end_data ,
u32 tag , const unsigned char * string , int len )
{
int data_len = end_data - data ;
int ret ;
if ( WARN ( tag > 30 , " ASN.1 tag can't be > 30 " ) )
return ERR_PTR ( - EINVAL ) ;
if ( ! string & & WARN ( len > 127 ,
" BUG: recode tag is too big (>127) " ) )
return ERR_PTR ( - EINVAL ) ;
if ( IS_ERR ( data ) )
return data ;
if ( ! string & & len > 0 ) {
/*
* we ' re recoding , so move back to the start of the
* tag and install a dummy length because the real
* data_len should be NULL
*/
data - = 2 ;
data_len = 2 ;
}
if ( data_len < 2 )
return ERR_PTR ( - EINVAL ) ;
* ( data + + ) = _tagn ( CONT , CONS , tag ) ;
data_len - - ;
ret = asn1_encode_length ( & data , & data_len , len ) ;
if ( ret < 0 )
return ERR_PTR ( ret ) ;
if ( ! string )
return data ;
if ( data_len < len )
return ERR_PTR ( - EINVAL ) ;
memcpy ( data , string , len ) ;
data + = len ;
return data ;
}
EXPORT_SYMBOL_GPL ( asn1_encode_tag ) ;
/**
* asn1_encode_octet_string ( ) - encode an ASN .1 OCTET STRING
* @ data : pointer to encode at
* @ end_data : end of data pointer , points one beyond last usable byte in @ data
* @ string : string to be encoded
* @ len : length of string
*
* Note ASN .1 octet strings may contain zeros , so the length is obligatory .
*/
unsigned char *
asn1_encode_octet_string ( unsigned char * data ,
const unsigned char * end_data ,
const unsigned char * string , u32 len )
{
int data_len = end_data - data ;
int ret ;
if ( IS_ERR ( data ) )
return data ;
/* need minimum of 2 bytes for tag and length of zero length string */
if ( data_len < 2 )
return ERR_PTR ( - EINVAL ) ;
* ( data + + ) = _tag ( UNIV , PRIM , OTS ) ;
data_len - - ;
ret = asn1_encode_length ( & data , & data_len , len ) ;
if ( ret )
return ERR_PTR ( ret ) ;
if ( data_len < len )
return ERR_PTR ( - EINVAL ) ;
memcpy ( data , string , len ) ;
data + = len ;
return data ;
}
EXPORT_SYMBOL_GPL ( asn1_encode_octet_string ) ;
/**
* asn1_encode_sequence ( ) - wrap a byte stream in an ASN .1 SEQUENCE
* @ data : pointer to encode at
* @ end_data : end of data pointer , points one beyond last usable byte in @ data
* @ seq : data to be encoded as a sequence
* @ len : length of the data to be encoded as a sequence
*
* Fill in a sequence . To encode in place , pass NULL for @ seq and - 1
* for @ len ; then call again once the length is known ( still with NULL
* for @ seq ) . In order to avoid having to keep both before and after
* pointers , the repeat expects to be called with @ data pointing to
* where the first encode placed it .
*/
unsigned char *
asn1_encode_sequence ( unsigned char * data , const unsigned char * end_data ,
const unsigned char * seq , int len )
{
int data_len = end_data - data ;
int ret ;
if ( ! seq & & WARN ( len > 127 ,
" BUG: recode sequence is too big (>127) " ) )
return ERR_PTR ( - EINVAL ) ;
if ( IS_ERR ( data ) )
return data ;
if ( ! seq & & len > = 0 ) {
/*
* we ' re recoding , so move back to the start of the
* sequence and install a dummy length because the
* real length should be NULL
*/
data - = 2 ;
data_len = 2 ;
}
if ( data_len < 2 )
return ERR_PTR ( - EINVAL ) ;
* ( data + + ) = _tag ( UNIV , CONS , SEQ ) ;
data_len - - ;
ret = asn1_encode_length ( & data , & data_len , len ) ;
if ( ret )
return ERR_PTR ( ret ) ;
if ( ! seq )
return data ;
if ( data_len < len )
return ERR_PTR ( - EINVAL ) ;
memcpy ( data , seq , len ) ;
data + = len ;
return data ;
}
EXPORT_SYMBOL_GPL ( asn1_encode_sequence ) ;
/**
* asn1_encode_boolean ( ) - encode a boolean value to ASN .1
* @ data : pointer to encode at
* @ end_data : end of data pointer , points one beyond last usable byte in @ data
* @ val : the boolean true / false value
*/
unsigned char *
asn1_encode_boolean ( unsigned char * data , const unsigned char * end_data ,
bool val )
{
int data_len = end_data - data ;
if ( IS_ERR ( data ) )
return data ;
/* booleans are 3 bytes: tag, length == 1 and value == 0 or 1 */
if ( data_len < 3 )
return ERR_PTR ( - EINVAL ) ;
* ( data + + ) = _tag ( UNIV , PRIM , BOOL ) ;
data_len - - ;
asn1_encode_length ( & data , & data_len , 1 ) ;
if ( val )
* ( data + + ) = 1 ;
else
* ( data + + ) = 0 ;
return data ;
}
EXPORT_SYMBOL_GPL ( asn1_encode_boolean ) ;
MODULE_LICENSE ( " GPL " ) ;