1*7304104dSAndroid Build Coastguard Worker /* Alpha specific symbolic name handling.
2*7304104dSAndroid Build Coastguard Worker Copyright (C) 2002-2011 Red Hat, Inc.
3*7304104dSAndroid Build Coastguard Worker This file is part of elfutils.
4*7304104dSAndroid Build Coastguard Worker Written by Ulrich Drepper <[email protected]>, 2002.
5*7304104dSAndroid Build Coastguard Worker
6*7304104dSAndroid Build Coastguard Worker This file is free software; you can redistribute it and/or modify
7*7304104dSAndroid Build Coastguard Worker it under the terms of either
8*7304104dSAndroid Build Coastguard Worker
9*7304104dSAndroid Build Coastguard Worker * the GNU Lesser General Public License as published by the Free
10*7304104dSAndroid Build Coastguard Worker Software Foundation; either version 3 of the License, or (at
11*7304104dSAndroid Build Coastguard Worker your option) any later version
12*7304104dSAndroid Build Coastguard Worker
13*7304104dSAndroid Build Coastguard Worker or
14*7304104dSAndroid Build Coastguard Worker
15*7304104dSAndroid Build Coastguard Worker * the GNU General Public License as published by the Free
16*7304104dSAndroid Build Coastguard Worker Software Foundation; either version 2 of the License, or (at
17*7304104dSAndroid Build Coastguard Worker your option) any later version
18*7304104dSAndroid Build Coastguard Worker
19*7304104dSAndroid Build Coastguard Worker or both in parallel, as here.
20*7304104dSAndroid Build Coastguard Worker
21*7304104dSAndroid Build Coastguard Worker elfutils is distributed in the hope that it will be useful, but
22*7304104dSAndroid Build Coastguard Worker WITHOUT ANY WARRANTY; without even the implied warranty of
23*7304104dSAndroid Build Coastguard Worker MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
24*7304104dSAndroid Build Coastguard Worker General Public License for more details.
25*7304104dSAndroid Build Coastguard Worker
26*7304104dSAndroid Build Coastguard Worker You should have received copies of the GNU General Public License and
27*7304104dSAndroid Build Coastguard Worker the GNU Lesser General Public License along with this program. If
28*7304104dSAndroid Build Coastguard Worker not, see <http://www.gnu.org/licenses/>. */
29*7304104dSAndroid Build Coastguard Worker
30*7304104dSAndroid Build Coastguard Worker #ifdef HAVE_CONFIG_H
31*7304104dSAndroid Build Coastguard Worker # include <config.h>
32*7304104dSAndroid Build Coastguard Worker #endif
33*7304104dSAndroid Build Coastguard Worker
34*7304104dSAndroid Build Coastguard Worker #include <elf.h>
35*7304104dSAndroid Build Coastguard Worker #include <stddef.h>
36*7304104dSAndroid Build Coastguard Worker #include <string.h>
37*7304104dSAndroid Build Coastguard Worker
38*7304104dSAndroid Build Coastguard Worker #define BACKEND alpha_
39*7304104dSAndroid Build Coastguard Worker #include "libebl_CPU.h"
40*7304104dSAndroid Build Coastguard Worker
41*7304104dSAndroid Build Coastguard Worker
42*7304104dSAndroid Build Coastguard Worker const char *
alpha_dynamic_tag_name(int64_t tag,char * buf,size_t len)43*7304104dSAndroid Build Coastguard Worker alpha_dynamic_tag_name (int64_t tag, char *buf __attribute__ ((unused)),
44*7304104dSAndroid Build Coastguard Worker size_t len __attribute__ ((unused)))
45*7304104dSAndroid Build Coastguard Worker {
46*7304104dSAndroid Build Coastguard Worker switch (tag)
47*7304104dSAndroid Build Coastguard Worker {
48*7304104dSAndroid Build Coastguard Worker case DT_ALPHA_PLTRO:
49*7304104dSAndroid Build Coastguard Worker return "ALPHA_PLTRO";
50*7304104dSAndroid Build Coastguard Worker default:
51*7304104dSAndroid Build Coastguard Worker break;
52*7304104dSAndroid Build Coastguard Worker }
53*7304104dSAndroid Build Coastguard Worker return NULL;
54*7304104dSAndroid Build Coastguard Worker }
55*7304104dSAndroid Build Coastguard Worker
56*7304104dSAndroid Build Coastguard Worker bool
alpha_dynamic_tag_check(int64_t tag)57*7304104dSAndroid Build Coastguard Worker alpha_dynamic_tag_check (int64_t tag)
58*7304104dSAndroid Build Coastguard Worker {
59*7304104dSAndroid Build Coastguard Worker return tag == DT_ALPHA_PLTRO;
60*7304104dSAndroid Build Coastguard Worker }
61*7304104dSAndroid Build Coastguard Worker
62*7304104dSAndroid Build Coastguard Worker /* Check for the simple reloc types. */
63*7304104dSAndroid Build Coastguard Worker Elf_Type
alpha_reloc_simple_type(Ebl * ebl,int type,int * addsub)64*7304104dSAndroid Build Coastguard Worker alpha_reloc_simple_type (Ebl *ebl __attribute__ ((unused)), int type,
65*7304104dSAndroid Build Coastguard Worker int *addsub __attribute__ ((unused)))
66*7304104dSAndroid Build Coastguard Worker {
67*7304104dSAndroid Build Coastguard Worker switch (type)
68*7304104dSAndroid Build Coastguard Worker {
69*7304104dSAndroid Build Coastguard Worker case R_ALPHA_REFLONG:
70*7304104dSAndroid Build Coastguard Worker return ELF_T_WORD;
71*7304104dSAndroid Build Coastguard Worker case R_ALPHA_REFQUAD:
72*7304104dSAndroid Build Coastguard Worker return ELF_T_XWORD;
73*7304104dSAndroid Build Coastguard Worker default:
74*7304104dSAndroid Build Coastguard Worker return ELF_T_NUM;
75*7304104dSAndroid Build Coastguard Worker }
76*7304104dSAndroid Build Coastguard Worker }
77*7304104dSAndroid Build Coastguard Worker
78*7304104dSAndroid Build Coastguard Worker
79*7304104dSAndroid Build Coastguard Worker /* Check whether SHF_MASKPROC flags are valid. */
80*7304104dSAndroid Build Coastguard Worker bool
alpha_machine_section_flag_check(GElf_Xword sh_flags)81*7304104dSAndroid Build Coastguard Worker alpha_machine_section_flag_check (GElf_Xword sh_flags)
82*7304104dSAndroid Build Coastguard Worker {
83*7304104dSAndroid Build Coastguard Worker return (sh_flags &~ (SHF_ALPHA_GPREL)) == 0;
84*7304104dSAndroid Build Coastguard Worker }
85*7304104dSAndroid Build Coastguard Worker
86*7304104dSAndroid Build Coastguard Worker bool
alpha_check_special_section(Ebl * ebl,int ndx,const GElf_Shdr * shdr,const char * sname)87*7304104dSAndroid Build Coastguard Worker alpha_check_special_section (Ebl *ebl,
88*7304104dSAndroid Build Coastguard Worker int ndx __attribute__ ((unused)),
89*7304104dSAndroid Build Coastguard Worker const GElf_Shdr *shdr,
90*7304104dSAndroid Build Coastguard Worker const char *sname __attribute__ ((unused)))
91*7304104dSAndroid Build Coastguard Worker {
92*7304104dSAndroid Build Coastguard Worker if ((shdr->sh_flags
93*7304104dSAndroid Build Coastguard Worker & (SHF_WRITE | SHF_EXECINSTR)) == (SHF_WRITE | SHF_EXECINSTR)
94*7304104dSAndroid Build Coastguard Worker && shdr->sh_addr != 0)
95*7304104dSAndroid Build Coastguard Worker {
96*7304104dSAndroid Build Coastguard Worker /* This is ordinarily flagged, but is valid for an old-style PLT.
97*7304104dSAndroid Build Coastguard Worker
98*7304104dSAndroid Build Coastguard Worker Look for the SHT_DYNAMIC section and the DT_PLTGOT tag in it.
99*7304104dSAndroid Build Coastguard Worker Its d_ptr should match the .plt section's sh_addr. */
100*7304104dSAndroid Build Coastguard Worker
101*7304104dSAndroid Build Coastguard Worker Elf_Scn *scn = NULL;
102*7304104dSAndroid Build Coastguard Worker while ((scn = elf_nextscn (ebl->elf, scn)) != NULL)
103*7304104dSAndroid Build Coastguard Worker {
104*7304104dSAndroid Build Coastguard Worker GElf_Shdr scn_shdr;
105*7304104dSAndroid Build Coastguard Worker if (likely (gelf_getshdr (scn, &scn_shdr) != NULL)
106*7304104dSAndroid Build Coastguard Worker && scn_shdr.sh_type == SHT_DYNAMIC
107*7304104dSAndroid Build Coastguard Worker && scn_shdr.sh_entsize != 0)
108*7304104dSAndroid Build Coastguard Worker {
109*7304104dSAndroid Build Coastguard Worker GElf_Addr pltgot = 0;
110*7304104dSAndroid Build Coastguard Worker Elf_Data *data = elf_getdata (scn, NULL);
111*7304104dSAndroid Build Coastguard Worker if (data != NULL)
112*7304104dSAndroid Build Coastguard Worker for (size_t i = 0; i < data->d_size / scn_shdr.sh_entsize; ++i)
113*7304104dSAndroid Build Coastguard Worker {
114*7304104dSAndroid Build Coastguard Worker GElf_Dyn dyn;
115*7304104dSAndroid Build Coastguard Worker if (unlikely (gelf_getdyn (data, i, &dyn) == NULL))
116*7304104dSAndroid Build Coastguard Worker break;
117*7304104dSAndroid Build Coastguard Worker if (dyn.d_tag == DT_PLTGOT)
118*7304104dSAndroid Build Coastguard Worker pltgot = dyn.d_un.d_ptr;
119*7304104dSAndroid Build Coastguard Worker else if (dyn.d_tag == DT_ALPHA_PLTRO && dyn.d_un.d_val != 0)
120*7304104dSAndroid Build Coastguard Worker return false; /* This PLT should not be writable. */
121*7304104dSAndroid Build Coastguard Worker }
122*7304104dSAndroid Build Coastguard Worker return pltgot == shdr->sh_addr;
123*7304104dSAndroid Build Coastguard Worker }
124*7304104dSAndroid Build Coastguard Worker }
125*7304104dSAndroid Build Coastguard Worker }
126*7304104dSAndroid Build Coastguard Worker
127*7304104dSAndroid Build Coastguard Worker return false;
128*7304104dSAndroid Build Coastguard Worker }
129*7304104dSAndroid Build Coastguard Worker
130*7304104dSAndroid Build Coastguard Worker /* Check whether given symbol's st_value and st_size are OK despite failing
131*7304104dSAndroid Build Coastguard Worker normal checks. */
132*7304104dSAndroid Build Coastguard Worker bool
alpha_check_special_symbol(Elf * elf,const GElf_Sym * sym,const char * name,const GElf_Shdr * destshdr)133*7304104dSAndroid Build Coastguard Worker alpha_check_special_symbol (Elf *elf __attribute__ ((unused)),
134*7304104dSAndroid Build Coastguard Worker const GElf_Sym *sym __attribute__ ((unused)),
135*7304104dSAndroid Build Coastguard Worker const char *name,
136*7304104dSAndroid Build Coastguard Worker const GElf_Shdr *destshdr __attribute__ ((unused)))
137*7304104dSAndroid Build Coastguard Worker {
138*7304104dSAndroid Build Coastguard Worker if (name == NULL)
139*7304104dSAndroid Build Coastguard Worker return false;
140*7304104dSAndroid Build Coastguard Worker
141*7304104dSAndroid Build Coastguard Worker if (strcmp (name, "_GLOBAL_OFFSET_TABLE_") == 0)
142*7304104dSAndroid Build Coastguard Worker /* On Alpha any place in the section is valid. */
143*7304104dSAndroid Build Coastguard Worker return true;
144*7304104dSAndroid Build Coastguard Worker
145*7304104dSAndroid Build Coastguard Worker return false;
146*7304104dSAndroid Build Coastguard Worker }
147*7304104dSAndroid Build Coastguard Worker
148*7304104dSAndroid Build Coastguard Worker /* Check whether only valid bits are set on the st_other symbol flag.
149*7304104dSAndroid Build Coastguard Worker Standard ST_VISIBILITY have already been masked off. */
150*7304104dSAndroid Build Coastguard Worker bool
alpha_check_st_other_bits(unsigned char st_other)151*7304104dSAndroid Build Coastguard Worker alpha_check_st_other_bits (unsigned char st_other)
152*7304104dSAndroid Build Coastguard Worker {
153*7304104dSAndroid Build Coastguard Worker return ((((st_other & STO_ALPHA_STD_GPLOAD) == STO_ALPHA_NOPV)
154*7304104dSAndroid Build Coastguard Worker || ((st_other & STO_ALPHA_STD_GPLOAD) == STO_ALPHA_STD_GPLOAD))
155*7304104dSAndroid Build Coastguard Worker && (st_other &~ STO_ALPHA_STD_GPLOAD) == 0);
156*7304104dSAndroid Build Coastguard Worker }
157