x86: tighten extend-to-32bit-address conditions
[binutils-gdb.git] / gas / output-file.c
index f94359ab6b0a50eedd77cfafaf22d13cca52bf8d..664bc88e9c1ffcb88a4f196df16d39617c7615db 100644 (file)
@@ -1,12 +1,11 @@
 /* output-file.c -  Deal with the output file
-   Copyright 1987, 1990, 1991, 1992, 1993, 1994, 1996, 1998, 1999, 2001,
-   2003, 2004, 2005 Free Software Foundation, Inc.
+   Copyright (C) 1987-2023 Free Software Foundation, Inc.
 
    This file is part of GAS, the GNU Assembler.
 
    GAS is free software; you can redistribute it and/or modify
    it under the terms of the GNU General Public License as published by
-   the Free Software Foundation; either version 2, or (at your option)
+   the Free Software Foundation; either version 3, or (at your option)
    any later version.
 
    GAS is distributed in the hope that it will be useful,
    the Free Software Foundation, 51 Franklin Street - Fifth Floor, Boston, MA
    02110-1301, USA.  */
 
-#include <stdio.h>
-
 #include "as.h"
-
+#include "subsegs.h"
+#include "sb.h"
+#include "macro.h"
 #include "output-file.h"
 
 #ifndef TARGET_MACH
 #define TARGET_MACH 0
 #endif
 
-#include "bfd.h"
 bfd *stdoutput;
 
 void
-output_file_create (char *name)
+output_file_create (const char *name)
 {
   if (name[0] == '-' && name[1] == '\0')
     as_fatal (_("can't open a bfd on stdout %s"), name);
 
   else if (!(stdoutput = bfd_openw (name, TARGET_FORMAT)))
     {
-      if (bfd_get_error () == bfd_error_invalid_target)
-       as_perror (_("Selected target format '%s' unknown"), TARGET_FORMAT);
+      bfd_error_type err = bfd_get_error ();
+
+      if (err == bfd_error_invalid_target)
+       as_fatal (_("selected target format '%s' unknown"), TARGET_FORMAT);
       else
-       as_perror (_("FATAL: can't create %s"), name);
-      exit (EXIT_FAILURE);
+       as_fatal (_("can't create %s: %s"), name, bfd_errmsg (err));
     }
 
   bfd_set_format (stdoutput, bfd_object);
@@ -53,15 +52,73 @@ output_file_create (char *name)
     stdoutput->flags |= BFD_TRADITIONAL_FORMAT;
 }
 
-void
-output_file_close (char *filename)
+static void
+stash_frchain_obs (asection *sec)
 {
-  /* Close the bfd.  */
-  if (bfd_close (stdoutput) == 0)
+  segment_info_type *info = seg_info (sec);
+  if (info)
     {
-      bfd_perror (filename);
-      as_perror (_("FATAL: can't close %s\n"), filename);
-      exit (EXIT_FAILURE);
+      struct frchain *frchp;
+      for (frchp = info->frchainP; frchp; frchp = frchp->frch_next)
+       obstack_ptr_grow (&notes, &frchp->frch_obstack);
+      info->frchainP = NULL;
     }
-  stdoutput = NULL;            /* Trust nobody!  */
+}
+
+void
+output_file_close (void)
+{
+  bool res;
+  bfd *obfd = stdoutput;
+  struct obstack **obs;
+  asection *sec;
+  const char *filename;
+
+  if (obfd == NULL)
+    return;
+
+  /* Prevent an infinite loop - if the close failed we will call as_fatal
+     which will call xexit() which may call this function again...  */
+  stdoutput = NULL;
+
+  /* We can't free obstacks attached to the output bfd sections before
+     closing the output bfd since data in those obstacks may need to
+     be accessed, but we can't access anything in the output bfd after
+     it is closed..  */
+  for (sec = obfd->sections; sec; sec = sec->next)
+    stash_frchain_obs (sec);
+  stash_frchain_obs (reg_section);
+  stash_frchain_obs (expr_section);
+  stash_frchain_obs (bfd_abs_section_ptr);
+  stash_frchain_obs (bfd_und_section_ptr);
+  obstack_ptr_grow (&notes, NULL);
+  obs = obstack_finish (&notes);
+
+  /* Close the bfd.  */
+  if (!flag_always_generate_output && had_errors ())
+    res = bfd_close_all_done (obfd);
+  else
+    res = bfd_close (obfd);
+  now_seg = NULL;
+  now_subseg = 0;
+
+  filename = out_file_name;
+  out_file_name = NULL;
+  if (!keep_it && filename)
+    unlink_if_ordinary (filename);
+
+#ifdef md_end
+  md_end ();
+#endif
+#ifdef obj_end
+  obj_end ();
+#endif
+  macro_end ();
+  expr_end ();
+  read_end ();
+  symbol_end ();
+  subsegs_end (obs);
+
+  if (!res)
+    as_fatal ("%s: %s", filename, bfd_errmsg (bfd_get_error ()));
 }