INTEGER.c
Go to the documentation of this file.
1 /*
2  * Copyright (c) 2003-2019 Lev Walkin <vlm@lionet.info>.
3  * All rights reserved.
4  * Redistribution and modifications are permitted subject to BSD license.
5  */
8 #include <errno.h>
9 #include <inttypes.h>
10 
11 /*
12  * INTEGER basic type description.
13  */
15  (ASN_TAG_CLASS_UNIVERSAL | (2 << 2))
16 };
19 #if !defined(ASN_DISABLE_PRINT_SUPPORT)
21 #else
22  0,
23 #endif /* !defined(ASN_DISABLE_PRINT_SUPPORT) */
26 #if !defined(ASN_DISABLE_BER_SUPPORT)
29 #else
30  0,
31  0,
32 #endif /* !defined(ASN_DISABLE_BER_SUPPORT) */
33 #if !defined(ASN_DISABLE_XER_SUPPORT)
36 #else
37  0,
38  0,
39 #endif /* !defined(ASN_DISABLE_XER_SUPPORT) */
40 #if !defined(ASN_DISABLE_JER_SUPPORT)
43 #else
44  0,
45  0,
46 #endif /* !defined(ASN_DISABLE_JER_SUPPORT) */
47 #if !defined(ASN_DISABLE_OER_SUPPORT)
48  INTEGER_decode_oer, /* OER decoder */
49  INTEGER_encode_oer, /* Canonical OER encoder */
50 #else
51  0,
52  0,
53 #endif /* !defined(ASN_DISABLE_OER_SUPPORT) */
54 #if !defined(ASN_DISABLE_UPER_SUPPORT)
55  INTEGER_decode_uper, /* Unaligned PER decoder */
56  INTEGER_encode_uper, /* Unaligned PER encoder */
57 #else
58  0,
59  0,
60 #endif /* !defined(ASN_DISABLE_UPER_SUPPORT) */
61 #if !defined(ASN_DISABLE_APER_SUPPORT)
62  INTEGER_decode_aper, /* Aligned PER decoder */
63  INTEGER_encode_aper, /* Aligned PER encoder */
64 #else
65  0,
66  0,
67 #endif /* !defined(ASN_DISABLE_APER_SUPPORT) */
68 #if !defined(ASN_DISABLE_RFILL_SUPPORT)
70 #else
71  0,
72 #endif /* !defined(ASN_DISABLE_RFILL_SUPPORT) */
73 0 /* Use generic outmost tag fetcher */
74 };
76  "INTEGER",
77  "INTEGER",
80  sizeof(asn_DEF_INTEGER_tags) / sizeof(asn_DEF_INTEGER_tags[0]),
81  asn_DEF_INTEGER_tags, /* Same as above */
82  sizeof(asn_DEF_INTEGER_tags) / sizeof(asn_DEF_INTEGER_tags[0]),
83  {
84 #if !defined(ASN_DISABLE_OER_SUPPORT)
85  0,
86 #endif /* !defined(ASN_DISABLE_OER_SUPPORT) */
87 #if !defined(ASN_DISABLE_UPER_SUPPORT) || !defined(ASN_DISABLE_APER_SUPPORT)
88  0,
89 #endif /* !defined(ASN_DISABLE_UPER_SUPPORT) || !defined(ASN_DISABLE_APER_SUPPORT) */
90 #if !defined(ASN_DISABLE_JER_SUPPORT)
91  0,
92 #endif /* !defined(ASN_DISABLE_JER_SUPPORT) */
94  },
95  0, 0, /* No members */
96  0 /* No specifics */
97 };
98 
99 /*
100  * INTEGER specific human-readable output.
101  */
102 ssize_t
103 INTEGER__dump(const asn_TYPE_descriptor_t *td, const INTEGER_t *st, asn_app_consume_bytes_f *cb, void *app_key, int plainOrXEROrJER) {
104  const asn_INTEGER_specifics_t *specs =
105  (const asn_INTEGER_specifics_t *)td->specifics;
106  char scratch[32];
107  uint8_t *buf = st->buf;
108  uint8_t *buf_end = st->buf + st->size;
109  intmax_t value = 0;
110  ssize_t wrote = 0;
111  char *p = NULL;
112  int ret = -1;
113 
114  if(specs && specs->field_unsigned)
115  ret = asn_INTEGER2umax(st, (uintmax_t *)&value);
116  else
117  ret = asn_INTEGER2imax(st, &value);
118 
119  /* Simple case: the integer size is small */
120  if(ret == 0) {
121  const asn_INTEGER_enum_map_t *el;
122  el = (value >= 0 || !specs || !specs->field_unsigned)
123  ? INTEGER_map_value2enum(specs, value) : 0;
124  if(el) {
125  if(plainOrXEROrJER == 0)
126  return asn__format_to_callback(cb, app_key,
127  "%" ASN_PRIdMAX " (%s)", value, el->enum_name);
128  else if (plainOrXEROrJER == 1)
129  return asn__format_to_callback(cb, app_key,
130  "<%s/>", el->enum_name);
131  else if (plainOrXEROrJER == 2)
132  return asn__format_to_callback(cb, app_key,
133  "\"%s\"", el->enum_name);
134  } else if(plainOrXEROrJER && specs && specs->strict_enumeration) {
135  ASN_DEBUG("ASN.1 forbids dealing with "
136  "unknown value of ENUMERATED type");
137  errno = EPERM;
138  return -1;
139  } else {
140  return asn__format_to_callback(cb, app_key,
141  (specs && specs->field_unsigned)
142  ? "%" ASN_PRIuMAX
143  : "%" ASN_PRIdMAX,
144  value);
145  }
146  } else if(plainOrXEROrJER && specs && specs->strict_enumeration) {
147  /*
148  * Here and earlier, we cannot encode the ENUMERATED values
149  * if there is no corresponding identifier.
150  */
151  ASN_DEBUG("ASN.1 forbids dealing with "
152  "unknown value of ENUMERATED type");
153  errno = EPERM;
154  return -1;
155  }
156 
157  /* Output in the long xx:yy:zz... format */
158  /* TODO: replace with generic algorithm (Knuth TAOCP Vol 2, 4.3.1) */
159  for(p = scratch; buf < buf_end; buf++) {
160  const char * const h2c = "0123456789ABCDEF";
161  if((p - scratch) >= (ssize_t)(sizeof(scratch) - 4)) {
162  /* Flush buffer */
163  if(cb(scratch, p - scratch, app_key) < 0)
164  return -1;
165  wrote += p - scratch;
166  p = scratch;
167  }
168  *p++ = h2c[*buf >> 4];
169  *p++ = h2c[*buf & 0x0F];
170  *p++ = 0x3a; /* ":" */
171  }
172  if(p != scratch)
173  p--; /* Remove the last ":" */
174 
175  wrote += p - scratch;
176  return (cb(scratch, p - scratch, app_key) < 0) ? -1 : wrote;
177 }
178 
179 static int
180 INTEGER__compar_value2enum(const void *kp, const void *am) {
181  long a = *(const long *)kp;
182  const asn_INTEGER_enum_map_t *el = (const asn_INTEGER_enum_map_t *)am;
183  long b = el->nat_value;
184  if(a < b) return -1;
185  else if(a == b) return 0;
186  else return 1;
187 }
188 
191  int count = specs ? specs->map_count : 0;
192  if(!count) return 0;
193  return (asn_INTEGER_enum_map_t *)bsearch(&value, specs->value2enum,
194  count, sizeof(specs->value2enum[0]),
196 }
197 
198 static intmax_t
199 asn__integer_convert(const uint8_t *b, const uint8_t *end) {
200  uintmax_t value;
201 
202  /* Perform the sign initialization */
203  /* Actually value = -(*b >> 7); gains nothing, yet unreadable! */
204  if((*b >> 7)) {
205  value = (uintmax_t)(-1);
206  } else {
207  value = 0;
208  }
209 
210  /* Conversion engine */
211  for(; b < end; b++) {
212  value = (value << 8) | *b;
213  }
214 
215  return value;
216 }
217 
218 int
219 asn_INTEGER2imax(const INTEGER_t *iptr, intmax_t *lptr) {
220  uint8_t *b, *end;
221  size_t size = 0;
222 
223  /* Sanity checking */
224  if(!iptr || !iptr->buf || !lptr) {
225  errno = EINVAL;
226  return -1;
227  }
228 
229  /* Cache the begin/end of the buffer */
230  b = iptr->buf; /* Start of the INTEGER buffer */
231  size = iptr->size;
232  end = b + size; /* Where to stop */
233 
234  if(size > sizeof(intmax_t)) {
235  uint8_t *end1 = end - 1;
236  /*
237  * Slightly more advanced processing,
238  * able to process INTEGERs with >sizeof(intmax_t) bytes
239  * when the actual value is small, e.g. for intmax_t == int32_t
240  * (0x0000000000abcdef INTEGER would yield a fine 0x00abcdef int32_t)
241  */
242  /* Skip out the insignificant leading bytes */
243  for(; b < end1; b++) {
244  switch(*b) {
245  case 0x00: if((b[1] & 0x80) == 0) continue; break;
246  case 0xff: if((b[1] & 0x80) != 0) continue; break;
247  }
248  break;
249  }
250 
251  size = end - b;
252  if(size > sizeof(intmax_t)) {
253  /* Still cannot fit the sizeof(intmax_t) */
254  errno = ERANGE;
255  return -1;
256  }
257  }
258 
259  /* Shortcut processing of a corner case */
260  if(end == b) {
261  *lptr = 0;
262  return 0;
263  }
264 
265  *lptr = asn__integer_convert(b, end);
266  return 0;
267 }
268 
269 /* FIXME: negative INTEGER values are silently interpreted as large unsigned ones. */
270 int
271 asn_INTEGER2umax(const INTEGER_t *iptr, uintmax_t *lptr) {
272  uint8_t *b, *end;
273  uintmax_t value;
274  size_t size;
275 
276  if(!iptr || !iptr->buf || !lptr) {
277  errno = EINVAL;
278  return -1;
279  }
280 
281  b = iptr->buf;
282  size = iptr->size;
283  end = b + size;
284 
285  /* If all extra leading bytes are zeroes, ignore them */
286  for(; size > sizeof(value); b++, size--) {
287  if(*b) {
288  /* Value won't fit into uintmax_t */
289  errno = ERANGE;
290  return -1;
291  }
292  }
293 
294  /* Conversion engine */
295  for(value = 0; b < end; b++)
296  value = (value << 8) | *b;
297 
298  *lptr = value;
299  return 0;
300 }
301 
302 int
303 asn_umax2INTEGER(INTEGER_t *st, uintmax_t value) {
304  uint8_t *buf;
305  uint8_t *end;
306  uint8_t *b;
307  int shr;
308 
309  if(value <= ((~(uintmax_t)0) >> 1)) {
310  return asn_imax2INTEGER(st, value);
311  }
312 
313  buf = (uint8_t *)MALLOC(1 + sizeof(value));
314  if(!buf) return -1;
315 
316  end = buf + (sizeof(value) + 1);
317  buf[0] = 0; /* INTEGERs are signed. 0-byte indicates positive. */
318  for(b = buf + 1, shr = (sizeof(value) - 1) * 8; b < end; shr -= 8, b++)
319  *b = (uint8_t)(value >> shr);
320 
321  if(st->buf) FREEMEM(st->buf);
322  st->buf = buf;
323  st->size = 1 + sizeof(value);
324 
325  return 0;
326 }
327 
328 int
329 asn_imax2INTEGER(INTEGER_t *st, intmax_t value) {
330  uint8_t *buf, *bp;
331  volatile uint8_t *p;
332  volatile uint8_t *pstart;
333  volatile uint8_t *pend1;
334  int littleEndian = 1; /* Run-time detection */
335  volatile int add;
336 
337  if(!st) {
338  errno = EINVAL;
339  return -1;
340  }
341 
342  buf = (uint8_t *)(long *)MALLOC(sizeof(value));
343  if(!buf) return -1;
344 
345  if(*(char *)&littleEndian) {
346  pstart = (uint8_t *)&value + sizeof(value) - 1;
347  pend1 = (uint8_t *)&value;
348  add = -1;
349  } else {
350  pstart = (uint8_t *)&value;
351  pend1 = pstart + sizeof(value) - 1;
352  add = 1;
353  }
354 
355  /*
356  * If the contents octet consists of more than one octet,
357  * then bits of the first octet and bit 8 of the second octet:
358  * a) shall not all be ones; and
359  * b) shall not all be zero.
360  */
361  for(p = pstart; p != pend1; p += add) {
362  switch(*p) {
363  case 0x00: if((*(p+add) & 0x80) == 0)
364  continue;
365  break;
366  case 0xff: if((*(p+add) & 0x80))
367  continue;
368  break;
369  }
370  break;
371  }
372  /* Copy the integer body */
373  for(bp = buf, pend1 += add; p != pend1; p += add)
374  *bp++ = *p;
375 
376  if(st->buf) FREEMEM(st->buf);
377  st->buf = buf;
378  st->size = bp - buf;
379 
380  return 0;
381 }
382 
383 int
384 asn_INTEGER2long(const INTEGER_t *iptr, long *l) {
385  intmax_t v;
386  if(asn_INTEGER2imax(iptr, &v) == 0) {
387  if(v < LONG_MIN || v > LONG_MAX) {
388  errno = ERANGE;
389  return -1;
390  }
391  *l = v;
392  return 0;
393  } else {
394  return -1;
395  }
396 }
397 
398 int
399 asn_INTEGER2ulong(const INTEGER_t *iptr, unsigned long *l) {
400  uintmax_t v;
401  if(asn_INTEGER2umax(iptr, &v) == 0) {
402  if(v > ULONG_MAX) {
403  errno = ERANGE;
404  return -1;
405  }
406  *l = v;
407  return 0;
408  } else {
409  return -1;
410  }
411 }
412 
413 int
414 asn_long2INTEGER(INTEGER_t *st, long value) {
415  return asn_imax2INTEGER(st, value);
416 }
417 
418 int
419 asn_ulong2INTEGER(INTEGER_t *st, unsigned long value) {
420  return asn_imax2INTEGER(st, value);
421 }
422 
423 int asn_INTEGER2int64(const INTEGER_t *st, int64_t *value) {
424  intmax_t v;
425  if(asn_INTEGER2imax(st, &v) == 0) {
426  if(v < INT64_MIN || v > INT64_MAX) {
427  errno = ERANGE;
428  return -1;
429  }
430  *value = v;
431  return 0;
432  } else {
433  return -1;
434  }
435 }
436 
437 int asn_INTEGER2uint64(const INTEGER_t *st, uint64_t *value) {
438  uintmax_t v;
439  if(asn_INTEGER2umax(st, &v) == 0) {
440  if(v > UINT64_MAX) {
441  errno = ERANGE;
442  return -1;
443  }
444  *value = v;
445  return 0;
446  } else {
447  return -1;
448  }
449 }
450 
451 int
452 asn_uint642INTEGER(INTEGER_t *st, uint64_t value) {
453  uint8_t *buf;
454  uint8_t *end;
455  uint8_t *b;
456  int shr;
457 
458  if(value <= INT64_MAX)
459  return asn_int642INTEGER(st, value);
460 
461  buf = (uint8_t *)MALLOC(1 + sizeof(value));
462  if(!buf) return -1;
463 
464  end = buf + (sizeof(value) + 1);
465  buf[0] = 0;
466  for(b = buf + 1, shr = (sizeof(value)-1)*8; b < end; shr -= 8, b++)
467  *b = (uint8_t)(value >> shr);
468 
469  if(st->buf) FREEMEM(st->buf);
470  st->buf = buf;
471  st->size = 1 + sizeof(value);
472 
473  return 0;
474 }
475 
476 int
477 asn_int642INTEGER(INTEGER_t *st, int64_t value) {
478  uint8_t *buf, *bp;
479  volatile uint8_t *p;
480  volatile uint8_t *pstart;
481  volatile uint8_t *pend1;
482  int littleEndian = 1; /* Run-time detection */
483  volatile int add;
484 
485  if(!st) {
486  errno = EINVAL;
487  return -1;
488  }
489 
490  buf = (uint8_t *)MALLOC(sizeof(value));
491  if(!buf) return -1;
492 
493  if(*(char *)&littleEndian) {
494  pstart = (uint8_t *)&value + sizeof(value) - 1;
495  pend1 = (uint8_t *)&value;
496  add = -1;
497  } else {
498  pstart = (uint8_t *)&value;
499  pend1 = pstart + sizeof(value) - 1;
500  add = 1;
501  }
502 
503  /*
504  * If the contents octet consists of more than one octet,
505  * then bits of the first octet and bit 8 of the second octet:
506  * a) shall not all be ones; and
507  * b) shall not all be zero.
508  */
509  for(p = pstart; p != pend1; p += add) {
510  switch(*p) {
511  case 0x00: if((*(p+add) & 0x80) == 0)
512  continue;
513  break;
514  case 0xff: if((*(p+add) & 0x80))
515  continue;
516  break;
517  }
518  break;
519  }
520  /* Copy the integer body */
521  for(pstart = p, bp = buf, pend1 += add; p != pend1; p += add)
522  *bp++ = *p;
523 
524  if(st->buf) FREEMEM(st->buf);
525  st->buf = buf;
526  st->size = bp - buf;
527 
528  return 0;
529 }
530 
531 /*
532  * Parse the number in the given string until the given *end position,
533  * returning the position after the last parsed character back using the
534  * same (*end) pointer.
535  * WARNING: This behavior is different from the standard strtol/strtoimax(3).
536  */
538 asn_strtoimax_lim(const char *str, const char **end, intmax_t *intp) {
539  int sign = 1;
540  intmax_t value;
541 
542  const intmax_t asn1_intmax_max = ((~(uintmax_t)0) >> 1);
543  const intmax_t upper_boundary = asn1_intmax_max / 10;
544  intmax_t last_digit_max = asn1_intmax_max % 10;
545 
546  if(str >= *end) return ASN_STRTOX_ERROR_INVAL;
547 
548  switch(*str) {
549  case '-':
550  last_digit_max++;
551  sign = -1;
552  /* FALL THROUGH */
553  case '+':
554  str++;
555  if(str >= *end) {
556  *end = str;
557  return ASN_STRTOX_EXPECT_MORE;
558  }
559  }
560 
561  for(value = 0; str < (*end); str++) {
562  if(*str >= 0x30 && *str <= 0x39) {
563  int d = *str - '0';
564  if(value < upper_boundary) {
565  value = value * 10 + d;
566  } else if(value == upper_boundary) {
567  if(d <= last_digit_max) {
568  if(sign > 0) {
569  value = value * 10 + d;
570  } else {
571  sign = 1;
572  value = -value * 10 - d;
573  }
574  str += 1;
575  if(str < *end) {
576  // If digits continue, we're guaranteed out of range.
577  *end = str;
578  if(*str >= 0x30 && *str <= 0x39) {
579  return ASN_STRTOX_ERROR_RANGE;
580  } else {
581  *intp = sign * value;
582  return ASN_STRTOX_EXTRA_DATA;
583  }
584  }
585  break;
586  } else {
587  *end = str;
588  return ASN_STRTOX_ERROR_RANGE;
589  }
590  } else {
591  *end = str;
592  return ASN_STRTOX_ERROR_RANGE;
593  }
594  } else {
595  *end = str;
596  *intp = sign * value;
597  return ASN_STRTOX_EXTRA_DATA;
598  }
599  }
600 
601  *end = str;
602  *intp = sign * value;
603  return ASN_STRTOX_OK;
604 }
605 
606 /*
607  * Parse the number in the given string until the given *end position,
608  * returning the position after the last parsed character back using the
609  * same (*end) pointer.
610  * WARNING: This behavior is different from the standard strtoul/strtoumax(3).
611  */
613 asn_strtoumax_lim(const char *str, const char **end, uintmax_t *uintp) {
614  uintmax_t value;
615 
616  const uintmax_t asn1_uintmax_max = ((~(uintmax_t)0));
617  const uintmax_t upper_boundary = asn1_uintmax_max / 10;
618  uintmax_t last_digit_max = asn1_uintmax_max % 10;
619 
620  if(str >= *end) return ASN_STRTOX_ERROR_INVAL;
621 
622  switch(*str) {
623  case '-':
624  return ASN_STRTOX_ERROR_INVAL;
625  case '+':
626  str++;
627  if(str >= *end) {
628  *end = str;
629  return ASN_STRTOX_EXPECT_MORE;
630  }
631  }
632 
633  for(value = 0; str < (*end); str++) {
634  if(*str >= 0x30 && *str <= 0x39) {
635  unsigned int d = *str - '0';
636  if(value < upper_boundary) {
637  value = value * 10 + d;
638  } else if(value == upper_boundary) {
639  if(d <= last_digit_max) {
640  value = value * 10 + d;
641  str += 1;
642  if(str < *end) {
643  // If digits continue, we're guaranteed out of range.
644  *end = str;
645  if(*str >= 0x30 && *str <= 0x39) {
646  return ASN_STRTOX_ERROR_RANGE;
647  } else {
648  *uintp = value;
649  return ASN_STRTOX_EXTRA_DATA;
650  }
651  }
652  break;
653  } else {
654  *end = str;
655  return ASN_STRTOX_ERROR_RANGE;
656  }
657  } else {
658  *end = str;
659  return ASN_STRTOX_ERROR_RANGE;
660  }
661  } else {
662  *end = str;
663  *uintp = value;
664  return ASN_STRTOX_EXTRA_DATA;
665  }
666  }
667 
668  *end = str;
669  *uintp = value;
670  return ASN_STRTOX_OK;
671 }
672 
674 asn_strtol_lim(const char *str, const char **end, long *lp) {
675  intmax_t value;
676  switch(asn_strtoimax_lim(str, end, &value)) {
678  return ASN_STRTOX_ERROR_RANGE;
680  return ASN_STRTOX_ERROR_INVAL;
682  return ASN_STRTOX_EXPECT_MORE;
683  case ASN_STRTOX_OK:
684  if(value >= LONG_MIN && value <= LONG_MAX) {
685  *lp = value;
686  return ASN_STRTOX_OK;
687  } else {
688  return ASN_STRTOX_ERROR_RANGE;
689  }
691  if(value >= LONG_MIN && value <= LONG_MAX) {
692  *lp = value;
693  return ASN_STRTOX_EXTRA_DATA;
694  } else {
695  return ASN_STRTOX_ERROR_RANGE;
696  }
697  }
698 
699  assert(!"Unreachable");
700  return ASN_STRTOX_ERROR_INVAL;
701 }
702 
704 asn_strtoul_lim(const char *str, const char **end, unsigned long *ulp) {
705  uintmax_t value;
706  switch(asn_strtoumax_lim(str, end, &value)) {
708  return ASN_STRTOX_ERROR_RANGE;
710  return ASN_STRTOX_ERROR_INVAL;
712  return ASN_STRTOX_EXPECT_MORE;
713  case ASN_STRTOX_OK:
714  if(value <= ULONG_MAX) {
715  *ulp = value;
716  return ASN_STRTOX_OK;
717  } else {
718  return ASN_STRTOX_ERROR_RANGE;
719  }
721  if(value <= ULONG_MAX) {
722  *ulp = value;
723  return ASN_STRTOX_EXTRA_DATA;
724  } else {
725  return ASN_STRTOX_ERROR_RANGE;
726  }
727  }
728 
729  assert(!"Unreachable");
730  return ASN_STRTOX_ERROR_INVAL;
731 }
732 
733 int
734 INTEGER_compare(const asn_TYPE_descriptor_t *td, const void *aptr,
735  const void *bptr) {
736  const INTEGER_t *a = aptr;
737  const INTEGER_t *b = bptr;
738 
739  (void)td;
740 
741  if(a && b) {
742  if(a->size && b->size) {
743  int sign_a = (a->buf[0] & 0x80) ? -1 : 1;
744  int sign_b = (b->buf[0] & 0x80) ? -1 : 1;
745 
746  if(sign_a < sign_b) return -1;
747  if(sign_a > sign_b) return 1;
748 
749  /* The shortest integer wins, unless comparing negatives */
750  if(a->size < b->size) {
751  return -1 * sign_a;
752  } else if(a->size > b->size) {
753  return 1 * sign_b;
754  }
755 
756  return sign_a * memcmp(a->buf, b->buf, a->size);
757  } else if(a->size) {
758  int sign = (a->buf[0] & 0x80) ? -1 : 1;
759  return (1) * sign;
760  } else if(b->size) {
761  int sign = (a->buf[0] & 0x80) ? -1 : 1;
762  return (-1) * sign;
763  } else {
764  return 0;
765  }
766  } else if(!a && !b) {
767  return 0;
768  } else if(!a) {
769  return -1;
770  } else {
771  return 1;
772  }
773 
774 }
775 
776 int
777 INTEGER_copy(const asn_TYPE_descriptor_t *td, void **aptr,
778  const void *bptr) {
779  (void)td;
780  INTEGER_t *a = *aptr;
781  const INTEGER_t *b = bptr;
782 
783  if(!b) {
784  if(a) {
785  FREEMEM(a->buf);
786  FREEMEM(a);
787  *aptr = 0;
788  }
789  return 0;
790  }
791 
792  if(!a) {
793  a = *aptr = CALLOC(1, sizeof(*a));
794  if(!a) return -1;
795  }
796 
797  if(b->size) {
798  uint8_t* buf = MALLOC(b->size);
799  if(!buf) return -1;
800  memcpy(buf, b->buf, b->size);
801  FREEMEM(a->buf);
802  a->buf = buf;
803  a->size = b->size;
804  } else {
805  FREEMEM(a->buf);
806  a->buf = 0;
807  a->size = 0;
808  }
809 
810  return 0;
811 }
INTEGER_compare
int INTEGER_compare(const asn_TYPE_descriptor_t *td, const void *aptr, const void *bptr)
Definition: INTEGER.c:734
ASN__PRIMITIVE_TYPE_s
Definition: asn_codecs_prim.h:14
asn_ulong2INTEGER
int asn_ulong2INTEGER(INTEGER_t *st, unsigned long value)
Definition: INTEGER.c:419
INTEGER_copy
int INTEGER_copy(const asn_TYPE_descriptor_t *td, void **aptr, const void *bptr)
Definition: INTEGER.c:777
INTEGER_free
#define INTEGER_free
Definition: INTEGER.h:43
asn_TYPE_operation_s
Definition: constr_TYPE.h:184
asn_strtoimax_lim
enum asn_strtox_result_e asn_strtoimax_lim(const char *str, const char **end, intmax_t *intp)
Definition: INTEGER.c:538
ASN_STRTOX_EXPECT_MORE
@ ASN_STRTOX_EXPECT_MORE
Definition: INTEGER.h:120
asn_DEF_INTEGER
asn_TYPE_descriptor_t asn_DEF_INTEGER
Definition: INTEGER.c:75
asn_imax2INTEGER
int asn_imax2INTEGER(INTEGER_t *st, intmax_t value)
Definition: INTEGER.c:329
ASN_PRIdMAX
#define ASN_PRIdMAX
Definition: asn_system.h:181
asn_strtol_lim
enum asn_strtox_result_e asn_strtol_lim(const char *str, const char **end, long *lp)
Definition: INTEGER.c:674
ber_tlv_tag_t
unsigned ber_tlv_tag_t
Definition: ber_tlv_tag.h:18
ASN__PRIMITIVE_TYPE_s::size
size_t size
Definition: asn_codecs_prim.h:16
asn_umax2INTEGER
int asn_umax2INTEGER(INTEGER_t *st, uintmax_t value)
Definition: INTEGER.c:303
INTEGER_map_value2enum
const asn_INTEGER_enum_map_t * INTEGER_map_value2enum(const asn_INTEGER_specifics_t *specs, long value)
Definition: INTEGER.c:190
ASN_TAG_CLASS_UNIVERSAL
@ ASN_TAG_CLASS_UNIVERSAL
Definition: ber_tlv_tag.h:13
ber_decode_primitive
ber_type_decoder_f ber_decode_primitive
Definition: asn_codecs_prim.h:22
asn_strtoumax_lim
enum asn_strtox_result_e asn_strtoumax_lim(const char *str, const char **end, uintmax_t *uintp)
Definition: INTEGER.c:613
asn_INTEGER2long
int asn_INTEGER2long(const INTEGER_t *iptr, long *l)
Definition: INTEGER.c:384
CALLOC
#define CALLOC(nmemb, size)
Definition: asn_internal.h:37
ASN_STRTOX_OK
@ ASN_STRTOX_OK
Definition: INTEGER.h:121
asn_INTEGER2ulong
int asn_INTEGER2ulong(const INTEGER_t *iptr, unsigned long *l)
Definition: INTEGER.c:399
INTEGER_encode_jer
jer_type_encoder_f INTEGER_encode_jer
Definition: INTEGER.h:66
FREEMEM
#define FREEMEM(ptr)
Definition: asn_internal.h:40
asn_INTEGER_specifics_s::strict_enumeration
int strict_enumeration
Definition: INTEGER.h:33
INTEGER_random_fill
asn_random_fill_f INTEGER_random_fill
Definition: INTEGER.h:84
asn_int642INTEGER
int asn_int642INTEGER(INTEGER_t *st, int64_t value)
Definition: INTEGER.c:477
INTEGER_decode_uper
per_type_decoder_f INTEGER_decode_uper
Definition: INTEGER.h:75
asn_TYPE_descriptor_s
Definition: constr_TYPE.h:224
INTEGER_decode_xer
xer_type_decoder_f INTEGER_decode_xer
Definition: INTEGER.h:60
asn_uint642INTEGER
int asn_uint642INTEGER(INTEGER_t *st, uint64_t value)
Definition: INTEGER.c:452
ASN__PRIMITIVE_TYPE_s::buf
uint8_t * buf
Definition: asn_codecs_prim.h:15
asn_TYPE_descriptor_s::specifics
const void * specifics
Definition: constr_TYPE.h:259
asn_OP_INTEGER
asn_TYPE_operation_t asn_OP_INTEGER
Definition: INTEGER.c:17
asn_INTEGER_specifics_s::map_count
int map_count
Definition: INTEGER.h:31
INTEGER_encode_xer
xer_type_encoder_f INTEGER_encode_xer
Definition: INTEGER.h:61
INTEGER_encode_uper
per_type_encoder_f INTEGER_encode_uper
Definition: INTEGER.h:76
INTEGER__dump
ssize_t INTEGER__dump(const asn_TYPE_descriptor_t *td, const INTEGER_t *st, asn_app_consume_bytes_f *cb, void *app_key, int plainOrXEROrJER)
Definition: INTEGER.c:103
asn_INTEGER_specifics_s
Definition: INTEGER.h:28
asn_INTEGER2imax
int asn_INTEGER2imax(const INTEGER_t *iptr, intmax_t *lptr)
Definition: INTEGER.c:219
asn_generic_no_constraint
asn_constr_check_f asn_generic_no_constraint
Definition: constraints.h:51
asn_INTEGER2int64
int asn_INTEGER2int64(const INTEGER_t *st, int64_t *value)
Definition: INTEGER.c:423
asn_INTEGER_enum_map_s::nat_value
long nat_value
Definition: INTEGER.h:22
INTEGER_decode_jer
jer_type_decoder_f INTEGER_decode_jer
Definition: INTEGER.h:65
ASN_STRTOX_ERROR_RANGE
@ ASN_STRTOX_ERROR_RANGE
Definition: INTEGER.h:118
INTEGER_encode_der
der_type_encoder_f INTEGER_encode_der
Definition: INTEGER.h:56
ASN_STRTOX_EXTRA_DATA
@ ASN_STRTOX_EXTRA_DATA
Definition: INTEGER.h:122
asn_INTEGER_enum_map_s::enum_name
const char * enum_name
Definition: INTEGER.h:24
INTEGER.h
MALLOC
#define MALLOC(size)
Definition: asn_internal.h:38
asn_INTEGER2uint64
int asn_INTEGER2uint64(const INTEGER_t *st, uint64_t *value)
Definition: INTEGER.c:437
asn_DEF_INTEGER_tags
static const ber_tlv_tag_t asn_DEF_INTEGER_tags[]
Definition: INTEGER.c:14
INTEGER_decode_aper
per_type_decoder_f INTEGER_decode_aper
Definition: INTEGER.h:79
INTEGER_decode_oer
oer_type_decoder_f INTEGER_decode_oer
Definition: INTEGER.h:70
asn_app_consume_bytes_f
int() asn_app_consume_bytes_f(const void *buffer, size_t size, void *application_specific_key)
Definition: asn_application.h:124
asn_internal.h
asn__format_to_callback
ssize_t asn__format_to_callback(int(*cb)(const void *, size_t, void *key), void *key, const char *fmt,...)
Definition: asn_internal.c:4
asn_strtoul_lim
enum asn_strtox_result_e asn_strtoul_lim(const char *str, const char **end, unsigned long *ulp)
Definition: INTEGER.c:704
INTEGER_encode_aper
per_type_encoder_f INTEGER_encode_aper
Definition: INTEGER.h:80
asn_INTEGER2umax
int asn_INTEGER2umax(const INTEGER_t *iptr, uintmax_t *lptr)
Definition: INTEGER.c:271
INTEGER__compar_value2enum
static int INTEGER__compar_value2enum(const void *kp, const void *am)
Definition: INTEGER.c:180
ASN_PRIuMAX
#define ASN_PRIuMAX
Definition: asn_system.h:180
asn_INTEGER_specifics_s::value2enum
const asn_INTEGER_enum_map_t * value2enum
Definition: INTEGER.h:29
asn_INTEGER_specifics_s::field_unsigned
int field_unsigned
Definition: INTEGER.h:35
asn_strtox_result_e
asn_strtox_result_e
Definition: INTEGER.h:117
INTEGER_print
asn_struct_print_f INTEGER_print
Definition: INTEGER.h:46
INTEGER_encode_oer
oer_type_encoder_f INTEGER_encode_oer
Definition: INTEGER.h:71
asn_long2INTEGER
int asn_long2INTEGER(INTEGER_t *st, long value)
Definition: INTEGER.c:414
asn__integer_convert
static intmax_t asn__integer_convert(const uint8_t *b, const uint8_t *end)
Definition: INTEGER.c:199
ASN_STRTOX_ERROR_INVAL
@ ASN_STRTOX_ERROR_INVAL
Definition: INTEGER.h:119
asn_INTEGER_enum_map_s
Definition: INTEGER.h:21


etsi_its_cpm_ts_coding
Author(s): Jean-Pierre Busch , Guido Küppers , Lennart Reiher
autogenerated on Sun May 18 2025 02:22:38