]> git.ipfire.org Git - thirdparty/gcc.git/commitdiff
c++: Add a late-writing step for modules
authorNathan Sidwell <nathan@acm.org>
Thu, 9 Jun 2022 15:14:31 +0000 (08:14 -0700)
committerNathan Sidwell <nathan@acm.org>
Fri, 10 Jun 2022 19:32:22 +0000 (12:32 -0700)
To add a module initializer optimization, we need to defer finishing writing
out the module file until the end of determining the dynamic initializers.
This is achieved by passing some saved-state from the main module writing
to a new function that completes it.

This patch merely adds the skeleton of that state and move things around,
allowing the finalization of the ELF file to be postponed.  None of the
contents writing is moved, or the init optimization added.

gcc/cp/
* cp-tree.h (fini_modules): Add some parameters.
(finish_module_processing): Return an opaque pointer.
* decl2.cc (c_parse_final_cleanups): Propagate a cookie from
finish_module_processing to fini_modules.
* module.cc (struct module_processing_cookie): New.
(finish_module_processing): Return a heap-allocated cookie.
(late_finish_module): New.  Finish out the module writing.
(fini_modules): Adjust.

gcc/cp/cp-tree.h
gcc/cp/decl2.cc
gcc/cp/module.cc

index f1294dac7d5b6a254582fcf572606015136e41a2..60d7b20159538362dd305992e2d4390a66b17dff 100644 (file)
@@ -7209,9 +7209,9 @@ extern void import_module (module_state *, location_t, bool export_p,
 extern void declare_module (module_state *, location_t, bool export_p,
                            tree attr, cpp_reader *);
 extern void init_modules (cpp_reader *);
-extern void fini_modules ();
+extern void fini_modules (cpp_reader *, void *cookie);
 extern void maybe_check_all_macros (cpp_reader *);
-extern void finish_module_processing (cpp_reader *);
+extern void *finish_module_processing (cpp_reader *);
 extern char const *module_name (unsigned, bool header_ok);
 extern bitmap get_import_bitmap ();
 extern bitmap visible_instantiation_path (bitmap *);
index ff1c36745cf4a48170353cdff2bff7c4f64e1a10..cc0b41324b3a1b5027f9dcbbb81bd57e2155306f 100644 (file)
@@ -5154,7 +5154,7 @@ c_parse_final_cleanups (void)
        reconsider = true;
     }
 
-  finish_module_processing (parse_in);
+  void *module_cookie = finish_module_processing (parse_in);
 
   lower_var_init ();
 
@@ -5238,7 +5238,7 @@ c_parse_final_cleanups (void)
       }
   pop_lang_context ();
 
-  fini_modules ();
+  fini_modules (parse_in, module_cookie);
 
   /* Generate any missing aliases.  */
   maybe_apply_pending_pragma_weaks ();
index 2b1877ea82e3c30fa8297df3b6a65d5046470844..51d774ae608c9d5f3b8142ff65d6b7da2b7e86f5 100644 (file)
@@ -19854,11 +19854,32 @@ maybe_check_all_macros (cpp_reader *reader)
   dump.pop (n);
 }
 
+// State propagated from finish_module_processing to fini_modules
+struct module_processing_cookie
+{
+  elf_out out;
+  char *cmi_name;
+  char *tmp_name;
+  bool began;
+
+  module_processing_cookie (char *cmi, char *tmp, int fd, int e)
+    : out (fd, e), cmi_name (cmi), tmp_name (tmp), began (false)
+  {
+  }
+  ~module_processing_cookie ()
+  {
+    XDELETEVEC (tmp_name);
+    XDELETEVEC (cmi_name);
+  }
+};
+
 /* Write the CMI, if we're a module interface.  */
 
-void
+void *
 finish_module_processing (cpp_reader *reader)
 {
+  module_processing_cookie *cookie = nullptr;
+
   if (header_module_p ())
     module_kind &= ~MK_EXPORTING;
 
@@ -19870,7 +19891,7 @@ finish_module_processing (cpp_reader *reader)
   else if (!flag_syntax_only)
     {
       int fd = -1;
-      int e = ENOENT;
+      int e = -1;
 
       timevar_start (TV_MODULE_EXPORT);
 
@@ -19879,7 +19900,7 @@ finish_module_processing (cpp_reader *reader)
       linemap_add (line_table, LC_ENTER, false, "", 0);
 
       /* We write to a tmpname, and then atomically rename.  */
-      const char *path = NULL;
+      char *cmi_name = NULL;
       char *tmp_name = NULL;
       module_state *state = (*modules)[0];
 
@@ -19888,9 +19909,9 @@ finish_module_processing (cpp_reader *reader)
       if (state->filename)
        {
          size_t len = 0;
-         path = maybe_add_cmi_prefix (state->filename, &len);
+         cmi_name = xstrdup (maybe_add_cmi_prefix (state->filename, &len));
          tmp_name = XNEWVEC (char, len + 3);
-         memcpy (tmp_name, path, len);
+         memcpy (tmp_name, cmi_name, len);
          strcpy (&tmp_name[len], "~");
 
          if (!errorcount)
@@ -19905,57 +19926,23 @@ finish_module_processing (cpp_reader *reader)
                create_dirs (tmp_name);
              }
          if (note_module_cmi_yes || state->inform_cmi_p)
-           inform (state->loc, "writing CMI %qs", path);
-         dump () && dump ("CMI is %s", path);
+           inform (state->loc, "writing CMI %qs", cmi_name);
+         dump () && dump ("CMI is %s", cmi_name);
        }
 
+      cookie = new module_processing_cookie (cmi_name, tmp_name, fd, e);
+
       if (errorcount)
        warning_at (state->loc, 0, "not writing module %qs due to errors",
                    state->get_flatname ());
-      else
+      else if (cookie->out.begin ())
        {
-         elf_out to (fd, e);
-         if (to.begin ())
-           {
-             auto loc = input_location;
-             /* So crashes finger-point the module decl.  */
-             input_location = state->loc;
-             state->write (&to, reader);
-             input_location = loc;
-           }
-         if (to.end ())
-           {
-             /* Some OS's do not replace NEWNAME if it already
-                exists.  This'll have a race condition in erroneous
-                concurrent builds.  */
-             unlink (path);
-             if (rename (tmp_name, path))
-               {
-                 dump () && dump ("Rename ('%s','%s') errno=%u", errno);
-                 to.set_error (errno);
-               }
-           }
-
-         if (to.get_error ())
-           {
-             error_at (state->loc, "failed to write compiled module: %s",
-                       to.get_error (state->filename));
-             state->note_cmi_name ();
-           }
-       }
-
-      if (!errorcount)
-       {
-         auto *mapper = get_mapper (cpp_main_loc (reader));
-
-         mapper->ModuleCompiled (state->get_flatname ());
-       }
-      else if (path)
-       {
-         /* We failed, attempt to erase all evidence we even tried.  */
-         unlink (tmp_name);
-         unlink (path);
-         XDELETEVEC (tmp_name);
+         cookie->began = true;
+         auto loc = input_location;
+         /* So crashes finger-point the module decl.  */
+         input_location = state->loc;
+         state->write (&cookie->out, reader);
+         input_location = loc;
        }
 
       dump.pop (n);
@@ -19974,11 +19961,67 @@ finish_module_processing (cpp_reader *reader)
                       (available_clusters + !available_clusters));
       dump.pop (n);
     }
+
+  return cookie;
+}
+
+// Do the final emission of a module.  At this point we know whether
+// the module static initializer is a NOP or not.
+
+static void
+late_finish_module (cpp_reader *reader, module_processing_cookie *cookie)
+{
+  timevar_start (TV_MODULE_EXPORT);
+
+  module_state *state = (*modules)[0];
+  unsigned n = dump.push (state);
+  state->announce ("finishing");
+
+  if (cookie->out.end () && cookie->cmi_name)
+    {
+      /* Some OS's do not replace NEWNAME if it already exists.
+        This'll have a race condition in erroneous concurrent
+        builds.  */
+      unlink (cookie->cmi_name);
+      if (rename (cookie->tmp_name, cookie->cmi_name))
+       {
+         dump () && dump ("Rename ('%s','%s') errno=%u",
+                          cookie->tmp_name, cookie->cmi_name, errno);
+         cookie->out.set_error (errno);
+       }
+    }
+
+  if (cookie->out.get_error () && cookie->began)
+    {
+      error_at (state->loc, "failed to write compiled module: %s",
+               cookie->out.get_error (state->filename));
+      state->note_cmi_name ();
+    }
+
+  if (!errorcount)
+    {
+      auto *mapper = get_mapper (cpp_main_loc (reader));
+      mapper->ModuleCompiled (state->get_flatname ());
+    }
+  else if (cookie->cmi_name)
+    {
+      /* We failed, attempt to erase all evidence we even tried.  */
+      unlink (cookie->tmp_name);
+      unlink (cookie->cmi_name);
+    }
+
+  delete cookie;
+  dump.pop (n);
+  timevar_stop (TV_MODULE_EXPORT);
 }
 
 void
-fini_modules ()
+fini_modules (cpp_reader *reader, void *cookie)
 {
+  if (cookie)
+    late_finish_module (reader,
+                       static_cast<module_processing_cookie *> (cookie));
+
   /* We're done with the macro tables now.  */
   vec_free (macro_exports);
   vec_free (macro_imports);