blob: 5a7c4d7f02774778872656d20a2074466b8e4403 [file] [log] [blame]
# This shell script emits a C file. -*- C -*-
# It does some substitutions.
if [ -z "$MACHINE" ]; then
fragment <<EOF
/* This file is generated by a shell script. DO NOT EDIT! */
/* Solaris 2 emulation code for ${EMULATION_NAME}
Copyright (C) 2010-2023 Free Software Foundation, Inc.
Written by Rainer Orth <ro@CeBiTec.Uni-Bielefeld.DE>
This file is part of the GNU Binutils.
This program 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 3 of the License, or
(at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program; if not, write to the Free Software
Foundation, Inc., 51 Franklin Street - Fifth Floor, Boston,
MA 02110-1301, USA. */
/* The Solaris 2 ABI requires some global symbols to be present in the
.dynsym table of executables and shared objects. If generating a
versioned shared object, they must always be bound to the base version.
The Solaris 2 ABI also requires two local symbols to be emitted for
every executable and shared object.
Cf. Linker and Libraries Guide, Ch. 2, Link-Editor, Generating the Output
File, p.63. */
static void
elf_solaris2_before_allocation (void)
/* Global symbols required by the Solaris 2 ABI. */
static const char *global_syms[] = {
/* Local symbols required by the Solaris 2 ABI. Already emitted by
emulparams/ */
static const char *local_syms[] = {
const char **sym;
/* Do this for both executables and shared objects. */
if (!bfd_link_relocatable (&link_info)
&& is_elf_hash_table (link_info.hash))
for (sym = global_syms; *sym != NULL; sym++)
struct elf_link_hash_entry *h;
/* Lookup symbol. */
h = elf_link_hash_lookup (elf_hash_table (&link_info), *sym,
false, false, false);
if (h == NULL)
/* Undo the hiding done by _bfd_elf_define_linkage_sym. */
h->forced_local = 0;
h->other &= ~STV_HIDDEN;
/* Emit it into the .dynamic section, too. */
bfd_elf_link_record_dynamic_symbol (&link_info, h);
for (sym = local_syms; *sym != NULL; sym++)
struct elf_link_hash_entry *h;
/* Lookup symbol. */
h = elf_link_hash_lookup (elf_hash_table (&link_info), *sym,
false, false, false);
if (h == NULL)
/* Turn it local. */
h->forced_local = 1;
/* Type should be STT_OBJECT, not STT_NOTYPE. */
h->type = STT_OBJECT;
/* Only do this if emitting a shared object and versioning is in place. */
if (bfd_link_dll (&link_info)
&& ((link_info.version_info != NULL
&& link_info.version_info->name[0] != '\0')
|| link_info.create_default_symver))
struct bfd_elf_version_expr *globals = NULL, *locals = NULL;
struct bfd_elf_version_tree *basever;
const char *soname;
for (sym = global_syms; *sym != NULL; sym++)
/* Create a version pattern for this symbol. Some of them start
off as local, others as global, so try both. */
globals = lang_new_vers_pattern (globals, *sym, NULL, true);
locals = lang_new_vers_pattern (locals, *sym, NULL, true);
/* New version node for those symbols. */
basever = lang_new_vers_node (globals, locals);
/* The version name matches what bfd_elf_size_dynamic_sections uses
for the base version. */
soname = bfd_elf_get_dt_soname (link_info.output_bfd);
if (soname == NULL)
soname = lbasename (bfd_get_filename (link_info.output_bfd));
/* Register the node. */
lang_register_vers_node (soname, basever, NULL);
/* Enforce base version. The encoded vd_ndx is vernum + 1. */
basever->vernum = 0;
gld${EMULATION_NAME}_before_allocation ();