summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorRoland McGrath <mcgrathr@google.com>2020-06-12 23:52:57 -0700
committerRoland McGrath <mcgrathr@google.com>2020-06-12 23:52:57 -0700
commit9242996ba250d0411efe6b00ee9a9893081d9137 (patch)
tree2d20f848d72886f5e8297c6b056a38c0952292bc
parent10059956eef2a85e0992d1df3aca304d380d0c43 (diff)
downloadbinutils-gdb-users/roland/ld-start-stop-visibility.tar.gz
gold, ld: Implement -z start-stop-visibility=... option.users/roland/ld-start-stop-visibility
gold/ * options.h (class General_options): Handle -z start-stop-visibility=. (General_options::start_stop_visibility): New public method. (General_options::start_stop_visibility_): New private member. * options.cc (General_options::General_options): Add initializer. (General_options::parse_start_stop_visibility): New function. * layout.cc (Layout::define_section_symbols): Use option setting. bfd/ * elflink.c (bfd_elf_define_start_stop): Use start_stop_visibility field of bfd_link_info. include/ * bfdlink.h (struct bfd_link_info): New field start_stop_visibility. ld/ * NEWS: Mention -z start-stop-visibility=... option for ELF. * ld.texi (Options): Document -z start-stop-visibility=... option. * ldmain.c (main): Initialize link_info.start_stop_visibility. * emultempl/elf.em (gld${EMULATION_NAME}_handle_option): Parse -z start-stop-visibility=... option.
-rw-r--r--bfd/ChangeLog7
-rw-r--r--bfd/elflink.c3
-rw-r--r--gold/ChangeLog10
-rw-r--r--gold/layout.cc5
-rw-r--r--gold/options.cc20
-rw-r--r--gold/options.h10
-rw-r--r--include/ChangeLog4
-rw-r--r--include/bfdlink.h7
-rw-r--r--ld/ChangeLog8
-rw-r--r--ld/NEWS3
-rw-r--r--ld/emultempl/elf.em15
-rw-r--r--ld/ld.texi13
-rw-r--r--ld/ldmain.c2
13 files changed, 100 insertions, 7 deletions
diff --git a/bfd/ChangeLog b/bfd/ChangeLog
index 6159d3aea3f..392a46281fa 100644
--- a/bfd/ChangeLog
+++ b/bfd/ChangeLog
@@ -1,3 +1,8 @@
+2020-06-12 Roland McGrath <mcgrathr@google.com>
+
+ * elflink.c (bfd_elf_define_start_stop): Use start_stop_visibility
+ field of bfd_link_info.
+
2020-06-11 Alan Modra <amodra@gmail.com>
PR 26107
@@ -330,7 +335,7 @@
(aout_link_add_symbols): e517df3dbf7 PR 19629 - Check for out of
range string table offsets.
531336e3a0b PR 20909 - Fix off-by-one error in check for an
- illegal string offset.
+ illegal string offset.
(aout_link_includes_newfunc): Add comment.
(pdp11_aout_link_input_section): ad756e3f9e6 - Return with an error
on unexpected relocation type rather than ASSERT.
diff --git a/bfd/elflink.c b/bfd/elflink.c
index 3e56a297f69..d7ed468045f 100644
--- a/bfd/elflink.c
+++ b/bfd/elflink.c
@@ -14829,7 +14829,8 @@ bfd_elf_define_start_stop (struct bfd_link_info *info,
else
{
if (ELF_ST_VISIBILITY (h->other) == STV_DEFAULT)
- h->other = (h->other & ~ELF_ST_VISIBILITY (-1)) | STV_PROTECTED;
+ h->other = ((h->other & ~ELF_ST_VISIBILITY (-1))
+ | info->start_stop_visibility);
if (was_dynamic)
bfd_elf_link_record_dynamic_symbol (info, h);
}
diff --git a/gold/ChangeLog b/gold/ChangeLog
index f3d37159362..5fa3b630801 100644
--- a/gold/ChangeLog
+++ b/gold/ChangeLog
@@ -1,3 +1,13 @@
+2020-06-12 Roland McGrath <mcgrathr@google.com>
+
+ Implement -z start-stop-visibility=... option.
+ * options.h (class General_options): Handle -z start-stop-visibility=.
+ (General_options::start_stop_visibility): New public method.
+ (General_options::start_stop_visibility_): New private member.
+ * options.cc (General_options::General_options): Add initializer.
+ (General_options::parse_start_stop_visibility): New function.
+ * layout.cc (Layout::define_section_symbols): Use option setting.
+
2020-06-06 Alan Modra <amodra@gmail.com>
* powerpc.cc: Update throughout for reloc renaming.
diff --git a/gold/layout.cc b/gold/layout.cc
index be437f3900e..bb7fd497b42 100644
--- a/gold/layout.cc
+++ b/gold/layout.cc
@@ -2474,6 +2474,7 @@ Layout::create_initial_dynamic_sections(Symbol_table* symtab)
void
Layout::define_section_symbols(Symbol_table* symtab)
{
+ const elfcpp::STV visibility = parameters->options().start_stop_visibility();
for (Section_list::const_iterator p = this->section_list_.begin();
p != this->section_list_.end();
++p)
@@ -2495,7 +2496,7 @@ Layout::define_section_symbols(Symbol_table* symtab)
0, // symsize
elfcpp::STT_NOTYPE,
elfcpp::STB_GLOBAL,
- elfcpp::STV_PROTECTED,
+ visibility,
0, // nonvis
false, // offset_is_from_end
true); // only_if_ref
@@ -2508,7 +2509,7 @@ Layout::define_section_symbols(Symbol_table* symtab)
0, // symsize
elfcpp::STT_NOTYPE,
elfcpp::STB_GLOBAL,
- elfcpp::STV_PROTECTED,
+ visibility,
0, // nonvis
true, // offset_is_from_end
true); // only_if_ref
diff --git a/gold/options.cc b/gold/options.cc
index 94867b361a2..9936619685f 100644
--- a/gold/options.cc
+++ b/gold/options.cc
@@ -760,6 +760,23 @@ General_options::parse_pop_state(const char*, const char*, Command_line*)
delete posdep;
}
+void
+General_options::parse_start_stop_visibility(char const*, char const* arg,
+ Command_line*)
+{
+ if (strcmp(arg, "default") == 0)
+ this->start_stop_visibility_ = elfcpp::STV_DEFAULT;
+ else if (strcmp(arg, "internal") == 0)
+ this->start_stop_visibility_ = elfcpp::STV_INTERNAL;
+ else if (strcmp(arg, "hidden") == 0)
+ this->start_stop_visibility_ = elfcpp::STV_HIDDEN;
+ else if (strcmp(arg, "protected") == 0)
+ this->start_stop_visibility_ = elfcpp::STV_PROTECTED;
+ else
+ gold_fatal(_("-z start-stop-visibility: must take one of the following "
+ "arguments: default, internal, hidden, protected"));
+}
+
} // End namespace gold.
namespace
@@ -997,7 +1014,8 @@ General_options::General_options()
fix_v4bx_(FIX_V4BX_NONE),
endianness_(ENDIANNESS_NOT_SET),
discard_locals_(DISCARD_SEC_MERGE),
- orphan_handling_enum_(ORPHAN_PLACE)
+ orphan_handling_enum_(ORPHAN_PLACE),
+ start_stop_visibility_(elfcpp::STV_PROTECTED)
{
// Turn off option registration once construction is complete.
gold::options::ready_to_register = false;
diff --git a/gold/options.h b/gold/options.h
index b2059d984cd..ed050c3b026 100644
--- a/gold/options.h
+++ b/gold/options.h
@@ -1500,6 +1500,10 @@ class General_options
N_("Don't mark variables read-only after relocation"));
DEFINE_uint64(stack_size, options::DASH_Z, '\0', 0,
N_("Set PT_GNU_STACK segment p_memsz to SIZE"), N_("SIZE"));
+ DEFINE_special(start_stop_visibility, options::DASH_Z, '\0',
+ N_("ELF symbol visibility for synthesized "
+ "__start_* and __stop_* symbols"),
+ N_("[default,internal,hidden,protected]"));
DEFINE_bool(text, options::DASH_Z, '\0', false,
N_("Do not permit relocations in read-only segments"),
N_("Permit relocations in read-only segments"));
@@ -1750,6 +1754,10 @@ class General_options
orphan_handling_enum() const
{ return this->orphan_handling_enum_; }
+ elfcpp::STV
+ start_stop_visibility() const
+ { return this->start_stop_visibility_; }
+
private:
// Don't copy this structure.
General_options(const General_options&);
@@ -1876,6 +1884,8 @@ class General_options
std::vector<Position_dependent_options*> options_stack_;
// Orphan handling option, decoded to an enum value.
Orphan_handling orphan_handling_enum_;
+ // Symbol visibility for __start_* / __stop_* magic symbols.
+ elfcpp::STV start_stop_visibility_;
};
// The position-dependent options. We use this to store the state of
diff --git a/include/ChangeLog b/include/ChangeLog
index 248866cbc94..f30e5e2a24a 100644
--- a/include/ChangeLog
+++ b/include/ChangeLog
@@ -1,3 +1,7 @@
+2020-06-12 Roland McGrath <mcgrathr@google.com>
+
+ * bfdlink.h (struct bfd_link_info): New field start_stop_visibility.
+
2020-06-12 Nelson Chu <nelson.chu@sifive.com>
* opcode/riscv-opc.h: Update the defined versions of CSR from
diff --git a/include/bfdlink.h b/include/bfdlink.h
index 34a0d2ec4ef..71634333830 100644
--- a/include/bfdlink.h
+++ b/include/bfdlink.h
@@ -542,10 +542,10 @@ struct bfd_link_info
Normally these optimizations are disabled by default but some targets
prefer to enable them by default. So this field is a tri-state variable.
The values are:
-
+
zero: Enable the optimizations (either from --relax being specified on
the command line or the backend's before_allocation emulation function.
-
+
positive: The user has requested that these optimizations be disabled.
(Via the --no-relax command line option).
@@ -649,6 +649,9 @@ struct bfd_link_info
/* May be used to set DT_FLAGS_1 for ELF. */
bfd_vma flags_1;
+ /* May be used to set ELF visibility for __start_* / __stop_. */
+ unsigned int start_stop_visibility;
+
/* Start and end of RELRO region. */
bfd_vma relro_start, relro_end;
diff --git a/ld/ChangeLog b/ld/ChangeLog
index 2a6f1b5489d..fc6a43b4b44 100644
--- a/ld/ChangeLog
+++ b/ld/ChangeLog
@@ -1,3 +1,11 @@
+2020-06-12 Roland McGrath <mcgrathr@google.com>
+
+ * NEWS: Mention -z start-stop-visibility=... option for ELF.
+ * ld.texi (Options): Document -z start-stop-visibility=... option.
+ * ldmain.c (main): Initialize link_info.start_stop_visibility.
+ * emultempl/elf.em (gld${EMULATION_NAME}_handle_option):
+ Parse -z start-stop-visibility=... option.
+
2020-06-11 Alan Modra <amodra@gmail.com>
* testsuite/ld-plugin/lto.exp (lto_link_tests): Move lto-6,
diff --git a/ld/NEWS b/ld/NEWS
index 485e1cf5b9a..6955937429b 100644
--- a/ld/NEWS
+++ b/ld/NEWS
@@ -25,6 +25,9 @@
searched relative to the directory of the linker script before other search
paths.
+* Add ELF linker command-line option `-z start-stop-visibility=...' to control
+ the visibility of synthetic `__start_SECNAME` and `__stop_SECNAME` symbols.
+
Changes in 2.34:
* The ld check for "PHDR segment not covered by LOAD segment" is more
diff --git a/ld/emultempl/elf.em b/ld/emultempl/elf.em
index c4979eb9538..c577e8b2e61 100644
--- a/ld/emultempl/elf.em
+++ b/ld/emultempl/elf.em
@@ -749,6 +749,21 @@ fragment <<EOF
{
link_info.flags_1 |= DF_1_GLOBAUDIT;
}
+ else if (CONST_STRNEQ (optarg, "start-stop-visibility="))
+ {
+ if (strcmp (optarg, "start-stop-visibility=default") == 0)
+ link_info.start_stop_visibility = STV_DEFAULT;
+ else if (strcmp (optarg, "start-stop-visibility=internal") == 0)
+ link_info.start_stop_visibility = STV_INTERNAL;
+ else if (strcmp (optarg, "start-stop-visibility=hidden") == 0)
+ link_info.start_stop_visibility = STV_HIDDEN;
+ else if (strcmp (optarg, "start-stop-visibility=protected") == 0)
+ link_info.start_stop_visibility = STV_PROTECTED;
+ else
+ einfo (_("%F%P: invalid visibility in \`-z %s'; "
+ "must be default, internal, hidden, or protected"),
+ optarg);
+ }
EOF
if test x"$GENERATE_SHLIB_SCRIPT" = xyes; then
diff --git a/ld/ld.texi b/ld/ld.texi
index bf474d4c62b..b89c1a57c0b 100644
--- a/ld/ld.texi
+++ b/ld/ld.texi
@@ -1373,6 +1373,19 @@ Specify a stack size for an ELF @code{PT_GNU_STACK} segment.
Specifying zero will override any default non-zero sized
@code{PT_GNU_STACK} segment creation.
+@item start-stop-visibility=@var{value}
+@cindex visibility
+@cindex ELF symbol visibility
+Specify the ELF symbol visibility for synthesized
+@code{__start_SECNAME} and @code{__stop_SECNAME} symbols (@pxref{Input
+Section Example}). @var{value} must be exactly @samp{default},
+@samp{internal}, @samp{hidden}, or @samp{protected}. If no @samp{-z
+start-stop-visibility} option is given, @samp{protected} is used for
+compatibility with historical practice. However, it's highly
+recommended to use @samp{-z start-stop-visibility=hidden} in new
+programs and shared libraries so that these symbols are not exported
+between shared objects, which is not usually what's intended.
+
@item text
@itemx notext
@itemx textoff
diff --git a/ld/ldmain.c b/ld/ldmain.c
index e2c559ea3e0..b0ce69f1186 100644
--- a/ld/ldmain.c
+++ b/ld/ldmain.c
@@ -27,6 +27,7 @@
#include "bfdlink.h"
#include "ctf-api.h"
#include "filenames.h"
+#include "elf/common.h"
#include "ld.h"
#include "ldmain.h"
@@ -307,6 +308,7 @@ main (int argc, char **argv)
#ifdef DEFAULT_NEW_DTAGS
link_info.new_dtags = DEFAULT_NEW_DTAGS;
#endif
+ link_info.start_stop_visibility = STV_PROTECTED;
ldfile_add_arch ("");
emulation = get_emulation (argc, argv);