Skip to content
Snippets Groups Projects
Commit 9fe8f723 authored by Howard Chu's avatar Howard Chu
Browse files

Add DER OID encoder/decoder

parent 23ba7dfa
No related branches found
No related tags found
No related merge requests found
......@@ -45,6 +45,40 @@ static ber_len_t ber_getnint LDAP_P((
ber_int_t *num,
ber_len_t len ));
/* out->bv_len should be the buffer size on input */
int
ber_decode_oid( BerValue *in, BerValue *out )
{
unsigned char *der = in->bv_val;
unsigned long val, val1;
int i, len;
char *ptr;
assert( in != NULL );
assert( out != NULL );
/* expands by 5/2, and we add dots - call it 3 */
if ( !out->bv_val || out->bv_len < in->bv_len * 3 )
return -1;
val1 = der[0] / 40;
val = der[0] - val1 * 40;
len = sprintf( out->bv_val, "%ld.%ld", val1, val );
ptr = out->bv_val + len;
val = 0;
for ( i=1; i<in->bv_len; i++ ) {
val = val << 7;
val |= der[i] & 0x7f;
if ( !( der[i] & 0x80 )) {
ptr += sprintf( ptr, ".%ld", val );
val = 0;
}
}
out->bv_len = ptr - out->bv_val;
return 0;
}
/* return the tag - LBER_DEFAULT returned means trouble */
ber_tag_t
ber_get_tag( BerElement *ber )
......
......@@ -177,6 +177,67 @@ ber_put_len( BerElement *ber, ber_len_t len, int nosos )
return rc == i ? i+1 : -1;
}
/* out->bv_len should be the buffer size on input */
int
ber_encode_oid( BerValue *in, BerValue *out )
{
unsigned char *der = out->bv_val;
unsigned long val, val1;
int i, len;
char *ptr, *end, *inend;
assert( in != NULL );
assert( out != NULL );
if ( !out->bv_val || out->bv_len < in->bv_len )
return -1;
/* OIDs must have at least two components */
if ( sscanf( in->bv_val, "%ld.%ld", &val, &val1 ) != 2 )
return -1;
val *= 40;
val += val1;
inend = in->bv_val + in->bv_len;
ptr = strchr( in->bv_val, '.' );
ptr = strchr( ptr+1, '.' );
if ( ptr )
++ptr;
else
ptr = inend;
for (;;) {
if ( !val ) {
*der++ = 0;
} else {
int hibit = 0;
i = sizeof(unsigned long) + 1;
len = i;
for (;val;) {
i--;
val1 = val & 0x7f;
val >>= 7;
der[i] = val1 | hibit;
hibit = 0x80;
}
if ( i ) {
len -= i;
memcpy( der, der+i, len );
}
der += len;
}
if ( ptr >= inend ) break;
val = strtol( ptr, &end, 10 );
if ( ptr == end ) break;
if ( *end && *end != '.' ) break;
ptr = end + 1;
}
out->bv_len = (char *)der - out->bv_val;
return 0;
}
static int
ber_put_int_or_enum(
BerElement *ber,
......
0% Loading or .
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment