Coverage Report

Created: 2026-05-11 07:54

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
35
{
71
  /* Points to name of symbol.  */
72
35
  char *name;
73
  /* Points to symbol.  */
74
35
  symbolS *symbolP;
75
35
  struct conditional_frame cframe;
76
35
  char c;
77
78
  /* Leading whitespace is part of operand.  */
79
35
  SKIP_WHITESPACE ();
80
35
  name = input_line_pointer;
81
82
35
  if (!is_name_beginner (*name) && *name != '"')
83
3
    {
84
3
      as_bad (_("invalid identifier for \".ifdef\""));
85
3
      obstack_1grow (&cond_obstack, 0);
86
3
      ignore_rest_of_line ();
87
3
      return;
88
3
    }
89
90
32
  c = get_symbol_name (& name);
91
32
  symbolP = symbol_find (name);
92
32
  (void) restore_line_pointer (c);
93
94
32
  initialize_cframe (&cframe);
95
96
32
  if (cframe.dead_tree)
97
0
    cframe.ignoring = 1;
98
32
  else
99
32
    {
100
32
      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
32
      is_defined =
106
32
  symbolP != NULL
107
0
  && (S_IS_DEFINED (symbolP) || symbol_equated_p (symbolP))
108
0
  && S_GET_SEGMENT (symbolP) != reg_section;
109
110
32
      cframe.ignoring = ! (test_defined ^ is_defined);
111
32
    }
112
113
32
  current_cframe = obstack_alloc (&cond_obstack, sizeof cframe);
114
32
  memcpy (current_cframe, &cframe, sizeof cframe);
115
116
32
  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
32
  demand_empty_rest_of_line ();
123
32
}
124
125
void
126
s_if (int arg)
127
16
{
128
16
  expressionS operand;
129
16
  struct conditional_frame cframe;
130
16
  int t;
131
16
  char *stop = NULL;
132
16
  char stopc = 0;
133
134
16
  if (flag_mri)
135
10
    stop = mri_comment_field (&stopc);
136
137
  /* Leading whitespace is part of operand.  */
138
16
  SKIP_WHITESPACE ();
139
140
16
  if (current_cframe != NULL && current_cframe->ignoring)
141
0
    {
142
0
      operand.X_add_number = 0;
143
0
      while (! is_end_of_stmt (*input_line_pointer))
144
0
  ++input_line_pointer;
145
0
    }
146
16
  else
147
16
    {
148
16
      expression_and_evaluate (&operand);
149
16
      if (operand.X_op != O_constant)
150
10
  as_bad (_("non-constant expression in \".if\" statement"));
151
16
    }
152
153
16
  switch ((operatorT) arg)
154
16
    {
155
0
    case O_eq: t = operand.X_add_number == 0; break;
156
0
    case O_ne: t = operand.X_add_number != 0; break;
157
0
    case O_lt: t = operand.X_add_number < 0; break;
158
13
    case O_le: t = operand.X_add_number <= 0; break;
159
0
    case O_ge: t = operand.X_add_number >= 0; break;
160
3
    case O_gt: t = operand.X_add_number > 0; break;
161
0
    default:
162
0
      abort ();
163
0
      return;
164
16
    }
165
166
  /* If the above error is signaled, this will dispatch
167
     using an undefined result.  No big deal.  */
168
16
  initialize_cframe (&cframe);
169
16
  cframe.ignoring = cframe.dead_tree || ! t;
170
16
  current_cframe = obstack_alloc (&cond_obstack, sizeof cframe);
171
16
  memcpy (current_cframe, & cframe, sizeof cframe);
172
173
16
  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
16
  if (flag_mri)
180
10
    mri_comment_end (stop, stopc);
181
182
16
  demand_empty_rest_of_line ();
183
16
}
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
684
{
223
684
  char *ret;
224
684
  char *s;
225
226
684
  SKIP_WHITESPACE ();
227
684
  s = ret = input_line_pointer;
228
684
  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
684
  else
245
684
    {
246
4.99k
      while (*input_line_pointer != terminator
247
4.65k
       && ! is_end_of_stmt (*input_line_pointer))
248
4.31k
  ++input_line_pointer;
249
684
      s = input_line_pointer;
250
726
      while (s > ret && is_whitespace (s[-1]))
251
42
  --s;
252
684
    }
253
254
684
  *len = s - ret;
255
684
  return ret;
256
684
}
257
258
/* The MRI IFC and IFNC pseudo-ops.  */
259
260
void
261
s_ifc (int arg)
262
342
{
263
342
  char *stop = NULL;
264
342
  char stopc = 0;
265
342
  char *s1, *s2;
266
342
  int len1, len2;
267
342
  int res;
268
342
  struct conditional_frame cframe;
269
270
342
  if (flag_mri)
271
5
    stop = mri_comment_field (&stopc);
272
273
342
  s1 = get_mri_string (',', &len1);
274
275
342
  if (*input_line_pointer != ',')
276
1
    as_bad (_("bad format for ifc or ifnc"));
277
341
  else
278
341
    ++input_line_pointer;
279
280
342
  s2 = get_mri_string (';', &len2);
281
282
342
  res = len1 == len2 && strncmp (s1, s2, len1) == 0;
283
284
342
  initialize_cframe (&cframe);
285
342
  cframe.ignoring = cframe.dead_tree || ! (res ^ arg);
286
342
  current_cframe = obstack_alloc (&cond_obstack, sizeof cframe);
287
342
  memcpy (current_cframe, &cframe, sizeof cframe);
288
  
289
342
 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
342
  if (flag_mri)
296
5
    mri_comment_end (stop, stopc);
297
298
342
  demand_empty_rest_of_line ();
299
342
}
300
301
void
302
s_elseif (int arg)
303
7
{
304
7
  if (current_cframe == NULL)
305
4
    {
306
4
      as_bad (_("\".elseif\" without matching \".if\""));
307
4
    }
308
3
  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
3
  else
319
3
    {
320
3
      current_cframe->else_file_line.file
321
3
        = as_where (&current_cframe->else_file_line.line);
322
323
3
      current_cframe->dead_tree |= !current_cframe->ignoring;
324
3
      current_cframe->ignoring = current_cframe->dead_tree;
325
3
    }
326
327
7
  if (current_cframe == NULL || current_cframe->ignoring)
328
4
    {
329
98
      while (! is_end_of_stmt (*input_line_pointer))
330
94
  ++input_line_pointer;
331
332
4
      if (current_cframe == NULL)
333
4
  return;
334
4
    }
335
3
  else
336
3
    {
337
3
      expressionS operand;
338
3
      int t;
339
340
      /* Leading whitespace is part of operand.  */
341
3
      SKIP_WHITESPACE ();
342
343
3
      expression_and_evaluate (&operand);
344
3
      if (operand.X_op != O_constant)
345
0
  as_bad (_("non-constant expression in \".elseif\" statement"));
346
347
3
      switch (arg)
348
3
  {
349
0
  case O_eq: t = operand.X_add_number == 0; break;
350
3
  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
3
  }
359
360
3
      current_cframe->ignoring = current_cframe->dead_tree || ! t;
361
3
    }
362
363
3
  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
3
  demand_empty_rest_of_line ();
374
3
}
375
376
void
377
s_endif (int arg ATTRIBUTE_UNUSED)
378
8
{
379
8
  struct conditional_frame *hold;
380
381
8
  if (current_cframe == NULL)
382
5
    {
383
5
      as_bad (_("\".endif\" without \".if\""));
384
5
    }
385
3
  else
386
3
    {
387
3
      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
3
      hold = current_cframe;
394
3
      current_cframe = current_cframe->previous_cframe;
395
3
      obstack_free (&cond_obstack, hold);
396
3
    }        /* if one pop too many */
397
398
8
  if (flag_mri)
399
6
    {
400
8
      while (! is_end_of_stmt (*input_line_pointer))
401
2
  ++input_line_pointer;
402
6
    }
403
404
8
  demand_empty_rest_of_line ();
405
8
}
406
407
void
408
s_else (int arg ATTRIBUTE_UNUSED)
409
6
{
410
6
  if (current_cframe == NULL)
411
6
    {
412
6
      as_bad (_("\".else\" without matching \".if\""));
413
6
    }
414
0
  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
0
  else
425
0
    {
426
0
      current_cframe->else_file_line.file
427
0
        = as_where (&current_cframe->else_file_line.line);
428
429
0
      current_cframe->ignoring =
430
0
  current_cframe->dead_tree | !current_cframe->ignoring;
431
432
0
      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
0
      current_cframe->else_seen = 1;
443
0
    }
444
445
6
  if (flag_mri)
446
6
    {
447
30
      while (! is_end_of_stmt (*input_line_pointer))
448
24
  ++input_line_pointer;
449
6
    }
450
451
6
  demand_empty_rest_of_line ();
452
6
}
453
454
void
455
s_ifeqs (int arg)
456
9
{
457
9
  char *s1, *s2;
458
9
  int len1, len2;
459
9
  int res;
460
9
  struct conditional_frame cframe;
461
462
9
  s1 = demand_copy_C_string (&len1);
463
464
9
  SKIP_WHITESPACE ();
465
9
  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
9
  ++input_line_pointer;
473
474
9
  s2 = demand_copy_C_string (&len2);
475
476
9
  res = len1 == len2 && strncmp (s1, s2, len1) == 0;
477
478
9
  initialize_cframe (&cframe);
479
9
  cframe.ignoring = cframe.dead_tree || ! (res ^ arg);
480
9
  current_cframe = obstack_alloc (&cond_obstack, sizeof cframe);
481
9
  memcpy (current_cframe, &cframe, sizeof cframe);
482
483
9
  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
9
  demand_empty_rest_of_line ();
490
9
}
491
492
int
493
ignore_input (void)
494
571k
{
495
571k
  char *s;
496
497
571k
  s = input_line_pointer;
498
499
571k
  if (NO_PSEUDO_DOT || flag_m68k_mri)
500
0
    {
501
0
      if (s[-1] != '.')
502
0
  --s;
503
0
    }
504
571k
  else
505
571k
    {
506
571k
      if (s[-1] != '.')
507
215k
  return current_cframe != NULL && current_cframe->ignoring;
508
571k
    }
509
510
  /* We cannot ignore certain pseudo ops.  */
511
355k
  switch (s[0])
512
355k
    {
513
21.7k
    case 'i': case  'I':
514
21.7k
      if (s[1] == 'f' || s[1] == 'F')
515
8.55k
  return 0;
516
13.2k
      break;
517
13.2k
    case 'e': case 'E':
518
6.09k
      if (!strncasecmp (s, "else", 4)
519
6.06k
    || !strncasecmp (s, "endif", 5)
520
5.87k
    || !strncasecmp (s, "endc", 4))
521
228
  return 0;
522
5.87k
      break;
523
235k
    case 'l': case 'L':
524
235k
      if (!strncasecmp (s, "linefile", 8))
525
230k
  return 0;
526
4.99k
      break;
527
355k
    }
528
529
116k
  return current_cframe != NULL && current_cframe->ignoring;
530
355k
}
531
532
static void
533
initialize_cframe (struct conditional_frame *cframe)
534
399
{
535
399
  memset (cframe, 0, sizeof (*cframe));
536
399
  cframe->if_file_line.file = as_where (&cframe->if_file_line.line);
537
399
  cframe->previous_cframe = current_cframe;
538
399
  cframe->dead_tree = current_cframe != NULL && current_cframe->ignoring;
539
399
  cframe->macro_nest = macro_nest;
540
399
}
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
510
{
550
510
  if (current_cframe != NULL && current_cframe->macro_nest >= nest)
551
24
    {
552
24
      if (nest >= 0)
553
6
  as_bad (_("end of macro inside conditional"));
554
18
      else
555
18
  as_bad (_("end of file inside conditional"));
556
557
24
      as_bad_where (current_cframe->if_file_line.file,
558
24
        current_cframe->if_file_line.line,
559
24
        _("here is the start of the unterminated conditional"));
560
24
      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
24
      cond_exit_macro (nest);
565
24
    }
566
510
}
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
27
{
575
423
  while (current_cframe != NULL && current_cframe->macro_nest >= nest)
576
396
    {
577
396
      struct conditional_frame *hold;
578
579
396
      hold = current_cframe;
580
396
      current_cframe = current_cframe->previous_cframe;
581
396
      obstack_free (&cond_obstack, hold);
582
396
    }
583
27
}