The FreeRADIUS server $Id: 15bac2a4c627c01d1aa2047687b3418955ac7f00 $
Loading...
Searching...
No Matches
edit.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: 64d211a250036ca5102ed65474d431be3773b387 $
19 *
20 * @file src/lib/util/edit.c
21 * @brief Functions to edit pair lists, and track undo operations
22 *
23 * This file implements an "edit list" for changing values of
24 * #fr_pair_t. After some investigation, it turns out that it's much
25 * easier to have an "undo list" than to track partially applied
26 * transactions. Tracking partial transactions means that none of
27 * the fr_pair_foo() functions will work, as some pairs are in the
28 * "old" list and some in the "new" list. Also, a transaction may
29 * still fail when we finalize it by moving the pairs around.
30 *
31 * In contrast, an "undo" list means that all of the fr_pair_foo()
32 * functions will work, as any list contains only "active" pairs.
33 * And we never need to "finalize" a transaction, as the lists are
34 * already in their final form. The only thing needed for
35 * finalization is to free the undo list. Which can never fail.
36 *
37 * Note that the functions here require the input VPs to already have
38 * the correct talloc parent! The only thing the edit list does is
39 * to record "undo" actions.
40 *
41 * The only exception to this is fr_edit_list_apply_list_assignment().
42 * Which does call talloc_steal, and then also frees any pairs which
43 * weren't applied to the LHS.
44 *
45 * @copyright 2021 Network RADIUS SAS (legal@networkradius.com)
46 */
47
48RCSID("$Id: 64d211a250036ca5102ed65474d431be3773b387 $")
49
50#include <freeradius-devel/util/value.h>
51#include <freeradius-devel/util/talloc.h>
52#include "edit.h"
53#include "calc.h"
54
55typedef enum {
57 FR_EDIT_DELETE, //!< delete a VP
58 FR_EDIT_VALUE, //!< edit a VP in place
59 FR_EDIT_CLEAR, //!< clear the children of a structural entry.
60 FR_EDIT_INSERT, //!< insert a VP into a list, after another one.
61 FR_EDIT_CHILD, //!< child edit list
63
64#if 0
65/*
66 * For debugging.
67 */
68static const char *edit_names[4] = {
69 "invalid",
70 "delete",
71 "value",
72 "clear",
73 "insert",
74};
75#endif
76
77/** Track one particular edit.
78 */
79typedef struct {
80 fr_edit_op_t op; //!< edit operation to perform
81 fr_dlist_t entry; //!< linked list of edits
82
83 fr_pair_t *vp; //!< pair edited, deleted, or inserted
84
85 union {
86 union {
87 fr_value_box_t data; //!< original data
88 fr_pair_list_t children; //!< original child list, for "clear"
89 fr_edit_list_t *child_edit;
90 };
91
92 struct {
93 fr_pair_list_t *list; //!< parent list
94 fr_pair_t *ref; //!< reference pair for delete, insert before/after
95 };
96 };
97} fr_edit_t;
98
99/** Track a series of edits.
100 *
101 */
103 /*
104 * List of undo changes to be made, in order.
105 */
107
108 fr_dlist_head_t ignore; //!< lists to ignore
109
110 /*
111 * VPs which were inserted, and then over-written by a
112 * later edit.
113 */
115
116 fr_edit_list_t *parent; //!< for nested transactions
117 fr_edit_t *e; //!< so we don't have to loop over parent edits on abort
118};
119
120typedef struct {
122 fr_pair_list_t *list; //!< list to ignore (never dereferenced)
124
125
127{
128 return fr_dlist_empty(&el->undo) && fr_dlist_empty(&el->ignore) && fr_pair_list_empty(&el->deleted_pairs);
129}
130
131/** Undo one particular edit.
132 */
133static int edit_undo(fr_edit_t *e)
134{
135 fr_pair_t *vp = e->vp;
136#ifndef NDEBUG
137 int rcode;
138#endif
139
140 fr_assert(vp != NULL);
142
143 switch (e->op) {
144 case FR_EDIT_INVALID:
145 return -1;
146
147 case FR_EDIT_VALUE:
148 fr_assert(fr_type_is_leaf(vp->vp_type));
149 if (!fr_type_is_fixed_size(vp->vp_type)) fr_value_box_clear(&vp->data);
150 fr_value_box_copy(vp, &vp->data, &e->data);
151 break;
152
153 case FR_EDIT_CLEAR:
155
156 fr_pair_list_free(&vp->vp_group);
157 fr_pair_list_append(&vp->vp_group, &e->children);
158 break;
159
160 case FR_EDIT_DELETE:
161 fr_assert(e->list != NULL);
162#ifndef NDEBUG
163 rcode =
164#endif
165 fr_pair_insert_after(e->list, e->ref, vp);
166 fr_assert(rcode == 0);
167 break;
168
169 case FR_EDIT_INSERT:
170 /*
171 * We can free the VP here, as any edits to its'
172 * children MUST come after the creation of the
173 * VP. And any deletion of VPs after this one
174 * must come after this VP was created.
175 */
176 fr_pair_delete(e->list, vp);
177 break;
178
179 case FR_EDIT_CHILD:
180 fr_edit_list_abort(e->child_edit);
181 break;
182 }
183
184 return 0;
185}
186
187/** Abort the entries in an edit list.
188 *
189 * After this call, the input list(s) are unchanged from before any
190 * edits were made.
191 *
192 * the caller does not have to call talloc_free(el);
193 */
195{
196 fr_edit_t *e;
197
198 if (!el) return;
199
200 /*
201 * All of these pairs are already in the edit list. They
202 * have the correct parent, and will be placed back into
203 * their correct location by edit_undo()
204 */
205 fr_pair_list_init(&el->deleted_pairs);
206
207 /*
208 * Undo edits in reverse order, as later edits depend on
209 * earlier ones. We don't have multiple edits of the
210 * same VP, but we can create a VP, and then later edit
211 * its children.
212 */
213 while ((e = fr_dlist_pop_tail(&el->undo)) != NULL) {
214 edit_undo(e);
215 /*
216 * Don't free "e", it will be cleaned up when we
217 * talloc_free(el). That should be somewhat
218 * faster than doing it incrementally.
219 */
220 }
221
222 /*
223 * There's a parent, we remove ourselves from the parent undo list.
224 */
225 if (el->parent) {
226 fr_dlist_remove(&el->parent->undo, el->e);
227 talloc_free(el->e);
228 }
229
231}
232
233/** Record one particular edit
234 *
235 * For INSERT / DELETE, this function will also insert / delete the
236 * VP.
237 *
238 * For VALUE changes, this function must be called BEFORE the value
239 * is changed. Once this function has been called, it is then safe
240 * to edit the value in place.
241 *
242 * Note that VALUE changes for structural types are allowed ONLY when
243 * using T_OP_SET, which over-writes previous values. For every
244 * other modification to structural types, we MUST instead call
245 * insert / delete on the vp_group.
246 */
248{
249 fr_edit_t *e;
250
251 fr_assert(el != NULL);
252 fr_assert(vp != NULL);
253
254 fr_assert(op != FR_EDIT_CHILD); /* only used by fr_edit_alloc() */
255
256 /*
257 * When we insert a structural type, we also want to
258 * not track edits to it's children. The "ignore list"
259 * allows us to see which lists don't have edits recorded.
260 *
261 * Perform the operation. We're not recording
262 * it, but we still need to do the work.
263 */
264 fr_dlist_foreach(&el->ignore, fr_edit_ignore_t, i) {
265 if (i->list != list) continue;
266
267 switch (op) {
268 /*
269 * No need to save the value.
270 */
271 case FR_EDIT_VALUE:
272 return 0;
273
274 /*
275 * No need to save the value.
276 */
277 case FR_EDIT_DELETE:
278 fr_pair_remove(list, vp);
279 return 0;
280
281 /*
282 * Delete all of the children.
283 */
284 case FR_EDIT_CLEAR:
285 if (!fr_type_is_structural(vp->vp_type)) return 0;
286
287 /*
288 * The VP is a child of an attribute which was previously inserted as part of
289 * this edit. We therefore allow the "clear" to clear it, even if it contains
290 * immutable children. Because this operation is equivalent to just never
291 * creating the children.
292 */
293
294 fr_pair_list_free(&vp->vp_group);
295 return 0;
296
297 /*
298 * Insert it, and perhaps save the list
299 * for a structural VP saying "don't
300 * record edits to this, either".
301 */
302 case FR_EDIT_INSERT:
303 if (fr_pair_insert_after(list, ref, vp) < 0) return -1;
304
305 /*
306 * Non-structural types don't have any other work to do.
307 */
308 if (!fr_type_is_structural(vp->vp_type)) return 0;
309
310 /*
311 * Otherwise we're inserting a VP which has a
312 * child list. Remember that we need to ignore
313 * edits to the children of this VP, too.
314 */
315 goto insert_ignore;
316
317 default:
318 return -1;
319 }
320 }
321
322 /*
323 * Catch NOOPs
324 */
325 if (op == FR_EDIT_CLEAR) {
327
328 if (fr_pair_list_empty(&vp->vp_group)) return 0;
329 }
330
331 /*
332 * Search for previous edits.
333 *
334 * @todo - if we're modifying values of a child VP, and
335 * it's parent is marked as INSERT, then we don't need to
336 * record FR_EDIT_VALUE changes to the children. It's
337 * not yet clear how best to track this.
338 */
339 for (e = fr_dlist_head(&el->undo);
340 e != NULL;
341 e = fr_dlist_next(&el->undo, e)) {
342 fr_assert(e->vp != NULL);
343
344 if (e->vp != vp) continue;
345
346 switch (op) {
347 case FR_EDIT_INVALID:
348 return -1;
349
350 /*
351 * We're editing a previous edit.
352 * There's no need to record anything
353 * new, as we've already recorded the
354 * original value.
355 *
356 * Note that we can insert a pair and
357 * then edit it. The undo list only
358 * saves the insert, as the later edit is
359 * irrelevant. If we're undoing, we
360 * simply delete the new attribute which
361 * was inserted.
362 */
363 case FR_EDIT_VALUE:
364 /*
365 * If we delete a pair, we can't later
366 * edit it. That indicates serious
367 * issues with the code.
368 *
369 * However, if we previously inserted
370 * this VP, then we don't need to record
371 * changes to its value. Similarly, if
372 * we had previously changed its value,
373 * we don't need to record that
374 * information again.
375 */
377 fr_assert(fr_type_is_leaf(vp->vp_type));
378 return 0;
379
380 /*
381 * We're inserting a new pair.
382 *
383 * We can't have previously edited this
384 * pair (inserted, deleted, or updated
385 * the value), as the pair is new!
386 */
387 case FR_EDIT_INSERT:
388 fr_assert(0);
389 return -1;
390
391 case FR_EDIT_CLEAR:
392 /*
393 * If we're clearing it, we MUST have
394 * previously inserted it. So just nuke
395 * it's children, as merging the
396 * operations of "insert with stuff" and
397 * then "clear" is just "insert empty
398 * pair".
399 *
400 * However, we don't yet delete the
401 * children, as there may be other edit
402 * operations which are referring to
403 * them.
404 */
407
408 fr_pair_list_append(&el->deleted_pairs, &vp->vp_group);
409 break;
410
411 /*
412 * We're being asked to delete something
413 * we previously inserted, or previously
414 * edited.
415 */
416 case FR_EDIT_DELETE:
417 /*
418 * We can't delete something which was
419 * already deleted.
420 */
422
423 /*
424 * We had previously inserted it. So
425 * just delete the insert operation, and
426 * delete the VP from the list.
427 *
428 * Other edits may refer to children of
429 * this pair. So we don't free the VP
430 * immediately, but instead reparent it
431 * to the edit list. So that when the
432 * edit list is freed, the VP will be
433 * freed.
434 */
435 if (e->op == FR_EDIT_INSERT) {
436 fr_assert(e->list == list);
437
438 fr_pair_remove(list, vp);
439 fr_pair_append(&el->deleted_pairs, vp);
440
441 fr_dlist_remove(&el->undo, e);
442 talloc_free(e);
443 return 0;
444 }
445
446 /*
447 * We had previously changed the value,
448 * but now we're going to delete it.
449 *
450 * Since it had previously existed, we
451 * have to reset its value to the
452 * original one, and then track the
453 * deletion.
454 */
455 edit_undo(e);
456
457 /*
458 * Rewrite the edit to be delete.
459 *
460 * And move the deletion to the tail of
461 * the edit list, because edits between
462 * "here" and the tail of the list may
463 * refer to "vp". If we leave the
464 * deletion in place, then subsequent
465 * edit list entries will refer to a VP
466 * which has been deleted!
467 */
468 e->op = FR_EDIT_DELETE;
469 fr_dlist_remove(&el->undo, e);
470 goto delete;
471
472 case FR_EDIT_CHILD: /* e->vp==NULL, so this should never happen */
473 fr_assert(0);
474 return -1;
475 }
476 } /* loop over existing edits */
477
478 /*
479 * No edit for this pair exists. Create a new edit entry.
480 */
481 e = talloc_zero(el, fr_edit_t);
482 if (!e) return -1;
483
484 e->op = op;
485 e->vp = vp;
486 fr_value_box_init_null(&e->data);
487
488 switch (op) {
489 case FR_EDIT_INVALID:
490 case FR_EDIT_CHILD:
491 fail:
492 talloc_free(e);
493 return -1;
494
495 case FR_EDIT_VALUE:
496 fr_assert(list == NULL);
497 fr_assert(ref == NULL);
498
499 fr_assert(fr_type_is_leaf(vp->vp_type));
500 fr_value_box_copy(e, &e->data, &vp->data);
501 break;
502
503 case FR_EDIT_CLEAR:
504 fr_assert(list == NULL);
505 fr_assert(ref == NULL);
506
508 fr_pair_list_init(&e->children);
509 fr_pair_list_append(&e->children, &vp->vp_group);
510 break;
511
512 case FR_EDIT_INSERT:
513 fr_assert(list != NULL);
514
515 /*
516 * There's no need to record "prev". On undo, we
517 * just delete this pair from the list.
518 */
519 e->list = list;
520 if (fr_pair_insert_after(list, ref, vp) < 0) goto fail;
521 break;
522
523 case FR_EDIT_DELETE:
524 delete:
525 fr_assert(list != NULL);
526 fr_assert(ref == NULL);
527
528 e->list = list;
529 e->ref = fr_pair_list_prev(list, vp);
530
531 fr_pair_remove(list, vp);
532 break;
533 }
534
535 fr_dlist_insert_tail(&el->undo, e);
536
537 /*
538 * Insert an "ignore" entry.
539 */
540 if ((op == FR_EDIT_INSERT) && fr_type_is_structural(vp->vp_type)) {
542
543 insert_ignore:
544 i = talloc_zero(el, fr_edit_ignore_t);
545 if (!i) return 0;
546
547 i->list = &vp->vp_group;
548 fr_dlist_insert_tail(&el->ignore, i);
549 }
550
551 return 0;
552}
553
554
555/** Insert a new VP after an existing one.
556 *
557 * This function mirrors fr_pair_insert_after().
558 *
559 * After this function returns, the new VP has been inserted into the
560 * list.
561 */
563{
564 if (!el) return fr_pair_insert_after(list, pos, vp);
565
566 return edit_record(el, FR_EDIT_INSERT, vp, list, pos);
567}
568
569/** Delete a VP
570 *
571 * This function mirrors fr_pair_delete()
572 *
573 * After this function returns, the VP has been removed from the list.
574 */
576{
577 if (fr_pair_immutable(vp)) {
578 fr_strerror_printf("Cannot modify immutable value for %s", vp->da->name);
579 return -1;
580 }
581
582 if (!el) {
583 fr_pair_delete(list, vp);
584 return 0;
585 }
586
587 return edit_record(el, FR_EDIT_DELETE, vp, list, NULL);
588}
589
590/** Delete VPs with a matching da
591 *
592 * This function mirrors fr_pair_delete_by_da()
593 */
595{
596 if (!el) {
597 fr_pair_delete_by_da(list, da);
598 return 0;
599 }
600
601 /*
602 * Delete all VPs with a matching da.
603 */
604 fr_pair_list_foreach(list, vp) {
605 if (fr_pair_immutable(vp)) continue;
606
607 if (vp->da != da) continue;
608
609 (void) fr_pair_remove(list, vp);
610
611 if (edit_record(el, FR_EDIT_DELETE, vp, list, NULL) < 0) return -1;
612 }
613
614 return 0;
615}
616
617
618/** Record the value of a leaf #fr_value_box_t
619 *
620 * After this function returns, it's safe to edit the pair.
621 */
623{
624 if (!el) return 0;
625
626 if (!fr_type_is_leaf(vp->vp_type)) return -1;
627
628 return edit_record(el, FR_EDIT_VALUE, vp, NULL, NULL);
629}
630
631/** Write a new value to the #fr_value_box_t
632 *
633 * After this function returns, the value has been updated.
634 */
636{
637 if (!fr_type_is_leaf(vp->vp_type)) return -1;
638
639 if (vp->vp_immutable) {
640 fr_strerror_printf("Cannot modify immutable value for %s", vp->da->name);
641 return -1;
642 }
643
644 if (el && (edit_record(el, FR_EDIT_VALUE, vp, NULL, NULL) < 0)) return -1;
645
646 if (!fr_type_is_fixed_size(vp->vp_type)) fr_value_box_clear(&vp->data);
647 fr_value_box_copy_shallow(NULL, &vp->data, box);
648 return 0;
649}
650
651/** Replace a pair with another one.
652 *
653 * This function mirrors fr_pair_replace().
654 *
655 * After this function returns, the new VP has replaced the old one,
656 * and the new one can be edited.
657 */
659{
660 if (to_replace->da != vp->da) return -1;
661
662 if (fr_pair_immutable(to_replace)) {
663 fr_strerror_printf("Cannot modify immutable value for %s", vp->da->name);
664 return -1;
665 }
666
667 if (!el) {
668 if (fr_pair_insert_after(list, to_replace, vp) < 0) return -1;
669 fr_pair_delete(list, to_replace);
670 return -1;
671 }
672
673 /*
674 * We call edit_record() twice, which involves two
675 * complete passes over the edit list. That's fine,
676 * either the edit list is small, OR we will eventually
677 * put the VPs to be edited into an RB tree.
678 */
679 if (edit_record(el, FR_EDIT_INSERT, vp, list, to_replace) < 0) return -1;
680
681 /*
682 * If deleting the old entry fails, then the new entry
683 * above MUST be the last member of the edit list. If
684 * it's not the last member, then it means that it
685 * already existed in the list (either VP list of edit
686 * list). The edit_record() function checks for that,
687 * and errors if so.
688 */
689 if (edit_record(el, FR_EDIT_DELETE, to_replace, list, NULL) < 0) {
690 fr_edit_t *e = fr_dlist_pop_tail(&el->undo);
691
692 fr_assert(e != NULL);
693 fr_assert(e->vp == vp);
694 talloc_free(e);
695 return -1;
696 }
697
698 return 0;
699}
700
701
702/** Free children of a structural pair.
703 *
704 * This function mirrors fr_pair_replace().
705 *
706 * After this function returns, the new VP has replaced the old one,
707 * and the new one can be edited.
708 */
710{
711 if (!fr_type_is_structural(vp->vp_type)) return -1;
712
713 if (!el) {
714 fr_pair_list_free(&vp->children);
715 return 0;
716 }
717
718 /*
719 * No children == do nothing.
720 */
721 if (fr_pair_list_empty(&vp->vp_group)) return 0;
722
723 /*
724 * Record the list, even if it's empty. That way if we
725 * later add children to it, the "undo" operation can
726 * reset the children list to be empty.
727 */
728 return edit_record(el, FR_EDIT_CLEAR, vp, NULL, NULL);
729}
730
731/** Finalize the edits when we destroy the edit list.
732 *
733 * Which in large part means freeing the VPs which have been deleted,
734 * or saved, and then deleting the edit list.
735 */
737{
738 fr_edit_t *e;
739
740 fr_assert(el != NULL);
741
742 for (e = fr_dlist_head(&el->undo);
743 e != NULL;
744 e = fr_dlist_next(&el->undo, e)) {
745 switch (e->op) {
746 case FR_EDIT_INVALID:
747 fr_assert(0);
748 break;
749
750 case FR_EDIT_INSERT:
751 break;
752
753 case FR_EDIT_DELETE:
754 fr_assert(e->vp != NULL);
755 talloc_free(e->vp);
756 break;
757
758 case FR_EDIT_CLEAR:
759 fr_pair_list_free(&e->children);
760 break;
761
762 case FR_EDIT_VALUE:
763 fr_assert(fr_type_is_leaf(e->vp->vp_type));
764 fr_value_box_clear(&e->data);
765 break;
766
767 case FR_EDIT_CHILD:
768 talloc_free(e->child_edit);
769 break;
770 }
771 }
772
773 fr_pair_list_free(&el->deleted_pairs);
774
776
777 return 0;
778}
779
780/** Allocate an edit list.
781 *
782 * Edit lists can be nested. If the child list commits, then it does nothing
783 * until the parent list commits. On the other hand, if the child list aborts,
784 * then the parent list might continue.
785 *
786 * @param ctx talloc context. Ignored if parent!=NULL
787 * @param hint how many edits we are likely to allocate
788 * @param parent a parent edit list
789 */
791{
793 fr_edit_t *e;
794
795 /*
796 * If we have nested transactions, then allocate this
797 * list in the context of the parent. Otherwise it will
798 * be freed too soon.
799 */
800 if (parent) ctx = parent;
801
802 el = talloc_zero_pooled_object(ctx, fr_edit_list_t, hint, hint * sizeof(fr_edit_t));
803 if (!el) return NULL;
804
805 fr_dlist_init(&el->undo, fr_edit_t, entry);
806 fr_dlist_init(&el->ignore, fr_edit_ignore_t, entry);
807
808 fr_pair_list_init(&el->deleted_pairs);
809
810 talloc_set_destructor(el, _edit_list_destructor);
811
812 el->parent = parent;
813
814 if (!parent) return el;
815
816 e = talloc_zero(parent, fr_edit_t);
817 if (!e) {
819 return NULL;
820 }
821
822 /*
823 * Insert the child into the tail of the current edit list.
824 */
825 e->op = FR_EDIT_CHILD;
826 e->vp = NULL;
827 e->child_edit = el;
828
829 fr_dlist_insert_tail(&parent->undo, e);
830 el->e = e;
831
832 return el;
833}
834
835/** Commit an edit list.
836 *
837 * If there are nested transactions, then this transaction is
838 * committed only when the parent transaction has been committed.
839 *
840 */
842{
843 if (el->parent) return;
844
846}
847
848/** Notes
849 *
850 * Unlike "update" sections, edits are _not_ hierarchical. If we're
851 * editing values a list, then the list has to exist. If we're
852 * inserting pairs in a list, then we find the lowest existing pair,
853 * and add pairs there.
854 *
855 * The functions tmpl_extents_find() and tmpl_extents_build_to_leaf_parent()
856 * should help us figure out where the VPs exist or not.
857 *
858 * The overall "update" algorithm is now:
859 *
860 * alloc(edit list)
861 *
862 * foreach entry in the things to do
863 * expand LHS if needed to local TMPL
864 * expand RHS if needed to local box / cursor / TMPL
865 *
866 * use LHS/RHS cursors to find VPs
867 * edit VPs, recording edits
868 *
869 * free temporary map
870 * commit(edit list)
871 */
872
873/**********************************************************************
874 *
875 * Now we have helper functions which use the edit list to get things
876 * done.
877 *
878 **********************************************************************/
879
880/** Insert a list after a particular point in another list.
881 *
882 * This function mirrors fr_pair_list_append(), but with a bit more
883 * control over where the to_insert list ends up.
884 *
885 * There's nothing magical about this function, it's just easier to
886 * have it here than in multiple places in the code.
887 */
889{
890 fr_pair_t *prev, *vp;
891
892 prev = pos;
893
894 if (!el) {
895 /*
896 * @todo - this should really be an O(1) dlist
897 * operation.
898 */
899 while ((vp = fr_pair_list_head(to_insert)) != NULL) {
900 (void) fr_pair_remove(to_insert, vp);
901 (void) fr_pair_insert_after(list, prev, vp);
902 prev = vp;
903 }
904
905 return 0;
906 }
907
908 /*
909 * We have to record each individual insert as a separate
910 * item. Some later edit may insert pairs in the middle
911 * of the ones we've added.
912 */
913 while ((vp = fr_pair_list_head(to_insert)) != NULL) {
914 (void) fr_pair_remove(to_insert, vp);
915
916 if (edit_record(el, FR_EDIT_INSERT, vp, list, prev) < 0) {
917 fr_pair_prepend(to_insert, vp); /* don't lose it! */
918 return -1;
919 }
920
921 prev = vp;
922 }
923
924 return 0;
925}
926
927/** Removes elements matching a list
928 *
929 * O(N^2) unfortunately.
930 */
932{
933 /*
934 * We have a list of VPs with operators and values. Those contain the list of things we want to
935 * be removed from the main "list".
936 */
937 fr_pair_list_foreach(to_remove, vp) {
938 fr_pair_t *found, *next;
939
940 /*
941 * @todo - do this recursively.
942 */
943 if (fr_type_is_structural(vp->vp_type)) continue;
944
945 /*
946 * Search the list to edit for VPs which match the ones we're trying to delete.
947 */
948 for (found = fr_pair_find_by_da(list, NULL, vp->da);
949 found != NULL;
950 found = next) {
951 int rcode;
952
953 next = fr_pair_find_by_da(list, found, vp->da);
954
955 /*
956 * It doesn't match, keep it. If it matches, delete it.
957 */
958 rcode = fr_value_box_cmp_op(vp->op, &found->data, &vp->data);
959 if (rcode < 0) return -1;
960
961 if (!rcode) continue;
962
963 if (fr_edit_list_pair_delete(el, list, found) < 0) return -1;
964 }
965 }
966
967 return 0;
968}
969
970/** Apply operators to pairs.
971 *
972 * := is "if found vp, call fr_edit_list_pair_replace(). Otherwise call fr_edit_list_insert_pair_tail()
973 * = is "if found vp, do nothing. Otherwise call fr_edit_list_insert_pair_tail()
974 *
975 */
977{
978 fr_value_box_t box;
979
980 switch (op) {
981 case T_OP_LE:
982 case T_OP_LT:
983 case T_OP_GT:
984 case T_OP_GE:
985 if (fr_value_calc_binary_op(vp, &box, FR_TYPE_BOOL, &vp->data, op, in) < 0) return -1;
986
987 if (box.vb_bool) return 0;
988
989 if (el && (fr_edit_list_save_pair_value(el, vp) < 0)) return -1;
990
992
993 /*
994 * The input type may be different, so we can't just copy it.
995 */
996 return fr_value_box_cast(vp, &vp->data, vp->vp_type, vp->data.enumv, in);
997
998 default:
999 break;
1000
1001 }
1002
1003 if (el && (fr_edit_list_save_pair_value(el, vp) < 0)) return -1;
1004
1005 return fr_value_calc_assignment_op(vp, &vp->data, op, in);
1006}
1007
1008#undef COPY
1009#define COPY(_x) do { if (copy) { \
1010 c = fr_pair_copy(dst, _x); \
1011 if (!c) return -1; \
1012 } else { \
1013 c = talloc_steal(dst, _x); \
1014 fr_pair_remove(src, c); \
1015 } \
1016 } while (0)
1017
1018#define NEXT_A do { a = an; an = fr_pair_list_next(&dst->children, a); } while (0)
1019#define NEXT_B do { b = bn; bn = fr_pair_list_next(src, b); } while (0)
1020
1021
1022/** A UNION B
1023 *
1024 */
1025static int list_union(fr_edit_list_t *el, fr_pair_t *dst, fr_pair_list_t *src, bool copy)
1026{
1027 fr_pair_t *a, *an;
1028 fr_pair_t *b, *bn;
1029 fr_pair_t *c;
1030
1031 /*
1032 * Prevent people from doing stupid things.
1033 * While it's technically possible to take a
1034 * UNION of structs, that would work ONLY when
1035 * the two structs had disjoint members.
1036 * e.g. {1, 3, 4} and {2, 5, 6}. That's too
1037 * complex to check right now, so we punt on the
1038 * problem.
1039 */
1040 if (dst->vp_type == FR_TYPE_STRUCT) {
1041 fr_strerror_printf("Cannot take union of STRUCT data types, it would break the structure");
1042 return -1;
1043 }
1044
1047
1048 PAIR_LIST_VERIFY(&dst->children);
1049 PAIR_LIST_VERIFY(src);
1050
1051 a = fr_pair_list_head(&dst->children);
1052 an = fr_pair_list_next(&dst->children, a);
1053 b = fr_pair_list_head(src);
1054 bn = fr_pair_list_next(src, b);
1055
1056 while (true) {
1057 int rcode;
1058
1059 /*
1060 * B is done, so we stop processing.
1061 */
1062 if (!b) break;
1063
1064 /*
1065 * A is done, so we can add in B at the end of A.
1066 */
1067 if (!a) {
1068 COPY(b);
1069
1070 if (fr_edit_list_insert_pair_tail(el, &dst->children, c) < 0) {
1071 return -1;
1072 }
1073
1074 NEXT_B;
1075 continue;
1076 }
1077
1078 /*
1079 * Compare the da's
1080 */
1081 rcode = fr_pair_cmp_by_parent_num(a, b);
1082
1083 /*
1084 * We've seen things in A which aren't in B, so
1085 * we just increment A.
1086 */
1087 if (rcode < 0) {
1088 NEXT_A;
1089 continue;
1090 }
1091
1092 /*
1093 * a > b
1094 *
1095 * This means that in the ordered set, the
1096 * equivalent to B does not exist. So we copy B
1097 * to after A.
1098 */
1099 if (rcode > 0) {
1100 COPY(b);
1101
1102 if (fr_edit_list_insert_pair_after(el, &dst->children, a, c) < 0) {
1103 return -1;
1104 }
1105
1106 NEXT_B;
1107 continue;
1108 }
1109
1110 fr_assert(rcode == 0);
1111
1112 /*
1113 * They're the same.
1114 */
1115 fr_assert(a->da == b->da);
1116
1117 /*
1118 * Union lists recursively.
1119 *
1120 * Note that this doesn't mean copying both VPs! We just merge their contents.
1121 */
1122 if (fr_type_is_structural(a->vp_type)) {
1123 rcode = list_union(el, a, &b->children, copy);
1124 if (rcode < 0) return rcode;
1125
1126 NEXT_A;
1127 NEXT_B;
1128 continue;
1129 }
1130
1131 /*
1132 * Process all identical attributes, but by
1133 * value. If the value is the same, we keep only
1134 * one. If the values are different, we keep
1135 * both.
1136 */
1137 while (a && b && (a->da == b->da)) {
1138 /*
1139 * Check if the values are the same. This
1140 * returns 0 for "equal", and non-zero for
1141 * anything else.
1142 */
1143 rcode = fr_value_box_cmp(&a->data, &b->data);
1144 if (rcode != 0) {
1145 COPY(b);
1146
1147 if (fr_edit_list_insert_pair_after(el, &dst->children, a, c) < 0) {
1148 return -1;
1149 }
1150 }
1151
1152 NEXT_A;
1153 NEXT_B;
1154 }
1155 }
1156
1157 return 0;
1158}
1159
1160/** A MERGE B
1161 *
1162 * with priority to A
1163 */
1164static int list_merge_lhs(fr_edit_list_t *el, fr_pair_t *dst, fr_pair_list_t *src, bool copy)
1165{
1166 fr_pair_t *a, *an;
1167 fr_pair_t *b, *bn;
1168 fr_pair_t *c;
1169
1172
1173 PAIR_LIST_VERIFY(&dst->children);
1174 PAIR_LIST_VERIFY(src);
1175
1176 a = fr_pair_list_head(&dst->children);
1177 an = fr_pair_list_next(&dst->children, a);
1178 b = fr_pair_list_head(src);
1179 bn = fr_pair_list_next(src, b);
1180
1181 while (true) {
1182 int rcode;
1183
1184 /*
1185 * B is done, so we stop processing.
1186 */
1187 if (!b) break;
1188
1189 /*
1190 * A is done, so we can add in B at the end of A.
1191 */
1192 if (!a) {
1193 COPY(b);
1194
1195 if (fr_edit_list_insert_pair_tail(el, &dst->children, c) < 0) {
1196 return -1;
1197 }
1198
1199 NEXT_B;
1200 continue;
1201 }
1202
1203 /*
1204 * Compare the da's
1205 */
1206 rcode = fr_pair_cmp_by_parent_num(a, b);
1207
1208 /*
1209 * We've seen things in A which aren't in B, so
1210 * we just increment A.
1211 */
1212 if (rcode < 0) {
1213 NEXT_A;
1214 continue;
1215 }
1216
1217 /*
1218 * a > b
1219 *
1220 * This means that in the ordered set, the
1221 * equivalent to B does not exist. So we copy B
1222 * to before A.
1223 */
1224 if (rcode > 0) {
1225 COPY(b);
1226
1227 if (fr_edit_list_insert_pair_before(el, &dst->children, a, c) < 0) {
1228 return -1;
1229 }
1230
1231 NEXT_B;
1232 continue;
1233 }
1234
1235 fr_assert(rcode == 0);
1236
1237 /*
1238 * They're the same.
1239 */
1240 fr_assert(a->da == b->da);
1241
1242 /*
1243 * Merge lists recursively.
1244 */
1245 if (fr_type_is_structural(a->vp_type)) {
1246 rcode = list_merge_lhs(el, a, &b->children, copy);
1247 if (rcode < 0) return rcode;
1248
1249 goto next_both;
1250 }
1251
1252 /*
1253 * We have both A and B, so we prefer A, which means just skipping B.
1254 */
1255
1256 next_both:
1257 NEXT_A;
1258 NEXT_B;
1259 }
1260
1261 return 0;
1262}
1263
1264/** A MERGE B
1265 *
1266 * with priority to B.
1267 */
1268static int list_merge_rhs(fr_edit_list_t *el, fr_pair_t *dst, fr_pair_list_t *src, bool copy)
1269{
1270 fr_pair_t *a, *an;
1271 fr_pair_t *b, *bn;
1272 fr_pair_t *c;
1273
1276
1277 PAIR_LIST_VERIFY(&dst->children);
1278 PAIR_LIST_VERIFY(src);
1279
1280 a = fr_pair_list_head(&dst->children);
1281 an = fr_pair_list_next(&dst->children, a);
1282 b = fr_pair_list_head(src);
1283 bn = fr_pair_list_next(src, b);
1284
1285 while (true) {
1286 int rcode;
1287
1288 /*
1289 * B is done, so we stop processing.
1290 */
1291 if (!b) break;
1292
1293 /*
1294 * A is done, so we can in B at the end of A.
1295 */
1296 if (!a) {
1297 COPY(b);
1298
1299 if (fr_edit_list_insert_pair_tail(el, &dst->children, c) < 0) {
1300 return -1;
1301 }
1302
1303 NEXT_B;
1304 continue;
1305 }
1306
1307 /*
1308 * Compare the da's
1309 */
1310 rcode = fr_pair_cmp_by_parent_num(a, b);
1311
1312 /*
1313 * We've seen things in A which aren't in B, so
1314 * we just increment A.
1315 */
1316 if (rcode < 0) {
1317 NEXT_A;
1318 continue;
1319 }
1320
1321 /*
1322 * a > b
1323 *
1324 * This means that in the ordered set, the
1325 * equivalent to B does not exist. So we copy B
1326 * to before A.
1327 */
1328 if (rcode > 0) {
1329 COPY(b);
1330
1331 if (fr_edit_list_insert_pair_before(el, &dst->children, a, c) < 0) {
1332 return -1;
1333 }
1334
1335 NEXT_B;
1336 continue;
1337 }
1338
1339 fr_assert(rcode == 0);
1340
1341 /*
1342 * They're the same.
1343 */
1344 fr_assert(a->da == b->da);
1345
1346 /*
1347 * Merge lists recursively.
1348 */
1349 if (fr_type_is_structural(a->vp_type)) {
1350 rcode = list_merge_rhs(el, a, &b->children, copy);
1351 if (rcode < 0) return rcode;
1352
1353 goto next_both;
1354 }
1355
1356 /*
1357 * We have both A and B, so we prefer B.
1358 */
1359 COPY(b);
1360 if (fr_edit_list_replace_pair(el, &dst->children, a, c) < 0) {
1361 return -1;
1362 }
1363
1364 next_both:
1365 NEXT_A;
1366 NEXT_B;
1367 }
1368
1369 return 0;
1370}
1371
1372/** A INTERSECTION B
1373 *
1374 */
1376{
1377 fr_pair_t *a, *an;
1378 fr_pair_t *b, *bn;
1379
1380 /*
1381 * Prevent people from doing stupid things.
1382 */
1383 if (dst->vp_type == FR_TYPE_STRUCT) {
1384 fr_strerror_printf("Cannot take intersection of STRUCT data types, it would break the structure");
1385 return -1;
1386 }
1387
1390
1391 a = fr_pair_list_head(&dst->children);
1392 an = fr_pair_list_next(&dst->children, a);
1393 b = fr_pair_list_head(src);
1394 bn = fr_pair_list_next(src, b);
1395
1396 while (true) {
1397 int rcode;
1398
1399 /*
1400 * A is done, so we can return. We don't need to
1401 * delete everything from B, as that will be
1402 * cleaned up by the caller when we exit.
1403 */
1404 if (!a) break;
1405
1406 /*
1407 * B is done, so we delete everything else in A.
1408 */
1409 if (!b) {
1410 delete_a:
1411 if (fr_edit_list_pair_delete(el, &dst->children, a) < 0) return -1;
1412 NEXT_A;
1413 continue;
1414 }
1415
1416 /*
1417 * Compare the da's
1418 */
1419 rcode = fr_pair_cmp_by_parent_num(a, b);
1420
1421 /*
1422 * a < b
1423 *
1424 * A gets removed.
1425 */
1426 if (rcode < 0) goto delete_a;
1427
1428 /*
1429 * a > b
1430 *
1431 * Skip forward in B until we have it better matching A.
1432 */
1433 if (rcode > 0) {
1434 NEXT_B;
1435 continue;
1436 }
1437
1438 fr_assert(rcode == 0);
1439
1440 /*
1441 * INTERSECT the children, and then leave A
1442 * alone, unless it's empty, in which case A
1443 * INTERSECT B is empty, so we also delete A.
1444 */
1445 if (fr_type_is_structural(a->vp_type)) {
1446 rcode = list_intersection(el, a, &b->children);
1447 if (rcode < 0) return rcode;
1448
1449 NEXT_B;
1450
1451 if (fr_pair_list_empty(&a->children)) goto delete_a;
1452
1453 NEXT_A;
1454 continue;
1455 }
1456
1457 /*
1458 * Process all identical attributes, but by
1459 * value.
1460 */
1461 while (a && b && (a->da == b->da)) {
1462 /*
1463 * Check if the values are the same. This
1464 * returns 0 for "equal", and non-zero for
1465 * anything else.
1466 */
1467 rcode = fr_value_box_cmp(&a->data, &b->data);
1468 if (rcode != 0) {
1469 if (fr_edit_list_pair_delete(el, &dst->children, a) < 0) return -1;
1470 }
1471
1472 NEXT_A;
1473 NEXT_B;
1474 }
1475 }
1476
1477 return 0;
1478}
1479
1480
1481/** Apply operators to lists.
1482 *
1483 * = is "if found vp, do nothing. Otherwise call fr_edit_list_insert_pair_tail()
1484 *
1485 * The src list is sorted, but is otherwise not modified.
1486 */
1488{
1489 fr_pair_list_t list;
1490
1491 if (!fr_type_is_structural(dst->vp_type)) {
1492 fr_strerror_printf("Cannot perform list assignment to non-structural type '%s'",
1493 fr_type_to_str(dst->vp_type));
1494 return -1;
1495 }
1496
1497#undef COPY
1498#define COPY do { if (copy) { \
1499 fr_pair_list_init(&list); \
1500 if (fr_pair_list_copy(dst, &list, src) < 0) return -1;\
1501 src = &list; \
1502 } else { \
1503 fr_pair_list_steal(dst, src); \
1504 } \
1505 } while (0)
1506
1507
1508 switch (op) {
1509 /*
1510 * Over-ride existing value (i.e. children) with
1511 * new list.
1512 */
1513 case T_OP_SET:
1514 if (&dst->children == src) return 0; /* A := A == A */
1515
1516 if (fr_edit_list_free_pair_children(el, dst) < 0) return -1;
1518
1519 case T_OP_ADD_EQ:
1520 if (&dst->children == src) {
1521 fr_strerror_printf("Cannot append list to itself");
1522 return -1;
1523 }
1524
1525 COPY;
1526 return fr_edit_list_insert_list_tail(el, &dst->children, src);
1527
1528 case T_OP_SUB_EQ:
1529 /*
1530 * foo -= foo --> {}
1531 */
1532 if (&dst->children == src) {
1533 fr_pair_t *vp;
1534
1535 while ((vp = fr_pair_list_head(&dst->children)) != NULL) {
1536 if (fr_edit_list_pair_delete(el, &dst->children, vp) < 0) return -1;
1537 }
1538
1539 return 0;
1540 }
1541
1542 return fr_edit_list_delete_list(el, &dst->children, src);
1543
1544 case T_OP_PREPEND:
1545 if (&dst->children == src) {
1546 fr_strerror_printf("Cannot prepend list to itself");
1547 return -1;
1548 }
1549
1550 COPY;
1551 return fr_edit_list_insert_list_head(el, &dst->children, src);
1552
1553 case T_OP_AND_EQ:
1554 if (&dst->children == src) return 0; /* A INTERSECTION A == A */
1555
1556 if (!fr_edit_list_empty(el)) {
1557 not_empty:
1558 fr_strerror_printf("Failed to perform %s - undo list is not empty", fr_tokens[op]);
1559 return -1;
1560 }
1561
1562 return list_intersection(el, dst, src);
1563
1564 case T_OP_OR_EQ:
1565 if (&dst->children == src) return 0; /* A UNION A == A */
1566
1567 if (!fr_edit_list_empty(el)) goto not_empty;
1568
1569 return list_union(el, dst, src, copy);
1570
1571 case T_OP_GE:
1572 if (&dst->children == src) return 0; /* A MERGE A == A */
1573
1574 if (!fr_edit_list_empty(el)) goto not_empty;
1575
1576 return list_merge_lhs(el, dst, src, copy);
1577
1578 case T_OP_LE:
1579 if (&dst->children == src) return 0; /* A MERGE A == A */
1580
1581 if (!fr_edit_list_empty(el)) goto not_empty;
1582
1583 return list_merge_rhs(el, dst, src, copy);
1584
1585 default:
1586 break;
1587 }
1588
1589 fr_strerror_printf("Invalid assignment operator %s for destination type %s",
1590 fr_tokens[op],
1591 fr_type_to_str(dst->vp_type));
1592 return -1;
1593}
#define RCSID(id)
Definition build.h:483
#define FALL_THROUGH
clang 10 doesn't recognised the FALL-THROUGH comment anymore
Definition build.h:322
int fr_value_calc_assignment_op(TALLOC_CTX *ctx, fr_value_box_t *dst, fr_token_t op, fr_value_box_t const *src)
Calculate DST OP SRC.
Definition calc.c:2399
int fr_value_calc_binary_op(TALLOC_CTX *ctx, fr_value_box_t *dst, fr_type_t hint, fr_value_box_t const *a, fr_token_t op, fr_value_box_t const *b)
Calculate DST = A OP B.
Definition calc.c:1894
static fr_slen_t in
Definition dict.h:824
#define fr_dlist_init(_head, _type, _field)
Initialise the head structure of a doubly linked list.
Definition dlist.h:260
static void * fr_dlist_head(fr_dlist_head_t const *list_head)
Return the HEAD item of a list or NULL if the list is empty.
Definition dlist.h:486
#define fr_dlist_foreach(_list_head, _type, _iter)
Iterate over the contents of a list.
Definition dlist.h:94
static void * fr_dlist_remove(fr_dlist_head_t *list_head, void *ptr)
Remove an item from the list.
Definition dlist.h:638
static bool fr_dlist_empty(fr_dlist_head_t const *list_head)
Check whether a list has any items.
Definition dlist.h:501
static void * fr_dlist_pop_tail(fr_dlist_head_t *list_head)
Remove the tail item in a list.
Definition dlist.h:688
static int fr_dlist_insert_tail(fr_dlist_head_t *list_head, void *ptr)
Insert an item into the tail of a list.
Definition dlist.h:378
static void * fr_dlist_next(fr_dlist_head_t const *list_head, void const *ptr)
Get the next item in a list.
Definition dlist.h:555
Head of a doubly linked list.
Definition dlist.h:51
Entry in a doubly linked list.
Definition dlist.h:41
talloc_free(reap)
@ FR_TYPE_STRUCT
like TLV, but without T or L, and fixed-width children
@ FR_TYPE_BOOL
A truth value.
int8_t fr_pair_cmp_by_parent_num(void const *a, void const *b)
Order attributes by their parent(s), attribute number, and tag.
Definition pair.c:1921
fr_pair_t * fr_pair_find_by_da(fr_pair_list_t const *list, fr_pair_t const *prev, fr_dict_attr_t const *da)
Find the first pair with a matching da.
Definition pair.c:693
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:1345
int fr_pair_delete_by_da(fr_pair_list_t *list, fr_dict_attr_t const *da)
Delete matching pairs from the specified list.
Definition pair.c:1689
void fr_pair_list_init(fr_pair_list_t *list)
Initialise a pair list header.
Definition pair.c:46
bool fr_pair_immutable(fr_pair_t const *vp)
Definition pair.c:2280
int fr_pair_delete(fr_pair_list_t *list, fr_pair_t *vp)
Remove fr_pair_t from a list and free.
Definition pair.c:1826
int fr_pair_insert_after(fr_pair_list_t *list, fr_pair_t *pos, fr_pair_t *to_add)
Add a VP after another VP.
Definition pair.c:1373
int fr_pair_prepend(fr_pair_list_t *list, fr_pair_t *to_add)
Add a VP to the start of the list.
Definition pair.c:1314
#define fr_assert(_expr)
Definition rad_assert.h:38
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
#define talloc_zero_pooled_object(_ctx, _type, _num_subobjects, _total_subobjects_size)
Definition talloc.h:177
char const * fr_tokens[T_TOKEN_LAST]
Definition token.c:78
enum fr_token fr_token_t
@ T_OP_SUB_EQ
Definition token.h:70
@ T_OP_AND_EQ
Definition token.h:74
@ T_OP_SET
Definition token.h:84
@ T_OP_ADD_EQ
Definition token.h:69
@ T_OP_LE
Definition token.h:100
@ T_OP_GE
Definition token.h:98
@ T_OP_GT
Definition token.h:99
@ T_OP_OR_EQ
Definition token.h:73
@ T_OP_LT
Definition token.h:101
@ T_OP_PREPEND
Definition token.h:85
static fr_event_list_t * el
int fr_edit_list_apply_list_assignment(fr_edit_list_t *el, fr_pair_t *dst, fr_token_t op, fr_pair_list_t *src, bool copy)
Apply operators to lists.
Definition edit.c:1487
int fr_edit_list_pair_delete(fr_edit_list_t *el, fr_pair_list_t *list, fr_pair_t *vp)
Delete a VP.
Definition edit.c:575
int fr_edit_list_replace_pair_value(fr_edit_list_t *el, fr_pair_t *vp, fr_value_box_t *box)
Write a new value to the fr_value_box_t.
Definition edit.c:635
static int edit_undo(fr_edit_t *e)
Undo one particular edit.
Definition edit.c:133
void fr_edit_list_commit(fr_edit_list_t *el)
Commit an edit list.
Definition edit.c:841
fr_edit_op_t op
edit operation to perform
Definition edit.c:80
fr_edit_list_t * parent
for nested transactions
Definition edit.c:116
int fr_edit_list_insert_pair_after(fr_edit_list_t *el, fr_pair_list_t *list, fr_pair_t *pos, fr_pair_t *vp)
Insert a new VP after an existing one.
Definition edit.c:562
#define NEXT_B
Definition edit.c:1019
fr_dlist_t entry
linked list of edits
Definition edit.c:81
int fr_edit_list_save_pair_value(fr_edit_list_t *el, fr_pair_t *vp)
Record the value of a leaf fr_value_box_t.
Definition edit.c:622
#define COPY(_x)
Definition edit.c:1009
static int _edit_list_destructor(fr_edit_list_t *el)
Finalize the edits when we destroy the edit list.
Definition edit.c:736
static int list_merge_lhs(fr_edit_list_t *el, fr_pair_t *dst, fr_pair_list_t *src, bool copy)
A MERGE B.
Definition edit.c:1164
static int edit_record(fr_edit_list_t *el, fr_edit_op_t op, fr_pair_t *vp, fr_pair_list_t *list, fr_pair_t *ref)
Record one particular edit.
Definition edit.c:247
static int fr_edit_list_delete_list(fr_edit_list_t *el, fr_pair_list_t *list, fr_pair_list_t *to_remove)
Removes elements matching a list.
Definition edit.c:931
fr_dlist_head_t undo
Definition edit.c:106
int fr_edit_list_insert_list_after(fr_edit_list_t *el, fr_pair_list_t *list, fr_pair_t *pos, fr_pair_list_t *to_insert)
Notes.
Definition edit.c:888
fr_dlist_t entry
Definition edit.c:121
void fr_edit_list_abort(fr_edit_list_t *el)
Abort the entries in an edit list.
Definition edit.c:194
int fr_edit_list_replace_pair(fr_edit_list_t *el, fr_pair_list_t *list, fr_pair_t *to_replace, fr_pair_t *vp)
Replace a pair with another one.
Definition edit.c:658
static int list_intersection(fr_edit_list_t *el, fr_pair_t *dst, fr_pair_list_t *src)
A INTERSECTION B.
Definition edit.c:1375
static int list_merge_rhs(fr_edit_list_t *el, fr_pair_t *dst, fr_pair_list_t *src, bool copy)
A MERGE B.
Definition edit.c:1268
fr_pair_t * vp
pair edited, deleted, or inserted
Definition edit.c:83
fr_dlist_head_t ignore
lists to ignore
Definition edit.c:108
fr_pair_list_t deleted_pairs
Definition edit.c:114
static bool fr_edit_list_empty(fr_edit_list_t *el)
Definition edit.c:126
fr_pair_list_t * list
list to ignore (never dereferenced)
Definition edit.c:122
fr_edit_op_t
Definition edit.c:55
@ FR_EDIT_CHILD
child edit list
Definition edit.c:61
@ FR_EDIT_INVALID
Definition edit.c:56
@ FR_EDIT_CLEAR
clear the children of a structural entry.
Definition edit.c:59
@ FR_EDIT_DELETE
delete a VP
Definition edit.c:57
@ FR_EDIT_VALUE
edit a VP in place
Definition edit.c:58
@ FR_EDIT_INSERT
insert a VP into a list, after another one.
Definition edit.c:60
#define NEXT_A
Definition edit.c:1018
int fr_edit_list_free_pair_children(fr_edit_list_t *el, fr_pair_t *vp)
Free children of a structural pair.
Definition edit.c:709
int fr_edit_list_apply_pair_assignment(fr_edit_list_t *el, fr_pair_t *vp, fr_token_t op, fr_value_box_t const *in)
Apply operators to pairs.
Definition edit.c:976
static int list_union(fr_edit_list_t *el, fr_pair_t *dst, fr_pair_list_t *src, bool copy)
A UNION B.
Definition edit.c:1025
int fr_edit_list_pair_delete_by_da(fr_edit_list_t *el, fr_pair_list_t *list, fr_dict_attr_t const *da)
Delete VPs with a matching da.
Definition edit.c:594
fr_edit_t * e
so we don't have to loop over parent edits on abort
Definition edit.c:117
fr_edit_list_t * fr_edit_list_alloc(TALLOC_CTX *ctx, int hint, fr_edit_list_t *parent)
Allocate an edit list.
Definition edit.c:790
Track a series of edits.
Definition edit.c:102
Track one particular edit.
Definition edit.c:79
Structures and prototypes for editing lists.
#define fr_edit_list_insert_pair_tail(_el, _list, _vp)
Definition edit.h:51
#define fr_edit_list_insert_list_tail(_el, _list, _to_insert)
Definition edit.h:74
#define fr_edit_list_insert_pair_before(_el, _list, _pos, _vp)
Definition edit.h:45
#define fr_edit_list_insert_list_head(_el, _list, _to_insert)
Definition edit.h:72
bool fr_pair_list_empty(fr_pair_list_t const *list)
Is a valuepair list empty.
#define PAIR_VERIFY(_x)
Definition pair.h:191
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.
fr_pair_t * fr_pair_list_next(fr_pair_list_t const *list, fr_pair_t const *item))
Get the next item in a valuepair list after a specific entry.
Definition pair_inline.c:70
fr_pair_t * fr_pair_remove(fr_pair_list_t *list, fr_pair_t *vp)
Remove fr_pair_t from a list without freeing.
Definition pair_inline.c:94
#define fr_pair_list_foreach(_list_head, _iter)
Iterate over the contents of a fr_pair_list_t.
Definition pair.h:261
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.
#define PAIR_LIST_VERIFY(_x)
Definition pair.h:194
fr_pair_t * fr_pair_list_prev(fr_pair_list_t const *list, fr_pair_t const *item))
Get the previous item in a valuepair list before a specific entry.
Definition pair_inline.c:83
fr_pair_t * fr_pair_list_head(fr_pair_list_t const *list)
Get the head of a valuepair list.
Definition pair_inline.c:43
static fr_slen_t parent
Definition pair.h:851
#define fr_strerror_printf(_fmt,...)
Log to thread local error buffer.
Definition strerror.h:64
#define fr_type_is_structural(_x)
Definition types.h:371
#define fr_type_is_fixed_size(_x)
Definition types.h:366
#define fr_type_is_leaf(_x)
Definition types.h:372
static char const * fr_type_to_str(fr_type_t type)
Return a static string containing the type name.
Definition types.h:433
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.
Definition value.c:3352
int8_t fr_value_box_cmp(fr_value_box_t const *a, fr_value_box_t const *b)
Compare two values.
Definition value.c:676
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.
Definition value.c:3740
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.
Definition value.c:929
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.
Definition value.c:3834
void fr_value_box_clear_value(fr_value_box_t *data)
Clear/free any existing value.
Definition value.c:3681
void fr_value_box_clear(fr_value_box_t *data)
Clear/free any existing value and metadata.
Definition value.c:3723
static fr_slen_t data
Definition value.h:1265
#define fr_value_box_init_null(_vb)
Initialise an empty/null box that will be filled later.
Definition value.h:593