1 /* PPC specific symbolic name handling.
2 Copyright (C) 2004, 2005, 2007, 2014, 2015 Red Hat, Inc.
3 This file is part of elfutils.
4 Written by Ulrich Drepper <drepper@redhat.com>, 2004.
6 This file is free software; you can redistribute it and/or modify
7 it under the terms of either
9 * the GNU Lesser General Public License as published by the Free
10 Software Foundation; either version 3 of the License, or (at
11 your option) any later version
15 * the GNU General Public License as published by the Free
16 Software Foundation; either version 2 of the License, or (at
17 your option) any later version
19 or both in parallel, as here.
21 elfutils is distributed in the hope that it will be useful, but
22 WITHOUT ANY WARRANTY; without even the implied warranty of
23 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
24 General Public License for more details.
26 You should have received copies of the GNU General Public License and
27 the GNU Lesser General Public License along with this program. If
28 not, see <http://www.gnu.org/licenses/>. */
40 #include "libebl_CPU.h"
43 /* Check for the simple reloc types. */
45 ppc_reloc_simple_type (Ebl *ebl __attribute__ ((unused)), int type,
46 int *addsub __attribute__ ((unused)))
61 /* Check whether machine flags are valid. */
63 ppc_machine_flag_check (GElf_Word flags)
65 return ((flags &~ (EF_PPC_EMB
67 | EF_PPC_RELOCATABLE_LIB)) == 0);
72 ppc_dynamic_tag_name (int64_t tag, char *buf __attribute__ ((unused)),
73 size_t len __attribute__ ((unused)))
89 ppc_dynamic_tag_check (int64_t tag)
91 return (tag == DT_PPC_GOT
92 || tag == DT_PPC_OPT);
96 /* Look for DT_PPC_GOT. */
98 find_dyn_got (Elf *elf, GElf_Addr *addr)
101 if (elf_getphdrnum (elf, &phnum) != 0)
104 for (size_t i = 0; i < phnum; ++i)
107 GElf_Phdr *phdr = gelf_getphdr (elf, i, &phdr_mem);
108 if (phdr == NULL || phdr->p_type != PT_DYNAMIC)
111 Elf_Scn *scn = gelf_offscn (elf, phdr->p_offset);
113 GElf_Shdr *shdr = gelf_getshdr (scn, &shdr_mem);
114 Elf_Data *data = elf_getdata (scn, NULL);
115 if (shdr != NULL && shdr->sh_type == SHT_DYNAMIC && data != NULL
116 && shdr->sh_entsize != 0)
117 for (unsigned int j = 0; j < shdr->sh_size / shdr->sh_entsize; ++j)
120 GElf_Dyn *dyn = gelf_getdyn (data, j, &dyn_mem);
121 if (dyn != NULL && dyn->d_tag == DT_PPC_GOT)
123 *addr = dyn->d_un.d_ptr;
128 /* There is only one PT_DYNAMIC entry. */
136 /* Check whether given symbol's st_value and st_size are OK despite failing
139 ppc_check_special_symbol (Elf *elf, const GElf_Sym *sym,
140 const char *name, const GElf_Shdr *destshdr)
145 if (strcmp (name, "_GLOBAL_OFFSET_TABLE_") == 0)
147 /* In -msecure-plt mode, DT_PPC_GOT is present and must match. */
149 if (find_dyn_got (elf, &gotaddr))
150 return sym->st_value == gotaddr;
152 /* In -mbss-plt mode, any place in the section is valid. */
157 if (elf_getshdrstrndx (elf, &shstrndx) != 0)
159 const char *sname = elf_strptr (elf, shstrndx, destshdr->sh_name);
163 /* Small data area. Normally points to .sdata, in which case we
164 check it is at an offset of 0x8000. It might however fall in the
165 .data section, in which case we cannot check the offset. The
166 size always should be zero. */
167 if (strcmp (name, "_SDA_BASE_") == 0)
168 return (((strcmp (sname, ".sdata") == 0
169 && sym->st_value == destshdr->sh_addr + 0x8000)
170 || strcmp (sname, ".data") == 0)
171 && sym->st_size == 0);
173 if (strcmp (name, "_SDA2_BASE_") == 0)
174 return (strcmp (sname, ".sdata2") == 0
175 && sym->st_value == destshdr->sh_addr + 0x8000
176 && sym->st_size == 0);
182 /* Check if backend uses a bss PLT in this file. */
184 ppc_bss_plt_p (Elf *elf)
187 return ! find_dyn_got (elf, &addr);