25RCSID(
"$Id: ef6f1ce808f5aebecdac92a8c814f067e8865a39 $")
27#include <freeradius-devel/util/dbuff.h>
28#include <freeradius-devel/util/encode.h>
29#include <freeradius-devel/util/proto.h>
30#include <freeradius-devel/util/sbuff.h>
31#include <freeradius-devel/util/struct.h>
32#include <freeradius-devel/util/time.h>
33#include <freeradius-devel/util/dict_ext.h>
35#include <freeradius-devel/io/test_point.h>
69#define fr_der_encode_enumerated fr_der_encode_integer
73static inline CC_HINT(always_inline)
ssize_t
174 for (i = 0, len = 0; i <
sizeof(
value); i++) {
184 }
else if (len == 1) {
190 if ((first_octet == 0xff && (
byte & 0x80)) || ((first_octet == 0x00) && (
byte >> 7 == 0))) {
191 if (i ==
sizeof(
value) - 1) {
298 if (
byte != 0)
break;
316 while ((last_byte != 0) && ((last_byte & 0x01) == 0)) {
340 if (
vp->vp_length == 0) {
392 if (
vp->vp_ip.prefix == 0) {
410 if (
vp->vp_ip.prefix == 32) {
418 len = (
vp->vp_ip.prefix + 0x07) >> 3;
481 if (
vp->vp_ip.prefix == 128) {
489 len = (
vp->vp_ip.prefix + 0x07) >> 3;
517 if (
vp->vp_ip.af == AF_INET) {
583 if (
vp->vp_length != 0) {
615 if ((*component < 2) && (
value > 40)) {
620 oid = *component * 40 +
value;
634 for (i = 63; i >= 0; i -= 7) {
637 part = (oid >> i) & 0x7f;
640 more = ((
uint8_t) (i > 0)) << 7;
656 unsigned long long oid;
692 for (start =
vp->vp_strvalue; *start !=
'\0'; start = end + 1) {
698 oid = strtoull(start, &end, 10);
699 if ((oid == ULLONG_MAX) || (*end && (*end !=
'.'))) {
705 if (slen < 0)
return -1;
860 ptr_pairs[i].
len = slen;
880 for (i = 0; i <
count; i++) {
904 char fmt_time[50] = { 0 };
959 for (i = 0; i <
sizeof(fmt_time); i++) {
960 if (fmt_time[i] ==
'\0') {
964 if ((fmt_time[i] ==
'-') || (fmt_time[i] ==
'T') || (fmt_time[i] ==
':')) {
967 while (fmt_time[j] !=
'\0') {
968 fmt_time[j] = fmt_time[j + 1];
989 char fmt_time[50] = { 0 };
1042 for (i = 0; i <
sizeof(fmt_time); i++) {
1043 if (fmt_time[i] ==
'\0') {
1047 if ((fmt_time[i] ==
'-') || (fmt_time[i] ==
'T') || (fmt_time[i] ==
':')) {
1050 while (fmt_time[j] !=
'\0') {
1051 fmt_time[j] = fmt_time[j + 1];
1060 if (fmt_time[i] ==
'.') {
1064 size_t j = strlen(fmt_time) - 2;
1066 while (fmt_time[j] ==
'0') {
1067 fmt_time[j] = fmt_time[j + 1];
1068 fmt_time[j + 1] =
'\0';
1076 fmt_time[i] = fmt_time[i + 1];
1077 fmt_time[i + 1] =
'\0';
1114 if (slen < 0)
return -1;
1126 size_t is_critical = 0;
1169 if (slen < 0)
return slen;
1199 if (slen < 0)
return slen;
1208 if (slen < 0)
return slen;
1228 if (!is_critical && (strcmp(child->
da->name,
"Critical") == 0)) {
1274 if (slen < 0)
return slen;
1292 if (slen < 0)
return slen;
1301 if (slen < 0)
return slen;
1307 if (slen < 0)
return slen;
1313 if (slen < 0)
return -1;
1322 "Encoded X509 extension");
1333 if (slen < 0)
return slen;
1367 if (slen < 0)
return slen;
1422 if (slen < 0)
return slen;
1428 if (slen < 0)
return slen;
1521 size_t tmp, datalen;
1528 if (datalen <= 0x7f) {
1536 for (tmp = datalen, len_len = 0; tmp != 0; tmp >>= 8) {
1580 for (i = len_len; i > 0; i--) {
1596static inline CC_HINT(always_inline)
ssize_t
1656 if (slen < 0)
return 0;
1717 if (slen < 0)
return slen;
1742 tag_class = flags->
class;
1752 if (slen < 0)
return slen;
1764 slen = func->
encode(&our_dbuff, cursor, uctx);
1766 if (slen < 0)
return slen;
1772 if (slen < 0)
return slen;
1807 if (!test_ctx)
return -1;
#define CMP_PREFER_SMALLER(_a, _b)
Evaluates to +1 for a > b, and -1 for a < b.
#define fr_dbuff_behind(_dbuff_or_marker)
How many bytes the dbuff or marker is behind its parent.
#define fr_dbuff_used(_dbuff_or_marker)
Return the number of bytes remaining between the start of the dbuff or marker and the current positio...
#define FR_DBUFF_ADVANCE_RETURN(_dbuff_or_marker, _len)
Advance the 'current' position in dbuff or marker by _len bytes returning if _len is out of range.
#define FR_DBUFF_EXTEND_LOWAT_OR_RETURN(_dbuff_or_marker, _lowat)
Extend if we're below _lowat and return if we can't extend above _lowat.
struct fr_dbuff_marker_s fr_dbuff_marker_t
A position marker associated with a dbuff.
#define fr_dbuff_current(_dbuff_or_marker)
Return the 'current' position of a dbuff or marker.
#define fr_dbuff_set(_dst, _src)
Set the 'current' position in a dbuff or marker using another dbuff or marker, a char pointer,...
#define fr_dbuff_init(_out, _start, _len_or_end)
Initialise an dbuff for encoding or decoding.
#define fr_dbuff_start(_dbuff_or_marker)
Return the 'start' position of a dbuff or marker.
#define fr_dbuff_remaining(_dbuff_or_marker)
Return the number of bytes remaining between the dbuff or marker and the end of the buffer.
static uint8_t * fr_dbuff_marker(fr_dbuff_marker_t *m, fr_dbuff_t *dbuff)
Initialises a new marker pointing to the 'current' position of the dbuff.
#define FR_DBUFF_IN_MEMCPY_RETURN(_dbuff_or_marker, _in, _inlen)
Copy exactly _inlen bytes into dbuff or marker returning if there's insufficient space.
#define fr_dbuff_in_memcpy(_dbuff_or_marker, _in, _inlen)
Copy exactly _inlen bytes into a dbuff or marker.
#define fr_dbuff_in(_dbuff_or_marker, _in)
Copy data from a fixed sized C type into a dbuff or marker.
#define FR_DBUFF_IN_RETURN(_dbuff_or_marker, _in)
Copy data from a fixed sized C type into a dbuff returning if there is insufficient space.
#define FR_DBUFF(_dbuff_or_marker)
Create a new dbuff pointing to the same underlying buffer.
#define fr_dbuff_move(_out, _in, _len)
Copy in as many bytes as possible from one dbuff or marker to another.
#define fr_dbuff_out(_out, _dbuff_or_marker)
Copy data from a dbuff or marker to a fixed sized C type.
#define fr_dbuff_set_end(_dst, _end)
Set a new 'end' position in a dbuff or marker.
#define FR_DBUFF_IN_BYTES_RETURN(_dbuff_or_marker,...)
Copy a byte sequence into a dbuff or marker returning if there's insufficient space.
static void * fr_dcursor_next(fr_dcursor_t *cursor)
Advanced the cursor to the next item.
static void fr_dcursor_copy(fr_dcursor_t *out, fr_dcursor_t const *in)
Copy cursor parameters and state.
static void * fr_dcursor_current(fr_dcursor_t *cursor)
Return the item the cursor current points to.
fr_der_tag_t
Enumeration describing the data types in a DER encoded structure.
@ FR_DER_TAG_IA5_STRING
String of IA5 (7bit) chars.
@ FR_DER_TAG_SEQUENCE
A sequence of DER encoded data (a structure).
@ FR_DER_TAG_SET
A set of DER encoded data (a structure).
@ FR_DER_TAG_INTEGER
Arbitrary width signed integer.
@ FR_DER_TAG_BOOLEAN
Boolean true/false.
@ FR_DER_TAG_UTF8_STRING
String of UTF8 chars.
@ FR_DER_TAG_UTC_TIME
A time in UTC "YYMMDDhhmmssZ" format.
@ FR_DER_TAG_GENERALIZED_TIME
A time in "YYYYMMDDHHMMSS[.fff]Z" format.
@ FR_DER_TAG_INVALID
Invalid tag.
@ FR_DER_TAG_NULL
An empty value.
@ FR_DER_TAG_OCTETSTRING
String of octets (length field specifies bytes).
@ FR_DER_TAG_VISIBLE_STRING
String of visible chars.
@ FR_DER_TAG_BITSTRING
String of bits (length field specifies bits).
@ FR_DER_TAG_T61_STRING
String of T61 (8bit) chars.
@ FR_DER_TAG_ENUMERATED
An enumerated value.
@ FR_DER_TAG_UNIVERSAL_STRING
String of universal chars.
@ FR_DER_TAG_PRINTABLE_STRING
String of printable chars.
@ FR_DER_TAG_GENERAL_STRING
String of general chars.
@ FR_DER_TAG_OID
Reference to an OID based attribute.
bool is_extensions
a list of X.509 extensions
#define fr_der_flag_der_type(_da)
fr_der_tag_t der_type
the DER type, which is different from the FreeRADIUS type
bool is_option
has an option defined
#define DER_BOOLEAN_TRUE
DER encoded boolean true value.
#define fr_der_flag_is_oid_and_value(_da)
#define DER_BOOLEAN_FALSE
DER encoded boolean false value.
#define fr_der_flag_is_set_of(_da)
#define DER_TAG_CLASS_MASK
Mask to extract the class from the tag.
#define fr_der_flag_max(_da)
#define DER_UTC_TIME_LEN
Length of the UTC time string.
@ FR_DER_TAG_CONSTRUCTED
This is a sequence or set, it contains children.
@ FR_DER_TAG_PRIMITIVE
This is a leaf value, it contains no children.
#define DER_TAG_NUM_MASK
Mask to extract the tag number from the tag.
#define fr_der_flag_is_oid_leaf(_da)
static fr_der_attr_flags_t const * fr_der_attr_flags(fr_dict_attr_t const *da)
bool is_choice
DER name "choice".
uint8_t option
an "attribute number" encoded in the tag field.
#define DER_TAG_CONSTRUCTED_MASK
Mask to check if the tag is constructed.
bool has_default_value
a default value exists
fr_der_tag_class_t class
tag Class
static ssize_t encode_value(fr_dbuff_t *dbuff, fr_da_stack_t *da_stack, int depth, fr_dcursor_t *cursor, void *encode_ctx)
Encodes the data portion of an attribute.
ssize_t fr_pair_cursor_to_network(fr_dbuff_t *dbuff, fr_da_stack_t *da_stack, unsigned int depth, fr_dcursor_t *cursor, void *encode_ctx, fr_encode_dbuff_t encode_pair)
@ FR_TYPE_IPV4_ADDR
32 Bit IPv4 Address.
@ FR_TYPE_IPV6_PREFIX
IPv6 Prefix.
@ FR_TYPE_MAX
Number of defined data types.
@ FR_TYPE_INT64
64 Bit signed integer.
@ FR_TYPE_IPV6_ADDR
128 Bit IPv6 Address.
@ FR_TYPE_IPV4_PREFIX
IPv4 Prefix.
@ FR_TYPE_COMBO_IP_ADDR
IPv4 or IPv6 address depending on length.
@ FR_TYPE_OCTETS
Raw octets.
static uint8_t depth(fr_minmax_heap_index_t i)
void fr_proto_da_stack_build(fr_da_stack_t *stack, fr_dict_attr_t const *da)
Build a complete DA stack from the da back to the root.
static fr_internal_encode_ctx_t encode_ctx
static int encode_test_ctx(void **out, TALLOC_CTX *ctx, UNUSED fr_dict_t const *dict)
static ssize_t encode_pair(fr_dbuff_t *dbuff, fr_dcursor_t *cursor, UNUSED void *encode_ctx)
fr_der_tag_t fr_type_to_der_tag_default(fr_type_t type)
static ssize_t fr_der_encode_ipv4_addr(fr_dbuff_t *dbuff, fr_dcursor_t *cursor, UNUSED fr_der_encode_ctx_t *encode_ctx)
static ssize_t fr_der_encode_sequence(fr_dbuff_t *dbuff, fr_dcursor_t *cursor, fr_der_encode_ctx_t *encode_ctx)
uint8_t * tmp_ctx
Temporary context for encoding.
static ssize_t fr_der_encode_octetstring(fr_dbuff_t *dbuff, fr_dcursor_t *cursor, UNUSED fr_der_encode_ctx_t *encode_ctx)
static const fr_der_tag_encode_t tag_funcs[FR_DER_TAG_MAX]
static int fr_der_encode_set_of_cmp(void const *one, void const *two)
static ssize_t fr_der_encode_choice(fr_dbuff_t *dbuff, fr_dcursor_t *cursor, fr_der_encode_ctx_t *encode_ctx)
Encode a CHOICE type.
ssize_t(* fr_der_encode_t)(fr_dbuff_t *dbuff, fr_dcursor_t *cursor, fr_der_encode_ctx_t *encode_ctx)
Function signature for DER encode functions.
static ssize_t fr_der_encode_ipv6_addr(fr_dbuff_t *dbuff, fr_dcursor_t *cursor, UNUSED fr_der_encode_ctx_t *encode_ctx)
static ssize_t fr_der_encode_oid(fr_dbuff_t *dbuff, fr_dcursor_t *cursor, UNUSED fr_der_encode_ctx_t *encode_ctx)
static ssize_t fr_der_encode_null(UNUSED fr_dbuff_t *dbuff, fr_dcursor_t *cursor, UNUSED fr_der_encode_ctx_t *encode_ctx)
static ssize_t fr_der_encode_boolean(fr_dbuff_t *dbuff, fr_dcursor_t *cursor, UNUSED fr_der_encode_ctx_t *encode_ctx)
static ssize_t fr_der_encode_oid_and_value(fr_dbuff_t *dbuff, fr_dcursor_t *cursor, fr_der_encode_ctx_t *encode_ctx)
static ssize_t fr_der_encode_oid_from_value(fr_dbuff_t *dbuff, uint64_t value, uint64_t *component, int *count)
static ssize_t fr_der_encode_proto(UNUSED TALLOC_CTX *ctx, fr_pair_list_t *vps, uint8_t *data, size_t data_len, void *encode_ctx)
static ssize_t fr_der_encode_generalized_time(fr_dbuff_t *dbuff, fr_dcursor_t *cursor, UNUSED fr_der_encode_ctx_t *encode_ctx)
size_t len
Length of the encoded item (tag + length + value)
static ssize_t fr_der_encode_len(fr_dbuff_t *dbuff, fr_dbuff_marker_t *length_start)
Encode the length field of a DER structure.
#define fr_der_encode_enumerated
static ssize_t fr_der_encode_string(fr_dbuff_t *dbuff, fr_dcursor_t *cursor, UNUSED fr_der_encode_ctx_t *encode_ctx)
fr_der_tag_constructed_t constructed
static ssize_t fr_der_encode_X509_extensions(fr_dbuff_t *dbuff, fr_dcursor_t *cursor, fr_der_encode_ctx_t *encode_ctx)
static int8_t fr_der_pair_cmp_by_da_tag(void const *a, void const *b)
Compare two pairs by their tag number.
static const fr_der_tag_encode_t type_funcs[FR_TYPE_MAX]
static ssize_t fr_der_encode_combo_ip(fr_dbuff_t *dbuff, fr_dcursor_t *cursor, UNUSED fr_der_encode_ctx_t *encode_ctx)
static ssize_t fr_der_encode_set(fr_dbuff_t *dbuff, fr_dcursor_t *cursor, fr_der_encode_ctx_t *encode_ctx)
fr_test_point_pair_encode_t der_tp_encode_pair
static ssize_t fr_der_encode_bitstring(fr_dbuff_t *dbuff, fr_dcursor_t *cursor, fr_der_encode_ctx_t *encode_ctx)
fr_test_point_proto_encode_t der_tp_encode_proto
static ssize_t fr_der_encode_tag(fr_dbuff_t *dbuff, fr_der_tag_t tag_num, fr_der_tag_class_t tag_class, fr_der_tag_constructed_t constructed)
Encode a DER tag.
static ssize_t fr_der_encode_ipv6_prefix(fr_dbuff_t *dbuff, fr_dcursor_t *cursor, UNUSED fr_der_encode_ctx_t *encode_ctx)
static ssize_t fr_der_encode_integer(fr_dbuff_t *dbuff, fr_dcursor_t *cursor, UNUSED fr_der_encode_ctx_t *encode_ctx)
static ssize_t fr_der_encode_utc_time(fr_dbuff_t *dbuff, fr_dcursor_t *cursor, UNUSED fr_der_encode_ctx_t *encode_ctx)
static ssize_t fr_der_encode_ipv4_prefix(fr_dbuff_t *dbuff, fr_dcursor_t *cursor, UNUSED fr_der_encode_ctx_t *encode_ctx)
uint8_t * data
Pointer to the start of the encoded item (beginning of the tag)
size_t fr_sbuff_shift(fr_sbuff_t *sbuff, size_t shift)
Shift the contents of the sbuff, returning the number of bytes we managed to shift.
#define FR_SBUFF_OUT(_start, _len_or_end)
static char buff[sizeof("18446744073709551615")+3]
ssize_t fr_struct_to_network(fr_dbuff_t *dbuff, fr_da_stack_t *da_stack, unsigned int depth, fr_dcursor_t *parent_cursor, void *encode_ctx, fr_encode_dbuff_t encode_value, fr_encode_dbuff_t encode_pair)
Stores an attribute, a value and various bits of other data.
fr_dict_attr_t const *_CONST da
Dictionary attribute defines the attribute number, vendor and type of the pair.
fr_test_point_ctx_alloc_t test_ctx
Allocate a test ctx for the encoder.
fr_test_point_ctx_alloc_t test_ctx
Allocate a test ctx for the encoder.
Entry point for pair encoders.
Entry point for protocol encoders.
fr_slen_t fr_unix_time_to_str(fr_sbuff_t *out, fr_unix_time_t time, fr_time_res_t res, bool utc)
Convert unix time to string.
static fr_pair_t * fr_pair_dcursor_child_iter_init(fr_dcursor_t *cursor, fr_pair_list_t const *list, fr_dcursor_t const *parent)
Initializes a child dcursor from a parent cursor, with an iteration function.
void fr_pair_list_sort(fr_pair_list_t *list, fr_cmp_t cmp)
Sort a doubly linked list of fr_pair_ts using merge sort.
#define fr_pair_dcursor_init(_cursor, _list)
Initialises a special dcursor with callbacks that will maintain the attr sublists correctly.
size_t fr_pair_list_num_elements(fr_pair_list_t const *list)
Get the length of a list of fr_pair_t.
#define FR_PROTO_HEX_DUMP(_data, _data_len, _fmt,...)
#define FR_PROTO_TRACE(_fmt,...)
#define FR_PROTO_STACK_PRINT(_stack, _depth)
Structure for holding the stack of dictionary attributes being encoded.
char const * fr_strerror(void)
Get the last library error.
#define fr_strerror_printf(_fmt,...)
Log to thread local error buffer.
#define fr_strerror_const(_msg)
#define fr_type_is_group(_x)
#define fr_type_is_structural(_x)
#define fr_type_is_tlv(_x)
static char const * fr_type_to_str(fr_type_t type)
Return a static string containing the type name.
#define fr_type_is_struct(_x)
int8_t fr_value_box_cmp(fr_value_box_t const *a, fr_value_box_t const *b)
Compare two values.
static size_t char ** out