libyang 5.0.8
libyang is YANG data modelling language parser and toolkit written (and providing API) in C.
Loading...
Searching...
No Matches
union.c
Go to the documentation of this file.
1
15
16#define _GNU_SOURCE /* strdup */
17
18#include "plugins_types.h"
19
20#include <assert.h>
21#include <stdint.h>
22#include <stdlib.h>
23#include <string.h>
24
25#include "libyang.h"
26
27/* additional internal headers for some useful simple macros */
28#include "compat.h"
29#include "ly_common.h"
30#include "lyb.h"
31#include "plugins_internal.h" /* LY_TYPE_*_STR */
32
44
45static void lyplg_type_free_union(const struct ly_ctx *ctx, struct lyd_value *value);
46
50#define LYPLG_UNION_TYPE_IDX_SIZE 1
51
62static LY_ERR
63union_subvalue_assignment(const void *value, uint32_t value_size_bits, void **original, uint32_t *orig_size_bits,
64 uint32_t *options)
65{
66 LY_ERR ret = LY_SUCCESS;
67
68 if (*options & LYPLG_TYPE_STORE_DYNAMIC) {
69 /* The allocated value is stored and spend. */
70 *original = (void *)value;
71 *options &= ~LYPLG_TYPE_STORE_DYNAMIC;
72 } else if (value_size_bits) {
73 /* Make a copy of the value. */
74 *original = calloc(1, LYPLG_BITS2BYTES(value_size_bits));
75 LY_CHECK_ERR_RET(!*original, ret = LY_EMEM, ret);
76 memcpy(*original, value, LYPLG_BITS2BYTES(value_size_bits));
77 } else {
78 /* Empty value. */
79 *original = strdup("");
80 LY_CHECK_ERR_RET(!*original, ret = LY_EMEM, ret);
81 }
82 *orig_size_bits = value_size_bits;
83
84 return ret;
85}
86
96static LY_ERR
97lyb_union_validate(const void *lyb_data, uint32_t lyb_data_size_bits, const struct lysc_type_union *type_u,
98 struct ly_err_item **err)
99{
100 uint32_t type_idx = 0;
101
102 /* basic validation */
103 if (lyb_data_size_bits < LYPLG_UNION_TYPE_IDX_SIZE * 8) {
104 return ly_err_new(err, LY_EVALID, LYVE_DATA, NULL, NULL, "Invalid LYB union value size %" PRIu32
105 " b (expected at least %" PRIu8 " b).", lyb_data_size_bits, LYPLG_UNION_TYPE_IDX_SIZE * 8);
106 }
107
108 /* get index in correct byte order */
109 memcpy(&type_idx, lyb_data, LYPLG_UNION_TYPE_IDX_SIZE);
110 type_idx = le32toh(type_idx);
111 if (type_idx >= LY_ARRAY_COUNT(type_u->types)) {
112 return ly_err_new(err, LY_EVALID, LYVE_DATA, NULL, NULL,
113 "Invalid LYB union type index %" PRIu32 " (type count %" LY_PRI_ARRAY_COUNT_TYPE ").",
114 type_idx, LY_ARRAY_COUNT(type_u->types));
115 }
116
117 return LY_SUCCESS;
118}
119
130static void
131lyb_parse_union(const void *lyb_data, uint32_t lyb_data_size_bits, uint32_t *type_idx, const void **lyb_value,
132 uint32_t *lyb_value_size_bits)
133{
134 uint32_t num = 0;
135
136 assert(lyb_data && !(lyb_value && !lyb_value_size_bits));
137
138 if (type_idx) {
139 memcpy(&num, lyb_data, LYPLG_UNION_TYPE_IDX_SIZE);
140 num = le32toh(num);
141
142 *type_idx = num;
143 }
144
145 if (lyb_value && lyb_value_size_bits && lyb_data_size_bits) {
146 /* Get lyb_value and its length. */
147 if (lyb_data_size_bits == LYPLG_UNION_TYPE_IDX_SIZE * 8) {
148 *lyb_value_size_bits = 0;
149 *lyb_value = "";
150 } else {
151 *lyb_value_size_bits = lyb_data_size_bits - LYPLG_UNION_TYPE_IDX_SIZE * 8;
152 *lyb_value = (char *)lyb_data + LYPLG_UNION_TYPE_IDX_SIZE;
153 }
154 }
155}
156
168static LY_ERR
169union_update_lref_err(struct ly_err_item *err, const struct lysc_type *type, const void *value, uint32_t value_size_bits)
170{
171 const struct lysc_type_leafref *lref;
172 char *valstr = NULL;
173 int r;
174
175 if (!err || (type->basetype != LY_TYPE_LEAFREF)) {
176 /* nothing to do */
177 return LY_SUCCESS;
178 }
179
180 lref = (const struct lysc_type_leafref *)type;
181
182 /* update error-app-tag */
183 free(err->apptag);
184 err->apptag = strdup("instance-required");
185 LY_CHECK_ERR_RET(!err->apptag, LOGMEM(NULL), LY_EMEM);
186
187 valstr = strndup((const char *)value, value_size_bits / 8);
188 LY_CHECK_ERR_RET(!valstr, LOGMEM(NULL), LY_EMEM);
189
190 /* update error-message */
191 free(err->msg);
192 r = asprintf(&err->msg, LY_ERRMSG_NOLREF_VAL, valstr, lyxp_get_expr(lref->path));
193 free(valstr);
194 LY_CHECK_ERR_RET(r == -1, LOGMEM(NULL), LY_EMEM);
195
196 return LY_SUCCESS;
197}
198
214static LY_ERR
215union_store_type(const struct ly_ctx *ctx, struct lysc_type_union *type_u, uint32_t type_idx, struct lyd_value_union *subvalue,
216 uint32_t options, ly_bool validate_tree, const struct lyd_node *ctx_node, const struct lyd_node *tree,
217 struct lys_glob_unres *unres, struct ly_err_item **err)
218{
219 LY_ERR rc = LY_SUCCESS;
220 struct lysc_type *type = type_u->types[type_idx];
221 const void *value = NULL;
222 ly_bool dynamic = 0;
223 LY_VALUE_FORMAT format;
224 void *prefix_data;
225 uint32_t value_size_bits = 0, opts = 0, ti;
226 struct lyplg_type *type_plg;
227
228 *err = NULL;
229
230 if (subvalue->format == LY_VALUE_LYB) {
231 lyb_parse_union(subvalue->original, subvalue->orig_size_bits, &ti, &value, &value_size_bits);
232 if (ti != type_idx) {
233 /* value of another type, first store the value properly and then use its JSON value for parsing */
234 type_plg = LYSC_GET_TYPE_PLG(type_u->types[ti]->plugin_ref);
235 rc = type_plg->store(ctx, type_u->types[ti], value, value_size_bits, LYPLG_TYPE_STORE_ONLY,
236 subvalue->format, subvalue->prefix_data, subvalue->hints, subvalue->ctx_node, &subvalue->value,
237 unres, err);
238 if ((rc != LY_SUCCESS) && (rc != LY_EINCOMPLETE)) {
239 /* clear any leftover/freed garbage */
240 memset(&subvalue->value, 0, sizeof subvalue->value);
241
242 /* if this is a leafref, lets make sure we propagate the appropriate error, and not a type validation failure */
243 union_update_lref_err(*err, type_u->types[ti], value, value_size_bits);
244 goto cleanup;
245 }
246
247 assert(subvalue->value.realtype);
248 value = LYSC_GET_TYPE_PLG(subvalue->value.realtype->plugin_ref)->print(ctx, &subvalue->value,
249 LY_VALUE_JSON, NULL, &dynamic, &value_size_bits);
250 assert(!(value_size_bits % 8));
251
252 /* to avoid leaks, free subvalue->value, but we need the value, which may be stored there */
253 if (!dynamic) {
254 value = strndup(value, value_size_bits / 8);
255 dynamic = 1;
256 }
257 type_plg->free(ctx, &subvalue->value);
258
259 format = LY_VALUE_JSON;
260 prefix_data = NULL;
261 } else {
262 format = subvalue->format;
263 prefix_data = subvalue->prefix_data;
264 }
265 } else {
266 value = subvalue->original;
267 value_size_bits = subvalue->orig_size_bits;
268 format = subvalue->format;
269 prefix_data = subvalue->prefix_data;
270 }
271
272 if (options & LYPLG_TYPE_STORE_ONLY) {
273 opts |= LYPLG_TYPE_STORE_ONLY;
274 }
275
276 type_plg = LYSC_GET_TYPE_PLG(type->plugin_ref);
277
278 rc = type_plg->store(ctx, type, value, value_size_bits, opts, format, prefix_data, subvalue->hints,
279 subvalue->ctx_node, &subvalue->value, unres, err);
280 if ((rc != LY_SUCCESS) && (rc != LY_EINCOMPLETE)) {
281 /* clear any leftover/freed garbage */
282 memset(&subvalue->value, 0, sizeof subvalue->value);
283
284 /* if this is a leafref, lets make sure we propagate the appropriate error, and not a type validation failure */
285 union_update_lref_err(*err, type, value, value_size_bits);
286 goto cleanup;
287 }
288
289 if (validate_tree && type_plg->validate_tree) {
290 /* we need the value validated in the data tree */
291 rc = type_plg->validate_tree(ctx, type, ctx_node, tree, &subvalue->value, err);
292 if (rc) {
293 /* validate failed, we need to free the stored value */
294 type_plg->free(ctx, &subvalue->value);
295 goto cleanup;
296 }
297 }
298
299cleanup:
300 if (dynamic) {
301 free((void *)value);
302 }
303 return rc;
304}
305
321static LY_ERR
322union_find_type(const struct ly_ctx *ctx, struct lysc_type_union *type_u, struct lyd_value_union *subvalue,
323 uint32_t options, ly_bool validate_tree, const struct lyd_node *ctx_node, const struct lyd_node *tree,
324 uint32_t *type_idx, struct lys_glob_unres *unres, struct ly_err_item **err)
325{
326 LY_ERR ret = LY_SUCCESS;
328 struct ly_err_item **errs = NULL, *e;
329 uint32_t *prev_lo, temp_lo = 0;
330 char *msg = NULL, *err_app_tag = NULL;
331 int msg_len = 0;
332 ly_bool use_err_app_tag = 0;
333 struct lyplg_type *type_plg;
334
335 *err = NULL;
336
337 /* alloc errors */
338 errs = calloc(LY_ARRAY_COUNT(type_u->types), sizeof *errs);
339 LY_CHECK_RET(!errs, LY_EMEM);
340
341 /* turn logging temporarily off */
342 prev_lo = ly_temp_log_options(&temp_lo);
343
344 /* use the first usable subtype to store the value */
345 for (u = 0; u < LY_ARRAY_COUNT(type_u->types); ++u) {
346 ret = union_store_type(ctx, type_u, u, subvalue, options, validate_tree, ctx_node, tree, unres, &e);
347 if ((ret == LY_SUCCESS) || (ret == LY_EINCOMPLETE)) {
348 break;
349 }
350
351 errs[u] = e;
352 }
353
354 if (u == LY_ARRAY_COUNT(type_u->types)) {
355 /* create the full error */
356 if (subvalue->format == LY_VALUE_LYB) {
357 msg_len = asprintf(&msg, "Invalid LYB union value - no matching subtype found:\n");
358 } else {
359 msg_len = asprintf(&msg, "Invalid union value \"%.*s\" - no matching subtype found:\n",
360 (int)subvalue->orig_size_bits / 8, (char *)subvalue->original);
361 }
362 if (msg_len == -1) {
363 LY_CHECK_ERR_GOTO(!errs, ret = LY_EMEM, cleanup);
364 /* for further actions in function msg_len is just 0 */
365 msg_len = 0;
366 }
367 for (u = 0; u < LY_ARRAY_COUNT(type_u->types); ++u) {
368 if (!errs[u]) {
369 /* no error for some reason */
370 continue;
371 }
372
373 /* use an app-tag if all the types set it or set none */
374 if (errs[u]->apptag) {
375 if (!err_app_tag) {
376 err_app_tag = strdup(errs[u]->apptag);
377 LY_CHECK_ERR_GOTO(!err_app_tag, ret = LY_EMEM, cleanup);
378 use_err_app_tag = 1;
379 } else if (strcmp(errs[u]->apptag, err_app_tag)) {
380 use_err_app_tag = 0;
381 }
382 }
383
384 type_plg = LYSC_GET_TYPE_PLG(type_u->types[u]->plugin_ref);
385
386 msg = ly_realloc(msg, msg_len + 4 + strlen(type_plg->id) + 2 + strlen(errs[u]->msg) + 2);
387 LY_CHECK_ERR_GOTO(!msg, ret = LY_EMEM, cleanup);
388 msg_len += sprintf(msg + msg_len, " %s: %s\n", type_plg->id, errs[u]->msg);
389 }
390
391 if (!use_err_app_tag) {
392 free(err_app_tag);
393 err_app_tag = NULL;
394 }
395 ret = ly_err_new(err, LY_EVALID, LYVE_DATA, NULL, err_app_tag, "%s", msg);
396 err_app_tag = NULL;
397 } else if (type_idx) {
398 *type_idx = u;
399 }
400
401cleanup:
402 for (u = 0; u < LY_ARRAY_COUNT(type_u->types); ++u) {
403 ly_err_free(errs[u]);
404 }
405 free(errs);
406 free(msg);
407 free(err_app_tag);
408 ly_temp_log_options(prev_lo);
409 return ret;
410}
411
426static LY_ERR
427lyb_fill_subvalue(const struct ly_ctx *ctx, struct lysc_type_union *type_u, const void *lyb_data, uint32_t lyb_data_size_bits,
428 void *prefix_data, struct lyd_value_union *subvalue, uint32_t *options, struct lys_glob_unres *unres,
429 struct ly_err_item **err)
430{
431 LY_ERR ret;
432 uint32_t lyb_value_size_bits = 0, type_idx;
433 const void *lyb_value = NULL;
434
435 ret = lyb_union_validate(lyb_data, lyb_data_size_bits, type_u, err);
436 LY_CHECK_RET(ret);
437
438 /* parse lyb_data and set the lyb_value and lyb_value_size_bits */
439 lyb_parse_union(lyb_data, lyb_data_size_bits, &type_idx, &lyb_value, &lyb_value_size_bits);
440
441 /* store lyb_data to subvalue */
442 ret = union_subvalue_assignment(lyb_data, lyb_data_size_bits, &subvalue->original, &subvalue->orig_size_bits, options);
443 LY_CHECK_RET(ret);
444
445 if (lyb_value) {
446 /* resolve prefix_data and set format */
447 ret = lyplg_type_prefix_data_new(ctx, lyb_value, LYPLG_BITS2BYTES(lyb_value_size_bits), LY_VALUE_LYB,
448 prefix_data, &subvalue->format, &subvalue->prefix_data);
449 LY_CHECK_RET(ret);
450 assert(subvalue->format == LY_VALUE_LYB);
451 } else {
452 /* lyb_parse_union() did not find lyb_value, just set format */
453 subvalue->format = LY_VALUE_LYB;
454 }
455
456 /* use the specific type to store the value */
457 ret = union_store_type(ctx, type_u, type_idx, subvalue, *options, 0, NULL, NULL, unres, err);
458
459 return ret;
460}
461
462static LY_ERR
463lyplg_type_store_union(const struct ly_ctx *ctx, const struct lysc_type *type, const void *value, uint32_t value_size_bits,
464 uint32_t options, LY_VALUE_FORMAT format, void *prefix_data, uint32_t hints, const struct lysc_node *ctx_node,
465 struct lyd_value *storage, struct lys_glob_unres *unres, struct ly_err_item **err)
466{
467 LY_ERR ret = LY_SUCCESS, r;
468 struct lysc_type_union *type_u = (struct lysc_type_union *)type;
469 struct lyd_value_union *subvalue;
470
471 *err = NULL;
472
473 /* init storage */
474 memset(storage, 0, sizeof *storage);
475 LYPLG_TYPE_VAL_INLINE_PREPARE(storage, subvalue);
476 LY_CHECK_ERR_GOTO(!subvalue, ret = LY_EMEM, cleanup);
477 storage->realtype = type;
478 subvalue->hints = hints;
479 subvalue->ctx_node = ctx_node;
480
481 if (format == LY_VALUE_LYB) {
482 ret = lyb_fill_subvalue(ctx, type_u, value, value_size_bits, prefix_data, subvalue, &options, unres, err);
483 LY_CHECK_GOTO((ret != LY_SUCCESS) && (ret != LY_EINCOMPLETE), cleanup);
484 } else {
485 /* store value to subvalue */
486 ret = union_subvalue_assignment(value, value_size_bits, &subvalue->original, &subvalue->orig_size_bits, &options);
487 LY_CHECK_GOTO(ret, cleanup);
488
489 /* store format-specific data for later prefix resolution */
491 &subvalue->format, &subvalue->prefix_data);
492 LY_CHECK_GOTO(ret, cleanup);
493
494 /* use the first usable and valid subtype to store the value */
495 ret = union_find_type(ctx, type_u, subvalue, options & ~LYPLG_TYPE_STORE_ONLY, 0, NULL, NULL, NULL, unres, err);
496 if (ret && (ret != LY_EINCOMPLETE) && (options & LYPLG_TYPE_STORE_ONLY)) {
497 /* we tried to find the actual type by validating the value but no type matched, so try to find any type */
498 ly_err_free(*err);
499 *err = NULL;
500 ret = union_find_type(ctx, type_u, subvalue, options, 0, NULL, NULL, NULL, unres, err);
501 }
502 LY_CHECK_GOTO(ret && (ret != LY_EINCOMPLETE), cleanup);
503 }
504
505 /* store canonical value, if any (use the specific type value) */
506 r = lydict_insert(ctx, subvalue->value._canonical, 0, &storage->_canonical);
507 LY_CHECK_ERR_GOTO(r, ret = r, cleanup);
508
509cleanup:
510 if (options & LYPLG_TYPE_STORE_DYNAMIC) {
511 free((void *)value);
512 }
513
514 if ((ret != LY_SUCCESS) && (ret != LY_EINCOMPLETE)) {
515 lyplg_type_free_union(ctx, storage);
516 }
517 return ret;
518}
519
520static LY_ERR
521lyplg_type_validate_tree_union(const struct ly_ctx *ctx, const struct lysc_type *type, const struct lyd_node *ctx_node,
522 const struct lyd_node *tree, struct lyd_value *storage, struct ly_err_item **err)
523{
524 LY_ERR rc = LY_SUCCESS;
525 struct lysc_type_union *type_u = (struct lysc_type_union *)type;
526 struct lyd_value_union *subvalue = storage->subvalue;
527 struct lyd_value orig = {0};
528 uint32_t type_idx;
529 ly_bool validated = 0;
530 struct lyplg_type *subvalue_type_plg;
531
532 *err = NULL;
533
534 /* because of types that do not store their own type as realtype (leafref), we are not able to call their
535 * validate callback (there is no way to get the type) but even if possible, the value may be invalid
536 * for the type, so we may have to perform union value storing again from scratch, but keep a value backup */
537 subvalue_type_plg = LYSC_GET_TYPE_PLG(subvalue->value.realtype->plugin_ref);
538 LY_CHECK_RET(subvalue_type_plg->duplicate(ctx, &subvalue->value, &orig));
539 subvalue_type_plg->free(ctx, &subvalue->value);
540
541 if (subvalue->format == LY_VALUE_LYB) {
542 /* use the specific type to store and validate the value */
543 lyb_parse_union(subvalue->original, 0, &type_idx, NULL, NULL);
544
545 if (union_store_type(ctx, type_u, type_idx, subvalue, 0, 1, ctx_node, tree, NULL, err)) {
546 /* validation failed, we need to try storing the value again */
547 ly_err_free(*err);
548 *err = NULL;
549 } else {
550 validated = 1;
551 }
552 }
553
554 if (!validated) {
555 /* use the first usable subtype to store and validate the value */
556 rc = union_find_type(ctx, type_u, subvalue, 0, 1, ctx_node, tree, NULL, NULL, err);
557 if (rc) {
558 /* validation failed, restore the previous value */
559 subvalue->value = orig;
560 return rc;
561 }
562 }
563
564 /* update the canonical value, if any generated */
565 lydict_remove(ctx, storage->_canonical);
566 LY_CHECK_RET(lydict_insert(ctx, subvalue->value._canonical, 0, &storage->_canonical));
567
568 /* free backup value */
569 LYSC_GET_TYPE_PLG(orig.realtype->plugin_ref)->free(ctx, &orig);
570 return LY_SUCCESS;
571}
572
573static LY_ERR
574lyplg_type_compare_union(const struct ly_ctx *ctx, const struct lyd_value *val1, const struct lyd_value *val2)
575{
576 if (val1->subvalue->value.realtype != val2->subvalue->value.realtype) {
577 return LY_ENOT;
578 }
579 return LYSC_GET_TYPE_PLG(val1->subvalue->value.realtype->plugin_ref)->compare(ctx,
580 &val1->subvalue->value, &val2->subvalue->value);
581}
582
583static int
584lyplg_type_sort_union(const struct ly_ctx *ctx, const struct lyd_value *val1, const struct lyd_value *val2)
585{
586 int rc;
588 struct lysc_type **types, *type;
589
590 if (val1->subvalue->value.realtype == val2->subvalue->value.realtype) {
591 return LYSC_GET_TYPE_PLG(val1->subvalue->value.realtype->plugin_ref)->sort(ctx,
592 &val1->subvalue->value, &val2->subvalue->value);
593 }
594
595 /* compare according to the order of types */
596 rc = 0;
597 types = ((struct lysc_type_union *)val1->realtype)->types;
598 LY_ARRAY_FOR(types, u) {
599 if (types[u]->basetype == LY_TYPE_LEAFREF) {
600 type = ((struct lysc_type_leafref *)types[u])->realtype;
601 } else {
602 type = types[u];
603 }
604
605 if (type == val1->subvalue->value.realtype) {
606 rc = 1;
607 break;
608 } else if (type == val2->subvalue->value.realtype) {
609 rc = -1;
610 break;
611 }
612 }
613 assert(rc);
614
615 return rc;
616}
617
630static const void *
631lyb_union_print(const struct ly_ctx *ctx, struct lysc_type_union *type_u, struct lyd_value_union *subvalue,
632 void *prefix_data, uint32_t *value_size_bits)
633{
634 void *ret = NULL;
635 LY_ERR r;
636 struct ly_err_item *err;
637 uint32_t num = 0, pval_size_bits, type_idx = 0;
638 ly_bool dynamic;
639 void *pval;
640
641 /* learn the type index, must succeed because have been called before */
642 if (!ctx) {
643 assert(subvalue->ctx_node);
644 ctx = subvalue->ctx_node->module->ctx;
645 }
646 LYSC_GET_TYPE_PLG(subvalue->value.realtype->plugin_ref)->free(ctx, &subvalue->value);
647 r = union_find_type(ctx, type_u, subvalue, 0, 0, NULL, NULL, &type_idx, NULL, &err);
649 LY_CHECK_RET((r != LY_SUCCESS) && (r != LY_EINCOMPLETE), NULL);
650
651 /* print subvalue in LYB format */
652 pval = (void *)LYSC_GET_TYPE_PLG(subvalue->value.realtype->plugin_ref)->print(NULL, &subvalue->value, LY_VALUE_LYB,
653 prefix_data, &dynamic, &pval_size_bits);
654 LY_CHECK_RET(!pval, NULL);
655
656 /* create LYB data */
657 *value_size_bits = LYPLG_UNION_TYPE_IDX_SIZE * 8 + pval_size_bits;
658 ret = malloc(LYPLG_BITS2BYTES(*value_size_bits));
659 LY_CHECK_RET(!ret, NULL);
660
661 num = htole32(type_idx);
662 memcpy(ret, &num, LYPLG_UNION_TYPE_IDX_SIZE);
663 memcpy((char *)ret + LYPLG_UNION_TYPE_IDX_SIZE, pval, LYPLG_BITS2BYTES(pval_size_bits));
664
665 if (dynamic) {
666 free(pval);
667 }
668
669 return ret;
670}
671
672static const void *
673lyplg_type_print_union(const struct ly_ctx *ctx, const struct lyd_value *value, LY_VALUE_FORMAT format,
674 void *prefix_data, ly_bool *dynamic, uint32_t *value_size_bits)
675{
676 const void *ret;
677 struct lyd_value_union *subvalue = value->subvalue;
678 struct lysc_type_union *type_u = (struct lysc_type_union *)value->realtype;
679 uint32_t lyb_data_size_bits = 0;
680
681 if ((format == LY_VALUE_LYB) && (subvalue->format == LY_VALUE_LYB)) {
682 /* The return value is already ready. */
683 *dynamic = 0;
684 if (value_size_bits) {
685 *value_size_bits = subvalue->orig_size_bits;
686 }
687 return subvalue->original;
688 } else if ((format == LY_VALUE_LYB) && (subvalue->format != LY_VALUE_LYB)) {
689 /* The return LYB data must be created. */
690 *dynamic = 1;
691 ret = lyb_union_print(ctx, type_u, subvalue, prefix_data, &lyb_data_size_bits);
692 if (value_size_bits) {
693 *value_size_bits = lyb_data_size_bits;
694 }
695 return ret;
696 }
697
698 assert(format != LY_VALUE_LYB);
699 ret = (void *)LYSC_GET_TYPE_PLG(subvalue->value.realtype->plugin_ref)->print(ctx, &subvalue->value,
700 format, prefix_data, dynamic, value_size_bits);
701 if (!value->_canonical && (format == LY_VALUE_CANON)) {
702 /* the canonical value is supposed to be stored now */
703 lydict_insert(ctx, subvalue->value._canonical, 0, (const char **)&value->_canonical);
704 }
705
706 return ret;
707}
708
709static LY_ERR
710lyplg_type_dup_union(const struct ly_ctx *ctx, const struct lyd_value *original, struct lyd_value *dup)
711{
712 LY_ERR ret = LY_SUCCESS;
713 struct lyd_value_union *orig_val = original->subvalue, *dup_val;
714
715 /* init dup value */
716 memset(dup, 0, sizeof *dup);
717 dup->realtype = original->realtype;
718
719 ret = lydict_insert(ctx, original->_canonical, 0, &dup->_canonical);
720 LY_CHECK_GOTO(ret, cleanup);
721
722 dup_val = calloc(1, sizeof *dup_val);
723 LY_CHECK_ERR_GOTO(!dup_val, LOGMEM(ctx); ret = LY_EMEM, cleanup);
724 dup->subvalue = dup_val;
725
726 ret = LYSC_GET_TYPE_PLG(orig_val->value.realtype->plugin_ref)->duplicate(ctx, &orig_val->value, &dup_val->value);
727 LY_CHECK_GOTO(ret, cleanup);
728
729 if (orig_val->orig_size_bits) {
730 dup_val->original = calloc(1, LYPLG_BITS2BYTES(orig_val->orig_size_bits));
731 LY_CHECK_ERR_GOTO(!dup_val->original, LOGMEM(ctx); ret = LY_EMEM, cleanup);
732 memcpy(dup_val->original, orig_val->original, LYPLG_BITS2BYTES(orig_val->orig_size_bits));
733 } else {
734 dup_val->original = strdup("");
735 LY_CHECK_ERR_GOTO(!dup_val->original, LOGMEM(ctx); ret = LY_EMEM, cleanup);
736 }
737 dup_val->orig_size_bits = orig_val->orig_size_bits;
738
739 dup_val->format = orig_val->format;
740 dup_val->ctx_node = orig_val->ctx_node;
741 dup_val->hints = orig_val->hints;
742 ret = lyplg_type_prefix_data_dup(ctx, orig_val->format, orig_val->prefix_data, &dup_val->prefix_data);
743 LY_CHECK_GOTO(ret, cleanup);
744
745cleanup:
746 if (ret) {
747 lyplg_type_free_union(ctx, dup);
748 }
749 return ret;
750}
751
752static void
753lyplg_type_free_union(const struct ly_ctx *ctx, struct lyd_value *value)
754{
755 struct lyd_value_union *val;
756
758 value->_canonical = NULL;
759 LYD_VALUE_GET(value, val);
760 if (val) {
761 if (val->value.realtype) {
762 LYSC_GET_TYPE_PLG(val->value.realtype->plugin_ref)->free(ctx, &val->value);
763 }
765 free(val->original);
766
768 }
769}
770
779 {
780 .module = "",
781 .revision = NULL,
782 .name = LY_TYPE_UNION_STR,
783
784 .plugin.id = "ly2 union",
785 .plugin.lyb_size = lyplg_type_lyb_size_variable_bits,
786 .plugin.store = lyplg_type_store_union,
787 .plugin.validate_value = NULL,
788 .plugin.validate_tree = lyplg_type_validate_tree_union,
789 .plugin.compare = lyplg_type_compare_union,
790 .plugin.sort = lyplg_type_sort_union,
791 .plugin.print = lyplg_type_print_union,
792 .plugin.duplicate = lyplg_type_dup_union,
793 .plugin.free = lyplg_type_free_union,
794 },
795 {0}
796};
libyang context handler.
LIBYANG_API_DECL LY_ERR lydict_insert(const struct ly_ctx *ctx, const char *value, size_t len, const char **str_p)
Insert string into dictionary. If the string is already present, only a reference counter is incremen...
LIBYANG_API_DECL LY_ERR lydict_remove(const struct ly_ctx *ctx, const char *value)
Remove specified string from the dictionary. It decrement reference counter for the string and if it ...
LY_ERR err
Definition log.h:299
char * apptag
Definition log.h:305
char * msg
Definition log.h:301
LY_ERR
libyang's error codes returned by the libyang functions.
Definition log.h:252
@ LYVE_DATA
Definition log.h:289
@ LY_EMEM
Definition log.h:254
@ LY_ENOT
Definition log.h:266
@ LY_EVALID
Definition log.h:260
@ LY_SUCCESS
Definition log.h:253
@ LY_EINCOMPLETE
Definition log.h:262
Libyang full error structure.
Definition log.h:297
LIBYANG_API_DECL uint32_t * ly_temp_log_options(uint32_t *opts)
Set temporary thread-safe (thread-specific) logger options overwriting those set by ly_log_options().
lyplg_type_store_clb store
const char * id
lyplg_type_dup_clb duplicate
lyplg_type_validate_tree_clb validate_tree
lyplg_type_free_clb free
#define LYPLG_TYPE_VAL_INLINE_PREPARE(storage, type_val)
Prepare value memory for storing a specific type value, may be allocated dynamically.
LIBYANG_API_DECL LY_ERR LIBYANG_API_DECL void ly_err_free(void *ptr)
Destructor for the error records created with ly_err_new().
#define LYPLG_TYPE_VAL_INLINE_DESTROY(type_val)
Destroy a prepared value.
LIBYANG_API_DECL LY_ERR lyplg_type_prefix_data_new(const struct ly_ctx *ctx, const void *value, uint32_t value_size, LY_VALUE_FORMAT format, const void *prefix_data, LY_VALUE_FORMAT *format_p, void **prefix_data_p)
Store used prefixes in a string into an internal libyang structure used in lyd_value.
LIBYANG_API_DECL LY_ERR ly_err_new(struct ly_err_item **err, LY_ERR ecode, LY_VECODE vecode, char *data_path, char *apptag, const char *err_format,...) _FORMAT_PRINTF(6
Create and fill error structure.
LIBYANG_API_DECL LY_ERR lyplg_type_prefix_data_dup(const struct ly_ctx *ctx, LY_VALUE_FORMAT format, const void *orig, void **dup)
Duplicate prefix data.
#define LYPLG_BITS2BYTES(bits)
Convert bits to bytes.
LIBYANG_API_DECL void lyplg_type_prefix_data_free(LY_VALUE_FORMAT format, void *prefix_data)
Free internal prefix data.
Hold type-specific functions for various operations with the data values.
LIBYANG_API_DECL void lyplg_type_lyb_size_variable_bits(const struct lysc_type *type, enum lyplg_lyb_size_type *size_type, uint32_t *fixed_size_bits)
Implementation of lyplg_type_lyb_size_clb for a type with variable length in bits.
#define LYPLG_TYPE_STORE_DYNAMIC
#define LYPLG_TYPE_STORE_ONLY
LY_DATA_TYPE basetype
struct lyxp_expr * path
struct lys_module * module
uintptr_t plugin_ref
struct lysc_type * realtype
struct lysc_type ** types
struct ly_ctx * ctx
LY_DATA_TYPE basetype
LIBYANG_API_DECL const char * lyxp_get_expr(const struct lyxp_expr *path)
Getter for original XPath expression from a parsed expression.
Compiled YANG data node.
#define LY_ARRAY_COUNT(ARRAY)
Get the number of records in the ARRAY.
Definition tree.h:148
#define LY_ARRAY_FOR(ARRAY,...)
Sized-array iterator (for-loop).
Definition tree.h:167
LY_VALUE_FORMAT
All kinds of supported value formats and prefix mappings to modules.
Definition tree.h:234
#define LY_PRI_ARRAY_COUNT_TYPE
Printing format specifier macro for LY_ARRAY_SIZE_TYPE values.
Definition tree.h:109
#define LY_ARRAY_COUNT_TYPE
Type (i.e. size) of the sized array's size counter.
Definition tree.h:104
@ LY_TYPE_LEAFREF
Definition tree.h:217
@ LY_VALUE_JSON
Definition tree.h:239
@ LY_VALUE_CANON
Definition tree.h:235
@ LY_VALUE_LYB
Definition tree.h:240
The main libyang public header.
uint8_t ly_bool
Type to indicate boolean value.
Definition log.h:36
API for (user) types plugins.
const struct lysc_type * realtype
Definition tree_data.h:544
void * prefix_data
Definition tree_data.h:604
uint32_t orig_size_bits
Definition tree_data.h:599
uint32_t hints
Definition tree_data.h:600
#define LYD_VALUE_GET(value, type_val)
Get the value in format specific to the type.
Definition tree_data.h:583
struct lyd_value value
Definition tree_data.h:596
const char * _canonical
Definition tree_data.h:541
LY_VALUE_FORMAT format
Definition tree_data.h:601
const struct lysc_node * ctx_node
Definition tree_data.h:605
Generic structure for a data node.
Definition tree_data.h:800
YANG data representation.
Definition tree_data.h:540
Special lyd_value structure for built-in union values.
Definition tree_data.h:595
void * ly_realloc(void *ptr, size_t size)
Wrapper for realloc() call. The only difference is that if it fails to allocate the requested memory,...
#define LOGMEM(CTX)
Definition tree_edit.h:22
#define LYPLG_UNION_TYPE_IDX_SIZE
Size in bytes of the used type index in the LYB Binary Format.
Definition union.c:50
const struct lyplg_type_record plugins_union[]
Plugin information for union type implementation.
Definition union.c:778