Coverage Report

Created: 2026-04-04 08:16

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/binutils-gdb/gas/subsegs.c
Line
Count
Source
1
/* subsegs.c - subsegments -
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
/* Segments & sub-segments.  */
22
23
#include "as.h"
24
25
#include "subsegs.h"
26
#include "obstack.h"
27
28
frchainS *frchain_now;
29
30
static struct obstack frchains;
31
32
static fragS dummy_frag;
33
34

35
void
36
subsegs_begin (void)
37
567
{
38
567
  obstack_begin (&frchains, chunksize);
39
567
#if __GNUC__ >= 2
40
567
  obstack_alignment_mask (&frchains) = __alignof__ (frchainS) - 1;
41
567
#endif
42
43
567
  frchain_now = NULL;   /* Warn new_subseg() that we are booting.  */
44
567
  frag_now = &dummy_frag;
45
567
}
46
47
void
48
subsegs_end (struct obstack **obs)
49
567
{
50
567
  if (!ENABLE_LEAK_CHECK)
51
0
    return;
52
7.76k
  for (; *obs; obs++)
53
7.19k
    _obstack_free (*obs, NULL);
54
567
  _obstack_free (&frchains, NULL);
55
567
  bfd_set_section_userdata (bfd_com_section_ptr, NULL);
56
567
  bfd_set_section_userdata (bfd_und_section_ptr, NULL);
57
567
  bfd_set_section_userdata (bfd_abs_section_ptr, NULL);
58
567
  bfd_set_section_userdata (bfd_ind_section_ptr, NULL);
59
  /* Reverse bfd_std_section_init, so the sections look as they did
60
     initially.  This, and clearing out userdata above, is so we don't
61
     leave dangling pointers into freed memory for oss-fuzz to mess
62
     with.  */
63
567
  asymbol *global_syms = bfd_com_section_ptr->symbol;
64
567
  bfd_und_section_ptr->used_by_bfd = NULL;
65
567
  bfd_und_section_ptr->symbol = global_syms + (bfd_und_section_ptr
66
567
                 - bfd_com_section_ptr);
67
567
  bfd_abs_section_ptr->used_by_bfd = NULL;
68
567
  bfd_abs_section_ptr->symbol = global_syms + (bfd_abs_section_ptr
69
567
                 - bfd_com_section_ptr);
70
567
}
71

72
static void
73
alloc_seginfo (segT seg)
74
7.15k
{
75
7.15k
  segment_info_type *seginfo;
76
77
7.15k
  seginfo = obstack_alloc (&notes, sizeof (*seginfo));
78
7.15k
  memset (seginfo, 0, sizeof (*seginfo));
79
7.15k
  bfd_set_section_userdata (seg, seginfo);
80
7.15k
}
81
/*
82
 *      subseg_change()
83
 *
84
 * Change the subsegment we are in, BUT DO NOT MAKE A NEW FRAG for the
85
 * subsegment. If we are already in the correct subsegment, change nothing.
86
 * This is used eg as a worker for subseg_set [which does make a new frag_now]
87
 * and for changing segments after we have read the source. We construct eg
88
 * fixSs even after the source file is read, so we do have to keep the
89
 * segment context correct.
90
 */
91
void
92
subseg_change (segT seg, int subseg)
93
15.4k
{
94
15.4k
  now_seg = seg;
95
15.4k
  now_subseg = subseg;
96
97
15.4k
  if (!seg_info (seg))
98
0
    alloc_seginfo (seg);
99
15.4k
}
100

101
static void
102
subseg_set_rest (segT seg, subsegT subseg)
103
15.4k
{
104
15.4k
  frchainS *frcP;   /* crawl frchain chain */
105
15.4k
  frchainS **lastPP;    /* address of last pointer */
106
15.4k
  frchainS *newP;   /* address of new frchain */
107
15.4k
  segment_info_type *seginfo;
108
109
15.4k
  mri_common_symbol = NULL;
110
111
15.4k
  if (frag_now && frchain_now)
112
14.8k
    frchain_now->frch_frag_now = frag_now;
113
114
15.4k
  gas_assert (frchain_now == 0
115
15.4k
    || frchain_now->frch_last == frag_now);
116
117
15.4k
  subseg_change (seg, (int) subseg);
118
119
15.4k
  seginfo = seg_info (seg);
120
121
  /* Should the section symbol be kept?  */
122
15.4k
  if (bfd_keep_unused_section_symbols (stdoutput))
123
0
    seg->symbol->flags |= BSF_SECTION_SYM_USED;
124
125
  /* Attempt to find or make a frchain for that subsection.
126
     We keep the list sorted by subsection number.  */
127
15.4k
  for (frcP = *(lastPP = &seginfo->frchainP);
128
15.5k
       frcP != NULL;
129
15.4k
       frcP = *(lastPP = &frcP->frch_next))
130
8.38k
    if (frcP->frch_subseg >= subseg)
131
8.28k
      break;
132
133
15.4k
  if (frcP == NULL || frcP->frch_subseg != subseg)
134
7.19k
    {
135
      /* This should be the only code that creates a frchainS.  */
136
137
7.19k
      newP = obstack_alloc (&frchains, sizeof (frchainS));
138
7.19k
      newP->frch_subseg = subseg;
139
7.19k
      newP->fix_root = NULL;
140
7.19k
      newP->fix_tail = NULL;
141
7.19k
      obstack_begin (&newP->frch_obstack, chunksize);
142
7.19k
#if __GNUC__ >= 2
143
7.19k
      obstack_alignment_mask (&newP->frch_obstack) = __alignof__ (fragS) - 1;
144
7.19k
#endif
145
7.19k
      newP->frch_frag_now = frag_alloc (&newP->frch_obstack, 0);
146
7.19k
      newP->frch_frag_now->fr_type = rs_fill;
147
7.19k
      newP->frch_cfi_data = NULL;
148
7.19k
      newP->frch_ginsn_data = NULL;
149
150
7.19k
      newP->frch_root = newP->frch_last = newP->frch_frag_now;
151
152
7.19k
      *lastPP = newP;
153
7.19k
      newP->frch_next = frcP;
154
7.19k
      frcP = newP;
155
7.19k
    }
156
157
15.4k
  frchain_now = frcP;
158
15.4k
  frag_now = frcP->frch_frag_now;
159
160
15.4k
  gas_assert (frchain_now->frch_last == frag_now);
161
15.4k
}
162
163
/*
164
 *      subseg_set(segT, subsegT)
165
 *
166
 * If you attempt to change to the current subsegment, nothing happens.
167
 *
168
 * In:  segT, subsegT code for new subsegment.
169
 *  frag_now -> incomplete frag for current subsegment.
170
 *  If frag_now==NULL, then there is no old, incomplete frag, so
171
 *  the old frag is not closed off.
172
 *
173
 * Out: now_subseg, now_seg updated.
174
 *  Frchain_now points to the (possibly new) struct frchain for this
175
 *  sub-segment.
176
 */
177
178
segT
179
subseg_get (const char *segname, int force_new)
180
7.72k
{
181
7.72k
  segT secptr;
182
7.72k
  const char *now_seg_name = now_seg ? bfd_section_name (now_seg) : 0;
183
184
7.72k
  if (!force_new
185
5.43k
      && now_seg_name
186
4.86k
      && (now_seg_name == segname
187
4.85k
    || !strcmp (now_seg_name, segname)))
188
17
    return now_seg;
189
190
7.70k
  if (!force_new)
191
5.41k
    secptr = bfd_make_section_old_way (stdoutput, segname);
192
2.29k
  else
193
2.29k
    secptr = bfd_make_section_anyway (stdoutput, segname);
194
195
7.70k
  if (!seg_info (secptr))
196
7.15k
    {
197
7.15k
      secptr->output_section = secptr;
198
7.15k
      alloc_seginfo (secptr);
199
7.15k
    }
200
7.70k
  return secptr;
201
7.72k
}
202
203
segT
204
subseg_new (const char *segname, subsegT subseg)
205
5.43k
{
206
5.43k
  segT secptr;
207
208
5.43k
  secptr = subseg_get (segname, 0);
209
5.43k
  subseg_set_rest (secptr, subseg);
210
5.43k
  return secptr;
211
5.43k
}
212
213
/* Like subseg_new, except a new section is always created, even if
214
   a section with that name already exists.  */
215
segT
216
subseg_force_new (const char *segname, subsegT subseg)
217
2.29k
{
218
2.29k
  segT secptr;
219
220
2.29k
  secptr = subseg_get (segname, 1);
221
2.29k
  subseg_set_rest (secptr, subseg);
222
2.29k
  return secptr;
223
2.29k
}
224
225
void
226
subseg_set (segT secptr, subsegT subseg)
227
8.80k
{
228
8.80k
  if (! (secptr == now_seg && subseg == now_subseg))
229
7.73k
    subseg_set_rest (secptr, subseg);
230
8.80k
  mri_common_symbol = NULL;
231
8.80k
}
232
233
#ifndef obj_sec_sym_ok_for_reloc
234
#define obj_sec_sym_ok_for_reloc(SEC) 0
235
#endif
236
237
symbolS *
238
section_symbol (segT sec)
239
4.66k
{
240
4.66k
  segment_info_type *seginfo = seg_info (sec);
241
4.66k
  symbolS *s;
242
243
4.66k
  if (seginfo == 0)
244
0
    abort ();
245
4.66k
  if (seginfo->sym)
246
566
    return seginfo->sym;
247
248
4.09k
#ifndef EMIT_SECTION_SYMBOLS
249
8.19k
#define EMIT_SECTION_SYMBOLS 1
250
4.09k
#endif
251
252
  /* A reference to the section (ie. an undefined symbol)
253
     should now become defined, but any other symbol that happens to
254
     have the same name as the section should not be modified.  Make
255
     sure an undefined_section symbol isn't equated to some other
256
     undefined symbol.  */
257
4.09k
  s = symbol_find (sec->symbol->name);
258
4.09k
  if (s == NULL
259
2.17k
      || S_GET_SEGMENT (s) != undefined_section
260
0
      || !symbol_constant_p (s))
261
4.09k
    {
262
4.09k
      if (!EMIT_SECTION_SYMBOLS || symbol_table_frozen)
263
0
  s = symbol_create (sec->symbol->name, sec, &zero_address_frag, 0);
264
4.09k
      else
265
4.09k
  s = symbol_new (sec->symbol->name, sec, &zero_address_frag, 0);
266
4.09k
    }
267
0
  else
268
0
    S_SET_SEGMENT (s, sec);
269
270
4.09k
  S_CLEAR_EXTERNAL (s);
271
272
  /* Use the BFD section symbol, if possible.  */
273
4.09k
  if (obj_sec_sym_ok_for_reloc (sec))
274
4.09k
    symbol_set_bfdsym (s, sec->symbol);
275
0
  else
276
0
    symbol_get_bfdsym (s)->flags |= BSF_SECTION_SYM;
277
278
4.09k
  seginfo->sym = s;
279
4.09k
  return s;
280
4.66k
}
281
282
/* Return whether the specified segment is thought to hold text.  */
283
284
int
285
subseg_text_p (segT sec)
286
824
{
287
824
  return (bfd_section_flags (sec) & SEC_CODE) != 0;
288
824
}
289
290
/* Return non zero if SEC has at least one byte of data.  It is
291
   possible that we'll return zero even on a non-empty section because
292
   we don't know all the fragment types, and it is possible that an
293
   fr_fix == 0 one still contributes data.  Think of this as
294
   seg_definitely_not_empty_p.  */
295
296
int
297
seg_not_empty_p (segT sec ATTRIBUTE_UNUSED)
298
1
{
299
1
  segment_info_type *seginfo = seg_info (sec);
300
1
  frchainS *chain;
301
1
  fragS *frag;
302
303
1
  if (!seginfo)
304
0
    return 0;
305
306
1
  for (chain = seginfo->frchainP; chain; chain = chain->frch_next)
307
1
    {
308
2
      for (frag = chain->frch_root; frag; frag = frag->fr_next)
309
1
  if (frag->fr_fix)
310
0
    return 1;
311
1
      if (obstack_next_free (&chain->frch_obstack)
312
1
    != chain->frch_last->fr_literal)
313
1
  return 1;
314
1
    }
315
0
  return 0;
316
1
}
317
318
void
319
subsegs_print_statistics (FILE *file)
320
0
{
321
0
  frchainS *frchp;
322
0
  asection *s;
323
324
  /* PR 20897 - check to see if the output bfd was actually created.  */
325
0
  if (stdoutput == NULL)
326
0
    return;
327
328
0
  fprintf (file, "frag chains:\n");
329
0
  for (s = stdoutput->sections; s; s = s->next)
330
0
    {
331
0
      segment_info_type *seginfo;
332
333
      /* Skip gas-internal sections.  */
334
0
      if (segment_name (s)[0] == '*')
335
0
  continue;
336
337
0
      seginfo = seg_info (s);
338
0
      if (!seginfo)
339
0
  continue;
340
341
0
      for (frchp = seginfo->frchainP; frchp; frchp = frchp->frch_next)
342
0
  {
343
0
    int count = 0;
344
0
    fragS *fragp;
345
346
0
    for (fragp = frchp->frch_root; fragp; fragp = fragp->fr_next)
347
0
      count++;
348
349
0
    fprintf (file, "\n");
350
0
    fprintf (file, "\t%p %-10s\t%10d frags\n", (void *) frchp,
351
0
       segment_name (s), count);
352
0
  }
353
0
    }
354
0
}
355
356
/* end of subsegs.c */