xref: /aosp_15_r20/external/elfutils/backends/alpha_symbol.c (revision 7304104da70ce23c86437a01be71edd1a2d7f37e)
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