The FreeRADIUS server $Id: 15bac2a4c627c01d1aa2047687b3418955ac7f00 $
Loading...
Searching...
No Matches
decode.c
Go to the documentation of this file.
1/*
2 * This library is free software; you can redistribute it and/or
3 * modify it under the terms of the GNU Lesser General Public
4 * License as published by the Free Software Foundation; either
5 * version 2.1 of the License, or (at your option) any later version.
6 *
7 * This library is distributed in the hope that it will be useful,
8 * but WITHOUT ANY WARRANTY; without even the implied warranty of
9 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
10 * Lesser General Public License for more details.
11 *
12 * You should have received a copy of the GNU Lesser General Public
13 * License along with this library; if not, write to the Free Software
14 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
15 */
16
17/**
18 * $Id: b7b7d42e7c834a156c837ce7f86c5b293eae6f37 $
19 *
20 * Because what we need is yet *ANOTHER* serialisation scheme.
21 *
22 * @file protocols/internal/decode.c
23 * @brief Functions to decode data in our internal structure.
24 *
25 * @copyright 2020 The FreeRADIUS server project
26 * @copyright 2020 Arran Cudbard-Bell (a.cudbardb@freeradius.org)
27 */
28
29#include <freeradius-devel/internal/internal.h>
30#include <freeradius-devel/io/pair.h>
31#include <freeradius-devel/io/test_point.h>
32#include <freeradius-devel/util/proto.h>
33
34static ssize_t internal_decode_pair(TALLOC_CTX *ctx, fr_pair_list_t *head, fr_dict_attr_t const *parent_da,
35 fr_dbuff_t *dbuff, void *decode_ctx);
36
37/** Decodes the value of an attribute, potentially producing a pair (added to the cursor)
38 *
39 */
40static ssize_t internal_decode_pair_value(TALLOC_CTX *ctx, fr_pair_list_t *head, fr_dict_attr_t const *parent_da,
41 fr_dbuff_t *dbuff,
42 bool tainted, UNUSED void *decode_ctx)
43{
45 ssize_t slen;
46 fr_dbuff_t work_dbuff = FR_DBUFF(dbuff);
47
48 vp = fr_pair_afrom_da(ctx, parent_da);
49 if (!vp) return PAIR_DECODE_OOM;
51
52 /*
53 * Zero length is fine here
54 */
55 slen = fr_value_box_from_network(vp, &vp->data, vp->vp_type, vp->da,
56 &work_dbuff, fr_dbuff_len(&work_dbuff), tainted);
57 if (slen < 0) {
59 return slen;
60 }
62
63 return fr_dbuff_set(dbuff, &work_dbuff);
64}
65
66/** Decode a group
67 *
68 */
69static ssize_t internal_decode_structural(TALLOC_CTX *ctx, fr_pair_list_t *head, fr_dict_attr_t const *parent_da,
70 fr_dbuff_t *dbuff, void *decode_ctx)
71{
73 ssize_t slen;
74 fr_dbuff_t work_dbuff = FR_DBUFF(dbuff);
75
76 FR_PROTO_TRACE("Decoding group - %s", parent_da->name);
77
78 vp = fr_pair_afrom_da(ctx, parent_da);
79 if (!vp) return PAIR_DECODE_OOM;
81
82 /*
83 * Decode all the children of this group
84 */
85 while (fr_dbuff_extend(&work_dbuff)) {
87 "Decoding child");
88
89 slen = internal_decode_pair(vp, &vp->vp_group, parent_da, &work_dbuff, decode_ctx);
90 if (slen <= 0) {
92 return slen;
93 }
94 }
96
97 return fr_dbuff_set(dbuff, &work_dbuff);
98}
99
100static ssize_t internal_decode_pair(TALLOC_CTX *ctx, fr_pair_list_t *out, fr_dict_attr_t const *parent_da,
101 fr_dbuff_t *dbuff, void *decode_ctx)
102{
103 ssize_t slen = 0;
104 fr_dict_attr_t const *da;
105 uint8_t enc_byte = 0, ext_byte = 0, type_field_size, len_field_size;
106 fr_dbuff_marker_t len_field, enc_field, ext_field;
107 uint64_t len = 0, type = 0;
108 size_t remaining, needed;
109 bool tainted, extended, unknown = false, internal = false;
110 fr_dbuff_t work_dbuff = FR_DBUFF(dbuff);
111
112 /*
113 * The first byte of each attribute describes the encoding format.
114 *
115 * tlen (type field len) - Describes how many byte(s) were used to encode the type.
116 * llen (length field len) - Describes how many byte(s) were used to encode the length.
117 * t (tainted) - This attribute was tainted when it was encoded,
118 * so should be marked tainted now.
119 * e (extended) - Process the next byte as an extension of the encoding
120 * field (allows for future extensions).
121 *
122 * 0 1 2 3
123 * 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1
124 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
125 * |tlen |llen |t|e| Type (min) | Length (min) | value...
126 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
127 *
128 */
129 remaining = fr_dbuff_extend_lowat(NULL, &work_dbuff, 3);
130 if (remaining < 3) {
131 fr_strerror_printf("%s: Insufficient data. Need %zu additional byte(s)",
132 __FUNCTION__, 3 - remaining);
133 return -fr_dbuff_len(&work_dbuff);
134 }
135
136 fr_dbuff_marker(&enc_field, &work_dbuff);
137 fr_dbuff_marker(&ext_field, &work_dbuff); /* Placed here to make static analysis happy */
138 FR_DBUFF_OUT_RETURN(&enc_byte, &work_dbuff);
139 type_field_size = ((enc_byte & FR_INTERNAL_MASK_TYPE) >> 5) + 1; /* bits 0-2 */
140 len_field_size = ((enc_byte & FR_INTERNAL_MASK_LEN) >> 2) + 1; /* bits 3-5 */
141
142 tainted = (enc_byte & FR_INTERNAL_FLAG_TAINTED) != 0; /* bit 6 */
143 extended = (enc_byte & FR_INTERNAL_FLAG_EXTENDED) != 0; /* bit 7 */
144
145 /* Processed first encoding byte */
146
147 needed = type_field_size + len_field_size + extended;
148 remaining = fr_dbuff_extend_lowat(NULL, &work_dbuff, needed);
149 if (remaining < needed) {
150 fr_strerror_printf("%s: Encoding byte invalid, fields overrun input data. "
151 "%zu byte(s) remaining, need %zu byte(s)",
152 __FUNCTION__, remaining, needed);
153 return -needed;
154 }
155
156 /*
157 * The second (optional) extension byte carries more flag information from the attribute.
158 *
159 * u (unknown attribute) - When this pair was converted from network to internal
160 * format, it was found to be badly formatted, or not
161 * match an existing dictionary definition.
162 * A new unknown DA should be allocated for this attribute
163 * and it should be treated as raw octets.
164 * i (internal attribute) - Resolve this attribute in the internal dictionary.
165 * - (currently unused) - Unused flag.
166 * e (extended) - Encoding definitions continue to a third byte.
167 *
168 * 0 1
169 * 0 1 2 3 4 5 6 7 8 9 0
170 * +-+-+-+-+-+-+-+-+-+-+
171 * |u|i|-|-|-|-|-|e|
172 * +-+-+-+-+-+-+-+-+-+-+
173 *
174 */
175 if (extended) {
176 fr_dbuff_set(&ext_field, &work_dbuff);
177 FR_DBUFF_OUT_RETURN(&ext_byte, &work_dbuff);
178 unknown = (ext_byte & FR_INTERNAL_FLAG_UNKNOWN) != 0;
179 internal = (ext_byte & FR_INTERNAL_FLAG_INTERNAL) != 0;
180 if (ext_byte & FR_INTERNAL_FLAG_EXTENDED) {
181 fr_strerror_printf("%s: Third extension byte not in use", __FUNCTION__);
183 }
184 }
185
186 FR_DBUFF_OUT_UINT64V_RETURN(&type, &work_dbuff, type_field_size);
187
188 /*
189 * This is the length of the start *after* the flags and
190 * type/length fields.
191 */
192 fr_dbuff_marker(&len_field, &work_dbuff);
193 FR_DBUFF_OUT_UINT64V_RETURN(&len, &work_dbuff, len_field_size);
194
195 remaining = fr_dbuff_extend_lowat(NULL, &work_dbuff, len);
196 if (remaining < len) {
197 fr_strerror_printf("%s: Length field value overruns input data. "
198 "%zu byte(s) remaining, need %zu byte(s)",
199 __FUNCTION__, remaining, (size_t) len);
200 return -(fr_dbuff_current(&len_field) - fr_dbuff_start(&work_dbuff));
201 }
202
203 /*
204 * Internal flag is only set on the outer attribute
205 * so it's fine to swap the parent_da.
206 */
207 if (internal) {
208 if (!parent_da->flags.is_root && !(parent_da->type == FR_TYPE_GROUP)) {
209 fr_strerror_printf("%s: Internal flag can only be set on top level attribute", __FUNCTION__);
211 }
212 parent_da = fr_dict_root(fr_dict_internal());
213 }
214
215 if (unknown || parent_da->flags.is_unknown) {
216 unknown:
217 FR_PROTO_TRACE("Unknown attribute %" PRIu64, type);
218 da = fr_dict_attr_unknown_raw_afrom_num(ctx, parent_da, type);
219 if (!da) return PAIR_DECODE_FATAL_ERROR;
220 unknown = true;
221 } else {
222 da = fr_dict_attr_child_by_num(parent_da, type);
223 if (!da) goto unknown;
224 }
225
226 FR_PROTO_TRACE("decode context changed %s -> %s", da->parent->name, da->name);
227
228 /*
229 * Set the end of our dbuff to match the length
230 * of the attribute.
231 */
232 fr_dbuff_set_end(&work_dbuff, fr_dbuff_current(&work_dbuff) + len);
233
234 switch (da->type) {
235 /*
236 * Structural types
237 *
238 * STRUCTs are encoded as TLVs, because the struct
239 * packing only applies to the original protocol, and not
240 * to our internal encoding.
241 */
243 if (fr_type_is_vsa(da->type)) {
244 if (unlikely(unknown)) {
245 fr_strerror_printf("%s: %s can't be marked as unknown", __FUNCTION__,
246 fr_type_to_str(da->type));
247 fr_dbuff_set(&work_dbuff, &ext_field);
248 error:
249 if (unknown) fr_dict_attr_unknown_free(&da);
250 return fr_pair_decode_slen(slen, fr_dbuff_start(&work_dbuff), fr_dbuff_current(&work_dbuff));
251 }
252 }
253 /*
254 * It's ok for this function to return 0
255 * we can have empty groups (i.e. groups
256 * with no children)
257 */
258 slen = internal_decode_structural(ctx, out, da, &work_dbuff, decode_ctx);
259 if (slen < 0) goto error;
260 break;
261
262 default:
263 /*
264 * It's ok for this function to return 0
265 * we can have zero length strings.
266 */
267 slen = internal_decode_pair_value(ctx, out, da, &work_dbuff, tainted, decode_ctx);
268 if (slen < 0) goto error;
269 }
270
271 return fr_dbuff_set(dbuff, &work_dbuff);
272}
273
274/** Create a single fr_pair_t and all its nesting
275 *
276 */
278 uint8_t const *data, size_t data_len, void *decode_ctx)
279{
280 return fr_internal_decode_pair_dbuff(ctx, list, parent, &FR_DBUFF_TMP(data, data_len), decode_ctx);
281}
282
284 fr_dbuff_t *dbuff, void *decode_ctx)
285{
286 fr_pair_list_t tmp;
287 ssize_t slen;
288 fr_dbuff_t work_dbuff = FR_DBUFF(dbuff);
289
290 fr_pair_list_init(&tmp);
291
292 slen = internal_decode_pair(ctx, &tmp, parent, &work_dbuff, decode_ctx);
293 if (slen <= 0) {
294 fr_pair_list_free(&tmp);
295 return slen;
296 }
297
299
300 return fr_dbuff_set(dbuff, &work_dbuff);
301}
302
303/** Retrieve all pairs from the dbuff
304 *
305 * @param ctx to create new pairs in
306 * @param out list to append pairs to
307 * @param parent attribute within which which to decode
308 * @param dbuff to parse
309 * @param decode_ctx to pass to decoder function
310 * @return
311 * - bytes of dbuff consumed
312 * - < 0 on error
313 */
315 fr_dbuff_t *dbuff, void *decode_ctx)
316{
317 ssize_t ret, len = 0;
318
319 while (fr_dbuff_remaining(dbuff)) {
320 ret = fr_internal_decode_pair_dbuff(ctx, out, parent, dbuff, decode_ctx);
321 if (ret < 0) return ret;
322 if (ret == 0) break;
323 len += ret;
324 }
325
326 return len;
327}
328
329/*
330 * Test points
331 */
#define unlikely(_x)
Definition build.h:402
#define UNUSED
Definition build.h:336
#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...
Definition dbuff.h:775
#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.
Definition dbuff.h:1867
#define fr_dbuff_len(_dbuff_or_marker)
The length of the underlying buffer.
Definition dbuff.h:784
struct fr_dbuff_marker_s fr_dbuff_marker_t
A position marker associated with a dbuff.
Definition dbuff.h:81
#define fr_dbuff_current(_dbuff_or_marker)
Return the 'current' position of a dbuff or marker.
Definition dbuff.h:919
#define fr_dbuff_set(_dst, _src)
Set the 'current' position in a dbuff or marker using another dbuff or marker, a char pointer,...
Definition dbuff.h:1012
#define fr_dbuff_start(_dbuff_or_marker)
Return the 'start' position of a dbuff or marker.
Definition dbuff.h:906
#define fr_dbuff_extend_lowat(_status, _dbuff_or_marker, _lowat)
Extend if we're below _lowat.
Definition dbuff.h:668
#define fr_dbuff_extend(_dbuff)
Extend if no space remains.
Definition dbuff.h:713
#define fr_dbuff_remaining(_dbuff_or_marker)
Return the number of bytes remaining between the dbuff or marker and the end of the buffer.
Definition dbuff.h:751
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.
Definition dbuff.h:1201
#define FR_DBUFF(_dbuff_or_marker)
Create a new dbuff pointing to the same underlying buffer.
Definition dbuff.h:230
#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.
Definition dbuff.h:1827
#define fr_dbuff_set_end(_dst, _end)
Set a new 'end' position in a dbuff or marker.
Definition dbuff.h:1059
#define FR_DBUFF_TMP(_start, _len_or_end)
Creates a compound literal to pass into functions which accept a dbuff.
Definition dbuff.h:522
static fr_dict_attr_t * fr_dict_attr_unknown_raw_afrom_num(TALLOC_CTX *ctx, fr_dict_attr_t const *parent, unsigned int attr)
Definition dict.h:611
fr_dict_attr_t const * fr_dict_root(fr_dict_t const *dict)
Return the root attribute of a dictionary.
Definition dict_util.c:2665
void fr_dict_attr_unknown_free(fr_dict_attr_t const **da)
Free dynamically allocated (unknown attributes)
fr_dict_t const * fr_dict_internal(void)
Definition dict_util.c:4928
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.
Definition dict_util.c:3593
talloc_free(hp)
#define FR_INTERNAL_FLAG_INTERNAL
Definition internal.h:33
#define FR_INTERNAL_MASK_TYPE
Definition internal.h:24
#define FR_INTERNAL_FLAG_EXTENDED
Definition internal.h:26
#define FR_INTERNAL_MASK_LEN
Definition internal.h:25
#define FR_INTERNAL_FLAG_UNKNOWN
Definition internal.h:32
#define FR_INTERNAL_FLAG_TAINTED
Definition internal.h:27
#define PAIR_DECODE_OOM
Fatal error - Out of memory.
Definition pair.h:45
#define PAIR_DECODE_FATAL_ERROR
Fatal error - Failed decoding the packet.
Definition pair.h:49
static ssize_t fr_pair_decode_slen(ssize_t slen, uint8_t const *start, uint8_t const *p)
Return the correct adjusted slen for errors.
Definition pair.h:57
@ FR_TYPE_GROUP
A grouping of other attributes.
long int ssize_t
unsigned char uint8_t
int fr_pair_append(fr_pair_list_t *list, fr_pair_t *to_add)
Add a VP to the end of the list.
Definition pair.c:1352
fr_pair_t * fr_pair_afrom_da(TALLOC_CTX *ctx, fr_dict_attr_t const *da)
Dynamically allocate a new attribute and assign a fr_dict_attr_t.
Definition pair.c:290
void fr_pair_list_init(fr_pair_list_t *list)
Initialise a pair list header.
Definition pair.c:46
ssize_t fr_internal_decode_list_dbuff(TALLOC_CTX *ctx, fr_pair_list_t *out, fr_dict_attr_t const *parent, fr_dbuff_t *dbuff, void *decode_ctx)
Retrieve all pairs from the dbuff.
Definition decode.c:314
static ssize_t internal_decode_pair_value(TALLOC_CTX *ctx, fr_pair_list_t *head, fr_dict_attr_t const *parent_da, fr_dbuff_t *dbuff, bool tainted, UNUSED void *decode_ctx)
Decodes the value of an attribute, potentially producing a pair (added to the cursor)
Definition decode.c:40
ssize_t fr_internal_decode_pair_dbuff(TALLOC_CTX *ctx, fr_pair_list_t *out, fr_dict_attr_t const *parent, fr_dbuff_t *dbuff, void *decode_ctx)
Definition decode.c:283
ssize_t fr_internal_decode_pair(TALLOC_CTX *ctx, fr_pair_list_t *list, fr_dict_attr_t const *parent, uint8_t const *data, size_t data_len, void *decode_ctx)
Create a single fr_pair_t and all its nesting.
Definition decode.c:277
fr_test_point_pair_decode_t internal_tp_decode_pair
Definition decode.c:333
static ssize_t internal_decode_structural(TALLOC_CTX *ctx, fr_pair_list_t *head, fr_dict_attr_t const *parent_da, fr_dbuff_t *dbuff, void *decode_ctx)
Decode a group.
Definition decode.c:69
static ssize_t internal_decode_pair(TALLOC_CTX *ctx, fr_pair_list_t *head, fr_dict_attr_t const *parent_da, fr_dbuff_t *dbuff, void *decode_ctx)
Definition decode.c:100
fr_aka_sim_id_type_t type
fr_pair_t * vp
Stores an attribute, a value and various bits of other data.
Definition pair.h:68
fr_dict_attr_t const *_CONST da
Dictionary attribute defines the attribute number, vendor and type of the pair.
Definition pair.h:69
fr_test_point_ctx_alloc_t test_ctx
Allocate a test ctx for the encoder.
Definition test_point.h:86
Entry point for pair decoders.
Definition test_point.h:85
static fr_slen_t head
Definition xlat.h:420
#define PAIR_ALLOCED(_x)
Definition pair.h:212
void fr_pair_list_free(fr_pair_list_t *list)
Free memory used by a valuepair list.
void fr_pair_list_append(fr_pair_list_t *dst, fr_pair_list_t *src)
Appends a list of fr_pair_t from a temporary list to a destination list.
static fr_slen_t parent
Definition pair.h:858
#define FR_PROTO_TRACE(_fmt,...)
Definition proto.h:41
#define FR_PROTO_HEX_MARKER(_data, _data_len, _slen, _fmt,...)
Definition proto.h:43
#define fr_strerror_printf(_fmt,...)
Log to thread local error buffer.
Definition strerror.h:64
#define fr_type_is_vsa(_x)
Definition types.h:374
#define FR_TYPE_STRUCTURAL
Definition types.h:316
static char const * fr_type_to_str(fr_type_t type)
Return a static string containing the type name.
Definition types.h:454
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.
Definition value.c:1892
static fr_slen_t data
Definition value.h:1340
static size_t char ** out
Definition value.h:1030