Coverage Report

Created: 2026-05-11 07:54

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/binutils-gdb/gas/output-file.c
Line
Count
Source
1
/* output-file.c -  Deal with the output file
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
18
   the Free Software Foundation, 51 Franklin Street - Fifth Floor, Boston, MA
19
   02110-1301, USA.  */
20
21
#include "as.h"
22
#include "subsegs.h"
23
#include "sb.h"
24
#include "macro.h"
25
#include "output-file.h"
26
27
#ifndef TARGET_MACH
28
#define TARGET_MACH 0
29
#endif
30
31
bfd *stdoutput;
32
33
void
34
output_file_create (const char *name)
35
478
{
36
478
  if (name[0] == '-' && name[1] == '\0')
37
0
    as_fatal (_("can't open a bfd on stdout %s"), name);
38
39
478
  else if (!(stdoutput = bfd_openw (name, TARGET_FORMAT)))
40
0
    {
41
0
      bfd_error_type err = bfd_get_error ();
42
43
0
      if (err == bfd_error_invalid_target)
44
0
  as_fatal (_("selected target format '%s' unknown"), TARGET_FORMAT);
45
0
      else
46
0
  as_fatal (_("can't create %s: %s"), name, bfd_errmsg (err));
47
0
    }
48
49
478
  bfd_set_format (stdoutput, bfd_object);
50
478
  bfd_set_arch_mach (stdoutput, TARGET_ARCH, TARGET_MACH);
51
478
  if (flag_traditional_format)
52
0
    stdoutput->flags |= BFD_TRADITIONAL_FORMAT;
53
478
}
54
55
static void
56
stash_frchain_obs (asection *sec)
57
5.21k
{
58
5.21k
  segment_info_type *info = seg_info (sec);
59
5.21k
  if (info)
60
5.21k
    {
61
5.21k
      struct frchain *frchp;
62
9.48k
      for (frchp = info->frchainP; frchp; frchp = frchp->frch_next)
63
4.27k
  obstack_ptr_grow (&notes, &frchp->frch_obstack);
64
5.21k
      info->frchainP = NULL;
65
5.21k
    }
66
5.21k
}
67
68
void
69
output_file_close (void)
70
478
{
71
478
  bool res;
72
478
  bfd *obfd = stdoutput;
73
478
  struct obstack **obs;
74
478
  asection *sec;
75
478
  const char *filename;
76
77
478
  if (obfd == NULL)
78
0
    return;
79
80
  /* Prevent an infinite loop - if the close failed we will call as_fatal
81
     which will call xexit() which may call this function again...  */
82
478
  stdoutput = NULL;
83
84
478
  if (ENABLE_LEAK_CHECK)
85
478
    {
86
      /* We can't free obstacks attached to the output bfd sections before
87
   closing the output bfd since data in those obstacks may need to
88
   be accessed, but we can't access anything in the output bfd after
89
   it is closed..  */
90
3.77k
      for (sec = obfd->sections; sec; sec = sec->next)
91
3.29k
  stash_frchain_obs (sec);
92
478
      stash_frchain_obs (reg_section);
93
478
      stash_frchain_obs (expr_section);
94
478
      stash_frchain_obs (bfd_abs_section_ptr);
95
478
      stash_frchain_obs (bfd_und_section_ptr);
96
478
      obstack_ptr_grow (&notes, NULL);
97
478
      obs = obstack_finish (&notes);
98
478
    }
99
0
  else
100
0
    obs = NULL;
101
102
  /* Close the bfd.  */
103
478
  if (!flag_always_generate_output && had_errors ())
104
478
    res = bfd_close_all_done (obfd);
105
0
  else
106
0
    res = bfd_close (obfd);
107
478
  now_seg = NULL;
108
478
  now_subseg = 0;
109
110
478
  filename = out_file_name;
111
478
  out_file_name = NULL;
112
478
  if (!keep_it && filename)
113
478
    unlink_if_ordinary (filename);
114
115
478
#ifdef md_end
116
478
  md_end ();
117
478
#endif
118
478
#ifdef obj_end
119
478
  obj_end ();
120
478
#endif
121
478
  macro_end ();
122
478
  expr_end ();
123
478
  read_end ();
124
478
  symbol_end ();
125
478
  subsegs_end (obs);
126
127
478
  if (!res)
128
0
    as_fatal ("%s: %s", filename, bfd_errmsg (bfd_get_error ()));
129
478
}