45RCSID(
"$Id: 27e48481175c476500fb5a774c9890ea04dd7f7a $")
48#include <freeradius-devel/util/value.h>
51#include <freeradius-devel/util/base16.h>
52#include <freeradius-devel/util/size.h>
62 "in_addr.s_addr has unexpected length");
64 "in6_addr.s6_addr has unexpected length");
66 "vb_ifid has unexpected length");
68 "vb_ether has unexpected length");
71 "datum.boolean has unexpected length");
73 "vb_uint8 has unexpected length");
75 "vb_uint16 has unexpected length");
77 "vb_uint32 has unexpected length");
79 "vb_uint64 has unexpected length");
82 "vb_int16 has unexpected length");
84 "vb_int16 has unexpected length");
86 "vb_int32 has unexpected length");
88 "vb_int64 has unexpected length");
91 "vb_float32 has unexpected length");
93 "vb_float64 has unexpected length");
105#define network_min_size(_x) (fr_value_box_network_sizes[_x][0])
106#define network_max_size(_x) (fr_value_box_network_sizes[_x][1])
378#pragma clang diagnostic ignored "-Wgnu-designator"
461 .name =
"unprintables",
555 L(
""),
L(
"\n"),
L(
"\r"),
L(
"\""))
561 L(
""),
L(
"\n"),
L(
"\r"),
L(
"'"))
567 L(
""),
L(
"\n"),
L(
"\r"),
L(
"/"))
573 L(
""),
L(
"\n"),
L(
"\r"),
L(
"`"))
582 L(
""),
L(
"\n"),
L(
"\r"),
L(
"\"\"\""))
588 L(
""),
L(
"\n"),
L(
"\r"),
L(
"'''"))
594 L(
""),
L(
"\n"),
L(
"\r"),
L(
"///"))
600 L(
""),
L(
"\n"),
L(
"\r"),
L(
"```"))
647 dst->vb_length = src->vb_length;
656 fr_value_box_list_init(&dst->vb_group);
681 dst->enumv = src->enumv;
682 dst->type = src->type;
683 dst->tainted = src->tainted;
684 dst->safe_for = src->safe_for;
685 dst->secret = src->secret;
686 fr_value_box_list_entry_init(dst);
702 if (a == b)
return 0;
713 if ((fpclassify(a) == FP_ZERO) || (fpclassify(b) == FP_ZERO)) {
715 if (diff < DBL_EPSILON)
return 0;
723 sum = fabs(a) + fabs(b);
728 if (sum < DBL_MIN)
goto check;
733 if ((diff / fmin(sum, DBL_MAX)) < DBL_EPSILON)
return 0;
750 if (a->type != b->type) {
764 if (a->vb_length < b->vb_length) {
765 length = a->vb_length;
767 length = b->vb_length;
780 if (a->secret || b->secret) {
783 cmp = memcmp(a->datum.ptr, b->datum.ptr, length);
785 if (cmp != 0)
return CMP(cmp, 0);
794 return CMP(a->vb_length, b->vb_length);
800#define RETURN(_type) return CMP(a->datum._type, b->datum._type)
801#define COMPARE(_type) return CMP(memcmp(&a->datum._type, &b->datum._type, sizeof(a->datum._type)), 0)
840 return float_cmp(a->vb_float32, b->vb_float32);
843 return float_cmp(a->vb_float64, b->vb_float64);
874 return CMP(a->vb_void, b->vb_void);
909 if (a_net == b_net) {
912 compare = memcmp(a, b, bytes);
918 if ((compare == 0) &&
980 if (common == 0)
return true;
985 if (common < 8)
break;
987 if (a[i] != b[i])
return false;
995 mask <<= (8 - common);
999 if ((a[i] &
mask) == ((b[i] &
mask))) {
1032 if (b->vb_ip.af != AF_INET)
goto fail_cmp_v4;
1039 if (b->vb_ip.af != AF_INET)
goto fail_cmp_v4;
1044 b->vb_ip.prefix, (
uint8_t const *) &b->vb_ipv4addr);
1056 if (b->vb_ip.af != AF_INET)
goto fail_cmp_v4;
1061 (
uint8_t const *) &a->vb_ipv4addr,
1062 32, (
uint8_t const *) &b->vb_ip.addr.v4);
1065 if (b->vb_ip.af != AF_INET)
goto fail_cmp_v4;
1070 (
uint8_t const *) &a->vb_ipv4addr,
1071 b->vb_ip.prefix, (
uint8_t const *) &b->vb_ipv4addr);
1081 if (b->vb_ip.af != AF_INET6)
goto fail_cmp_v6;
1088 if (b->vb_ip.af != AF_INET6)
goto fail_cmp_v6;
1093 b->vb_ip.prefix, (
uint8_t const *) &b->vb_ip.addr.v6);
1105 if (b->vb_ip.af != AF_INET6)
goto fail_cmp_v6;
1110 (
uint8_t const *) &a->vb_ip.addr.v6,
1111 128, (
uint8_t const *) &b->vb_ip.addr.v6);
1114 if (b->vb_ip.af != AF_INET6)
goto fail_cmp_v6;
1119 (
uint8_t const *) &a->vb_ip.addr.v6,
1120 b->vb_ip.prefix, (
uint8_t const *) &b->vb_ip.addr.v6);
1128 if (a->vb_ip.af != b->vb_ip.af)
goto fail_cmp_v4;
1133 if (a->vb_ip.af != b->vb_ip.af)
goto fail_cmp_v4;
1135 if (a->vb_ip.af == AF_INET)
goto cmp_prefix_v4;
1168 return (compare == 0);
1171 return (compare != 0);
1174 return (compare < 0);
1177 return (compare > 0);
1180 return (compare <= 0);
1183 return (compare >= 0);
1339 switch (src->type) {
1391 switch (src->type) {
1393 dst->vb_uint16 = htons(src->vb_uint16);
1398 dst->vb_uint32 = htonl(src->vb_uint32);
1403 dst->vb_uint64 = htonll(src->vb_uint64);
1407 dst->vb_int16 = htons(src->vb_int16);
1411 dst->vb_int32 = htonl(src->vb_int32);
1415 dst->vb_int64 = htonll(src->vb_int64);
1440 switch (
value->type) {
1446 if (
value->enumv->flags.length) {
1447 return value->enumv->flags.length;
1457 if (
value->vb_length > UINT16_MAX)
return UINT16_MAX;
1460 return value->vb_length;
1467 if (
value->enumv)
return value->enumv->flags.length;
1543 switch (
value->type) {
1546 max =
value->vb_length;
1553 if (
value->enumv->flags.length) {
1557 if (max < value->enumv->flags.length) {
1562 }
else if (max >
value->enumv->flags.length) {
1566 max =
value->enumv->flags.length;
1578 if (max > UINT16_MAX) max = UINT16_MAX;
1617 switch (
value->type) {
1622 sizeof(
value->vb_ipv4addr));
1632 sizeof(
value->vb_ipv4addr));
1653 switch (
value->vb_ip.af) {
1668 switch (
value->vb_ip.af) {
1720 if (
value->vb_attr->depth != 1) {
1722 value->vb_attr->depth,
value->vb_attr->name);
1726 switch (
value->vb_attr->flags.length) {
1729 base.vb_uint8 =
value->vb_attr->attr;
1734 base.vb_uint16 =
value->vb_attr->attr;
1739 base.vb_uint32 =
value->vb_attr->attr;
1744 value->vb_attr->flags.length,
value->vb_attr->name);
1765 if (!
value->enumv) {
1768 res =
value->enumv->flags.flag_time_res;
1772 if (!
value->enumv) {
1775 }
else switch (
value->enumv->flags.length) {
1777 if (date > UINT16_MAX) date = UINT16_MAX;
1783 if (date > UINT32_MAX) date = UINT32_MAX;
1802 if (
value->enumv) res =
value->enumv->flags.flag_time_res;
1806 if (!
value->enumv) {
1809 }
else if (!
value->enumv->flags.is_unsigned) {
1810 switch (
value->enumv->flags.length) {
1812 if (date < INT16_MIN) {
1814 }
else if (date > INT16_MAX) {
1822 if (date < INT32_MIN) {
1824 }
else if (date > INT32_MAX) {
1838 switch (
value->enumv->flags.length) {
1842 }
else if (date > UINT16_MAX) {
1851 }
else if (date > UINT32_MAX) {
1936 "Expected length >= %zu bytes, got %zu bytes",
1946 if (enumv && !enumv->flags.array) {
1948 "Expected length <= %zu bytes, got %zu bytes",
1961 size_t newlen = len;
1968 if (enumv->flags.length) {
1969 newlen = enumv->flags.length;
1992 if (newlen > len)
return -(newlen + offset);
2050 dst->vb_ip.scope_id = scope_id;
2066 dst->vb_ip.scope_id = scope_id;
2096 dst->datum.boolean = (val != 0);
2147 fr_strerror_const(
"No enumv (i.e. root) passed to fr_value_box_from_network for type 'attribute'");
2161 switch (enumv->flags.length) {
2183 if (!dst->vb_attr) {
2185 if (!dst->vb_attr)
return -1;
2205 length = enumv->flags.length;
2213 if (len > length)
return -(length);
2235 length = enumv->flags.length;
2243 if (len > length)
return -(length);
2247 if (!enumv || !enumv->flags.is_unsigned) {
2259 if (tmp > INT64_MAX) tmp = INT64_MAX;
2292 AF_INET, 32, 32, 0, 4,
2296 AF_INET, 0, 32, 0, 4,
2300 AF_INET6, 128, 128, 16, 16,
2304 AF_INET6, 0, 128, 0, 16,
2326 int prefix_len,
uint8_t const *
data,
size_t data_len,
2327 bool fixed,
bool tainted)
2365 fr_strerror_printf(
"Invalid prefix length '%d' - it requires %u bytes of data, and there are only %zu bytes of data",
2383 if (enumv && enumv->flags.array) {
2413 .prefix = prefix_len,
2417 if (!data_len)
return 0;
2419 fr_assert(data_len <=
sizeof(dst->vb_ip.addr));
2421 memcpy((
uint8_t *) &dst->vb_ip.addr,
data, data_len);
2448 void const *src,
size_t len)
2461 memcpy(&dst->datum, src, len);
2465 if (len !=
sizeof(
struct in_addr)) {
2472 memcpy(&dst->vb_ipv4addr, src, len);
2476 if (len !=
sizeof(
struct in6_addr)) {
2483 memcpy(&dst->vb_ipv6addr, src, len);
2518 switch (
value->type) {
2520 if (*outlen < 8)
return -1;
2535 if (slen < 0)
return -1;
2544 *outlen =
value->vb_ip.prefix;
2550 *outlen =
value->vb_length * 8;
2555 *outlen =
sizeof(
value->vb_ether) * 8;
2586 "destination type size %zu",
2599 if (!src->vb_length)
return 0;
2601 ptr = (
uint8_t *) &dst->datum;
2614 memcpy(ptr, src->vb_octets, src->vb_length);
2625 0x00, 0x00, 0x00, 0x00, 0xff, 0xff };
2646 switch (src->type) {
2654 (
char const *)src->vb_octets, src->vb_length, src->tainted);
2658 dst,
UNCONST(fr_value_box_list_t *, &src->vb_group),
2698 switch (src->type) {
2705 (
uint8_t const *)src->vb_strvalue, src->vb_length, src->tainted);
2709 dst,
UNCONST(fr_value_box_list_t *, &src->vb_group),
2718 (
uint8_t const *)&src->vb_ipv4addr,
2719 sizeof(src->vb_ipv4addr), src->tainted);
2729 sizeof(src->vb_ipv4addr) + 1, src->tainted) < 0)
return -1;
2731 bin[0] = src->vb_ip.prefix;
2732 memcpy(&bin[1], (
uint8_t const *)&src->vb_ipv4addr,
sizeof(src->vb_ipv4addr));
2741 (
uint8_t const *)src->vb_ipv6addr,
2742 sizeof(src->vb_ipv6addr), src->tainted);
2752 sizeof(src->vb_ipv6addr) + 2, src->tainted) < 0)
return -1;
2753 bin[0] = src->vb_ip.scope_id;
2754 bin[1] = src->vb_ip.prefix;
2755 memcpy(&bin[2], src->vb_ipv6addr,
sizeof(src->vb_ipv6addr));
2798#define CAST_IP_FIX_COMBO \
2799 case FR_TYPE_COMBO_IP_ADDR: \
2800 if (src->vb_ip.af == AF_INET) { \
2801 src_type = FR_TYPE_IPV4_ADDR; \
2802 } else if (src->vb_ip.af == AF_INET6) { \
2803 src_type = FR_TYPE_IPV6_ADDR; \
2806 case FR_TYPE_COMBO_IP_PREFIX: \
2807 if (src->vb_ip.af == AF_INET) { \
2808 src_type = FR_TYPE_IPV4_PREFIX; \
2809 } else if (src->vb_ip.af == AF_INET6) { \
2810 src_type = FR_TYPE_IPV6_PREFIX; \
2851 src->vb_strvalue, src->vb_length,
2864 dst->vb_ip.af = AF_INET;
2865 dst->vb_ip.prefix = 32;
2866 dst->vb_ip.scope_id = 0;
2878 memcpy(&dst->vb_ip.addr.v4, &src->vb_ipv6addr[
sizeof(
v4_v6_map)],
2879 sizeof(dst->vb_ip.addr.v4));
2884 if (src->vb_ip.prefix != 32) {
2886 "cast to IP address types",
2895 memcpy(&dst->vb_ip.addr.v4, &src->vb_ip.addr.v4,
sizeof(dst->vb_ip.addr.v4));
2899 if (src->vb_ip.prefix != 128) {
2900 fr_strerror_printf(
"Invalid cast from %s to %s. Only /128 (not /%i) prefixes may be "
2901 "cast to IP address types",
2907 if (memcmp(&src->vb_ipv6addr,
v4_v6_map,
sizeof(
v4_v6_map)) != 0)
goto bad_v6_prefix_map;
2908 memcpy(&dst->vb_ip.addr.v4, &src->vb_ipv6addr[
sizeof(
v4_v6_map)],
2909 sizeof(dst->vb_ip.addr.v4));
2913 if (src->vb_length !=
sizeof(dst->vb_ipv4addr)) {
2914 fr_strerror_printf(
"Invalid cast from %s to %s. Needed octet string of length %zu, got %zu",
2917 sizeof(dst->vb_ipv4addr), src->vb_length);
2920 memcpy(&dst->vb_ip.addr.v4, src->vb_octets,
sizeof(dst->vb_ipv4addr));
2927 net = ntohl(src->vb_uint32);
2928 memcpy(&dst->vb_ip.addr.v4, (
uint8_t *)&net,
sizeof(dst->vb_ipv4addr));
2965 src->vb_strvalue, src->vb_length,
2978 dst->vb_ip.af = AF_INET;
2979 dst->vb_ip.scope_id = 0;
2983 dst->vb_ip.prefix = src->vb_ip.prefix;
2987 memcpy(&dst->vb_ip, &src->vb_ip,
sizeof(dst->vb_ip));
3001 memcpy(&dst->vb_ipv4addr, &src->vb_ipv6addr[
sizeof(
v4_v6_map)],
3002 sizeof(dst->vb_ipv4addr));
3003 dst->vb_ip.prefix = 32;
3007 if (memcmp(src->vb_ipv6addr,
v4_v6_map,
sizeof(
v4_v6_map)) != 0)
goto bad_v6_prefix_map;
3009 if (src->vb_ip.prefix < (
sizeof(
v4_v6_map) << 3)) {
3010 fr_strerror_printf(
"Invalid cast from %s to %s. Expected prefix >= %u bits got %u bits",
3013 (
unsigned int)(
sizeof(
v4_v6_map) << 3), src->vb_ip.prefix);
3016 memcpy(&dst->vb_ipv4addr, &src->vb_ipv6addr[
sizeof(
v4_v6_map)],
3017 sizeof(dst->vb_ipv4addr));
3022 dst->vb_ip.prefix = src->vb_ip.prefix - (
sizeof(
v4_v6_map) << 3);
3026 if (src->vb_length !=
sizeof(dst->vb_ipv4addr) + 1) {
3027 fr_strerror_printf(
"Invalid cast from %s to %s. Needed octet string of length %zu, got %zu",
3030 sizeof(dst->vb_ipv4addr) + 1, src->vb_length);
3033 dst->vb_ip.prefix = src->vb_octets[0];
3034 memcpy(&dst->vb_ip.addr.v4, &src->vb_octets[1],
sizeof(dst->vb_ipv4addr));
3041 net = ntohl(src->vb_uint32);
3042 memcpy(&dst->vb_ip.addr.v4, (
uint8_t *)&net,
sizeof(dst->vb_ipv4addr));
3043 dst->vb_ip.prefix = 32;
3074 static_assert((
sizeof(
v4_v6_map) +
sizeof(src->vb_ip.addr.v4)) <=
3075 sizeof(src->vb_ip.addr.v6),
"IPv6 storage too small");
3083 src->vb_strvalue, src->vb_length,
3096 dst->vb_ip.af = AF_INET6;
3097 dst->vb_ip.prefix = 128;
3102 uint8_t *p = dst->vb_ipv6addr;
3107 memcpy(p, (
uint8_t const *)&src->vb_ipv4addr,
sizeof(src->vb_ipv4addr));
3108 dst->vb_ip.scope_id = 0;
3114 uint8_t *p = dst->vb_ipv6addr;
3116 if (src->vb_ip.prefix != 32) {
3118 "cast to IP address types",
3128 memcpy(p, (
uint8_t const *)&src->vb_ipv4addr,
sizeof(src->vb_ipv4addr));
3129 dst->vb_ip.scope_id = 0;
3134 if (src->vb_ip.prefix != 128) {
3135 fr_strerror_printf(
"Invalid cast from %s to %s. Only /128 (not /%i) prefixes may be "
3136 "cast to IP address types",
3145 memcpy(dst->vb_ipv6addr, src->vb_ipv6addr,
3146 sizeof(dst->vb_ipv6addr));
3147 dst->vb_ip.scope_id = src->vb_ip.scope_id;
3151 if (src->vb_length !=
sizeof(dst->vb_ipv6addr)) {
3152 fr_strerror_printf(
"Invalid cast from %s to %s. Needed octet string of length %zu, got %zu",
3155 sizeof(dst->vb_ipv6addr), src->vb_length);
3158 memcpy(&dst->vb_ipv6addr, src->vb_octets,
sizeof(dst->vb_ipv6addr));
3194 src->vb_strvalue, src->vb_length,
3207 dst->vb_ip.af = AF_INET6;
3212 uint8_t *p = dst->vb_ipv6addr;
3217 memcpy(p, (
uint8_t const *)&src->vb_ipv4addr,
sizeof(src->vb_ipv4addr));
3218 dst->vb_ip.prefix = 128;
3219 dst->vb_ip.scope_id = 0;
3225 uint8_t *p = dst->vb_ipv6addr;
3230 memcpy(p, (
uint8_t const *)&src->vb_ipv4addr,
sizeof(src->vb_ipv4addr));
3231 dst->vb_ip.prefix = (
sizeof(
v4_v6_map) << 3) + src->vb_ip.prefix;
3232 dst->vb_ip.scope_id = 0;
3237 dst->vb_ip.prefix = src->vb_ip.prefix;
3241 dst->vb_ip.prefix = 128;
3243 memcpy(dst->vb_ipv6addr, src->vb_ipv6addr,
3244 sizeof(dst->vb_ipv6addr));
3245 dst->vb_ip.scope_id = src->vb_ip.scope_id;
3249 if (src->vb_length != (
sizeof(dst->vb_ipv6addr) + 2)) {
3250 fr_strerror_printf(
"Invalid cast from %s to %s. Needed octet string of length %zu, got %zu",
3253 sizeof(dst->vb_ipv6addr) + 2, src->vb_length);
3256 dst->vb_ip.scope_id = src->vb_octets[0];
3257 dst->vb_ip.prefix = src->vb_octets[1];
3258 memcpy(&dst->vb_ipv6addr, src->vb_octets + 2,
sizeof(dst->vb_ipv6addr));
3287 switch (src->type) {
3290 src->vb_strvalue, src->vb_length,
3305 switch (src->type) {
3314 if ((array[0] != 0) || (array[1] != 0))
return -1;
3316 memcpy(dst->vb_ether, &array[2], 6);
3345 switch (src->type) {
3348 src->vb_strvalue, src->vb_length,
3366 switch (src->type) {
3368 dst->vb_bool = (src->vb_int8 != 0);
3372 dst->vb_bool = (src->vb_uint8 != 0);
3376 dst->vb_bool = (src->vb_int16 != 0);
3380 dst->vb_bool = (src->vb_uint16 != 0);
3384 dst->vb_bool = (src->vb_int32 != 0);
3388 dst->vb_bool = (src->vb_uint32 != 0);
3392 dst->vb_bool = (src->vb_int64 != 0);
3396 dst->vb_bool = (src->vb_uint64 != 0);
3400 dst->vb_bool = (src->vb_size != 0);
3408 dst->vb_bool = (fpclassify(src->vb_float32) != FP_ZERO);
3412 dst->vb_bool = (fpclassify(src->vb_float64) != FP_ZERO);
3435#define SIGN_BIT_HIGH(_int, _len) ((((uint64_t)1) << (((_len) << 3) - 1)) & (_int))
3436#define SIGN_PROMOTE(_int, _len) ((_len) < sizeof(_int) ? \
3437 (_int) | (~((__typeof__(_int))0)) << ((_len) << 3) : (_int))
3439#if !defined(NDEBUG) || defined(STATIC_ANALYZER)
3445 "invalid source type len, expected > 0, got %zu",
3451 "Invalid cast from %s to %s. "
3452 "invalid source type len, expected <= %zu, got %zu",
3455 sizeof(uint64_t), len))
return -1;
3458 switch (src->type) {
3471 if (src->enumv) res = src->enumv->flags.flag_time_res;
3487 if (src->enumv) res = src->enumv->flags.flag_time_res;
3494#ifdef WORDS_BIGENDIAN
3495 memcpy(((
uint8_t *)&tmp) + (
sizeof(tmp) - len),
3512 if ((int64_t)tmp <
min) {
3514 "outside value range %"PRId64
"-%"PRIu64,
3523 "outside value range 0-%"PRIu64,
3536 if (dst->enumv) res = dst->enumv->flags.flag_time_res;
3550 if (dst->enumv) res = dst->enumv->flags.flag_time_res;
3561#ifdef WORDS_BIGENDIAN
3586 switch (src->type) {
3589 src->vb_strvalue, src->vb_length,
3617 memcpy(&tmp.vb_uint32, &src->vb_ip.addr.v4,
sizeof(tmp.vb_uint32));
3638 memcpy(((
uint8_t *)&tmp.vb_uint64) + (
sizeof(tmp.vb_uint64) -
sizeof(src->vb_ether)),
3639 &src->vb_ether,
sizeof(src->vb_ether));
3640#ifndef WORDS_BIGENDIAN
3680 dst->vb_uint8 = src->vb_float32;
3684 dst->vb_uint16 = src->vb_float32;
3688 dst->vb_uint32 = src->vb_float32;
3692 dst->vb_uint64 = src->vb_float32;
3696 dst->vb_int8 = src->vb_float32;
3700 dst->vb_int16 = src->vb_float32;
3704 dst->vb_int32 = src->vb_float32;
3708 dst->vb_int64 = src->vb_float32;
3712 dst->vb_size = src->vb_float32;
3718 sec = src->vb_float32;
3720 nsec = ((src->vb_float32 *
NSEC) - ((
float) sec));
3733 sec = src->vb_float32;
3735 nsec = ((src->vb_float32 * res) - ((
double) sec));
3739 if (fail)
goto overflow;
3755 dst->vb_uint8 = src->vb_float64;
3759 dst->vb_uint16 = src->vb_float64;
3763 dst->vb_uint32 = src->vb_float64;
3767 dst->vb_uint64 = src->vb_float64;
3771 dst->vb_int8 = src->vb_float64;
3775 dst->vb_int16 = src->vb_float64;
3779 dst->vb_int32 = src->vb_float64;
3783 dst->vb_int64 = src->vb_float64;
3787 dst->vb_size = src->vb_float64;
3793 sec = src->vb_float64;
3795 nsec = ((src->vb_float64 *
NSEC) - ((
double) sec));
3808 sec = src->vb_float64;
3810 nsec = ((src->vb_float64 * res) - ((
double) sec));
3814 if (fail)
goto overflow;
3845 switch (src->type) {
3848 num = (double) src->vb_float32;
3856 num = src->vb_float64;
3871 num = src->vb_int16;
3875 num = src->vb_int32;
3879 num = src->vb_int64;
3883 num = src->vb_uint8;
3887 num = src->vb_uint16;
3891 num = src->vb_uint32;
3895 num = src->vb_uint64;
3926 dst->vb_float32 = num;
3928 dst->vb_float64 = num;
3988 if (dst_type == src->type) {
3992 if (ret < 0)
return ret;
3994 if (dst_enumv) dst->enumv = dst_enumv;
4050 dst->enumv = dst_enumv;
4066 if (when > INT64_MAX) {
4072 dst->enumv = dst_enumv;
4101 switch (src->vb_attr->flags.length) {
4104 dst->vb_uint8 = src->vb_attr->attr;
4109 dst->vb_uint16 = src->vb_attr->attr;
4114 dst->vb_uint32 = src->vb_attr->attr;
4119 src->vb_attr->flags.length, src->vb_attr->name);
4145 src->vb_strvalue, src->vb_length,
4152 fr_strerror_printf(
"Invalid cast from %s to %s. Source is length %zu is smaller than "
4153 "destination type size %zu",
4163 "destination type size %zu",
4178 tmp.type = dst_type;
4179 dst->enumv = dst_enumv;
4189 dst->enumv = dst_enumv;
4217 fr_value_box_entry_t entry = vb->entry;
4223 if (vb->type == dst_type) {
4224 vb->enumv = dst_enumv;
4263#define O(_x, _y) case FR_TYPE_##_x: return vb->vb_##_y
4298 switch (ipaddr->
af) {
4313 memcpy(&dst->vb_ip, ipaddr,
sizeof(dst->vb_ip));
4334 memcpy(dst, &src->vb_ip,
sizeof(*dst));
4347 switch (
data->type) {
4364 while ((vb = fr_value_box_list_pop_head(&
data->vb_group)) != NULL) {
4382 memset(&
data->datum, 0,
sizeof(
data->datum));
4410 switch (src->type) {
4435 dst->vb_strvalue = str;
4444 if (src->vb_length) {
4445 bin = talloc_memdup(ctx, src->vb_octets, src->vb_length);
4450 talloc_set_type(bin,
uint8_t);
4452 bin = talloc_array(ctx,
uint8_t, 0);
4454 dst->vb_octets = bin;
4465 while ((child = fr_value_box_list_next(&src->vb_group, child))) {
4475 fr_value_box_list_talloc_free(&dst->vb_group);
4486 fr_value_box_list_insert_tail(&dst->vb_group,
new);
4495 if (src->vb_attr->flags.is_unknown) {
4497 if (!dst->vb_attr)
return -1;
4500 dst->vb_attr = src->vb_attr;
4534 switch (src->type) {
4541 dst->datum.ptr = ctx ? talloc_reference(ctx, src->datum.ptr) : src->datum.ptr;
4546 dst->vb_attr = src->vb_attr;
4551 dst->vb_void = src->vb_void;
4568 switch (src->type) {
4576 str = talloc_steal(ctx, src->vb_strvalue);
4581 talloc_set_type(str,
char);
4582 dst->vb_strvalue = str;
4584 memset(&src->datum, 0,
sizeof(src->datum));
4592 bin = talloc_steal(ctx, src->vb_octets);
4597 talloc_set_type(bin,
uint8_t);
4599 dst->vb_octets = bin;
4601 memset(&src->datum, 0,
sizeof(src->datum));
4609 while ((child = fr_value_box_list_pop_head(&src->vb_group))) {
4610 child = talloc_steal(ctx, child);
4615 fr_value_box_list_insert_tail(&dst->vb_group, child);
4634 char const *src,
bool tainted)
4645 dst->vb_strvalue = str;
4666 len = strlen(vb->vb_strvalue);
4667 str = talloc_realloc(ctx,
UNCONST(
char *, vb->vb_strvalue),
char, len + 1);
4672 vb->vb_strvalue = str;
4673 vb->vb_length = len;
4691 char const *
fmt, va_list ap)
4700 if (!str)
return -1;
4703 dst->vb_strvalue = str;
4722 char const *
fmt, ...)
4744 char const *src,
bool tainted)
4747 dst->vb_strvalue = src;
4748 dst->vb_length = strlen(src);
4762 vb->vb_strvalue = src;
4763 vb->vb_length = len < 0 ? strlen(src) : (
size_t)len;
4779 size_t len,
bool tainted)
4783 str = talloc_zero_array(ctx,
char, len + 1);
4791 dst->vb_strvalue = str;
4792 dst->vb_length = len;
4819 if (dstlen == len)
return 0;
4821 str = talloc_realloc(ctx,
UNCONST(
char *, dst->vb_strvalue),
char, len + 1);
4831 memset(str + dstlen,
'\0', (len - dstlen) + 1);
4835 dst->vb_strvalue = str;
4836 dst->vb_length = len;
4853 char const *src,
size_t len,
bool tainted)
4870 dst->vb_strvalue = str;
4871 dst->vb_length = len;
4881 str = talloc_array(ctx,
char, len + 1);
4894 dst->vb_strvalue = str;
4895 dst->vb_length = len;
4916 char const *src,
bool tainted)
4922 len = talloc_array_length(src);
4923 if ((len == 0) || (src[len - 1] !=
'\0')) {
4940 char const *src,
size_t len,
bool tainted)
4943 dst->vb_strvalue = src;
4944 dst->vb_length = len;
4961 char const *src,
bool tainted)
4967 len = talloc_array_length(src);
4968 if ((len == 0) || (src[len - 1] !=
'\0')) {
4974 dst->vb_strvalue = ctx ? talloc_reference(ctx, src) : src;
4975 dst->vb_length = len - 1;
5000 size_t len,
bool tainted)
5004 bin = talloc_array(ctx,
uint8_t, len);
5009 talloc_set_type(bin,
uint8_t);
5012 dst->vb_octets = bin;
5013 dst->vb_length = len;
5039 dstlen = talloc_array_length(dst->vb_octets);
5040 if (dstlen == len)
return 0;
5050 bin = talloc_array(ctx,
uint8_t, 0);
5066 if (dstlen < len) memset(bin + dstlen, 0x00, len - dstlen);
5067 dst->vb_octets = bin;
5068 dst->vb_length = len;
5094 uint8_t const *src,
size_t len,
bool tainted)
5104 bin = talloc_memdup(ctx, src, len);
5109 talloc_set_type(bin,
uint8_t);
5112 dst->vb_octets = bin;
5113 dst->vb_length = len;
5123 bin = talloc_size(ctx, len);
5133 talloc_set_type(bin,
uint8_t);
5136 dst->vb_octets = bin;
5137 dst->vb_length = len;
5156 uint8_t const *src,
bool tainted)
5178 uint8_t const *src,
size_t len,
bool tainted)
5181 dst->vb_octets = src;
5182 dst->vb_length = len;
5196 uint8_t const *src,
bool tainted)
5201 dst->vb_octets = ctx ? talloc_reference(ctx, src) : src;
5202 dst->vb_length = talloc_array_length(src);
5213 dst->vb_cursor = cursor;
5214 dst->vb_cursor_name =
name;
5226 dst->vb_void =
UNCONST(
void *, ptr);
5248 if (!da->flags.has_value) {
5273 vb->vb_uint8 = vb->vb_uint8 ==
UINT8_MAX ? 0 : vb->vb_uint8 + 1;
5277 vb->vb_uint16 = vb->vb_uint16 == UINT16_MAX ? 0 : vb->vb_uint16 + 1;
5281 vb->vb_uint32 = vb->vb_uint32 == UINT32_MAX ? 0 : vb->vb_uint32 + 1;
5285 vb->vb_uint64 = vb->vb_uint64 == UINT64_MAX ? 0 : vb->vb_uint64 + 1;
5289 vb->vb_int8 = vb->vb_int8 == INT8_MAX ? INT8_MIN : vb->vb_int8 + 1;
5293 vb->vb_int16 = vb->vb_int16 == INT16_MAX ? INT16_MIN : vb->vb_int16 + 1;
5297 vb->vb_int32 = vb->vb_int32 == INT32_MAX ? INT32_MIN : vb->vb_int32 + 1;
5301 vb->vb_int64 = vb->vb_int64 == INT64_MAX ? INT64_MIN : vb->vb_int64 + 1;
5322static inline CC_HINT(always_inline)
5325 fr_sbuff_t *
in, fr_sbuff_parse_rules_t const *rules,
bool tainted)
5391 rules->escapes ? rules->escapes->chr :
'\0');
5403 fr_sbuff_parse_error_to_strerror(
err);
5427 static fr_sbuff_parse_rules_t default_rules;
5434 if (!rules) rules = &default_rules;
5441 if (dst_enumv && dst_enumv->flags.has_value && (dst_type !=
FR_TYPE_ATTR)) {
5458 if (!rules->escapes) {
5462 fr_sbuff_marker(&m, &our_in);
5467 fr_sbuff_marker_release(&m);
5469 if (!len)
goto parse;
5503 rules->terminals, rules->escapes);
5505 fr_sbuff_set_to_start(&our_in);
5511 fr_sbuff_set_to_start(&our_in);
5534 if (!dst_enumv || !unescaped) {
5537 if (
unlikely(fr_sbuff_out_aunescape_until(ctx, &
buff, &our_in, SIZE_MAX,
5538 rules->terminals, rules->escapes) < 0)) {
5564 if (!dst_enumv || !unescaped) {
5569 fr_sbuff_out_aunescape_until(ctx, &
buff, &our_in, SIZE_MAX,
5570 rules->terminals, rules->escapes);
5597 bin = talloc_zero_array(ctx,
uint8_t, 0);
5612 fr_sbuff_marker(&hex_start, &our_in);
5629 if ((hex_len & 0x01) != 0) {
5655 if (!name_len)
return 0;
5667 "for non-prefix types", addr.
prefix);
5671 memcpy(&dst->vb_ip, &addr,
sizeof(dst->vb_ip));
5678 if (!name_len)
return 0;
5688 if (!name_len)
return 0;
5704 if (addr.
prefix != 128) {
5707 "for non-prefix types", addr.
prefix);
5711 memcpy(&dst->vb_ip, &addr,
sizeof(dst->vb_ip));
5718 if (!name_len)
return 0;
5728 if (!name_len)
return 0;
5740 if ((addr.
af == AF_INET) && (addr.
prefix != 32)) {
5741 goto fail_ipv4_prefix;
5744 if ((addr.
af == AF_INET6) && (addr.
prefix != 128)) {
5745 goto fail_ipv6_prefix;
5748 memcpy(&dst->vb_ip, &addr,
sizeof(dst->vb_ip));
5755 if (!name_len)
return 0;
5785 if (slen >= 0)
return slen;
5791 if (!rules->escapes) {
5797 dst->vb_bool = (stmp != 0);
5803 dst->vb_bool = (utmp != 0);
5809 "\"yes\", \"no\", \"true\", \"false\" or any unquoted integer");
5849 fr_sbuff_set_to_start(&our_in);
5854 fr_sbuff_parse_error_to_strerror(
err);
5897 false, rules->terminals);
5898 if (slen < 0)
return slen;
5924 fr_strerror_printf(
"Can only start from data type 'tlv' for data type 'attribute', and not from %s", dst_enumv->name);
5939 fr_sbuff_marker(&m, &our_in);
5945 fr_sbuff_marker_release(&m);
5956 if (!dst->vb_attr) {
5963 if (dst->vb_attr->dict != dst_enumv->dict) {
5996 if (slen <= 0)
goto invalid_attr;
6012 if (rules && rules->terminals) {
6016 rules->terminals, rules->escapes);
6017 if (len >=
sizeof(
buffer))
goto too_small;
6048 dst->enumv = dst_enumv;
6065 dst->type = dst_type;
6066 dst->tainted =
false;
6072 dst->enumv = dst_enumv;
6073 fr_value_box_list_entry_init(dst);
6084 fr_sbuff_parse_rules_t prules = { .escapes = erules };
6087 if (slen <= 0)
return slen;
6090 fr_strerror_printf(
"Failed parsing '%s'. %zu bytes of trailing data after string value \"%pV\"",
6094 return (slen -
inlen) - 1;
6121 if (
data->enumv &&
data->enumv->flags.has_value) {
6131 switch (
data->type) {
6134 data->vb_strvalue,
data->vb_length, e_rules);
6171 data->vb_ether[0],
data->vb_ether[1],
6172 data->vb_ether[2],
data->vb_ether[3],
6173 data->vb_ether[4],
data->vb_ether[5]);
6224 if (
data->enumv) res =
data->enumv->flags.flag_time_res;
6237 bool is_unsigned =
false;
6240 res =
data->enumv->flags.flag_time_res;
6241 is_unsigned =
data->enumv->flags.is_unsigned;
6257 if (!e_rules || (!e_rules->do_oct && !e_rules->do_hex)) {
6268 NULL, &our_out,
UNCONST(fr_value_box_list_t *, &
data->vb_group),
6269 ", ", (
sizeof(
", ") - 1), e_rules,
6285 switch (
data->enumv->type) {
6361 switch (
data->type) {
6405 if (fr_value_box_list_empty(list))
return 0;
6412 if (!flatten)
goto print;
6414 sep, sep_len, e_rules,
6415 proc_action, safe_for, flatten);
6424 box_safe_for = safe_for;
6426 slen =
fr_sbuff_in_escape(&our_sbuff, (
char const *)vb->vb_strvalue, vb->vb_length, e_rules);
6446 if (e_rules) box_safe_for = safe_for;
6450 if (slen < 0)
return slen;
6455 if (sep && fr_value_box_list_next(list, vb)) {
6457 if (slen < 0)
return slen;
6469 (safety->safe_for != box_safe_for)) {
6471 safety->safe_for = box_safe_for;
6477 safety->tainted |= vb->tainted;
6478 safety->secret |= vb->secret;
6515 uint8_t const *sep,
size_t sep_len,
6519 TALLOC_CTX *tmp_ctx = NULL;
6522 if (fr_value_box_list_empty(list))
return 0;
6527 if (!flatten)
goto cast;
6530 proc_action, flatten);
6549 if (!tmp_ctx) tmp_ctx = talloc_pool(NULL, 1024);
6571 if (sep && fr_value_box_list_next(list, vb)) {
6573 if (slen < 0)
goto error;
6620 fr_dbuff_uctx_talloc_t dbuff_tctx;
6627 fr_value_box_entry_t entry;
6629 if (fr_value_box_list_empty(list)) {
6638 if ((fr_value_box_list_num_elements(list) == 1) && (head_vb ==
out) && (head_vb->type ==
type))
return 0;
6642 if (
unlikely(!fr_sbuff_init_talloc(ctx, &sbuff, &sbuff_tctx, 256, max_size)))
return -1;
6664 if (
out == head_vb) {
6701 fr_value_box_list_insert_head(list, head_vb);
6716 proc_action, flatten) < 0) {
6717 fr_value_box_list_insert_head(list, head_vb);
6729 fr_value_box_list_insert_head(list,
out);
6755 proc_action, flatten) < 0)
goto error;
6813 ret = escape->
func(vb, uctx);
6819 if (!ret) vb->safe_for = escape->
safe_for;
6820 vb->tainted =
false;
6869 .always_escape =
false,
6909 .always_escape =
false,
6916 if (rcode < 0)
return rcode;
6921 if (rcode < 0)
return rcode;
6923 fr_assert(fr_value_box_list_num_elements(&vb->vb_group) == 0);
6935 if (slen < 0)
return -1;
6978 fr_value_box_list_remove(&child->vb_group, grandchild);
6979 if (steal) talloc_steal(ctx, grandchild);
6980 fr_value_box_list_insert_before(list, child, grandchild);
7001 char *aggr, *td = NULL;
7002 TALLOC_CTX *pool = NULL;
7004 if (!vb)
return NULL;
7007 if (!aggr)
return NULL;
7008 if (!fr_value_box_list_next(list, vb))
return aggr;
7014 pool = talloc_pool(NULL, 255);
7017 while ((vb = fr_value_box_list_next(list, vb))) {
7018 char *str, *new_aggr;
7051 char *aggr, *td = NULL;
7052 TALLOC_CTX *pool = NULL;
7054 if (!vb)
return NULL;
7061 if (!aggr)
return NULL;
7062 if (!fr_value_box_list_next(list, vb))
return aggr;
7068 pool = talloc_pool(NULL, 255);
7071 while ((vb = fr_value_box_list_next(list, vb))) {
7072 char *str, *new_aggr;
7106 return fr_hash(vb->vb_strvalue, vb->vb_length);
7109 return fr_hash(vb->vb_octets, vb->vb_length);
7112 return fr_hash(&vb->vb_attr,
sizeof(vb->vb_attr));
7139 while ((in_p = fr_value_box_list_next(
in, in_p))) {
7145 fr_value_box_list_talloc_free(
out);
7167 while ((vb = fr_value_box_list_next(
head, vb))) {
7169 if (vb->tainted)
return true;
7183 while ((vb = fr_value_box_list_next(
head, vb))) {
7198 while ((vb = fr_value_box_list_next(
head, vb))) {
7200 vb->tainted =
false;
7226 if (!vb->vb_length) {
7228 fr_fatal_assert_msg(!vb->vb_strvalue || (talloc_array_length(vb->vb_strvalue) == 1),
"CONSISTENCY CHECK FAILED %s[%d]: fr_value_box_t strvalue field "
7229 "wasn non-NULL, but length was %u",
file,
line, vb->vb_length);
7234 fr_fatal_assert_msg(vb->vb_strvalue,
"CONSISTENCY CHECK FAILED %s[%d]: fr_value_box_t strvalue field "
7237 "CONSISTENCY CHECK FAILED %s[%i]: fr_value_box_t strvalue field "
7238 "not null terminated",
file,
line);
7240 size_t len = talloc_array_length(vb->vb_strvalue);
7244 fr_fatal_assert_fail(
"CONSISTENCY CHECK FAILED %s[%d]: Expected fr_value_box_t->vb_strvalue talloc buffer "
7245 "len >= %zu, got %zu",
7246 file,
line, vb->vb_length + 1, len);
7252 if (!vb->vb_length) {
7254 fr_fatal_assert_msg(!vb->vb_octets || (talloc_array_length(vb->vb_octets) == 0),
"CONSISTENCY CHECK FAILED %s[%d]: fr_value_box_t octets field "
7255 "wasn non-NULL, but length was %u",
file,
line, vb->vb_length);
7260 fr_fatal_assert_msg(vb->vb_octets,
"CONSISTENCY CHECK FAILED %s[%d]: fr_value_box_t octets field "
7265 fr_fatal_assert_msg(vb->vb_void,
"CONSISTENCY CHECK FAILED %s[%d]: fr_value_box_t ptr field "
7274 fr_fatal_assert_msg(vb->vb_attr,
"CONSISTENCY CHECK FAILED %s[%d]: fr_value_box_t vb_attr field "
7303 vb->safe_for = safe_for;
7332 vb->safe_for = safe_for;
7342 if (
out ==
in)
return;
7344 out->safe_for =
in->safe_for;
7345 out->tainted =
in->tainted;
7346 out->secret =
in->secret;
7356 out->tainted =
in->tainted;
7357 out->secret =
in->secret;
7364 if (
out ==
in)
return;
7372 (
out->safe_for !=
in->safe_for)) {
7380 out->safe_for =
in->safe_for;
7387 out->tainted |=
in->tainted;
7388 out->secret |=
in->secret;
7414 return (fr_value_box_list_num_elements(&
in->vb_group) > 0);
7421 return (
in->vb_length > 0);
7444#define INFO_INDENT(_fmt, ...) fprintf(fp, "%*s" _fmt "\n", depth * 2, " ", ## __VA_ARGS__)
7490 vb->secret ?
"s" :
"-",
7491 vb->tainted ?
"t" :
"-",
7496 vb->secret ?
"s" :
"-",
7497 vb->tainted ?
"t" :
"-",
static int const char char buffer[256]
static int const char * fmt
#define fr_base16_encode(_out, _in)
#define fr_base16_decode(_err, _out, _in, _no_trailing)
#define UNCONST(_type, _ptr)
Remove const qualification from a pointer.
#define L(_str)
Helper for initialising arrays of string literals.
#define FALL_THROUGH
clang 10 doesn't recognised the FALL-THROUGH comment anymore
#define SIZEOF_MEMBER(_t, _m)
#define CMP(_a, _b)
Same as CMP_PREFER_SMALLER use when you don't really care about ordering, you just want an ordering.
static fr_atomic_queue_t ** aq
static size_t min(size_t x, size_t y)
int fr_dbuff_trim_talloc(fr_dbuff_t *dbuff, size_t len)
Trim a talloced dbuff to the minimum length required to represent the contained string.
#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_OUT_UINT64V_RETURN(_num, _dbuff_or_marker, _len)
Read bytes from a dbuff or marker and interpret them as a network order unsigned integer.
#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_OUT_INT64V_RETURN(_num, _dbuff_or_marker, _len)
Read bytes from a dbuff or marker and interpret them as a network order unsigned integer.
#define fr_dbuff_buff(_dbuff_or_marker)
Return the underlying buffer in a dbuff or one of marker.
#define fr_dbuff_out_memcpy(_out, _dbuff_or_marker, _outlen)
Copy exactly _outlen bytes from the dbuff.
#define FR_DBUFF_MEMSET_RETURN(_dbuff_or_marker, _c, _inlen)
Set _inlen bytes of a dbuff or marker to _c returning if there is insufficient space.
#define FR_DBUFF_OUT_MEMCPY_RETURN(_out, _dbuff_or_marker, _outlen)
Copy outlen bytes from the dbuff returning if there's insufficient data in 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_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_OUT_RETURN(_out, _dbuff_or_marker)
Copy data from a dbuff or marker to a fixed sized C type returning if there is insufficient data.
static fr_dbuff_t * fr_dbuff_init_talloc(TALLOC_CTX *ctx, fr_dbuff_t *dbuff, fr_dbuff_uctx_talloc_t *tctx, size_t init, size_t max)
Initialise a special dbuff which automatically extends as additional data is written.
#define FR_DBUFF_IN_BYTES_RETURN(_dbuff_or_marker,...)
Copy a byte sequence into a dbuff or marker returning if there's insufficient space.
#define FR_DBUFF_TMP(_start, _len_or_end)
Creates a compound literal to pass into functions which accept a dbuff.
#define fr_fatal_assert_fail(_msg,...)
Calls panic_action ifndef NDEBUG, else logs error and causes the server to exit immediately with code...
#define fr_cond_assert(_x)
Calls panic_action ifndef NDEBUG, else logs error and evaluates to value of _x.
#define fr_assert_msg(_x, _msg,...)
Calls panic_action ifndef NDEBUG, else logs error and causes the server to exit immediately with code...
#define fr_assert_fail(_msg,...)
Calls panic_action ifndef NDEBUG, else logs error.
#define fr_cond_assert_msg(_x, _fmt,...)
Calls panic_action ifndef NDEBUG, else logs error and evaluates to value of _x.
#define fr_fatal_assert_msg(_x, _fmt,...)
Calls panic_action ifndef NDEBUG, else logs error and causes the server to exit immediately with code...
#define da_is_length_field16(_da)
bool const fr_dict_attr_nested_allowed_chars[SBUFF_CHAR_CLASS]
Characters allowed in a nested dictionary attribute name.
static fr_dict_attr_t * fr_dict_attr_unknown_copy(TALLOC_CTX *ctx, fr_dict_attr_t const *da)
#define da_is_length_field8(_da)
int fr_dict_protocol_reference(fr_dict_attr_t const **da_p, fr_dict_attr_t const *root, fr_sbuff_t *in)
Resolve a reference string to a dictionary attribute.
bool const fr_dict_enum_allowed_chars[SBUFF_CHAR_CLASS]
Characters that are allowed in dictionary enumeration value names.
fr_slen_t fr_dict_attr_by_oid_substr(fr_dict_attr_err_t *err, fr_dict_attr_t const **out, fr_dict_attr_t const *parent, fr_sbuff_t *in, fr_sbuff_term_t const *tt))
Resolve an attribute using an OID string.
static fr_dict_attr_t * fr_dict_attr_unknown_raw_afrom_num(TALLOC_CTX *ctx, fr_dict_attr_t const *parent, unsigned int attr)
fr_dict_attr_t const * fr_dict_root(fr_dict_t const *dict)
Return the root attribute of a dictionary.
fr_value_box_t const * value
Enum value (what name maps to).
char const * fr_dict_enum_name_by_value(fr_dict_attr_t const *da, fr_value_box_t const *value)
Lookup the name of an enum value in a fr_dict_attr_t.
fr_slen_t fr_dict_attr_unknown_afrom_oid_substr(TALLOC_CTX *ctx, fr_dict_attr_t const **out, fr_dict_attr_t const *parent, fr_sbuff_t *in, fr_type_t type))
Create a fr_dict_attr_t from an ASCII attribute and value.
@ FR_DICT_ATTR_EXT_REF
Attribute references another attribute and/or dictionary.
#define FR_DICT_ATTR_OID_PRINT_RETURN(...)
fr_dict_attr_t const * fr_dict_attr_child_by_num(fr_dict_attr_t const *parent, unsigned int attr)
Check if a child attribute exists in a parent using an attribute number.
fr_dict_enum_value_t const * fr_dict_enum_by_name(fr_dict_attr_t const *da, char const *name, ssize_t len)
static int8_t fr_dict_attr_cmp(fr_dict_attr_t const *a, fr_dict_attr_t const *b)
Value of an enumerated attribute.
fr_dict_attr_ref_type_t type
The state of the reference.
static void * fr_dict_attr_ext(fr_dict_attr_t const *da, fr_dict_attr_ext_t ext)
@ FR_DICT_ATTR_REF_ROOT
only for FR_TYPE_ATTR, point to the default root for enums
Attribute extension - Holds a reference to an attribute in another dictionary.
static int fr_dlist_insert_tail(fr_dlist_head_t *list_head, void *ptr)
Insert an item into the tail of a list.
uint32_t fr_hash(void const *data, size_t size)
int fr_ipaddr_is_prefix(fr_ipaddr_t const *ipaddr)
Determine if an address is a prefix.
char * fr_inet_ntop_prefix(char out[static FR_IPADDR_PREFIX_STRLEN], size_t outlen, fr_ipaddr_t const *addr)
Print a fr_ipaddr_t as a CIDR style network prefix.
int fr_inet_pton6(fr_ipaddr_t *out, char const *value, ssize_t inlen, bool resolve, bool fallback, bool mask)
Parse an IPv6 address or IPv6 prefix in presentation format (and others)
bool fr_hostname_lookups
hostname -> IP lookups?
int fr_inet_pton(fr_ipaddr_t *out, char const *value, ssize_t inlen, int af, bool resolve, bool mask)
Simple wrapper to decide whether an IP value is v4 or v6 and call the appropriate parser.
int fr_ipaddr_is_inaddr_any(fr_ipaddr_t const *ipaddr)
Determine if an address is the INADDR_ANY address for its address family.
char * fr_inet_ntop(char out[static FR_IPADDR_STRLEN], size_t outlen, fr_ipaddr_t const *addr)
Print the address portion of a fr_ipaddr_t.
int8_t fr_ipaddr_cmp(fr_ipaddr_t const *a, fr_ipaddr_t const *b)
Compare two ip addresses.
void fr_ipaddr_mask(fr_ipaddr_t *addr, uint8_t prefix)
Zeroes out the host portion of an fr_ipaddr_t.
char * fr_inet_ifid_ntop(char *out, size_t outlen, uint8_t const *ifid)
Print an interface-id in standard colon notation.
uint8_t * fr_inet_ifid_pton(uint8_t out[static 8], char const *ifid_str)
Convert interface-id in colon notation to 8 byte binary form.
uint8_t prefix
Prefix length - Between 0-32 for IPv4 and 0-128 for IPv6.
uint8_t addr[6]
Ethernet address.
Struct to represent an ethernet address.
#define fr_multiply(_out, _a, _b)
Multiplies two integers together.
static const uint8_t * zero
size_t fr_sbuff_out_unescape_until(fr_sbuff_t *out, fr_sbuff_t *in, size_t len, fr_sbuff_term_t const *tt, fr_sbuff_unescape_rules_t const *u_rules)
@ FR_TYPE_TIME_DELTA
A period of time measured in nanoseconds.
@ FR_TYPE_FLOAT32
Single precision floating point.
@ FR_TYPE_IPV4_ADDR
32 Bit IPv4 Address.
@ FR_TYPE_INT8
8 Bit signed integer.
@ FR_TYPE_TLV
Contains nested attributes.
@ FR_TYPE_ETHERNET
48 Bit Mac-Address.
@ FR_TYPE_IPV6_PREFIX
IPv6 Prefix.
@ FR_TYPE_STRING
String of printable characters.
@ FR_TYPE_MAX
Number of defined data types.
@ FR_TYPE_NULL
Invalid (uninitialised) attribute type.
@ FR_TYPE_UINT16
16 Bit unsigned integer.
@ FR_TYPE_INT64
64 Bit signed integer.
@ FR_TYPE_INT16
16 Bit signed integer.
@ FR_TYPE_DATE
Unix time stamp, always has value >2^31.
@ FR_TYPE_COMBO_IP_PREFIX
IPv4 or IPv6 address prefix depending on length.
@ FR_TYPE_VALUE_BOX
A boxed value.
@ FR_TYPE_UINT8
8 Bit unsigned integer.
@ FR_TYPE_UINT32
32 Bit unsigned integer.
@ FR_TYPE_STRUCT
like TLV, but without T or L, and fixed-width children
@ FR_TYPE_INT32
32 Bit signed integer.
@ FR_TYPE_VENDOR
Attribute that represents a vendor in the attribute tree.
@ FR_TYPE_UINT64
64 Bit unsigned integer.
@ FR_TYPE_IPV6_ADDR
128 Bit IPv6 Address.
@ FR_TYPE_IPV4_PREFIX
IPv4 Prefix.
@ FR_TYPE_BOOL
A truth value.
@ FR_TYPE_SIZE
Unsigned integer capable of representing any memory address on the local system.
@ FR_TYPE_VSA
Vendor-Specific, for RADIUS attribute 26.
@ FR_TYPE_COMBO_IP_ADDR
IPv4 or IPv6 address depending on length.
@ FR_TYPE_IFID
Interface ID.
@ FR_TYPE_OCTETS
Raw octets.
@ FR_TYPE_GROUP
A grouping of other attributes.
@ FR_TYPE_FLOAT64
Double precision floating point.
int fr_inet_pton4(fr_ipaddr_t *out, char const *value, ssize_t inlen, bool resolve, bool fallback, bool mask_bits)
@ FR_SBUFF_PARSE_ERROR_NOT_FOUND
String does not contain a token matching the output type.
@ FR_SBUFF_PARSE_OK
No error.
static uint8_t depth(fr_minmax_heap_index_t i)
int fr_digest_cmp(uint8_t const *a, uint8_t const *b, size_t length)
Do a comparison of two authentication digests by comparing the FULL data.
void * memset_explicit(void *ptr, int ch, size_t len)
static unsigned int fr_bytes_from_bits(unsigned int bits)
Convert bits (as in prefix length) to bytes, rounding up.
static uint64_t fr_nbo_to_uint64(uint8_t const data[static sizeof(uint64_t)])
Read an unsigned 64bit integer from wire format (big endian)
static void fr_nbo_from_uint64(uint8_t out[static sizeof(uint64_t)], uint64_t num)
Write out an unsigned 64bit integer in wire format (big endian)
char * fr_vasprintf(TALLOC_CTX *ctx, char const *fmt, va_list ap)
size_t fr_sbuff_adv_past_allowed(fr_sbuff_t *sbuff, size_t len, bool const allowed[static SBUFF_CHAR_CLASS], fr_sbuff_term_t const *tt)
Wind position past characters in the allowed set.
int fr_sbuff_trim_talloc(fr_sbuff_t *sbuff, size_t len)
Trim a talloced sbuff to the minimum length required to represent the contained string.
ssize_t fr_sbuff_in_escape(fr_sbuff_t *sbuff, char const *in, size_t inlen, fr_sbuff_escape_rules_t const *e_rules)
Print an escaped string to an sbuff.
bool const sbuff_char_class_hex[SBUFF_CHAR_CLASS]
bool const sbuff_char_class_uint[SBUFF_CHAR_CLASS]
bool const sbuff_char_class_hostname[SBUFF_CHAR_CLASS]
bool fr_sbuff_is_terminal(fr_sbuff_t *in, fr_sbuff_term_t const *tt)
Efficient terminal string search.
ssize_t fr_sbuff_in_bstrncpy(fr_sbuff_t *sbuff, char const *str, size_t len)
Copy bytes into the sbuff up to the first \0.
size_t fr_sbuff_adv_until(fr_sbuff_t *sbuff, size_t len, fr_sbuff_term_t const *tt, char escape_chr)
Wind position until we hit a character in the terminal set.
size_t fr_sbuff_out_bstrncpy(fr_sbuff_t *out, fr_sbuff_t *in, size_t len)
Copy as many bytes as possible from a sbuff to a sbuff.
bool fr_sbuff_next_if_char(fr_sbuff_t *sbuff, char c)
Return true if the current char matches, and if it does, advance.
#define fr_sbuff_start(_sbuff_or_marker)
#define fr_sbuff_adv_past_str_literal(_sbuff, _needle)
#define FR_SBUFF_IN_CHAR_RETURN(_sbuff,...)
#define fr_sbuff_set(_dst, _src)
#define FR_SBUFF_IN(_start, _len_or_end)
#define fr_sbuff_adv_past_strcase_literal(_sbuff, _needle)
#define fr_sbuff_current(_sbuff_or_marker)
char chr
Character at the start of an escape sequence.
#define FR_SBUFF_IN_ESCAPE_BUFFER_RETURN(...)
#define FR_SBUFF_TERMS(...)
Initialise a terminal structure with a list of sorted strings.
char const * name
Name for rule set to aid we debugging.
#define FR_SBUFF_IN_STRCPY_LITERAL_RETURN(_sbuff, _str)
#define fr_sbuff_extend(_sbuff_or_marker)
#define fr_sbuff_buff(_sbuff_or_marker)
#define FR_SBUFF_RETURN(_func, _sbuff,...)
#define fr_sbuff_is_char(_sbuff_or_marker, _c)
#define FR_SBUFF_ERROR_RETURN(_sbuff_or_marker)
#define FR_SBUFF_SET_RETURN(_dst, _src)
#define fr_sbuff_is_digit(_sbuff_or_marker)
#define FR_SBUFF_IN_SPRINTF_RETURN(...)
#define SBUFF_CHAR_UNPRINTABLES_EXTENDED
#define FR_SBUFF(_sbuff_or_marker)
#define fr_sbuff_advance(_sbuff_or_marker, _len)
#define fr_sbuff_out(_err, _out, _in)
#define FR_SBUFF_IN_ESCAPE_RETURN(...)
#define fr_sbuff_remaining(_sbuff_or_marker)
#define FR_SBUFF_OUT(_start, _len_or_end)
#define SBUFF_CHAR_UNPRINTABLES_LOW
#define fr_sbuff_used(_sbuff_or_marker)
#define FR_SBUFF_TERM(_str)
Initialise a terminal structure with a single string.
#define FR_SBUFF_IN_STRCPY_RETURN(...)
#define FR_SBUFF_TALLOC_THREAD_LOCAL(_out, _init, _max)
Talloc sbuff extension structure.
Set of parsing rules for *unescape_until functions.
fr_slen_t fr_size_from_str(size_t *out, fr_sbuff_t *in)
Parse a size string with optional unit.
fr_slen_t fr_size_to_str(fr_sbuff_t *out, size_t in)
Print a size string with unit.
static char buff[sizeof("18446744073709551615")+3]
PUBLIC int snprintf(char *string, size_t length, char *format, va_alist)
fr_aka_sim_id_type_t type
#define fr_table_str_by_value(_table, _number, _def)
Convert an integer to a string.
char * talloc_buffer_append_variadic_buffer(TALLOC_CTX *ctx, char *to, int argc,...)
Concatenate to + ...
uint8_t * talloc_typed_memdup(TALLOC_CTX *ctx, uint8_t const *in, size_t inlen)
Call talloc_memdup, setting the type on the new chunk correctly.
char * talloc_bstrndup(TALLOC_CTX *ctx, char const *in, size_t inlen)
Binary safe strndup function.
#define talloc_get_type_abort_const
static int talloc_const_free(void const *ptr)
Free const'd memory.
#define talloc_strdup(_ctx, _str)
static size_t talloc_strlen(char const *s)
Returns the length of a talloc array containing a string.
fr_table_num_ordered_t const fr_time_precision_table[]
fr_slen_t fr_time_delta_from_substr(fr_time_delta_t *out, fr_sbuff_t *in, fr_time_res_t hint, bool no_trailing, fr_sbuff_term_t const *tt)
Create fr_time_delta_t from a string.
int fr_unix_time_from_str(fr_unix_time_t *date, char const *date_str, fr_time_res_t hint)
Convert string in various formats to a fr_unix_time_t.
int64_t fr_time_scale(int64_t t, fr_time_res_t hint)
Scale an input time to NSEC, clamping it at max / min.
fr_slen_t fr_time_delta_to_str(fr_sbuff_t *out, fr_time_delta_t delta, fr_time_res_t res, bool is_unsigned)
Print fr_time_delta_t to a string with an appropriate suffix.
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.
int64_t const fr_time_multiplier_by_res[]
static fr_time_delta_t fr_time_delta_from_integer(bool *overflow, int64_t integer, fr_time_res_t res)
static int64_t fr_time_delta_to_integer(fr_time_delta_t delta, fr_time_res_t res)
static fr_unix_time_t fr_unix_time_from_nsec(int64_t nsec)
static int64_t fr_time_delta_unwrap(fr_time_delta_t time)
static int8_t fr_time_delta_cmp(fr_time_delta_t a, fr_time_delta_t b)
Compare two fr_time_delta_t values.
#define fr_time_delta_isneg(_a)
#define fr_time_delta_wrap(_time)
#define fr_unix_time_wrap(_time)
fr_time_res_t
The base resolution for print parse operations.
static fr_unix_time_t fr_unix_time_from_integer(bool *overflow, int64_t integer, fr_time_res_t res)
static int8_t fr_unix_time_cmp(fr_unix_time_t a, fr_unix_time_t b)
Compare two fr_unix_time_t values.
static uint64_t fr_unix_time_unwrap(fr_unix_time_t time)
static int64_t fr_unix_time_to_integer(fr_unix_time_t delta, fr_time_res_t res)
const char fr_token_quote[T_TOKEN_LAST]
Convert tokens back to a quoting character.
@ T_SOLIDUS_QUOTED_STRING
void fr_strerror_clear(void)
Clears all pending messages from the talloc pools.
#define fr_strerror_printf(_fmt,...)
Log to thread local error buffer.
#define fr_strerror_printf_push(_fmt,...)
Add a message to an existing stack of messages at the tail.
#define fr_strerror_const(_msg)
#define FR_TYPE_VARIABLE_SIZE
#define FR_TYPE_STRUCTURAL_EXCEPT_GROUP
#define fr_type_is_non_leaf(_x)
#define fr_type_is_group(_x)
#define fr_type_is_variable_size(_x)
#define fr_type_is_structural(_x)
@ FR_TYPE_VALUE_BOX_CURSOR
cursor over a fr_value_box_t
@ FR_TYPE_UNION
A union of limited children.
@ FR_TYPE_ATTR
A contains an attribute reference.
@ FR_TYPE_PAIR_CURSOR
cursor over a fr_pair_t
#define fr_type_is_fixed_size(_x)
#define FR_TYPE_STRUCTURAL
#define fr_type_is_ip(_x)
#define FR_TYPE_INTEGER_EXCEPT_BOOL
#define fr_type_is_leaf(_x)
static char const * fr_type_to_str(fr_type_t type)
Return a static string containing the type name.
#define FR_TYPE_FIXED_SIZE
int fr_value_box_bstrndup_dbuff(TALLOC_CTX *ctx, fr_value_box_t *dst, fr_dict_attr_t const *enumv, fr_dbuff_t *dbuff, size_t len, bool tainted)
void fr_value_box_list_verify(char const *file, int line, fr_value_box_list_t const *list)
void fr_value_box_memdup_buffer_shallow(TALLOC_CTX *ctx, fr_value_box_t *dst, fr_dict_attr_t const *enumv, uint8_t const *src, bool tainted)
Assign a talloced buffer to a box, but don't copy it.
size_t const fr_value_box_field_sizes[]
How many bytes wide each of the value data fields are.
int fr_value_box_hton(fr_value_box_t *dst, fr_value_box_t const *src)
Performs byte order reversal for types that need it.
size_t fr_value_box_network_length(fr_value_box_t const *value)
Get the size of the value held by the fr_value_box_t.
int fr_value_box_vasprintf(TALLOC_CTX *ctx, fr_value_box_t *dst, fr_dict_attr_t const *enumv, bool tainted, char const *fmt, va_list ap)
Print a formatted string using our internal printf wrapper and assign it to a value box.
void fr_value_box_set_void_shallow(fr_value_box_t *dst, void const *ptr)
Assign a void pointer to a box.
static void _fr_value_box_list_debug(FILE *fp, fr_value_box_list_t const *head, int depth)
#define INFO_INDENT(_fmt,...)
fr_sbuff_unescape_rules_t const fr_value_unescape_single
void fr_value_box_mark_unsafe(fr_value_box_t *vb)
Mark a value-box as "unsafe".
ssize_t fr_value_box_list_concat_as_string(fr_value_box_t *safety, fr_sbuff_t *sbuff, fr_value_box_list_t *list, char const *sep, size_t sep_len, fr_sbuff_escape_rules_t const *e_rules, fr_value_box_list_action_t proc_action, fr_value_box_safe_for_t safe_for, bool flatten)
Concatenate a list of value boxes together.
int fr_value_box_strtrim(TALLOC_CTX *ctx, fr_value_box_t *vb)
Trim the length of the string buffer to match the length of the C string.
uint32_t fr_value_box_hash(fr_value_box_t const *vb)
Hash the contents of a value box.
ssize_t fr_value_box_print(fr_sbuff_t *out, fr_value_box_t const *data, fr_sbuff_escape_rules_t const *e_rules)
Print one boxed value to a string.
fr_sbuff_escape_rules_t const fr_value_escape_double
fr_sbuff_parse_rules_t const value_parse_rules_single_3quoted
static fr_slen_t fr_value_box_from_numeric_substr(fr_value_box_t *dst, fr_type_t dst_type, fr_dict_attr_t const *dst_enumv, fr_sbuff_t *in, fr_sbuff_parse_rules_t const *rules, bool tainted)
Convert integer encoded as string to a fr_value_box_t type.
fr_sbuff_escape_rules_t const fr_value_escape_backtick
static int fr_value_box_cast_to_strvalue(TALLOC_CTX *ctx, fr_value_box_t *dst, fr_type_t dst_type, fr_dict_attr_t const *dst_enumv, fr_value_box_t const *src)
Convert any supported type to a string.
int fr_value_box_escape_erules(fr_value_box_t *vb, void *uctx)
Escape a value-box in place using the supplied fr_sbuff_escape_rules_t in uctx.
fr_sbuff_escape_rules_t const fr_value_escape_secret
Escape secret fields by simply mashing all data to '.
fr_sbuff_parse_rules_t const value_parse_rules_double_unquoted
char * fr_value_box_list_aprint_secure(TALLOC_CTX *ctx, fr_value_box_list_t const *list, char const *delim, fr_sbuff_escape_rules_t const *e_rules)
Concatenate the string representations of a list of value boxes together hiding "secret" values.
fr_sbuff_parse_rules_t const value_parse_rules_solidus_quoted
ssize_t fr_value_box_from_network(TALLOC_CTX *ctx, fr_value_box_t *dst, fr_type_t type, fr_dict_attr_t const *enumv, fr_dbuff_t *dbuff, size_t len, bool tainted)
Decode a fr_value_box_t from serialized binary data.
int fr_value_box_mem_alloc(TALLOC_CTX *ctx, uint8_t **out, fr_value_box_t *dst, fr_dict_attr_t const *enumv, size_t len, bool tainted)
Pre-allocate an octets buffer for filling by the caller.
int fr_value_box_memdup_buffer(TALLOC_CTX *ctx, fr_value_box_t *dst, fr_dict_attr_t const *enumv, uint8_t const *src, bool tainted)
Copy a talloced buffer to a fr_value_box_t.
fr_sbuff_escape_rules_t const fr_value_escape_unprintables
#define network_min_size(_x)
Sanity checks.
int fr_value_box_bstrdup_buffer(TALLOC_CTX *ctx, fr_value_box_t *dst, fr_dict_attr_t const *enumv, char const *src, bool tainted)
Copy a nul terminated talloced buffer to a fr_value_box_t.
int fr_value_box_cast(TALLOC_CTX *ctx, fr_value_box_t *dst, fr_type_t dst_type, fr_dict_attr_t const *dst_enumv, fr_value_box_t const *src)
Convert one type of fr_value_box_t to another.
int fr_value_box_asprintf(TALLOC_CTX *ctx, fr_value_box_t *dst, fr_dict_attr_t const *enumv, bool tainted, char const *fmt,...)
Print a formatted string using our internal printf wrapper and assign it to a value box.
fr_sbuff_parse_rules_t const * value_parse_rules_quoted[T_TOKEN_LAST]
Parse rules for quoted strings.
char * fr_value_box_list_aprint(TALLOC_CTX *ctx, fr_value_box_list_t const *list, char const *delim, fr_sbuff_escape_rules_t const *e_rules)
Concatenate the string representations of a list of value boxes together.
int fr_value_box_mem_realloc(TALLOC_CTX *ctx, uint8_t **out, fr_value_box_t *dst, size_t len)
Change the length of a buffer already allocated to a value box.
static size_t const fr_value_box_network_sizes[FR_TYPE_MAX+1][2]
fr_sbuff_escape_rules_t const fr_value_escape_solidus
static int fr_value_box_cast_to_float(UNUSED TALLOC_CTX *ctx, fr_value_box_t *dst, fr_type_t dst_type, fr_dict_attr_t const *dst_enumv, fr_value_box_t const *src)
Convert any value to a floating point value.
fr_sbuff_unescape_rules_t const * fr_value_unescape_by_quote[T_TOKEN_LAST]
int8_t fr_value_box_cmp(fr_value_box_t const *a, fr_value_box_t const *b)
Compare two values.
#define SIGN_BIT_HIGH(_int, _len)
size_t const fr_value_box_offsets[]
Where the value starts in the fr_value_box_t.
static void _fr_value_box_debug(FILE *fp, fr_value_box_t const *vb, int depth, int idx)
#define CAST_IP_FIX_COMBO
void fr_value_box_list_untaint(fr_value_box_list_t *head)
Untaint every list member (and their children)
fr_sbuff_parse_rules_t const value_parse_rules_bareword_unquoted
Default formatting rules.
static int fr_value_box_cast_to_ipv4addr(TALLOC_CTX *ctx, fr_value_box_t *dst, fr_type_t dst_type, fr_dict_attr_t const *dst_enumv, fr_value_box_t const *src)
Convert any supported type to an IPv4 address.
static const fr_value_box_ipaddr_sizes_t ipaddr_sizes[FR_TYPE_MAX]
int fr_value_box_copy(TALLOC_CTX *ctx, fr_value_box_t *dst, const fr_value_box_t *src)
Copy value data verbatim duplicating any buffers.
fr_sbuff_parse_rules_t const value_parse_rules_single_unquoted
int fr_value_box_cmp_op(fr_token_t op, fr_value_box_t const *a, fr_value_box_t const *b)
Compare two attributes using an operator.
int fr_value_box_list_escape_in_place(fr_value_box_list_t *list, fr_value_box_escape_t const *escape, void *uctx)
Escape a list of value boxes in place.
fr_sbuff_parse_rules_t const * value_parse_rules_unquoted_char[SBUFF_CHAR_CLASS]
uint64_t fr_value_box_as_uint64(fr_value_box_t const *vb)
Return a uint64_t from a fr_value_box_t.
bool fr_value_box_is_truthy(fr_value_box_t const *in)
Check truthiness of values.
int fr_value_box_cast_in_place(TALLOC_CTX *ctx, fr_value_box_t *vb, fr_type_t dst_type, fr_dict_attr_t const *dst_enumv)
Convert one type of fr_value_box_t to another in place.
void fr_value_box_set_cursor_shallow(fr_value_box_t *dst, fr_type_t type, void *cursor, char const *name)
fr_sbuff_parse_rules_t const value_parse_rules_single_quoted
static uint8_t const v4_v6_map[]
v4 to v6 mapping prefix
void fr_value_box_memdup_shallow(fr_value_box_t *dst, fr_dict_attr_t const *enumv, uint8_t const *src, size_t len, bool tainted)
Assign a buffer to a box, but don't copy it.
void fr_value_box_copy_shallow(TALLOC_CTX *ctx, fr_value_box_t *dst, fr_value_box_t const *src)
Perform a shallow copy of a value_box.
ssize_t fr_value_box_from_str(TALLOC_CTX *ctx, fr_value_box_t *dst, fr_type_t dst_type, fr_dict_attr_t const *dst_enumv, char const *in, size_t inlen, fr_sbuff_unescape_rules_t const *erules)
ssize_t fr_value_box_list_concat_as_octets(fr_value_box_t *safety, fr_dbuff_t *dbuff, fr_value_box_list_t *list, uint8_t const *sep, size_t sep_len, fr_value_box_list_action_t proc_action, bool flatten)
Concatenate a list of value boxes together.
static int fr_value_box_cast_to_octets(TALLOC_CTX *ctx, fr_value_box_t *dst, fr_type_t dst_type, fr_dict_attr_t const *dst_enumv, fr_value_box_t const *src)
Convert any supported type to octets.
void fr_value_box_increment(fr_value_box_t *vb)
Increment a boxed value.
void _fr_value_box_mark_safe_for(fr_value_box_t *vb, fr_value_box_safe_for_t safe_for)
Mark a value-box as "safe", of a particular type.
size_t fr_value_str_unescape(fr_sbuff_t *out, fr_sbuff_t *in, size_t inlen, char quote)
Convert a string value with escape sequences into its binary form.
void fr_value_box_clear_value(fr_value_box_t *data)
Clear/free any existing value.
void fr_value_box_set_attr(fr_value_box_t *dst, fr_dict_attr_t const *da)
fr_sbuff_unescape_rules_t const fr_value_unescape_backtick
void fr_value_box_verify(char const *file, int line, fr_value_box_t const *vb)
Validation function to check that a fr_value_box_t is correctly initialised.
fr_sbuff_parse_rules_t const * value_parse_rules_3quoted[T_TOKEN_LAST]
int fr_value_box_strdup(TALLOC_CTX *ctx, fr_value_box_t *dst, fr_dict_attr_t const *enumv, char const *src, bool tainted)
Copy a nul terminated string to a fr_value_box_t.
#define network_max_size(_x)
void fr_value_box_strdup_shallow_replace(fr_value_box_t *vb, char const *src, ssize_t len)
Free the existing buffer (if talloced) associated with the valuebox, and replace it with a new one.
fr_sbuff_unescape_rules_t const fr_value_unescape_double
ssize_t fr_value_box_print_quoted(fr_sbuff_t *out, fr_value_box_t const *data, fr_token_t quote)
Print one boxed value to a string with quotes (where needed)
fr_sbuff_parse_rules_t const value_parse_rules_double_3quoted
static int fr_value_box_cast_to_integer(TALLOC_CTX *ctx, fr_value_box_t *dst, fr_type_t dst_type, fr_dict_attr_t const *dst_enumv, fr_value_box_t const *src)
Convert any value to a signed or unsigned integer.
static int fr_value_box_cast_to_ipv6prefix(TALLOC_CTX *ctx, fr_value_box_t *dst, fr_type_t dst_type, fr_dict_attr_t const *dst_enumv, fr_value_box_t const *src)
Convert any supported type to an IPv6 address.
ssize_t fr_value_box_from_memory(TALLOC_CTX *ctx, fr_value_box_t *dst, fr_type_t type, fr_dict_attr_t const *enumv, void const *src, size_t len)
Decode a fr_value_box_t from a C type in memory.
void fr_value_box_safety_copy_changed(fr_value_box_t *out, fr_value_box_t const *in)
Copy the safety values from one box to another.
int fr_value_box_ipaddr(fr_value_box_t *dst, fr_dict_attr_t const *enumv, fr_ipaddr_t const *ipaddr, bool tainted)
Assign a fr_value_box_t value from an fr_ipaddr_t.
void fr_value_box_list_taint(fr_value_box_list_t *head)
Taint every list member (and their children)
static int fr_value_box_cidr_cmp_op(fr_token_t op, int bytes, uint8_t a_net, uint8_t const *a, uint8_t b_net, uint8_t const *b)
static void fr_value_box_copy_meta(fr_value_box_t *dst, fr_value_box_t const *src)
Copy flags and type data from one value box to another.
void fr_value_box_list_mark_safe_for(fr_value_box_list_t *list, fr_value_box_safe_for_t safe_for)
Set the escaped flag for all value boxes in a list.
static int fr_value_box_fixed_size_from_octets(fr_value_box_t *dst, fr_type_t dst_type, fr_dict_attr_t const *dst_enumv, fr_value_box_t const *src)
Convert octets to a fixed size value box value.
static int fr_value_box_cast_to_bool(TALLOC_CTX *ctx, fr_value_box_t *dst, fr_type_t dst_type, fr_dict_attr_t const *dst_enumv, fr_value_box_t const *src)
Convert any supported type to a bool.
int fr_value_unbox_ipaddr(fr_ipaddr_t *dst, fr_value_box_t *src)
Unbox an IP address performing a type check.
int fr_value_box_escape_in_place_erules(TALLOC_CTX *ctx, fr_value_box_t *vb, fr_sbuff_escape_rules_t const *erules)
Escape a value-box in place using sbuff escaping rules, and mark it safe-for.
fr_sbuff_parse_rules_t const value_parse_rules_bareword_quoted
void fr_value_box_safety_merge(fr_value_box_t *out, fr_value_box_t const *in)
Merge safety results.
fr_sbuff_parse_rules_t const value_parse_rules_backtick_3quoted
fr_sbuff_escape_rules_t const fr_value_escape_single
static uint64_t const fr_value_box_integer_max[]
void fr_value_box_strdup_shallow(fr_value_box_t *dst, fr_dict_attr_t const *enumv, char const *src, bool tainted)
Assign a buffer containing a nul terminated string to a box, but don't copy it.
void fr_value_box_list_debug(FILE *fp, fr_value_box_list_t const *head)
Print a list of value boxes as info messages.
fr_sbuff_parse_rules_t const * value_parse_rules_quoted_char[SBUFF_CHAR_CLASS]
fr_sbuff_parse_rules_t const value_parse_rules_solidus_unquoted
fr_sbuff_parse_rules_t const value_parse_rules_backtick_quoted
fr_sbuff_parse_rules_t const * value_parse_rules_unquoted[T_TOKEN_LAST]
Parse rules for non-quoted strings.
static int fr_value_box_cast_to_ipv4prefix(TALLOC_CTX *ctx, fr_value_box_t *dst, fr_type_t dst_type, fr_dict_attr_t const *dst_enumv, fr_value_box_t const *src)
Convert any supported type to an IPv6 address.
static int fr_value_box_cast_to_ethernet(TALLOC_CTX *ctx, fr_value_box_t *dst, fr_type_t dst_type, fr_dict_attr_t const *dst_enumv, fr_value_box_t const *src)
Convert any supported type to an ethernet address.
void fr_value_box_safety_copy(fr_value_box_t *out, fr_value_box_t const *in)
Copy the safety values from one box to another.
fr_sbuff_parse_rules_t const value_parse_rules_backtick_unquoted
ssize_t fr_value_box_ipaddr_from_network(fr_value_box_t *dst, fr_type_t type, fr_dict_attr_t const *enumv, int prefix_len, uint8_t const *data, size_t data_len, bool fixed, bool tainted)
Decode a fr_value_box_t of type IP address / prefix.
fr_sbuff_parse_rules_t const value_parse_rules_double_quoted
fr_sbuff_escape_rules_t const * erules
int fr_value_box_bstr_alloc(TALLOC_CTX *ctx, char **out, fr_value_box_t *dst, fr_dict_attr_t const *enumv, size_t len, bool tainted)
Alloc and assign an empty \0 terminated string to a fr_value_box_t.
#define SIGN_PROMOTE(_int, _len)
fr_sbuff_parse_rules_t const value_parse_rules_solidus_3quoted
static int _value_box_escape_rules(fr_value_box_t *vb, void *uctx)
int fr_value_box_steal(TALLOC_CTX *ctx, fr_value_box_t *dst, fr_value_box_t *src)
Copy value data verbatim moving any buffers to the specified context.
static int fr_value_box_cast_unsupported(fr_type_t dst, fr_type_t src)
int fr_value_box_to_key(uint8_t **out, size_t *outlen, fr_value_box_t const *value)
Get a key from a value box.
void fr_value_box_flatten(TALLOC_CTX *ctx, fr_value_box_list_t *list, bool steal, bool free)
Removes a single layer of nesting, moving all children into the parent list.
static int8_t float_cmp(double a, double b)
Compare two floating point numbers for equality.
int fr_value_box_list_acopy(TALLOC_CTX *ctx, fr_value_box_list_t *out, fr_value_box_list_t const *in)
Do a full copy of a list of value boxes.
void fr_value_box_clear(fr_value_box_t *data)
Clear/free any existing value and metadata.
bool fr_value_box_list_tainted(fr_value_box_list_t const *head)
Check to see if any list members (or their children) are tainted.
ssize_t fr_value_box_to_network(fr_dbuff_t *dbuff, fr_value_box_t const *value)
Encode a single value box, serializing its contents in generic network format.
static int64_t const fr_value_box_integer_min[]
int fr_value_box_bstr_realloc(TALLOC_CTX *ctx, char **out, fr_value_box_t *dst, size_t len)
Change the length of a buffer already allocated to a value box.
int fr_value_box_bstrndup(TALLOC_CTX *ctx, fr_value_box_t *dst, fr_dict_attr_t const *enumv, char const *src, size_t len, bool tainted)
Copy a string to to a fr_value_box_t.
int fr_value_box_memdup_dbuff(TALLOC_CTX *ctx, fr_value_box_t *dst, fr_dict_attr_t const *enumv, fr_dbuff_t *dbuff, size_t len, bool tainted)
fr_sbuff_unescape_rules_t const * fr_value_unescape_by_char[SBUFF_CHAR_CLASS]
void fr_value_box_debug(FILE *fp, fr_value_box_t const *vb)
Print the value of a box as info messages.
int fr_regex_cmp_op(fr_token_t op, fr_value_box_t const *a, fr_value_box_t const *b)
Compare two boxes using an operator.
fr_sbuff_unescape_rules_t const fr_value_unescape_solidus
fr_sbuff_escape_rules_t const * fr_value_escape_by_quote[T_TOKEN_LAST]
int fr_value_box_bstrdup_buffer_shallow(TALLOC_CTX *ctx, fr_value_box_t *dst, fr_dict_attr_t const *enumv, char const *src, bool tainted)
Assign a talloced buffer containing a nul terminated string to a box, but don't copy it.
size_t fr_value_substr_unescape(fr_sbuff_t *out, fr_sbuff_t *in, size_t inlen, char quote)
Convert a string value with escape sequences into its binary form.
ssize_t fr_value_box_from_substr(TALLOC_CTX *ctx, fr_value_box_t *dst, fr_type_t dst_type, fr_dict_attr_t const *dst_enumv, fr_sbuff_t *in, fr_sbuff_parse_rules_t const *rules)
Convert string value to a fr_value_box_t type.
static fr_dict_attr_t const * fr_value_box_attr_enumv(fr_dict_attr_t const *da)
int fr_value_box_escape_in_place(fr_value_box_t *vb, fr_value_box_escape_t const *escape, void *uctx)
Escape a single value box in place.
void fr_value_box_bstrndup_shallow(fr_value_box_t *dst, fr_dict_attr_t const *enumv, char const *src, size_t len, bool tainted)
Assign a string to to a fr_value_box_t.
static int fr_value_box_cast_to_ipv6addr(TALLOC_CTX *ctx, fr_value_box_t *dst, fr_type_t dst_type, fr_dict_attr_t const *dst_enumv, fr_value_box_t const *src)
Convert any supported type to an IPv6 address.
fr_sbuff_escape_rules_t const * fr_value_escape_by_char[SBUFF_CHAR_CLASS]
int fr_value_box_memdup(TALLOC_CTX *ctx, fr_value_box_t *dst, fr_dict_attr_t const *enumv, uint8_t const *src, size_t len, bool tainted)
Copy a buffer to a fr_value_box_t.
static int fr_value_box_cast_integer_to_integer(UNUSED TALLOC_CTX *ctx, fr_value_box_t *dst, fr_type_t dst_type, fr_dict_attr_t const *dst_enumv, fr_value_box_t const *src)
Convert any signed or unsigned integer type to any other signed or unsigned integer type.
int fr_value_box_list_concat_in_place(TALLOC_CTX *ctx, fr_value_box_t *out, fr_value_box_list_t *list, fr_type_t type, fr_value_box_list_action_t proc_action, bool flatten, size_t max_size)
Concatenate a list of value boxes.
fr_value_box_list_action_t
Actions to perform when we process a box in a list.
@ FR_VALUE_BOX_LIST_NONE
Do nothing to processed boxes.
@ FR_VALUE_BOX_LIST_REMOVE
Remove the box from the input list.
#define vb_should_free(_action)
#define vb_should_free_value(_action)
#define vb_should_remove(_action)
static int fr_value_box_memcpy_out(void *out, fr_value_box_t const *vb)
Copy the value of a value box to a field in a C struct.
#define fr_value_box_mark_safe_for(_box, _safe_for)
static fr_slen_t fr_value_box_aprint(TALLOC_CTX *ctx, char **out, fr_value_box_t const *data, fr_sbuff_escape_rules_t const *e_rules) 1(fr_value_box_print
static bool fr_value_box_contains_secret(fr_value_box_t const *box)
#define FR_VALUE_BOX_NET_ERROR
Special value to indicate fr_value_box_from_network experienced a general error.
static uint8_t * fr_value_box_raw(fr_value_box_t const *vb, fr_type_t type)
Return a pointer to the "raw" value from a value-box.
#define fr_box_strvalue_len(_val, _len)
#define FR_VALUE_BOX_MAGIC
#define fr_value_box_init_null(_vb)
Initialise an empty/null box that will be filled later.
#define fr_value_box_is_safe_for(_box, _safe_for)
static size_t char fr_sbuff_t size_t inlen
fr_value_box_safe_for_t safe_for
#define FR_VALUE_BOX_SAFE_FOR_NONE
uintptr_t fr_value_box_safe_for_t
Escaping that's been applied to a value box.
#define fr_value_box_alloc_null(_ctx)
Allocate a value box for later use with a value assignment function.
fr_value_box_escape_func_t func
static always_inline int fr_value_box_ethernet_addr(fr_value_box_t *dst, fr_dict_attr_t const *enumv, fr_ethernet_t const *src, bool tainted)
#define fr_value_box_init(_vb, _type, _enumv, _tainted)
Initialise a fr_value_box_t.
#define fr_value_box_list_foreach(_list_head, _iter)
#define FR_VALUE_BOX_NET_OOM
Special value to indicate fr_value_box_from_network hit an out of memory error.
static size_t char ** out
#define FR_VALUE_BOX_SAFE_FOR_ANY