Coverage Report

Created: 2023-06-29 07:09

/src/binutils-gdb/gas/output-file.c
Line
Count
Source (jump to first uncovered line)
1
/* output-file.c -  Deal with the output file
2
   Copyright (C) 1987-2023 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
633
{
36
633
  if (name[0] == '-' && name[1] == '\0')
37
0
    as_fatal (_("can't open a bfd on stdout %s"), name);
38
39
633
  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
633
  bfd_set_format (stdoutput, bfd_object);
50
633
  bfd_set_arch_mach (stdoutput, TARGET_ARCH, TARGET_MACH);
51
633
  if (flag_traditional_format)
52
0
    stdoutput->flags |= BFD_TRADITIONAL_FORMAT;
53
633
}
54
55
static void
56
stash_frchain_obs (asection *sec)
57
8.36k
{
58
8.36k
  segment_info_type *info = seg_info (sec);
59
8.36k
  if (info)
60
8.36k
    {
61
8.36k
      struct frchain *frchp;
62
16.0k
      for (frchp = info->frchainP; frchp; frchp = frchp->frch_next)
63
7.71k
  obstack_ptr_grow (&notes, &frchp->frch_obstack);
64
8.36k
      info->frchainP = NULL;
65
8.36k
    }
66
8.36k
}
67
68
void
69
output_file_close (void)
70
633
{
71
633
  bool res;
72
633
  bfd *obfd = stdoutput;
73
633
  struct obstack **obs;
74
633
  asection *sec;
75
633
  const char *filename;
76
77
633
  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
633
  stdoutput = NULL;
83
84
  /* We can't free obstacks attached to the output bfd sections before
85
     closing the output bfd since data in those obstacks may need to
86
     be accessed, but we can't access anything in the output bfd after
87
     it is closed..  */
88
6.46k
  for (sec = obfd->sections; sec; sec = sec->next)
89
5.82k
    stash_frchain_obs (sec);
90
633
  stash_frchain_obs (reg_section);
91
633
  stash_frchain_obs (expr_section);
92
633
  stash_frchain_obs (bfd_abs_section_ptr);
93
633
  stash_frchain_obs (bfd_und_section_ptr);
94
633
  obstack_ptr_grow (&notes, NULL);
95
633
  obs = obstack_finish (&notes);
96
97
  /* Close the bfd.  */
98
633
  if (!flag_always_generate_output && had_errors ())
99
633
    res = bfd_close_all_done (obfd);
100
0
  else
101
0
    res = bfd_close (obfd);
102
633
  now_seg = NULL;
103
633
  now_subseg = 0;
104
105
633
  filename = out_file_name;
106
633
  out_file_name = NULL;
107
633
  if (!keep_it && filename)
108
633
    unlink_if_ordinary (filename);
109
110
633
#ifdef md_end
111
633
  md_end ();
112
633
#endif
113
633
#ifdef obj_end
114
633
  obj_end ();
115
633
#endif
116
633
  macro_end ();
117
633
  expr_end ();
118
633
  read_end ();
119
633
  symbol_end ();
120
633
  subsegs_end (obs);
121
122
633
  if (!res)
123
0
    as_fatal ("%s: %s", filename, bfd_errmsg (bfd_get_error ()));
124
633
}