Coverage Report

Created: 2026-04-04 08:16

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/binutils-gdb/gas/cond.c
Line
Count
Source
1
/* cond.c - conditional assembly pseudo-ops, and .include
2
   Copyright (C) 1990-2026 Free Software Foundation, Inc.
3
4
   This file is part of GAS, the GNU Assembler.
5
6
   GAS is free software; you can redistribute it and/or modify
7
   it under the terms of the GNU General Public License as published by
8
   the Free Software Foundation; either version 3, or (at your option)
9
   any later version.
10
11
   GAS is distributed in the hope that it will be useful,
12
   but WITHOUT ANY WARRANTY; without even the implied warranty of
13
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
14
   GNU General Public License for more details.
15
16
   You should have received a copy of the GNU General Public License
17
   along with GAS; see the file COPYING.  If not, write to the Free
18
   Software Foundation, 51 Franklin Street - Fifth Floor, Boston, MA
19
   02110-1301, USA.  */
20
21
#include "as.h"
22
#include "sb.h"
23
#include "macro.h"
24
25
#include "obstack.h"
26
27
/* This is allocated to grow and shrink as .ifdef/.endif pairs are
28
   scanned.  */
29
struct obstack cond_obstack;
30
31
struct file_line
32
{
33
  const char *file;
34
  unsigned int line;
35
};
36
37
/* We push one of these structures for each .if, and pop it at the
38
   .endif.  */
39
40
struct conditional_frame
41
{
42
  /* The source file & line number of the "if".  */
43
  struct file_line if_file_line;
44
  /* The source file & line of the "else".  */
45
  struct file_line else_file_line;
46
  /* The previous conditional.  */
47
  struct conditional_frame *previous_cframe;
48
  /* Have we seen an else yet?  */
49
  int else_seen;
50
  /* Whether we are currently ignoring input.  */
51
  int ignoring;
52
  /* Whether a conditional at a higher level is ignoring input.
53
     Set also when a branch of an "if .. elseif .." tree has matched
54
     to prevent further matches.  */
55
  int dead_tree;
56
  /* Macro nesting level at which this conditional was created.  */
57
  int macro_nest;
58
};
59
60
static void initialize_cframe (struct conditional_frame *cframe);
61
static char *get_mri_string (int, int *);
62
63
static struct conditional_frame *current_cframe = NULL;
64
65
/* Performs the .ifdef (test_defined == 1) and
66
   the .ifndef (test_defined == 0) pseudo op.  */
67
68
void
69
s_ifdef (int test_defined)
70
1.15k
{
71
  /* Points to name of symbol.  */
72
1.15k
  char *name;
73
  /* Points to symbol.  */
74
1.15k
  symbolS *symbolP;
75
1.15k
  struct conditional_frame cframe;
76
1.15k
  char c;
77
78
  /* Leading whitespace is part of operand.  */
79
1.15k
  SKIP_WHITESPACE ();
80
1.15k
  name = input_line_pointer;
81
82
1.15k
  if (!is_name_beginner (*name) && *name != '"')
83
574
    {
84
574
      as_bad (_("invalid identifier for \".ifdef\""));
85
574
      obstack_1grow (&cond_obstack, 0);
86
574
      ignore_rest_of_line ();
87
574
      return;
88
574
    }
89
90
578
  c = get_symbol_name (& name);
91
578
  symbolP = symbol_find (name);
92
578
  (void) restore_line_pointer (c);
93
94
578
  initialize_cframe (&cframe);
95
96
578
  if (cframe.dead_tree)
97
0
    cframe.ignoring = 1;
98
578
  else
99
578
    {
100
578
      int is_defined;
101
102
      /* Use the same definition of 'defined' as .equiv so that a symbol
103
   which has been referenced but not yet given a value/address is
104
   considered to be undefined.  */
105
578
      is_defined =
106
578
  symbolP != NULL
107
0
  && (S_IS_DEFINED (symbolP) || symbol_equated_p (symbolP))
108
0
  && S_GET_SEGMENT (symbolP) != reg_section;
109
110
578
      cframe.ignoring = ! (test_defined ^ is_defined);
111
578
    }
112
113
578
  current_cframe = obstack_alloc (&cond_obstack, sizeof cframe);
114
578
  memcpy (current_cframe, &cframe, sizeof cframe);
115
116
578
  if (LISTING_SKIP_COND ()
117
0
      && cframe.ignoring
118
0
      && (cframe.previous_cframe == NULL
119
0
    || ! cframe.previous_cframe->ignoring))
120
0
    listing_list (2);
121
122
578
  demand_empty_rest_of_line ();
123
578
}
124
125
void
126
s_if (int arg)
127
204
{
128
204
  expressionS operand;
129
204
  struct conditional_frame cframe;
130
204
  int t;
131
204
  char *stop = NULL;
132
204
  char stopc = 0;
133
134
204
  if (flag_mri)
135
139
    stop = mri_comment_field (&stopc);
136
137
  /* Leading whitespace is part of operand.  */
138
204
  SKIP_WHITESPACE ();
139
140
204
  if (current_cframe != NULL && current_cframe->ignoring)
141
61
    {
142
61
      operand.X_add_number = 0;
143
1.32k
      while (! is_end_of_stmt (*input_line_pointer))
144
1.26k
  ++input_line_pointer;
145
61
    }
146
143
  else
147
143
    {
148
143
      expression_and_evaluate (&operand);
149
143
      if (operand.X_op != O_constant)
150
4
  as_bad (_("non-constant expression in \".if\" statement"));
151
143
    }
152
153
204
  switch ((operatorT) arg)
154
204
    {
155
0
    case O_eq: t = operand.X_add_number == 0; break;
156
202
    case O_ne: t = operand.X_add_number != 0; break;
157
1
    case O_lt: t = operand.X_add_number < 0; break;
158
1
    case O_le: t = operand.X_add_number <= 0; break;
159
0
    case O_ge: t = operand.X_add_number >= 0; break;
160
0
    case O_gt: t = operand.X_add_number > 0; break;
161
0
    default:
162
0
      abort ();
163
0
      return;
164
204
    }
165
166
  /* If the above error is signaled, this will dispatch
167
     using an undefined result.  No big deal.  */
168
204
  initialize_cframe (&cframe);
169
204
  cframe.ignoring = cframe.dead_tree || ! t;
170
204
  current_cframe = obstack_alloc (&cond_obstack, sizeof cframe);
171
204
  memcpy (current_cframe, & cframe, sizeof cframe);
172
173
204
  if (LISTING_SKIP_COND ()
174
0
      && cframe.ignoring
175
0
      && (cframe.previous_cframe == NULL
176
0
    || ! cframe.previous_cframe->ignoring))
177
0
    listing_list (2);
178
179
204
  if (flag_mri)
180
139
    mri_comment_end (stop, stopc);
181
182
204
  demand_empty_rest_of_line ();
183
204
}
184
185
/* Performs the .ifb (test_blank == 1) and
186
   the .ifnb (test_blank == 0) pseudo op.  */
187
188
void
189
s_ifb (int test_blank)
190
0
{
191
0
  struct conditional_frame cframe;
192
193
0
  initialize_cframe (&cframe);
194
195
0
  if (cframe.dead_tree)
196
0
    cframe.ignoring = 1;
197
0
  else
198
0
    {
199
0
      int is_eol;
200
201
0
      SKIP_WHITESPACE ();
202
0
      is_eol = is_end_of_stmt (*input_line_pointer);
203
0
      cframe.ignoring = (test_blank == !is_eol);
204
0
    }
205
206
0
  current_cframe = obstack_alloc (&cond_obstack, sizeof cframe);
207
0
  memcpy (current_cframe, &cframe, sizeof cframe);
208
209
0
  if (LISTING_SKIP_COND ()
210
0
      && cframe.ignoring
211
0
      && (cframe.previous_cframe == NULL
212
0
    || ! cframe.previous_cframe->ignoring))
213
0
    listing_list (2);
214
215
0
  ignore_rest_of_line ();
216
0
}
217
218
/* Get a string for the MRI IFC or IFNC pseudo-ops.  */
219
220
static char *
221
get_mri_string (int terminator, int *len)
222
914
{
223
914
  char *ret;
224
914
  char *s;
225
226
914
  SKIP_WHITESPACE ();
227
914
  s = ret = input_line_pointer;
228
914
  if (*input_line_pointer == '\'')
229
0
    {
230
0
      ++s;
231
0
      ++input_line_pointer;
232
0
      while (! is_end_of_stmt (*input_line_pointer))
233
0
  {
234
0
    *s++ = *input_line_pointer++;
235
0
    if (s[-1] == '\'')
236
0
      {
237
0
        if (*input_line_pointer != '\'')
238
0
    break;
239
0
        ++input_line_pointer;
240
0
      }
241
0
  }
242
0
      SKIP_WHITESPACE ();
243
0
    }
244
914
  else
245
914
    {
246
6.66k
      while (*input_line_pointer != terminator
247
6.20k
       && ! is_end_of_stmt (*input_line_pointer))
248
5.74k
  ++input_line_pointer;
249
914
      s = input_line_pointer;
250
970
      while (s > ret && is_whitespace (s[-1]))
251
56
  --s;
252
914
    }
253
254
914
  *len = s - ret;
255
914
  return ret;
256
914
}
257
258
/* The MRI IFC and IFNC pseudo-ops.  */
259
260
void
261
s_ifc (int arg)
262
457
{
263
457
  char *stop = NULL;
264
457
  char stopc = 0;
265
457
  char *s1, *s2;
266
457
  int len1, len2;
267
457
  int res;
268
457
  struct conditional_frame cframe;
269
270
457
  if (flag_mri)
271
457
    stop = mri_comment_field (&stopc);
272
273
457
  s1 = get_mri_string (',', &len1);
274
275
457
  if (*input_line_pointer != ',')
276
0
    as_bad (_("bad format for ifc or ifnc"));
277
457
  else
278
457
    ++input_line_pointer;
279
280
457
  s2 = get_mri_string (';', &len2);
281
282
457
  res = len1 == len2 && strncmp (s1, s2, len1) == 0;
283
284
457
  initialize_cframe (&cframe);
285
457
  cframe.ignoring = cframe.dead_tree || ! (res ^ arg);
286
457
  current_cframe = obstack_alloc (&cond_obstack, sizeof cframe);
287
457
  memcpy (current_cframe, &cframe, sizeof cframe);
288
  
289
457
 if (LISTING_SKIP_COND ()
290
0
      && cframe.ignoring
291
0
      && (cframe.previous_cframe == NULL
292
0
    || ! cframe.previous_cframe->ignoring))
293
0
    listing_list (2);
294
295
457
  if (flag_mri)
296
457
    mri_comment_end (stop, stopc);
297
298
457
  demand_empty_rest_of_line ();
299
457
}
300
301
void
302
s_elseif (int arg)
303
10
{
304
10
  if (current_cframe == NULL)
305
10
    {
306
10
      as_bad (_("\".elseif\" without matching \".if\""));
307
10
    }
308
0
  else if (current_cframe->else_seen)
309
0
    {
310
0
      as_bad (_("\".elseif\" after \".else\""));
311
0
      as_bad_where (current_cframe->else_file_line.file,
312
0
        current_cframe->else_file_line.line,
313
0
        _("here is the previous \".else\""));
314
0
      as_bad_where (current_cframe->if_file_line.file,
315
0
        current_cframe->if_file_line.line,
316
0
        _("here is the previous \".if\""));
317
0
    }
318
0
  else
319
0
    {
320
0
      current_cframe->else_file_line.file
321
0
        = as_where (&current_cframe->else_file_line.line);
322
323
0
      current_cframe->dead_tree |= !current_cframe->ignoring;
324
0
      current_cframe->ignoring = current_cframe->dead_tree;
325
0
    }
326
327
10
  if (current_cframe == NULL || current_cframe->ignoring)
328
10
    {
329
100
      while (! is_end_of_stmt (*input_line_pointer))
330
90
  ++input_line_pointer;
331
332
10
      if (current_cframe == NULL)
333
10
  return;
334
10
    }
335
0
  else
336
0
    {
337
0
      expressionS operand;
338
0
      int t;
339
340
      /* Leading whitespace is part of operand.  */
341
0
      SKIP_WHITESPACE ();
342
343
0
      expression_and_evaluate (&operand);
344
0
      if (operand.X_op != O_constant)
345
0
  as_bad (_("non-constant expression in \".elseif\" statement"));
346
347
0
      switch (arg)
348
0
  {
349
0
  case O_eq: t = operand.X_add_number == 0; break;
350
0
  case O_ne: t = operand.X_add_number != 0; break;
351
0
  case O_lt: t = operand.X_add_number < 0; break;
352
0
  case O_le: t = operand.X_add_number <= 0; break;
353
0
  case O_ge: t = operand.X_add_number >= 0; break;
354
0
  case O_gt: t = operand.X_add_number > 0; break;
355
0
  default:
356
0
    abort ();
357
0
    return;
358
0
  }
359
360
0
      current_cframe->ignoring = current_cframe->dead_tree || ! t;
361
0
    }
362
363
0
  if (LISTING_SKIP_COND ()
364
0
      && (current_cframe->previous_cframe == NULL
365
0
    || ! current_cframe->previous_cframe->ignoring))
366
0
    {
367
0
      if (! current_cframe->ignoring)
368
0
  listing_list (1);
369
0
      else
370
0
  listing_list (2);
371
0
    }
372
373
0
  demand_empty_rest_of_line ();
374
0
}
375
376
void
377
s_endif (int arg ATTRIBUTE_UNUSED)
378
12
{
379
12
  struct conditional_frame *hold;
380
381
12
  if (current_cframe == NULL)
382
6
    {
383
6
      as_bad (_("\".endif\" without \".if\""));
384
6
    }
385
6
  else
386
6
    {
387
6
      if (LISTING_SKIP_COND ()
388
0
    && current_cframe->ignoring
389
0
    && (current_cframe->previous_cframe == NULL
390
0
        || ! current_cframe->previous_cframe->ignoring))
391
0
  listing_list (1);
392
393
6
      hold = current_cframe;
394
6
      current_cframe = current_cframe->previous_cframe;
395
6
      obstack_free (&cond_obstack, hold);
396
6
    }        /* if one pop too many */
397
398
12
  if (flag_mri)
399
8
    {
400
8
      while (! is_end_of_stmt (*input_line_pointer))
401
0
  ++input_line_pointer;
402
8
    }
403
404
12
  demand_empty_rest_of_line ();
405
12
}
406
407
void
408
s_else (int arg ATTRIBUTE_UNUSED)
409
63
{
410
63
  if (current_cframe == NULL)
411
3
    {
412
3
      as_bad (_("\".else\" without matching \".if\""));
413
3
    }
414
60
  else if (current_cframe->else_seen)
415
0
    {
416
0
      as_bad (_("duplicate \".else\""));
417
0
      as_bad_where (current_cframe->else_file_line.file,
418
0
        current_cframe->else_file_line.line,
419
0
        _("here is the previous \".else\""));
420
0
      as_bad_where (current_cframe->if_file_line.file,
421
0
        current_cframe->if_file_line.line,
422
0
        _("here is the previous \".if\""));
423
0
    }
424
60
  else
425
60
    {
426
60
      current_cframe->else_file_line.file
427
60
        = as_where (&current_cframe->else_file_line.line);
428
429
60
      current_cframe->ignoring =
430
60
  current_cframe->dead_tree | !current_cframe->ignoring;
431
432
60
      if (LISTING_SKIP_COND ()
433
0
    && (current_cframe->previous_cframe == NULL
434
0
        || ! current_cframe->previous_cframe->ignoring))
435
0
  {
436
0
    if (! current_cframe->ignoring)
437
0
      listing_list (1);
438
0
    else
439
0
      listing_list (2);
440
0
  }
441
442
60
      current_cframe->else_seen = 1;
443
60
    }
444
445
63
  if (flag_mri)
446
2
    {
447
5
      while (! is_end_of_stmt (*input_line_pointer))
448
3
  ++input_line_pointer;
449
2
    }
450
451
63
  demand_empty_rest_of_line ();
452
63
}
453
454
void
455
s_ifeqs (int arg)
456
6
{
457
6
  char *s1, *s2;
458
6
  int len1, len2;
459
6
  int res;
460
6
  struct conditional_frame cframe;
461
462
6
  s1 = demand_copy_C_string (&len1);
463
464
6
  SKIP_WHITESPACE ();
465
6
  if (*input_line_pointer != ',')
466
0
    {
467
0
      as_bad (_(".ifeqs syntax error"));
468
0
      ignore_rest_of_line ();
469
0
      return;
470
0
    }
471
472
6
  ++input_line_pointer;
473
474
6
  s2 = demand_copy_C_string (&len2);
475
476
6
  res = len1 == len2 && strncmp (s1, s2, len1) == 0;
477
478
6
  initialize_cframe (&cframe);
479
6
  cframe.ignoring = cframe.dead_tree || ! (res ^ arg);
480
6
  current_cframe = obstack_alloc (&cond_obstack, sizeof cframe);
481
6
  memcpy (current_cframe, &cframe, sizeof cframe);
482
483
6
  if (LISTING_SKIP_COND ()
484
0
      && cframe.ignoring
485
0
      && (cframe.previous_cframe == NULL
486
0
    || ! cframe.previous_cframe->ignoring))
487
0
    listing_list (2);
488
489
6
  demand_empty_rest_of_line ();
490
6
}
491
492
int
493
ignore_input (void)
494
396k
{
495
396k
  char *s;
496
497
396k
  s = input_line_pointer;
498
499
396k
  if (NO_PSEUDO_DOT || flag_m68k_mri)
500
0
    {
501
0
      if (s[-1] != '.')
502
0
  --s;
503
0
    }
504
396k
  else
505
396k
    {
506
396k
      if (s[-1] != '.')
507
140k
  return current_cframe != NULL && current_cframe->ignoring;
508
396k
    }
509
510
  /* We cannot ignore certain pseudo ops.  */
511
256k
  switch (s[0])
512
256k
    {
513
8.14k
    case 'i': case  'I':
514
8.14k
      if (s[1] == 'f' || s[1] == 'F')
515
2.79k
  return 0;
516
5.34k
      break;
517
7.57k
    case 'e': case 'E':
518
7.57k
      if (!strncasecmp (s, "else", 4)
519
7.47k
    || !strncasecmp (s, "endif", 5)
520
7.45k
    || !strncasecmp (s, "endc", 4))
521
197
  return 0;
522
7.37k
      break;
523
159k
    case 'l': case 'L':
524
159k
      if (!strncasecmp (s, "linefile", 8))
525
152k
  return 0;
526
6.96k
      break;
527
256k
    }
528
529
100k
  return current_cframe != NULL && current_cframe->ignoring;
530
256k
}
531
532
static void
533
initialize_cframe (struct conditional_frame *cframe)
534
1.24k
{
535
1.24k
  memset (cframe, 0, sizeof (*cframe));
536
1.24k
  cframe->if_file_line.file = as_where (&cframe->if_file_line.line);
537
1.24k
  cframe->previous_cframe = current_cframe;
538
1.24k
  cframe->dead_tree = current_cframe != NULL && current_cframe->ignoring;
539
1.24k
  cframe->macro_nest = macro_nest;
540
1.24k
}
541
542
/* Give an error if a conditional is unterminated inside a macro or
543
   the assembly as a whole.  If NEST is non negative, we are being
544
   called because of the end of a macro expansion.  If NEST is
545
   negative, we are being called at the of the input files.  */
546
547
void
548
cond_finish_check (int nest)
549
778
{
550
778
  if (current_cframe != NULL && current_cframe->macro_nest >= nest)
551
20
    {
552
20
      if (nest >= 0)
553
2
  as_bad (_("end of macro inside conditional"));
554
18
      else
555
18
  as_bad (_("end of file inside conditional"));
556
557
20
      as_bad_where (current_cframe->if_file_line.file,
558
20
        current_cframe->if_file_line.line,
559
20
        _("here is the start of the unterminated conditional"));
560
20
      if (current_cframe->else_seen)
561
0
  as_bad_where (current_cframe->else_file_line.file,
562
0
          current_cframe->else_file_line.line,
563
0
          _("here is the \"else\" of the unterminated conditional"));
564
20
      cond_exit_macro (nest);
565
20
    }
566
778
}
567
568
/* This function is called when we exit out of a macro.  We assume
569
   that any conditionals which began within the macro are correctly
570
   nested, and just pop them off the stack.  */
571
572
void
573
cond_exit_macro (int nest)
574
20
{
575
1.25k
  while (current_cframe != NULL && current_cframe->macro_nest >= nest)
576
1.23k
    {
577
1.23k
      struct conditional_frame *hold;
578
579
1.23k
      hold = current_cframe;
580
1.23k
      current_cframe = current_cframe->previous_cframe;
581
1.23k
      obstack_free (&cond_obstack, hold);
582
1.23k
    }
583
20
}