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-file.c
Line
Count
Source
1
/* input_file.c - Deal with Input Files -
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
/* Confines all details of reading source bytes to this module.
22
   All O/S specific crocks should live here.
23
   What we lose in "efficiency" we gain in modularity.
24
   Note we don't need to #include the "as.h" file. No common coupling!  */
25
26
#include "as.h"
27
#include "input-file.h"
28
#include "safe-ctype.h"
29
30
/* This variable is non-zero if the file currently being read should be
31
   preprocessed by app.  It is zero if the file can be read straight in.  */
32
int preprocess = 0;
33
34
/* This code opens a file, then delivers BUFFER_SIZE character
35
   chunks of the file on demand.
36
   BUFFER_SIZE is supposed to be a number chosen for speed.
37
   The caller only asks once what BUFFER_SIZE is, and asks before
38
   the nature of the input files (if any) is known.  */
39
40
737
#define BUFFER_SIZE (32 * 1024)
41
42
/* We use static data: the data area is not sharable.  */
43
44
static FILE *f_in;
45
static const char *file_name;
46
47
/* Struct for saving the state of this module for file includes.  */
48
struct saved_file
49
  {
50
    FILE * f_in;
51
    const char * file_name;
52
    int    preprocess;
53
    char * app_save;
54
  };
55

56
/* These hooks accommodate most operating systems.  */
57
58
void
59
input_file_begin (void)
60
433
{
61
433
  f_in = NULL;
62
433
}
63
64
void
65
input_file_end (void)
66
433
{
67
433
}
68
69
/* Return BUFFER_SIZE.  */
70
size_t
71
input_file_buffer_size (void)
72
320
{
73
320
  return (BUFFER_SIZE);
74
320
}
75
76
/* Push the state of our input, returning a pointer to saved info that
77
   can be restored with input_file_pop ().  */
78
79
char *
80
input_file_push (void)
81
113
{
82
113
  struct saved_file *saved;
83
84
113
  saved = XNEW (struct saved_file);
85
86
113
  saved->f_in = f_in;
87
113
  saved->file_name = file_name;
88
113
  saved->preprocess = preprocess;
89
113
  if (preprocess)
90
113
    saved->app_save = app_push ();
91
92
  /* Initialize for new file.  */
93
113
  input_file_begin ();
94
95
113
  return (char *) saved;
96
113
}
97
98
void
99
input_file_pop (char *arg)
100
113
{
101
113
  struct saved_file *saved = (struct saved_file *) arg;
102
103
113
  input_file_end ();    /* Close out old file.  */
104
105
113
  f_in = saved->f_in;
106
113
  file_name = saved->file_name;
107
113
  preprocess = saved->preprocess;
108
113
  if (preprocess)
109
113
    app_pop (saved->app_save);
110
111
113
  free (arg);
112
113
}
113

114
/* Open the specified file, "" means stdin.  Filename must not be null.  */
115
116
void
117
input_file_open (const char *filename,
118
     int pre)
119
219
{
120
219
  int c;
121
219
  char buf[80];
122
123
219
  preprocess = pre;
124
125
219
  gas_assert (filename != 0); /* Filename may not be NULL.  */
126
219
  if (filename[0])
127
219
    {
128
219
      f_in = fopen (filename, FOPEN_RT);
129
219
      file_name = filename;
130
219
    }
131
0
  else
132
0
    {
133
      /* Use stdin for the input file.  */
134
0
      f_in = stdin;
135
      /* For error messages.  */
136
0
      file_name = _("{standard input}");
137
0
    }
138
139
219
  if (f_in == NULL)
140
0
    {
141
0
      as_bad (_("can't open %s for reading: %s"),
142
0
        file_name, xstrerror (errno));
143
0
      return;
144
0
    }
145
146
219
  c = getc (f_in);
147
148
219
  if (ferror (f_in))
149
12
    {
150
12
      as_bad (_("can't read from %s: %s"),
151
12
        file_name, xstrerror (errno));
152
153
12
      fclose (f_in);
154
12
      f_in = NULL;
155
12
      return;
156
12
    }
157
158
  /* Check for an empty input file.  */
159
207
  if (feof (f_in))
160
0
    {
161
0
      fclose (f_in);
162
0
      f_in = NULL;
163
0
      return;
164
0
    }
165
207
  gas_assert (c != EOF);
166
167
207
  if (strchr (line_comment_chars, '#')
168
207
      ? c == '#'
169
207
      : c && strchr (line_comment_chars, c))
170
8
    {
171
      /* Begins with comment, may not want to preprocess.  */
172
8
      int lead = c;
173
174
8
      c = getc (f_in);
175
8
      if (c == 'N')
176
4
  {
177
4
    char *p = fgets (buf, sizeof (buf), f_in);
178
4
    if (p && startswith (p, "O_APP") && is_end_of_line (p[5]))
179
0
      preprocess = 0;
180
4
    if (!p || !strchr (p, '\n'))
181
3
      ungetc (lead, f_in);
182
1
    else
183
1
      ungetc ('\n', f_in);
184
4
  }
185
4
      else if (c == 'A')
186
0
  {
187
0
    char *p = fgets (buf, sizeof (buf), f_in);
188
0
    if (p && startswith (p, "PP") && is_end_of_line (p[2]))
189
0
      preprocess = 1;
190
0
    if (!p || !strchr (p, '\n'))
191
0
      ungetc (lead, f_in);
192
0
    else
193
0
      ungetc ('\n', f_in);
194
0
  }
195
4
      else if (c == '\n')
196
0
  ungetc ('\n', f_in);
197
4
      else
198
4
  ungetc (lead, f_in);
199
8
    }
200
199
  else
201
199
    ungetc (c, f_in);
202
207
}
203
204
/* Close input file.  */
205
206
void
207
input_file_close (void)
208
207
{
209
  /* Don't close a null file pointer.  */
210
207
  if (f_in != NULL)
211
0
    fclose (f_in);
212
213
207
  f_in = 0;
214
207
}
215
216
/* This function is passed to do_scrub_chars.  */
217
218
static size_t
219
input_file_get (char *buf, size_t buflen)
220
704
{
221
704
  size_t size;
222
223
704
  if (feof (f_in))
224
497
    return 0;
225
226
207
  size = fread (buf, sizeof (char), buflen, f_in);
227
207
  if (ferror (f_in))
228
0
    as_bad (_("can't read from %s: %s"), file_name, xstrerror (errno));
229
207
  return size;
230
704
}
231
232
/* Read a buffer from the input file.  */
233
234
char *
235
input_file_give_next_buffer (char *where /* Where to place 1st character of new buffer.  */)
236
443
{
237
443
  char *return_value;   /* -> Last char of what we read, + 1.  */
238
443
  size_t size;
239
240
443
  if (f_in == NULL)
241
26
    return 0;
242
  /* fflush (stdin); could be done here if you want to synchronise
243
     stdin and stdout, for the case where our input file is stdin.
244
     Since the assembler shouldn't do any output to stdout, we
245
     don't bother to synch output and input.  */
246
417
  if (preprocess)
247
417
    size = do_scrub_chars (input_file_get, where, BUFFER_SIZE,
248
417
                           multibyte_handling == multibyte_warn);
249
0
  else
250
0
    {
251
0
      size = input_file_get (where, BUFFER_SIZE);
252
253
0
      if (multibyte_handling == multibyte_warn)
254
0
  {
255
0
    const unsigned char *start = (const unsigned char *) where;
256
257
0
    (void) scan_for_multibyte_characters (start, start + size,
258
0
            true /* Generate warnings */);
259
0
  }
260
0
    }
261
262
417
  if (size)
263
210
    return_value = where + size;
264
207
  else
265
207
    {
266
207
      if (fclose (f_in))
267
0
  as_warn (_("can't close %s: %s"), file_name, xstrerror (errno));
268
269
207
      f_in = NULL;
270
207
      return_value = 0;
271
207
    }
272
273
417
  return return_value;
274
443
}