Coverage Report

Created: 2026-03-10 08:46

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/binutils-gdb/gas/input-scrub.c
Line
Count
Source
1
/* input_scrub.c - Break up input buffers into whole numbers of lines.
2
   Copyright (C) 1987-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 "filenames.h"
23
#include "input-file.h"
24
#include "sb.h"
25
#include "listing.h"
26
27
/*
28
 * O/S independent module to supply buffers of sanitised source code
29
 * to rest of assembler.  We get sanitised input data of arbitrary length.
30
 * We break these buffers on line boundaries, recombine pieces that
31
 * were broken across buffers, and return a buffer of full lines to
32
 * the caller.
33
 * The last partial line begins the next buffer we build and return to caller.
34
 * The buffer returned to caller is preceded by BEFORE_STRING and followed
35
 * by AFTER_STRING, as sentinels. The last character before AFTER_STRING
36
 * is a newline.
37
 * Also looks after line numbers, for e.g. error messages.
38
 */
39
40
/*
41
 * We don't care how filthy our buffers are, but our callers assume
42
 * that the following sanitation has already been done.
43
 *
44
 * No comments, reduce a comment to a space.
45
 * Reduce a tab to a space unless it is 1st char of line.
46
 * All multiple tabs and spaces collapsed into 1 char. Tab only
47
 *   legal if 1st char of line.
48
 * # line file statements converted to .line x;.file y; statements.
49
 * Escaped newlines at end of line: remove them but add as many newlines
50
 *   to end of statement as you removed in the middle, to synch line numbers.
51
 */
52

53
320
#define BEFORE_STRING ("\n")
54
222
#define AFTER_STRING ("\0")  /* memcpy of 0 chars might choke.  */
55
1.46k
#define BEFORE_SIZE (1)
56
456
#define AFTER_SIZE  (1)
57
58
#ifndef TC_EOL_IN_INSN
59
210
#define TC_EOL_IN_INSN(P) 0
60
#endif
61
62
static char *buffer_start;  /*->1st char of full buffer area.  */
63
static char *partial_where; /*->after last full line in buffer.  */
64
static size_t partial_size; /* >=0. Number of chars in partial line in buffer.  */
65
66
/* Because we need AFTER_STRING just after last full line, it clobbers
67
   1st part of partial line. So we preserve 1st part of partial line
68
   here.  */
69
static char save_source[AFTER_SIZE];
70
71
/* The size of the input buffer we concatenate
72
   input_file_give_next_buffer chunks into.  Excludes the BEFORE and
73
   AFTER counts.  */
74
static size_t buffer_length;
75
76
/* The index into an sb structure we are reading from.  -1 if none.  */
77
static size_t sb_index = -1;
78
79
/* If we are reading from an sb structure, this is it.  */
80
static sb from_sb;
81
82
/* Should we do a conditional check on from_sb? */
83
static enum expansion from_sb_expansion = expanding_none;
84
85
/* The number of nested sb structures we have included.  */
86
int macro_nest;
87
88
/* We can have more than one source file open at once, though the info for all
89
   but the latest one are saved off in a struct input_save.  These files remain
90
   open, so we are limited by the number of open files allowed by the
91
   underlying OS. We may also sequentially read more than one source file in an
92
   assembly.  */
93
94
/* We must track the physical file and line number for error messages. We also
95
   track a "logical" file and line number corresponding to (C?)  compiler
96
   source line numbers.  Whenever we open a file we must fill in
97
   physical_input_file. So if it is NULL we have not opened any files yet.  */
98
99
static const char *physical_input_file;
100
static const char *logical_input_file;
101
102
/* 1-origin line number in a source file.  */
103
/* A line ends in '\n' or eof.  */
104
static unsigned int physical_input_line;
105
static unsigned int logical_input_line;
106
107
/* Indicator whether the origin of an update was a .linefile directive. */
108
static bool is_linefile;
109
110
/* Struct used to save the state of the input handler during include files */
111
struct input_save {
112
  char *              buffer_start;
113
  char *              partial_where;
114
  size_t              partial_size;
115
  char                save_source[AFTER_SIZE];
116
  size_t              buffer_length;
117
  const char *        physical_input_file;
118
  const char *        logical_input_file;
119
  unsigned int        physical_input_line;
120
  unsigned int        logical_input_line;
121
  bool                is_linefile;
122
  size_t              sb_index;
123
  sb                  from_sb;
124
  enum expansion      from_sb_expansion; /* Should we do a conditional check?  */
125
  struct input_save * next_saved_file;  /* Chain of input_saves.  */
126
  char *              input_file_save;  /* Saved state of input routines.  */
127
  char *              saved_position; /* Caller's saved position in buf.  */
128
};
129
130
static struct input_save *input_scrub_push (char *saved_position);
131
static char *input_scrub_pop (struct input_save *arg);
132
133
/* Saved information about the file that .include'd this one.  When we hit EOF,
134
   we automatically pop to that file.  */
135
136
static struct input_save *next_saved_file;
137
138
/* Initialize input buffering.  */
139
140
static void
141
input_scrub_reinit (void)
142
320
{
143
320
  input_file_begin ();    /* Reinitialize! */
144
320
  logical_input_line = -1u;
145
320
  logical_input_file = NULL;
146
320
  sb_index = -1;
147
148
320
  buffer_length = input_file_buffer_size () * 2;
149
320
  buffer_start = XNEWVEC (char, BEFORE_SIZE + AFTER_SIZE + 1 + buffer_length);
150
320
  memcpy (buffer_start, BEFORE_STRING, (int) BEFORE_SIZE);
151
320
}
152
153
/* Finish off old buffers.  */
154
155
static void
156
input_scrub_free (void)
157
320
{
158
320
  if (sb_index != (size_t) -1)
159
101
    {
160
101
      sb_kill (&from_sb);
161
101
      sb_index = -1;
162
101
    }
163
320
  free (buffer_start);
164
320
  buffer_start = NULL;
165
320
  input_file_end ();
166
320
}
167
168
/* Push the state of input reading and scrubbing so that we can #include.
169
   The return value is a 'void *' (fudged for old compilers) to a save
170
   area, which can be restored by passing it to input_scrub_pop().  */
171
172
static struct input_save *
173
input_scrub_push (char *saved_position)
174
113
{
175
113
  struct input_save *saved;
176
177
113
  saved = XNEW (struct input_save);
178
179
113
  saved->saved_position = saved_position;
180
113
  saved->buffer_start = buffer_start;
181
113
  saved->partial_where = partial_where;
182
113
  saved->partial_size = partial_size;
183
113
  saved->buffer_length = buffer_length;
184
113
  saved->physical_input_file = physical_input_file;
185
113
  saved->logical_input_file = logical_input_file;
186
113
  saved->physical_input_line = physical_input_line;
187
113
  saved->logical_input_line = logical_input_line;
188
113
  saved->is_linefile = is_linefile;
189
113
  saved->sb_index = sb_index;
190
113
  saved->from_sb = from_sb;
191
113
  saved->from_sb_expansion = from_sb_expansion;
192
113
  memcpy (saved->save_source, save_source, sizeof (save_source));
193
113
  saved->next_saved_file = next_saved_file;
194
113
  saved->input_file_save = input_file_push ();
195
196
113
  input_scrub_reinit ();
197
198
113
  return saved;
199
113
}
200
201
static char *
202
input_scrub_pop (struct input_save *saved)
203
113
{
204
113
  char *saved_position;
205
206
113
  input_scrub_free ();
207
208
113
  input_file_pop (saved->input_file_save);
209
113
  saved_position = saved->saved_position;
210
113
  buffer_start = saved->buffer_start;
211
113
  buffer_length = saved->buffer_length;
212
213
  /* When expanding an #APP / #NO_APP block, original lines are re-
214
     processed, so whatever they did to physical file/line needs
215
     retaining.  If logical file/line weren't changed, the logical
216
     line number will want bumping by a corresponding value.  */
217
113
  if (from_sb_expansion != expanding_app)
218
113
    {
219
113
      if (logical_input_file == 0 && logical_input_line == -1u
220
35
    && saved->logical_input_line != -1u)
221
12
  saved->logical_input_line
222
12
    += physical_input_line - saved->physical_input_line;
223
113
      physical_input_file = saved->physical_input_file;
224
113
      physical_input_line = saved->physical_input_line;
225
113
    }
226
113
  logical_input_file = saved->logical_input_file;
227
113
  logical_input_line = saved->logical_input_line;
228
229
113
  is_linefile = saved->is_linefile;
230
113
  sb_index = saved->sb_index;
231
113
  from_sb = saved->from_sb;
232
113
  from_sb_expansion = saved->from_sb_expansion;
233
113
  partial_where = saved->partial_where;
234
113
  partial_size = saved->partial_size;
235
113
  next_saved_file = saved->next_saved_file;
236
113
  memcpy (save_source, saved->save_source, sizeof (save_source));
237
238
113
  free (saved);
239
113
  return saved_position;
240
113
}
241

242
void
243
input_scrub_begin (void)
244
207
{
245
207
  know (strlen (BEFORE_STRING) == BEFORE_SIZE);
246
207
  know (strlen (AFTER_STRING) == AFTER_SIZE
247
207
  || (AFTER_STRING[0] == '\0' && AFTER_SIZE == 1));
248
249
207
  physical_input_file = NULL; /* No file read yet.  */
250
207
  next_saved_file = NULL; /* At EOF, don't pop to any other file */
251
207
  macro_nest = 0;
252
207
  input_scrub_reinit ();
253
207
  do_scrub_begin (flag_m68k_mri);
254
207
}
255
256
void
257
input_scrub_end (void)
258
207
{
259
207
  while (next_saved_file != NULL)
260
0
    input_scrub_pop (next_saved_file);
261
207
  input_scrub_free ();
262
207
}
263
264
/* Start reading input from a new file.
265
   Return start of caller's part of buffer.  */
266
267
char *
268
input_scrub_new_file (const char *filename)
269
219
{
270
219
  input_file_open (filename, !flag_no_comments);
271
219
  physical_input_file = filename[0] ? filename : _("{standard input}");
272
219
  physical_input_line = 0;
273
274
219
  partial_size = 0;
275
219
  return (buffer_start + BEFORE_SIZE);
276
219
}
277
278
/* Include a file from the current file.  Save our state, cause it to
279
   be restored on EOF, and begin handling a new file.  Same result as
280
   input_scrub_new_file.  */
281
282
char *
283
input_scrub_include_file (const char *filename, char *position)
284
12
{
285
12
  next_saved_file = input_scrub_push (position);
286
12
  from_sb_expansion = expanding_none;
287
12
  return input_scrub_new_file (filename);
288
12
}
289
290
/* Start getting input from an sb structure.  This is used when
291
   expanding a macro.  */
292
293
void
294
input_scrub_include_sb (sb *from, char *position, enum expansion expansion)
295
101
{
296
101
  int newline;
297
298
101
  if (expansion != expanding_app)
299
101
    {
300
101
      if (macro_nest > max_macro_nest)
301
0
  as_fatal (_("macros nested too deeply"));
302
101
      ++macro_nest;
303
101
    }
304
305
#ifdef md_macro_start
306
  if (expansion == expanding_macro)
307
    {
308
      md_macro_start ();
309
    }
310
#endif
311
312
101
  next_saved_file = input_scrub_push (position);
313
314
  /* Allocate sufficient space: from->len plus optional newline
315
     plus two ".linefile " directives, plus a little more for other
316
     expansion.  */
317
101
  newline = from->len >= 1 && from->ptr[0] != '\n';
318
101
  sb_build (&from_sb, from->len + newline + 2 * sizeof (".linefile") + 30);
319
101
  from_sb_expansion = expansion;
320
101
  if (newline)
321
78
    {
322
      /* Add the sentinel required by read.c.  */
323
78
      sb_add_char (&from_sb, '\n');
324
78
    }
325
101
  sb_scrub_and_add_sb (&from_sb, from);
326
327
  /* Make sure the parser looks at defined contents when it scans for
328
     e.g. end-of-line at the end of a macro.  */
329
101
  sb_terminate (&from_sb);
330
331
101
  sb_index = 1;
332
333
  /* These variables are reset by input_scrub_push.  Restore them
334
     since we are, after all, still at the same point in the file.  */
335
101
  logical_input_line = next_saved_file->logical_input_line;
336
101
  logical_input_file = next_saved_file->logical_input_file;
337
101
}
338
339
void
340
input_scrub_close (void)
341
207
{
342
207
  input_file_close ();
343
207
  physical_input_line = 0;
344
207
  logical_input_line = -1u;
345
207
}
346
347
char *
348
input_scrub_next_buffer (char **bufp)
349
622
{
350
622
  char *limit;    /*->just after last char of buffer.  */
351
352
622
  if (sb_index != (size_t) -1)
353
179
    {
354
179
      if (sb_index >= from_sb.len)
355
101
  {
356
101
    if (from_sb_expansion == expanding_macro)
357
2
      {
358
2
        cond_finish_check (macro_nest);
359
#ifdef md_macro_end
360
        /* Allow the target to clean up per-macro expansion
361
           data.  */
362
        md_macro_end ();
363
#endif
364
2
      }
365
101
    if (from_sb_expansion != expanding_app)
366
101
      --macro_nest;
367
101
    partial_where = NULL;
368
101
    partial_size = 0;
369
101
    if (next_saved_file != NULL)
370
101
      *bufp = input_scrub_pop (next_saved_file);
371
101
    return partial_where;
372
101
  }
373
374
78
      partial_where = from_sb.ptr + from_sb.len;
375
78
      partial_size = 0;
376
78
      *bufp = from_sb.ptr + sb_index;
377
78
      sb_index = from_sb.len;
378
78
      return partial_where;
379
179
    }
380
381
443
  if (partial_size)
382
12
    {
383
12
      memmove (buffer_start + BEFORE_SIZE, partial_where, partial_size);
384
12
      memcpy (buffer_start + BEFORE_SIZE, save_source, AFTER_SIZE);
385
12
    }
386
387
443
  while (1)
388
443
    {
389
443
      char *p;
390
443
      char *start = buffer_start + BEFORE_SIZE + partial_size;
391
392
443
      *bufp = buffer_start + BEFORE_SIZE;
393
443
      limit = input_file_give_next_buffer (start);
394
443
      if (!limit)
395
233
  {
396
233
    if (!partial_size)
397
      /* End of this file.  */
398
221
      break;
399
400
12
    as_warn (_("end of file not at end of a line; newline inserted"));
401
12
    p = buffer_start + BEFORE_SIZE + partial_size;
402
12
    *p++ = '\n';
403
12
    limit = p;
404
12
  }
405
210
      else
406
210
  {
407
    /* Terminate the buffer to avoid confusing TC_EOL_IN_INSN.  */
408
210
    *limit = '\0';
409
410
    /* Find last newline.  */
411
5.73k
    for (p = limit - 1; *p != '\n' || TC_EOL_IN_INSN (p); --p)
412
5.52k
      if (p < start)
413
0
        goto read_more;
414
210
    ++p;
415
210
  }
416
417
      /* We found a newline in the newly read chars.  */
418
222
      partial_where = p;
419
222
      partial_size = limit - p;
420
421
      /* Save the fragment after that last newline.  */
422
222
      memcpy (save_source, partial_where, AFTER_SIZE);
423
222
      memcpy (partial_where, AFTER_STRING, AFTER_SIZE);
424
222
      return partial_where;
425
426
0
    read_more:
427
      /* Didn't find a newline.  Read more text.  */
428
0
      partial_size = limit - (buffer_start + BEFORE_SIZE);
429
0
      if (buffer_length - input_file_buffer_size () < partial_size)
430
0
  {
431
    /* Increase the buffer when it doesn't have room for the
432
       next block of input.  */
433
0
    buffer_length *= 2;
434
0
    buffer_start = XRESIZEVEC (char, buffer_start,
435
0
             (buffer_length
436
0
              + BEFORE_SIZE + AFTER_SIZE + 1));
437
0
  }
438
0
    }
439
440
  /* Tell the listing we've finished the file.  */
441
221
  LISTING_EOF ();
442
443
  /* If we should pop to another file at EOF, do it.  */
444
221
  partial_where = NULL;
445
221
  if (next_saved_file)
446
12
    *bufp = input_scrub_pop (next_saved_file);
447
448
221
  return partial_where;
449
443
}
450

451
/* The remaining part of this file deals with line numbers, error
452
   messages and so on.  Return TRUE if we opened any file.  */
453
454
int
455
seen_at_least_1_file (void)
456
0
{
457
0
  return (physical_input_file != NULL);
458
0
}
459
460
void
461
bump_line_counters (void)
462
78.4k
{
463
78.4k
  if (sb_index == (size_t) -1 || from_sb_expansion == expanding_app)
464
4.01k
    ++physical_input_line;
465
466
78.4k
  if (logical_input_line != -1u)
467
74.6k
    ++logical_input_line;
468
78.4k
}
469

470
/* Tells us what the new logical line number and file are.
471
   If the line_number is -1, we don't change the current logical line
472
   number.
473
   If fname is NULL, we don't change the current logical file name, unless
474
   bit 3 of flags is set.
475
   Returns nonzero if the filename actually changes.  */
476
477
void
478
new_logical_line_flags (const char *fname, /* DON'T destroy it!  We point to it!  */
479
      int line_number,
480
      int flags)
481
8.09k
{
482
8.09k
  switch (flags)
483
8.09k
    {
484
7.16k
    case 0:
485
7.16k
      break;
486
904
    case 1:
487
904
      if (line_number != -1)
488
0
  abort ();
489
904
      break;
490
904
    case 1 << 1:
491
25
    case 1 << 2:
492
      /* FIXME: we could check that include nesting is correct.  */
493
25
      break;
494
0
    case 1 << 3:
495
      /* s_linefile conditionally decrements the line depending on
496
   whether '\n' is seen.  */
497
0
      if (line_number < 0
498
0
    || (line_number == 0 && input_line_pointer[-1] != '\n'))
499
0
  line_number = physical_input_line;
500
0
      if (fname != NULL)
501
0
  abort ();
502
0
      if (next_saved_file == NULL)
503
0
  fname = physical_input_file;
504
0
      else if (next_saved_file->logical_input_file)
505
0
  fname = next_saved_file->logical_input_file;
506
0
      else
507
0
  fname = next_saved_file->physical_input_file;
508
0
      break;
509
0
    default:
510
0
      abort ();
511
8.09k
    }
512
513
8.09k
  is_linefile = flags != 1 && (flags != 0 || fname);
514
515
8.09k
  if (line_number >= 0)
516
7.08k
    logical_input_line = line_number;
517
1.00k
  else if (line_number == -1 && fname && !*fname && (flags & (1 << 2)))
518
0
    {
519
0
      logical_input_file = physical_input_file;
520
0
      logical_input_line = physical_input_line;
521
0
      fname = NULL;
522
0
    }
523
524
8.09k
  if (fname
525
7.92k
      && (logical_input_file == NULL
526
7.88k
    || filename_cmp (logical_input_file, fname)))
527
182
    logical_input_file = fname;
528
8.09k
}
529
530
void
531
new_logical_line (const char *fname, int line_number)
532
140
{
533
140
  new_logical_line_flags (fname, line_number, 0);
534
140
}
535
536
void
537
as_report_context (void)
538
120k
{
539
120k
  const struct input_save *saved;
540
120k
  enum expansion expansion = from_sb_expansion;
541
120k
  int indent = 1;
542
543
120k
  if (!macro_nest)
544
12.4k
    return;
545
546
264k
  for (saved = next_saved_file; saved; saved = saved->next_saved_file)
547
156k
    {
548
156k
      if (expansion != expanding_macro)
549
155k
  /* Nothing.  */;
550
684
      else if (saved->logical_input_file != NULL
551
0
         && saved->logical_input_line != -1u)
552
0
  as_info_where (saved->logical_input_file, saved->logical_input_line,
553
0
           indent, _("macro invoked from here"));
554
684
      else
555
684
  as_info_where (saved->physical_input_file, saved->physical_input_line,
556
684
           indent, _("macro invoked from here"));
557
558
156k
      expansion = saved->from_sb_expansion;
559
156k
      ++indent;
560
156k
    }
561
108k
}
562

563
/* Return the current physical input file name and line number, if known  */
564
565
const char *
566
as_where_physical (unsigned int *linep)
567
15.7k
{
568
15.7k
  if (physical_input_file != NULL)
569
15.7k
    {
570
15.7k
      if (linep != NULL)
571
15.7k
  *linep = physical_input_line;
572
15.7k
      return physical_input_file;
573
15.7k
    }
574
575
0
  if (linep != NULL)
576
0
    *linep = 0;
577
0
  return NULL;
578
15.7k
}
579
580
/* Return the file name and line number at the top most macro
581
   invocation, unless .file / .line were used inside a macro.  */
582
583
const char *
584
as_where (unsigned int *linep)
585
28.1k
{
586
28.1k
  const char *file = as_where_top (linep);
587
588
28.1k
  if (macro_nest && is_linefile)
589
23.8k
    {
590
23.8k
      const struct input_save *saved;
591
23.8k
      enum expansion expansion = from_sb_expansion;
592
593
47.7k
      for (saved = next_saved_file; saved; saved = saved->next_saved_file)
594
23.8k
  {
595
23.8k
    if (expansion != expanding_macro)
596
23.8k
      /* Nothing.  */;
597
8
    else if (saved->logical_input_file != NULL
598
0
       && (linep == NULL || saved->logical_input_line != -1u))
599
0
      {
600
0
        if (linep != NULL)
601
0
    *linep = saved->logical_input_line;
602
0
        file = saved->logical_input_file;
603
0
      }
604
8
    else if (saved->physical_input_file != NULL)
605
8
      {
606
8
        if (linep != NULL)
607
8
    *linep = saved->physical_input_line;
608
8
        file = saved->physical_input_file;
609
8
      }
610
611
23.8k
    expansion = saved->from_sb_expansion;
612
23.8k
  }
613
23.8k
    }
614
615
28.1k
  return file;
616
28.1k
}
617
618
/* Return the current file name and line number.  */
619
620
const char *
621
as_where_top (unsigned int *linep)
622
149k
{
623
149k
  if (logical_input_file != NULL
624
133k
      && (linep == NULL || logical_input_line != -1u))
625
133k
    {
626
133k
      if (linep != NULL)
627
133k
  *linep = logical_input_line;
628
133k
      return logical_input_file;
629
133k
    }
630
631
15.7k
  return as_where_physical (linep);
632
149k
}