Coverage Report

Created: 2025-08-29 06:57

/src/libtasn1/lib/parser_aux.c
Line
Count
Source (jump to first uncovered line)
1
/*
2
 * Copyright (C) 2000-2025 Free Software Foundation, Inc.
3
 *
4
 * This file is part of LIBTASN1.
5
 *
6
 * The LIBTASN1 library is free software; you can redistribute it
7
 * and/or modify it under the terms of the GNU Lesser General Public
8
 * License as published by the Free Software Foundation; either
9
 * version 2.1 of the License, or (at your option) any later version.
10
 *
11
 * This library is distributed in the hope that it will be useful, but
12
 * WITHOUT ANY WARRANTY; without even the implied warranty of
13
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
14
 * Lesser General Public License for more details.
15
 *
16
 * You should have received a copy of the GNU Lesser General Public
17
 * License along with this library; if not, see
18
 * <https://www.gnu.org/licenses/>.
19
 */
20
21
#include <config.h>
22
23
#include <limits.h>   /* WORD_BIT */
24
25
#include "int.h"
26
#include "parser_aux.h"
27
#include "gstr.h"
28
#include "structure.h"
29
#include "element.h"
30
#include "c-ctype.h"
31
32
char _asn1_identifierMissing[ASN1_MAX_NAME_SIZE + 1]; /* identifier name not found */
33
34
/* Return a hash of the N bytes of X using the method described by
35
   Bruno Haible in https://www.haible.de/bruno/hashfunc.html.
36
   Note that while many hash functions reduce their result via modulo
37
   to a 0..table_size-1 range, this function does not do that.
38
39
   This implementation has been changed from size_t -> unsigned int. */
40
41
#ifdef __clang__
42
__attribute__((no_sanitize ("integer")))
43
#endif
44
     _GL_ATTRIBUTE_PURE static unsigned int _asn1_hash_name (const char *x)
45
95.7k
{
46
95.7k
  const unsigned char *s = (unsigned char *) x;
47
95.7k
  unsigned h = 0;
48
49
176k
  while (*s)
50
80.3k
    h = (*s++) + ((h << 9) | (h >> (WORD_BIT - 9)));
51
52
95.7k
  return h;
53
95.7k
}
54
55
/******************************************************/
56
/* Function : _asn1_add_static_node                   */
57
/* Description: creates a new NODE_ASN element and    */
58
/* puts it in the list pointed by e_list.       */
59
/* Parameters:                                        */
60
/*   e_list: of type list_type; must be NULL initially */
61
/*   type: type of the new element (see ASN1_ETYPE_   */
62
/*         and CONST_ constants).                     */
63
/* Return: pointer to the new element.                */
64
/******************************************************/
65
asn1_node
66
_asn1_add_static_node (list_type **e_list, unsigned int type)
67
42.4k
{
68
42.4k
  list_type *p;
69
42.4k
  asn1_node punt;
70
71
42.4k
  punt = calloc (1, sizeof (struct asn1_node_st));
72
42.4k
  if (punt == NULL)
73
0
    return NULL;
74
75
42.4k
  p = malloc (sizeof (list_type));
76
42.4k
  if (p == NULL)
77
0
    {
78
0
      free (punt);
79
0
      return NULL;
80
0
    }
81
82
42.4k
  p->node = punt;
83
42.4k
  p->next = *e_list;
84
42.4k
  *e_list = p;
85
86
42.4k
  punt->type = type;
87
88
42.4k
  return punt;
89
42.4k
}
90
91
static int
92
_asn1_add_static_node2 (list_type **e_list, asn1_node node)
93
48.6k
{
94
48.6k
  list_type *p;
95
96
48.6k
  p = malloc (sizeof (list_type));
97
48.6k
  if (p == NULL)
98
0
    {
99
0
      return -1;
100
0
    }
101
102
48.6k
  p->node = node;
103
48.6k
  p->next = *e_list;
104
48.6k
  *e_list = p;
105
106
48.6k
  return 0;
107
48.6k
}
108
109
/**
110
 * asn1_find_node:
111
 * @pointer: NODE_ASN element pointer.
112
 * @name: null terminated string with the element's name to find.
113
 *
114
 * Searches for an element called @name starting from @pointer.  The
115
 * name is composed by different identifiers separated by dots.  When
116
 * *@pointer has a name, the first identifier must be the name of
117
 * *@pointer, otherwise it must be the name of one child of *@pointer.
118
 *
119
 * Returns: the search result, or %NULL if not found.
120
 **/
121
asn1_node
122
asn1_find_node (asn1_node_const pointer, const char *name)
123
10.5k
{
124
10.5k
  asn1_node_const p;
125
10.5k
  char *n_end, n[ASN1_MAX_NAME_SIZE + 1];
126
10.5k
  const char *n_start;
127
10.5k
  unsigned int nsize;
128
10.5k
  unsigned int nhash;
129
10.5k
  const struct asn1_node_array_st *numbered_children;
130
131
10.5k
  if (pointer == NULL)
132
0
    return NULL;
133
134
10.5k
  if (name == NULL)
135
0
    return NULL;
136
137
10.5k
  p = pointer;
138
10.5k
  n_start = name;
139
140
10.5k
  if (name[0] == '?' && name[1] == 'C' && p->name[0] == '?')
141
209
    {       /* ?CURRENT */
142
209
      n_start = strchr (n_start, '.');
143
209
      if (n_start)
144
208
  n_start++;
145
209
    }
146
10.3k
  else if (p->name[0] != 0)
147
9.56k
    {       /* has *pointer got a name ? */
148
9.56k
      n_end = strchr (n_start, '.');  /* search the first dot */
149
9.56k
      if (n_end)
150
9.54k
  {
151
9.54k
    nsize = n_end - n_start;
152
9.54k
    if (nsize >= sizeof (n))
153
0
      return NULL;
154
155
9.54k
    memcpy (n, n_start, nsize);
156
9.54k
    n[nsize] = 0;
157
9.54k
    n_start = n_end;
158
9.54k
    n_start++;
159
160
9.54k
    nhash = _asn1_hash_name (n);
161
9.54k
  }
162
15
      else
163
15
  {
164
15
    _asn1_str_cpy (n, sizeof (n), n_start);
165
15
    nhash = _asn1_hash_name (n);
166
167
15
    n_start = NULL;
168
15
  }
169
170
9.61k
      while (p)
171
9.56k
  {
172
9.56k
    if (nhash == p->name_hash && (!strcmp (p->name, n)))
173
9.51k
      break;
174
50
    else
175
50
      p = p->right;
176
9.56k
  }     /* while */
177
178
9.56k
      if (p == NULL)
179
50
  return NULL;
180
9.56k
    }
181
770
  else
182
770
    {       /* *pointer doesn't have a name */
183
770
      if (n_start[0] == 0)
184
1
  return (asn1_node) p;
185
770
    }
186
187
21.2k
  while (n_start)
188
11.2k
    {       /* Has the end of NAME been reached? */
189
11.2k
      n_end = strchr (n_start, '.');  /* search the next dot */
190
11.2k
      if (n_end)
191
769
  {
192
769
    nsize = n_end - n_start;
193
769
    if (nsize >= sizeof (n))
194
0
      return NULL;
195
196
769
    memcpy (n, n_start, nsize);
197
769
    n[nsize] = 0;
198
769
    n_start = n_end;
199
769
    n_start++;
200
201
769
    nhash = _asn1_hash_name (n);
202
769
  }
203
10.4k
      else
204
10.4k
  {
205
10.4k
    _asn1_str_cpy (n, sizeof (n), n_start);
206
10.4k
    nhash = _asn1_hash_name (n);
207
10.4k
    n_start = NULL;
208
10.4k
  }
209
210
11.2k
      if (p->down == NULL)
211
5
  return NULL;
212
213
11.2k
      numbered_children = &p->numbered_children;
214
11.2k
      p = p->down;
215
11.2k
      if (p == NULL)
216
0
  return NULL;
217
218
      /* The identifier "?LAST" indicates the last element
219
         in the right chain. */
220
11.2k
      if (n[0] == '?' && n[1] == 'L')  /* ?LAST */
221
1.02k
  {
222
11.6k
    while (p->right)
223
10.6k
      p = p->right;
224
1.02k
  }
225
10.2k
      else
226
10.2k
  {     /* no "?LAST" */
227
10.2k
    if (n[0] == '?' && c_isdigit (n[1]))
228
658
      {
229
658
        long position = strtol (n + 1, NULL, 10);
230
658
        if (position > 0 && position < LONG_MAX)
231
138
    p = _asn1_node_array_get (numbered_children, position - 1);
232
658
      }
233
52.8k
    while (p)
234
52.4k
      {
235
52.4k
        if (p->name_hash == nhash && !strcmp (p->name, n))
236
9.78k
    break;
237
42.6k
        else
238
42.6k
    p = p->right;
239
52.4k
      }
240
10.2k
  }
241
11.2k
      if (p == NULL)
242
429
  return NULL;
243
11.2k
    }        /* while */
244
245
10.0k
  return (asn1_node) p;
246
10.4k
}
247
248
249
/******************************************************************/
250
/* Function : _asn1_set_value                                     */
251
/* Description: sets the field VALUE in a NODE_ASN element. The   */
252
/*              previous value (if exist) will be lost            */
253
/* Parameters:                                                    */
254
/*   node: element pointer.                                       */
255
/*   value: pointer to the value that you want to set.            */
256
/*   len: character number of value.                              */
257
/* Return: pointer to the NODE_ASN element.                       */
258
/******************************************************************/
259
asn1_node
260
_asn1_set_value (asn1_node node, const void *value, unsigned int len)
261
74.4k
{
262
74.4k
  if (node == NULL)
263
0
    return node;
264
74.4k
  if (node->value)
265
2.44k
    {
266
2.44k
      if (node->value != node->small_value)
267
400
  free (node->value);
268
2.44k
      node->value = NULL;
269
2.44k
      node->value_len = 0;
270
2.44k
    }
271
272
74.4k
  if (!len)
273
0
    return node;
274
275
74.4k
  if (len < sizeof (node->small_value))
276
66.7k
    {
277
66.7k
      node->value = node->small_value;
278
66.7k
    }
279
7.71k
  else
280
7.71k
    {
281
7.71k
      node->value = malloc (len);
282
7.71k
      if (node->value == NULL)
283
0
  return NULL;
284
7.71k
    }
285
74.4k
  node->value_len = len;
286
287
74.4k
  memcpy (node->value, value, len);
288
74.4k
  return node;
289
74.4k
}
290
291
/******************************************************************/
292
/* Function : _asn1_set_value_lv                                  */
293
/* Description: sets the field VALUE in a NODE_ASN element. The   */
294
/*              previous value (if exist) will be lost. The value */
295
/*    given is stored as an length-value format (LV     */
296
/* Parameters:                                                    */
297
/*   node: element pointer.                                       */
298
/*   value: pointer to the value that you want to set.            */
299
/*   len: character number of value.                              */
300
/* Return: pointer to the NODE_ASN element.                       */
301
/******************************************************************/
302
asn1_node
303
_asn1_set_value_lv (asn1_node node, const void *value, unsigned int len)
304
0
{
305
0
  int len2;
306
0
  void *temp;
307
308
0
  if (node == NULL)
309
0
    return node;
310
311
0
  asn1_length_der (len, NULL, &len2);
312
0
  temp = malloc (len + len2);
313
0
  if (temp == NULL)
314
0
    return NULL;
315
316
0
  asn1_octet_der (value, len, temp, &len2);
317
0
  return _asn1_set_value_m (node, temp, len2);
318
0
}
319
320
/* the same as _asn1_set_value except that it sets an already malloc'ed
321
 * value.
322
 */
323
asn1_node
324
_asn1_set_value_m (asn1_node node, void *value, unsigned int len)
325
0
{
326
0
  if (node == NULL)
327
0
    return node;
328
329
0
  if (node->value)
330
0
    {
331
0
      if (node->value != node->small_value)
332
0
  free (node->value);
333
0
      node->value = NULL;
334
0
      node->value_len = 0;
335
0
    }
336
337
0
  if (!len)
338
0
    return node;
339
340
0
  node->value = value;
341
0
  node->value_len = len;
342
343
0
  return node;
344
0
}
345
346
/******************************************************************/
347
/* Function : _asn1_append_value                                  */
348
/* Description: appends to the field VALUE in a NODE_ASN element. */
349
/*                        */
350
/* Parameters:                                                    */
351
/*   node: element pointer.                                       */
352
/*   value: pointer to the value that you want to be appended.    */
353
/*   len: character number of value.                              */
354
/* Return: pointer to the NODE_ASN element.                       */
355
/******************************************************************/
356
asn1_node
357
_asn1_append_value (asn1_node node, const void *value, unsigned int len)
358
0
{
359
0
  if (node == NULL)
360
0
    return node;
361
362
0
  if (node->value == NULL)
363
0
    return _asn1_set_value (node, value, len);
364
365
0
  if (len == 0)
366
0
    return node;
367
368
0
  if (node->value == node->small_value)
369
0
    {
370
      /* value is in node */
371
0
      int prev_len = node->value_len;
372
0
      node->value_len += len;
373
0
      node->value = malloc (node->value_len);
374
0
      if (node->value == NULL)
375
0
  {
376
0
    node->value_len = 0;
377
0
    return NULL;
378
0
  }
379
380
0
      if (prev_len > 0)
381
0
  memcpy (node->value, node->small_value, prev_len);
382
383
0
      memcpy (&node->value[prev_len], value, len);
384
385
0
      return node;
386
0
    }
387
0
  else        /* if (node->value != NULL && node->value != node->small_value) */
388
0
    {
389
      /* value is allocated */
390
0
      int prev_len = node->value_len;
391
0
      node->value_len += len;
392
393
0
      node->value = _asn1_realloc (node->value, node->value_len);
394
0
      if (node->value == NULL)
395
0
  {
396
0
    node->value_len = 0;
397
0
    return NULL;
398
0
  }
399
400
0
      memcpy (&node->value[prev_len], value, len);
401
402
0
      return node;
403
0
    }
404
0
}
405
406
/******************************************************************/
407
/* Function : _asn1_set_name                                      */
408
/* Description: sets the field NAME in a NODE_ASN element. The    */
409
/*              previous value (if exist) will be lost            */
410
/* Parameters:                                                    */
411
/*   node: element pointer.                                       */
412
/*   name: a null terminated string with the name that you want   */
413
/*         to set.                                                */
414
/* Return: pointer to the NODE_ASN element.                       */
415
/******************************************************************/
416
asn1_node
417
_asn1_set_name (asn1_node node, const char *name)
418
74.9k
{
419
74.9k
  if (node == NULL)
420
0
    return node;
421
422
74.9k
  _asn1_str_cpy (node->name, sizeof (node->name), name ? name : "");
423
74.9k
  node->name_hash = _asn1_hash_name (node->name);
424
425
74.9k
  return node;
426
74.9k
}
427
428
/******************************************************************/
429
/* Function : _asn1_cpy_name                                      */
430
/* Description: copies the field NAME in a NODE_ASN element.      */
431
/* Parameters:                                                    */
432
/*   dst: a dest element pointer.                                 */
433
/*   src: a source element pointer.                               */
434
/* Return: pointer to the NODE_ASN element.                       */
435
/******************************************************************/
436
asn1_node
437
_asn1_cpy_name (asn1_node dst, asn1_node_const src)
438
0
{
439
0
  if (dst == NULL)
440
0
    return dst;
441
442
0
  if (src == NULL)
443
0
    {
444
0
      dst->name[0] = 0;
445
0
      dst->name_hash = _asn1_hash_name (dst->name);
446
0
      return dst;
447
0
    }
448
449
0
  _asn1_str_cpy (dst->name, sizeof (dst->name), src->name);
450
0
  dst->name_hash = src->name_hash;
451
452
0
  return dst;
453
0
}
454
455
/******************************************************************/
456
/* Function : _asn1_set_right                                     */
457
/* Description: sets the field RIGHT in a NODE_ASN element.       */
458
/* Parameters:                                                    */
459
/*   node: element pointer.                                       */
460
/*   right: pointer to a NODE_ASN element that you want be pointed*/
461
/*          by NODE.                                              */
462
/* Return: pointer to *NODE.                                      */
463
/******************************************************************/
464
asn1_node
465
_asn1_set_right (asn1_node node, asn1_node right)
466
122k
{
467
122k
  if (node == NULL)
468
0
    return node;
469
122k
  node->right = right;
470
122k
  if (right)
471
115k
    right->left = node;
472
122k
  return node;
473
122k
}
474
475
476
/******************************************************************/
477
/* Function : _asn1_get_last_right                                */
478
/* Description: return the last element along the right chain.    */
479
/* Parameters:                                                    */
480
/*   node: starting element pointer.                              */
481
/* Return: pointer to the last element along the right chain.     */
482
/******************************************************************/
483
asn1_node
484
_asn1_get_last_right (asn1_node_const node)
485
23.1k
{
486
23.1k
  asn1_node_const p;
487
488
23.1k
  if (node == NULL)
489
0
    return NULL;
490
23.1k
  p = node;
491
1.55M
  while (p->right)
492
1.53M
    p = p->right;
493
23.1k
  return (asn1_node) p;
494
23.1k
}
495
496
/******************************************************************/
497
/* Function : _asn1_remove_node                                   */
498
/* Description: gets free the memory allocated for an NODE_ASN    */
499
/*              element (not the elements pointed by it).         */
500
/* Parameters:                                                    */
501
/*   node: NODE_ASN element pointer.                              */
502
/*   flags: ASN1_DELETE_FLAG_*                                    */
503
/******************************************************************/
504
void
505
_asn1_remove_node (asn1_node node, unsigned int flags)
506
92.4k
{
507
92.4k
  if (node == NULL)
508
1.37k
    return;
509
510
91.0k
  if (node->value != NULL)
511
72.0k
    {
512
72.0k
      if (flags & ASN1_DELETE_FLAG_ZEROIZE)
513
0
  {
514
0
    safe_memset (node->value, 0, node->value_len);
515
0
  }
516
517
72.0k
      if (node->value != node->small_value)
518
7.31k
  free (node->value);
519
72.0k
    }
520
521
91.0k
  free (node->numbered_children.nodes);
522
91.0k
  free (node);
523
91.0k
}
524
525
/******************************************************************/
526
/* Function : _asn1_find_up                                       */
527
/* Description: return the father of the NODE_ASN element.        */
528
/* Parameters:                                                    */
529
/*   node: NODE_ASN element pointer.                              */
530
/* Return: Null if not found.                                     */
531
/******************************************************************/
532
asn1_node
533
_asn1_find_up (asn1_node_const node)
534
59.6k
{
535
59.6k
  asn1_node_const p;
536
537
59.6k
  if (node == NULL)
538
0
    return NULL;
539
540
59.6k
  p = node;
541
542
177k
  while ((p->left != NULL) && (p->left->right == p))
543
117k
    p = p->left;
544
545
59.6k
  return p->left;
546
59.6k
}
547
548
static unsigned
549
_asn1_is_up (asn1_node_const up_cand, asn1_node_const down)
550
5.63k
{
551
5.63k
  asn1_node_const d, u;
552
553
5.63k
  if (up_cand == NULL || down == NULL)
554
0
    return 0;
555
556
5.63k
  d = down;
557
558
11.2k
  while ((u = _asn1_find_up (d)) != NULL && u != d)
559
5.63k
    {
560
5.63k
      if (u == up_cand)
561
0
  return 1;
562
5.63k
      d = u;
563
5.63k
    }
564
565
5.63k
  return 0;
566
5.63k
}
567
568
/******************************************************************/
569
/* Function : _asn1_delete_node_from_list                         */
570
/* Description: deletes the list element given                    */
571
/******************************************************************/
572
void
573
_asn1_delete_node_from_list (list_type *list, asn1_node node)
574
5.62k
{
575
5.62k
  list_type *p = list;
576
577
718k
  while (p)
578
713k
    {
579
713k
      if (p->node == node)
580
5.62k
  p->node = NULL;
581
713k
      p = p->next;
582
713k
    }
583
5.62k
}
584
585
/******************************************************************/
586
/* Function : _asn1_delete_list                                   */
587
/* Description: deletes the list elements (not the elements       */
588
/*  pointed by them).                                             */
589
/******************************************************************/
590
void
591
_asn1_delete_list (list_type *e_list)
592
451
{
593
451
  list_type *p;
594
595
26.7k
  while (e_list)
596
26.3k
    {
597
26.3k
      p = e_list;
598
26.3k
      e_list = e_list->next;
599
26.3k
      free (p);
600
26.3k
    }
601
451
}
602
603
/******************************************************************/
604
/* Function : _asn1_delete_list_and nodes                         */
605
/* Description: deletes the list elements and the elements        */
606
/*  pointed by them.                                              */
607
/******************************************************************/
608
void
609
_asn1_delete_list_and_nodes (list_type *e_list)
610
1.94k
{
611
1.94k
  list_type *p;
612
613
66.7k
  while (e_list)
614
64.7k
    {
615
64.7k
      p = e_list;
616
64.7k
      e_list = e_list->next;
617
64.7k
      _asn1_remove_node (p->node, 0);
618
64.7k
      free (p);
619
64.7k
    }
620
1.94k
}
621
622
623
char *
624
_asn1_ltostr (int64_t v, char str[LTOSTR_MAX_SIZE])
625
0
{
626
0
  uint64_t d, r;
627
0
  char temp[LTOSTR_MAX_SIZE];
628
0
  int count, k, start;
629
0
  uint64_t val;
630
631
0
  if (v < 0)
632
0
    {
633
0
      str[0] = '-';
634
0
      start = 1;
635
0
      val = -((uint64_t) v);
636
0
    }
637
0
  else
638
0
    {
639
0
      val = v;
640
0
      start = 0;
641
0
    }
642
643
0
  count = 0;
644
0
  do
645
0
    {
646
0
      d = val / 10;
647
0
      r = val - d * 10;
648
0
      temp[start + count] = '0' + (char) r;
649
0
      count++;
650
0
      val = d;
651
0
    }
652
0
  while (val && ((start + count) < LTOSTR_MAX_SIZE - 1));
653
654
0
  for (k = 0; k < count; k++)
655
0
    str[k + start] = temp[start + count - k - 1];
656
0
  str[count + start] = 0;
657
0
  return str;
658
0
}
659
660
661
/******************************************************************/
662
/* Function : _asn1_change_integer_value                          */
663
/* Description: converts into DER coding the value assign to an   */
664
/*   INTEGER constant.                                            */
665
/* Parameters:                                                    */
666
/*   node: root of an ASN1element.                                */
667
/* Return:                                                        */
668
/*   ASN1_ELEMENT_NOT_FOUND if NODE is NULL,                       */
669
/*   otherwise ASN1_SUCCESS                                             */
670
/******************************************************************/
671
int
672
_asn1_change_integer_value (asn1_node node)
673
602
{
674
602
  asn1_node p;
675
602
  unsigned char val[SIZEOF_UNSIGNED_LONG_INT];
676
602
  unsigned char val2[SIZEOF_UNSIGNED_LONG_INT + 1];
677
602
  int len;
678
679
602
  if (node == NULL)
680
0
    return ASN1_ELEMENT_NOT_FOUND;
681
682
602
  p = node;
683
14.8k
  while (p)
684
14.2k
    {
685
14.2k
      if ((type_field (p->type) == ASN1_ETYPE_INTEGER)
686
14.2k
    && (p->type & CONST_ASSIGN))
687
1.82k
  {
688
1.82k
    if (p->value)
689
1.82k
      {
690
1.82k
        _asn1_convert_integer (p->value, val, sizeof (val), &len);
691
1.82k
        asn1_octet_der (val, len, val2, &len);
692
1.82k
        _asn1_set_value (p, val2, len);
693
1.82k
      }
694
1.82k
  }
695
696
14.2k
      if (p->down)
697
4.38k
  {
698
4.38k
    p = p->down;
699
4.38k
  }
700
9.84k
      else
701
9.84k
  {
702
9.84k
    if (p == node)
703
0
      p = NULL;
704
9.84k
    else if (p->right)
705
6.39k
      p = p->right;
706
3.45k
    else
707
3.45k
      {
708
4.38k
        while (1)
709
4.38k
    {
710
4.38k
      p = _asn1_find_up (p);
711
4.38k
      if (p == node)
712
602
        {
713
602
          p = NULL;
714
602
          break;
715
602
        }
716
3.77k
      if (p && p->right)
717
2.84k
        {
718
2.84k
          p = p->right;
719
2.84k
          break;
720
2.84k
        }
721
3.77k
    }
722
3.45k
      }
723
9.84k
  }
724
14.2k
    }
725
726
602
  return ASN1_SUCCESS;
727
602
}
728
729
48.6k
#define MAX_CONSTANTS 1024
730
/******************************************************************/
731
/* Function : _asn1_expand_object_id                              */
732
/* Description: expand the IDs of an OBJECT IDENTIFIER constant.  */
733
/* Parameters:                                                    */
734
/*   list: root of an object list                                 */
735
/*   node: root of an ASN1 element.                               */
736
/* Return:                                                        */
737
/*   ASN1_ELEMENT_NOT_FOUND if NODE is NULL,                      */
738
/*   otherwise ASN1_SUCCESS                                       */
739
/******************************************************************/
740
int
741
_asn1_expand_object_id (list_type **list, asn1_node node)
742
602
{
743
602
  asn1_node p, p2, p3, p4, p5;
744
602
  char name_root[ASN1_MAX_NAME_SIZE], name2[2 * ASN1_MAX_NAME_SIZE + 1];
745
602
  int move, tlen, tries;
746
602
  unsigned max_constants;
747
748
602
  if (node == NULL)
749
0
    return ASN1_ELEMENT_NOT_FOUND;
750
751
602
  _asn1_str_cpy (name_root, sizeof (name_root), node->name);
752
753
602
  p = node;
754
602
  move = DOWN;
755
602
  tries = 0;
756
757
32.6k
  while (!((p == node) && (move == UP)))
758
32.1k
    {
759
32.1k
      if (move != UP)
760
29.2k
  {
761
29.2k
    if ((type_field (p->type) == ASN1_ETYPE_OBJECT_ID)
762
29.2k
        && (p->type & CONST_ASSIGN))
763
7.03k
      {
764
7.03k
        p2 = p->down;
765
7.03k
        if (p2 && (type_field (p2->type) == ASN1_ETYPE_CONSTANT))
766
6.38k
    {
767
6.38k
      if (p2->value && !c_isdigit (p2->value[0]))
768
5.71k
        {
769
5.71k
          _asn1_str_cpy (name2, sizeof (name2), name_root);
770
5.71k
          _asn1_str_cat (name2, sizeof (name2), ".");
771
5.71k
          _asn1_str_cat (name2, sizeof (name2),
772
5.71k
             (char *) p2->value);
773
5.71k
          p3 = asn1_find_node (node, name2);
774
5.71k
          if (!p3 || _asn1_is_up (p2, p3) ||
775
5.71k
        (type_field (p3->type) != ASN1_ETYPE_OBJECT_ID) ||
776
5.71k
        !(p3->type & CONST_ASSIGN))
777
92
      return ASN1_ELEMENT_NOT_FOUND;
778
779
5.62k
          _asn1_set_down (p, p2->right);
780
5.62k
          if (p2->down)
781
0
      _asn1_delete_structure (*list, &p2->down, 0);
782
5.62k
          _asn1_delete_node_from_list (*list, p2);
783
5.62k
          _asn1_remove_node (p2, 0);
784
5.62k
          p2 = p;
785
5.62k
          p4 = p3->down;
786
5.62k
          max_constants = 0;
787
54.2k
          while (p4)
788
48.6k
      {
789
48.6k
        if (type_field (p4->type) == ASN1_ETYPE_CONSTANT)
790
48.6k
          {
791
48.6k
            max_constants++;
792
48.6k
            if (max_constants == MAX_CONSTANTS)
793
13
        return ASN1_RECURSION;
794
795
48.6k
            p5 =
796
48.6k
        _asn1_add_single_node (ASN1_ETYPE_CONSTANT);
797
48.6k
            _asn1_set_name (p5, p4->name);
798
48.6k
            if (p4->value)
799
45.3k
        {
800
45.3k
          tlen = _asn1_strlen (p4->value);
801
45.3k
          if (tlen > 0)
802
44.3k
            _asn1_set_value (p5, p4->value, tlen + 1);
803
45.3k
        }
804
48.6k
            _asn1_add_static_node2 (list, p5);
805
806
48.6k
            if (p2 == p)
807
3.38k
        {
808
3.38k
          _asn1_set_right (p5, p->down);
809
3.38k
          _asn1_set_down (p, p5);
810
3.38k
        }
811
45.2k
            else
812
45.2k
        {
813
45.2k
          _asn1_set_right (p5, p2->right);
814
45.2k
          _asn1_set_right (p2, p5);
815
45.2k
        }
816
48.6k
            p2 = p5;
817
48.6k
          }
818
48.6k
        p4 = p4->right;
819
48.6k
      }
820
5.61k
          move = DOWN;
821
822
5.61k
          tries++;
823
5.61k
          if (tries >= EXPAND_OBJECT_ID_MAX_RECURSION)
824
44
      return ASN1_RECURSION;
825
826
5.56k
          continue;
827
5.61k
        }
828
6.38k
    }
829
7.03k
      }
830
23.4k
    move = DOWN;
831
23.4k
  }
832
2.95k
      else
833
2.95k
  move = RIGHT;
834
835
26.4k
      tries = 0;
836
26.4k
      if (move == DOWN)
837
23.4k
  {
838
23.4k
    if (p->down)
839
3.55k
      p = p->down;
840
19.9k
    else
841
19.9k
      move = RIGHT;
842
23.4k
  }
843
844
26.4k
      if (p == node)
845
0
  {
846
0
    move = UP;
847
0
    continue;
848
0
  }
849
850
26.4k
      if (move == RIGHT)
851
22.8k
  {
852
22.8k
    if (p && p->right)
853
19.4k
      p = p->right;
854
3.40k
    else
855
3.40k
      move = UP;
856
22.8k
  }
857
26.4k
      if (move == UP)
858
3.40k
  p = _asn1_find_up (p);
859
26.4k
    }
860
861
  /*******************************/
862
  /*       expand DEFAULT        */
863
  /*******************************/
864
453
  p = node;
865
453
  move = DOWN;
866
867
25.4k
  while (!((p == node) && (move == UP)))
868
24.9k
    {
869
24.9k
      if (move != UP)
870
22.0k
  {
871
22.0k
    if ((type_field (p->type) == ASN1_ETYPE_OBJECT_ID) &&
872
22.0k
        (p->type & CONST_DEFAULT))
873
814
      {
874
814
        p2 = p->down;
875
814
        if (p2 && (type_field (p2->type) == ASN1_ETYPE_DEFAULT))
876
814
    {
877
814
      _asn1_str_cpy (name2, sizeof (name2), name_root);
878
814
      _asn1_str_cat (name2, sizeof (name2), ".");
879
814
      if (p2->value)
880
814
        _asn1_str_cat (name2, sizeof (name2), (char *) p2->value);
881
814
      p3 = asn1_find_node (node, name2);
882
814
      if (!p3 || (type_field (p3->type) != ASN1_ETYPE_OBJECT_ID)
883
814
          || !(p3->type & CONST_ASSIGN))
884
1
        return ASN1_ELEMENT_NOT_FOUND;
885
813
      p4 = p3->down;
886
813
      name2[0] = 0;
887
3.21k
      while (p4)
888
2.40k
        {
889
2.40k
          if (type_field (p4->type) == ASN1_ETYPE_CONSTANT)
890
2.40k
      {
891
2.40k
        if (p4->value == NULL)
892
1
          return ASN1_VALUE_NOT_FOUND;
893
894
2.40k
        if (name2[0])
895
1.78k
          _asn1_str_cat (name2, sizeof (name2), ".");
896
2.40k
        _asn1_str_cat (name2, sizeof (name2),
897
2.40k
           (char *) p4->value);
898
2.40k
      }
899
2.40k
          p4 = p4->right;
900
2.40k
        }
901
812
      tlen = strlen (name2);
902
812
      if (tlen > 0)
903
618
        _asn1_set_value (p2, name2, tlen + 1);
904
812
    }
905
814
      }
906
22.0k
    move = DOWN;
907
22.0k
  }
908
2.89k
      else
909
2.89k
  move = RIGHT;
910
911
24.9k
      if (move == DOWN)
912
22.0k
  {
913
22.0k
    if (p->down)
914
3.34k
      p = p->down;
915
18.7k
    else
916
18.7k
      move = RIGHT;
917
22.0k
  }
918
919
24.9k
      if (p == node)
920
0
  {
921
0
    move = UP;
922
0
    continue;
923
0
  }
924
925
24.9k
      if (move == RIGHT)
926
21.6k
  {
927
21.6k
    if (p && p->right)
928
18.2k
      p = p->right;
929
3.34k
    else
930
3.34k
      move = UP;
931
21.6k
  }
932
24.9k
      if (move == UP)
933
3.34k
  p = _asn1_find_up (p);
934
24.9k
    }
935
936
451
  return ASN1_SUCCESS;
937
453
}
938
939
940
/******************************************************************/
941
/* Function : _asn1_type_set_config                               */
942
/* Description: sets the CONST_SET and CONST_NOT_USED properties  */
943
/*   in the fields of the SET elements.                           */
944
/* Parameters:                                                    */
945
/*   node: root of an ASN1 element.                               */
946
/* Return:                                                        */
947
/*   ASN1_ELEMENT_NOT_FOUND if NODE is NULL,                       */
948
/*   otherwise ASN1_SUCCESS                                             */
949
/******************************************************************/
950
int
951
_asn1_type_set_config (asn1_node node)
952
1.02k
{
953
1.02k
  asn1_node p, p2;
954
1.02k
  int move;
955
956
1.02k
  if (node == NULL)
957
0
    return ASN1_ELEMENT_NOT_FOUND;
958
959
1.02k
  p = node;
960
1.02k
  move = DOWN;
961
962
22.0k
  while (!((p == node) && (move == UP)))
963
20.9k
    {
964
20.9k
      if (move != UP)
965
16.6k
  {
966
16.6k
    if (type_field (p->type) == ASN1_ETYPE_SET)
967
449
      {
968
449
        p2 = p->down;
969
4.33k
        while (p2)
970
3.88k
    {
971
3.88k
      if (type_field (p2->type) != ASN1_ETYPE_TAG)
972
3.66k
        p2->type |= CONST_SET | CONST_NOT_USED;
973
3.88k
      p2 = p2->right;
974
3.88k
    }
975
449
      }
976
16.6k
    move = DOWN;
977
16.6k
  }
978
4.36k
      else
979
4.36k
  move = RIGHT;
980
981
20.9k
      if (move == DOWN)
982
16.6k
  {
983
16.6k
    if (p->down)
984
5.39k
      p = p->down;
985
11.2k
    else
986
11.2k
      move = RIGHT;
987
16.6k
  }
988
989
20.9k
      if (p == node)
990
0
  {
991
0
    move = UP;
992
0
    continue;
993
0
  }
994
995
20.9k
      if (move == RIGHT)
996
15.5k
  {
997
15.5k
    if (p && p->right)
998
10.1k
      p = p->right;
999
5.39k
    else
1000
5.39k
      move = UP;
1001
15.5k
  }
1002
20.9k
      if (move == UP)
1003
5.39k
  p = _asn1_find_up (p);
1004
20.9k
    }
1005
1006
1.02k
  return ASN1_SUCCESS;
1007
1.02k
}
1008
1009
1010
/******************************************************************/
1011
/* Function : _asn1_check_identifier                              */
1012
/* Description: checks the definitions of all the identifiers     */
1013
/*   and the first element of an OBJECT_ID (e.g. {pkix 0 4}).     */
1014
/*   The _asn1_identifierMissing global variable is filled if     */
1015
/*   necessary.                                                   */
1016
/* Parameters:                                                    */
1017
/*   node: root of an ASN1 element.                               */
1018
/* Return:                                                        */
1019
/*   ASN1_ELEMENT_NOT_FOUND      if NODE is NULL,                 */
1020
/*   ASN1_IDENTIFIER_NOT_FOUND   if an identifier is not defined, */
1021
/*   otherwise ASN1_SUCCESS                                       */
1022
/******************************************************************/
1023
int
1024
_asn1_check_identifier (asn1_node_const node)
1025
1.02k
{
1026
1.02k
  asn1_node_const p, p2;
1027
1.02k
  char name2[ASN1_MAX_NAME_SIZE * 2 + 2];
1028
1029
1.02k
  if (node == NULL)
1030
0
    return ASN1_ELEMENT_NOT_FOUND;
1031
1032
1.02k
  p = node;
1033
16.3k
  while (p)
1034
15.7k
    {
1035
15.7k
      if (p->value && type_field (p->type) == ASN1_ETYPE_IDENTIFIER)
1036
1.91k
  {
1037
1.91k
    _asn1_str_cpy (name2, sizeof (name2), node->name);
1038
1.91k
    _asn1_str_cat (name2, sizeof (name2), ".");
1039
1.91k
    _asn1_str_cat (name2, sizeof (name2), (char *) p->value);
1040
1.91k
    p2 = asn1_find_node (node, name2);
1041
1.91k
    if (p2 == NULL)
1042
394
      {
1043
394
        if (p->value)
1044
394
    _asn1_str_cpy (_asn1_identifierMissing,
1045
394
             sizeof (_asn1_identifierMissing),
1046
394
             (char *) p->value);
1047
0
        else
1048
0
    _asn1_strcpy (_asn1_identifierMissing, "(null)");
1049
394
        return ASN1_IDENTIFIER_NOT_FOUND;
1050
394
      }
1051
1.91k
  }
1052
13.8k
      else if ((type_field (p->type) == ASN1_ETYPE_OBJECT_ID) &&
1053
13.8k
         (p->type & CONST_DEFAULT))
1054
837
  {
1055
837
    p2 = p->down;
1056
837
    if (p2 && (type_field (p2->type) == ASN1_ETYPE_DEFAULT))
1057
837
      {
1058
837
        _asn1_str_cpy (name2, sizeof (name2), node->name);
1059
837
        if (p2->value)
1060
820
    {
1061
820
      _asn1_str_cat (name2, sizeof (name2), ".");
1062
820
      _asn1_str_cat (name2, sizeof (name2), (char *) p2->value);
1063
820
      _asn1_str_cpy (_asn1_identifierMissing,
1064
820
         sizeof (_asn1_identifierMissing),
1065
820
         (char *) p2->value);
1066
820
    }
1067
17
        else
1068
17
    _asn1_strcpy (_asn1_identifierMissing, "(null)");
1069
1070
837
        p2 = asn1_find_node (node, name2);
1071
837
        if (!p2 || (type_field (p2->type) != ASN1_ETYPE_OBJECT_ID) ||
1072
837
      !(p2->type & CONST_ASSIGN))
1073
23
    return ASN1_IDENTIFIER_NOT_FOUND;
1074
814
        else
1075
814
    _asn1_identifierMissing[0] = 0;
1076
837
      }
1077
837
  }
1078
13.0k
      else if ((type_field (p->type) == ASN1_ETYPE_OBJECT_ID) &&
1079
13.0k
         (p->type & CONST_ASSIGN))
1080
1.50k
  {
1081
1.50k
    p2 = p->down;
1082
1.50k
    if (p2 && (type_field (p2->type) == ASN1_ETYPE_CONSTANT))
1083
1.50k
      {
1084
1.50k
        if (p2->value && !c_isdigit (p2->value[0]))
1085
1.25k
    {
1086
1.25k
      _asn1_str_cpy (name2, sizeof (name2), node->name);
1087
1.25k
      _asn1_str_cat (name2, sizeof (name2), ".");
1088
1.25k
      _asn1_str_cat (name2, sizeof (name2), (char *) p2->value);
1089
1.25k
      _asn1_str_cpy (_asn1_identifierMissing,
1090
1.25k
         sizeof (_asn1_identifierMissing),
1091
1.25k
         (char *) p2->value);
1092
1093
1.25k
      p2 = asn1_find_node (node, name2);
1094
1.25k
      if (!p2 || (type_field (p2->type) != ASN1_ETYPE_OBJECT_ID)
1095
1.25k
          || !(p2->type & CONST_ASSIGN))
1096
10
        return ASN1_IDENTIFIER_NOT_FOUND;
1097
1.24k
      else
1098
1.24k
        _asn1_identifierMissing[0] = 0;
1099
1.25k
    }
1100
1.50k
      }
1101
1.50k
  }
1102
1103
15.3k
      if (p->down)
1104
4.94k
  {
1105
4.94k
    p = p->down;
1106
4.94k
  }
1107
10.3k
      else if (p->right)
1108
6.88k
  p = p->right;
1109
3.49k
      else
1110
3.49k
  {
1111
4.42k
    while (p)
1112
4.42k
      {
1113
4.42k
        p = _asn1_find_up (p);
1114
4.42k
        if (p == node)
1115
602
    {
1116
602
      p = NULL;
1117
602
      break;
1118
602
    }
1119
3.82k
        if (p && p->right)
1120
2.89k
    {
1121
2.89k
      p = p->right;
1122
2.89k
      break;
1123
2.89k
    }
1124
3.82k
      }
1125
3.49k
  }
1126
15.3k
    }
1127
1128
602
  return ASN1_SUCCESS;
1129
1.02k
}
1130
1131
1132
/******************************************************************/
1133
/* Function : _asn1_set_default_tag                               */
1134
/* Description: sets the default IMPLICIT or EXPLICIT property in */
1135
/*   the tagged elements that don't have this declaration.        */
1136
/* Parameters:                                                    */
1137
/*   node: pointer to a DEFINITIONS element.                      */
1138
/* Return:                                                        */
1139
/*   ASN1_ELEMENT_NOT_FOUND if NODE is NULL or not a pointer to   */
1140
/*     a DEFINITIONS element,                                     */
1141
/*   otherwise ASN1_SUCCESS                                       */
1142
/******************************************************************/
1143
int
1144
_asn1_set_default_tag (asn1_node node)
1145
1.02k
{
1146
1.02k
  asn1_node p;
1147
1148
1.02k
  if ((node == NULL) || (type_field (node->type) != ASN1_ETYPE_DEFINITIONS))
1149
0
    return ASN1_ELEMENT_NOT_FOUND;
1150
1151
1.02k
  p = node;
1152
17.6k
  while (p)
1153
16.6k
    {
1154
16.6k
      if ((type_field (p->type) == ASN1_ETYPE_TAG) &&
1155
16.6k
    !(p->type & CONST_EXPLICIT) && !(p->type & CONST_IMPLICIT))
1156
1.17k
  {
1157
1.17k
    if (node->type & CONST_EXPLICIT)
1158
557
      p->type |= CONST_EXPLICIT;
1159
620
    else
1160
620
      p->type |= CONST_IMPLICIT;
1161
1.17k
  }
1162
1163
16.6k
      if (p->down)
1164
5.39k
  {
1165
5.39k
    p = p->down;
1166
5.39k
  }
1167
11.2k
      else if (p->right)
1168
6.91k
  p = p->right;
1169
4.31k
      else
1170
4.31k
  {
1171
5.39k
    while (1)
1172
5.39k
      {
1173
5.39k
        p = _asn1_find_up (p);
1174
5.39k
        if (p == node)
1175
1.02k
    {
1176
1.02k
      p = NULL;
1177
1.02k
      break;
1178
1.02k
    }
1179
4.36k
        if (p && p->right)
1180
3.28k
    {
1181
3.28k
      p = p->right;
1182
3.28k
      break;
1183
3.28k
    }
1184
4.36k
      }
1185
4.31k
  }
1186
16.6k
    }
1187
1188
1.02k
  return ASN1_SUCCESS;
1189
1.02k
}