aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorJason Merrill <jason@redhat.com>2025-04-01 19:22:18 -0400
committerJason Merrill <jason@redhat.com>2025-04-03 11:19:30 -0400
commitacfe19f03bccd1f3183fd5d70d7a3c8c85e62fac (patch)
treedf1fade65c7da69b2569565ef19eb845dc82c104
parentbd5597156ca0c7d6fb50c6fe92a7abe6717cb2b5 (diff)
downloadgcc-acfe19f03bccd1f3183fd5d70d7a3c8c85e62fac.zip
gcc-acfe19f03bccd1f3183fd5d70d7a3c8c85e62fac.tar.gz
gcc-acfe19f03bccd1f3183fd5d70d7a3c8c85e62fac.tar.bz2
c++/modules: inline loaded at eof
std/format/string.cc and a few other libstdc++ tests were failing with module std with undefined references to __failed_to_parse_format_spec. This turned out to be because since r15-8012 we don't end up calling note_vague_linkage_fn for functions loaded after at_eof is set. But once import_export_decl decides on COMDAT linkage, we should be able to just clear DECL_EXTERNAL and let cgraph take it from there. I initially made this change in import_export_decl, but decided that for GCC 15 it would be safer to limit the change to modules. For GCC 16 I'd like to do away with DECL_NOT_REALLY_EXTERN entirely, it's been obsolete since cgraphunit in 2003. gcc/cp/ChangeLog: * module.cc (module_state::read_cluster) (post_load_processing): Clear DECL_EXTERNAL if DECL_COMDAT.
-rw-r--r--gcc/cp/module.cc13
1 files changed, 13 insertions, 0 deletions
diff --git a/gcc/cp/module.cc b/gcc/cp/module.cc
index ce22b2e..89deabb 100644
--- a/gcc/cp/module.cc
+++ b/gcc/cp/module.cc
@@ -16679,6 +16679,15 @@ module_state::read_cluster (unsigned snum)
#endif
cfun->returns_struct = aggr;
expand_or_defer_fn (decl);
+
+ /* If we first see this function after at_eof, it doesn't get
+ note_vague_linkage_fn from tentative_decl_linkage, so the loop in
+ c_parse_final_cleanups won't consider it. But with DECL_COMDAT we
+ can just clear DECL_EXTERNAL and let cgraph decide.
+ FIXME handle this outside module.cc after GCC 15. */
+ if (at_eof && DECL_COMDAT (decl) && DECL_EXTERNAL (decl)
+ && DECL_NOT_REALLY_EXTERN (decl))
+ DECL_EXTERNAL (decl) = false;
}
}
@@ -19159,6 +19168,10 @@ post_load_processing ()
gcc_checking_assert (DECL_MAYBE_IN_CHARGE_CDTOR_P (decl));
expand_or_defer_fn (decl);
+ /* As in module_state::read_cluster. */
+ if (at_eof && DECL_COMDAT (decl) && DECL_EXTERNAL (decl)
+ && DECL_NOT_REALLY_EXTERN (decl))
+ DECL_EXTERNAL (decl) = false;
}
cfun = old_cfun;