# This shell script emits a C file. -*- C -*- # Copyright (C) 2003-2019 Free Software Foundation, Inc. # # 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 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the # 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. # # This file is sourced from elf.em, and defines extra powerpc32-elf # specific routines. # fragment <next) { if (os->constraint == SPECIAL && strcmp (os->name, ".plt") == 0) { if (num_plt < 2) plt_os[num_plt] = os; ++num_plt; } if (os->constraint == SPECIAL && strcmp (os->name, ".got") == 0) { if (num_got < 2) got_os[num_got] = os; ++num_got; } } keep_new = new_plt == 1 ? 0 : -1; if (num_plt == 2) { plt_os[0]->constraint = keep_new; plt_os[1]->constraint = ~keep_new; } if (num_got == 2) { if (old_got) keep_new = -1; got_os[0]->constraint = keep_new; got_os[1]->constraint = ~keep_new; } } after_check_relocs_default (); } EOF fi fragment <sections; o != NULL; o = o->next) { if ((o->flags & (SEC_ALLOC | SEC_CODE)) != (SEC_ALLOC | SEC_CODE)) continue; if (o->rawsize == 0) continue; if (low > o->vma) low = o->vma; if (high < o->vma + o->rawsize - 1) high = o->vma + o->rawsize - 1; } if (high > low && high - low > (1 << 25) - 1) params.branch_trampolines = 1; } if (params.branch_trampolines || params.ppc476_workaround || params.pic_fixup > 0) ENABLE_RELAXATION; } /* Replaces default zero fill padding in executable sections with "ba 0" instructions. This works around the ppc476 icache bug if we have a function pointer tail call near the end of a page, some small amount of padding, then the function called at the beginning of the next page. If the "ba 0" is ever executed we should hit a segv, so it's almost as good as an illegal instruction (zero). */ static void no_zero_padding (lang_statement_union_type *l) { if (l->header.type == lang_padding_statement_enum && l->padding_statement.size != 0 && l->padding_statement.output_section != NULL && (l->padding_statement.output_section->flags & SEC_CODE) != 0 && l->padding_statement.fill->size == 0) { struct _ppc_fill_type { size_t size; unsigned char data[4]; }; static struct _ppc_fill_type fill_be = { 4, {0x48, 0, 0, 2} }; static struct _ppc_fill_type fill_le = { 4, {2, 0, 0, 0x48} }; if (bfd_big_endian (link_info.output_bfd)) l->padding_statement.fill = (struct _fill_type *) &fill_be; else l->padding_statement.fill = (struct _fill_type *) &fill_le; } } static void ppc_finish (void) { if (params.ppc476_workaround) lang_for_each_statement (no_zero_padding); if (!ppc_finish_symbols (&link_info)) einfo (_("%X%P: ppc_finish_symbols problem %E\n")); finish_default (); } EOF if grep -q 'ld_elf32_spu_emulation' ldemul-list.h; then fragment < 5) einfo (_("%F%P: invalid --plt-align `%s'\''\n"), optarg); params.plt_stub_align = val; } else params.plt_stub_align = 5; break; case OPTION_NO_PLT_ALIGN: params.plt_stub_align = 0; break; case OPTION_NO_INLINE_OPT: no_inline_opt = 1; break; case OPTION_OLD_GOT: old_got = 1; break; case OPTION_TRADITIONAL_FORMAT: notlsopt = 1; params.no_tls_get_addr_opt = 1; return FALSE; case OPTION_PPC476_WORKAROUND: params.ppc476_workaround = 1; if (optarg != NULL) { char *end; params.pagesize = strtoul (optarg, &end, 0); if (*end || (params.pagesize < 4096 && params.pagesize != 0) || params.pagesize != (params.pagesize & -params.pagesize)) einfo (_("%F%P: invalid pagesize `%s'\''\n"), optarg); } break; case OPTION_NO_PPC476_WORKAROUND: params.ppc476_workaround = 0; break; case OPTION_NO_PICFIXUP: params.pic_fixup = -1; break; case OPTION_VLE_RELOC_FIXUP: params.vle_reloc_fixup = 1; break; ' # Put these extra ppc32elf routines in ld_${EMULATION_NAME}_emulation # LDEMUL_CREATE_OUTPUT_SECTION_STATEMENTS=ppc_after_open_output if test -z "$VXWORKS_BASE_EM_FILE" ; then LDEMUL_AFTER_CHECK_RELOCS=ppc_after_check_relocs fi LDEMUL_BEFORE_ALLOCATION=ppc_before_allocation LDEMUL_FINISH=ppc_finish